smithy4s.http.HttpUnaryClientCodecs.scala Maven / Gradle / Ivy
/*
* Copyright 2021-2024 Disney Streaming
*
* Licensed under the Tomorrow Open Source Technology License, Version 1.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://disneystreaming.github.io/TOST-1.0.txt
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package smithy4s
package http
import smithy4s.client.UnaryClientCodecs
import smithy4s.codecs.{BlobEncoder, BlobDecoder}
import smithy4s.codecs.{Decoder}
import smithy4s.codecs.Writer
import smithy4s.codecs.PayloadError
import smithy4s.schema.CachedSchemaCompiler
import smithy4s.schema.OperationSchema
import smithy4s.capability.MonadThrowLike
import smithy4s.kinds.PolyFunction5
// scalafmt: { maxColumn = 120 }
object HttpUnaryClientCodecs {
def builder[F[_]](implicit F: MonadThrowLike[F]): Builder[F, HttpRequest[Blob], HttpResponse[Blob]] =
HttpUnaryClientCodecsBuilderImpl[F, HttpRequest[Blob], HttpResponse[Blob]](
operationPreprocessor = PolyFunction5.identity,
baseRequest = _ => F.raiseError(new Exception("Undefined base request")),
requestBodyEncoders = BlobEncoder.noop,
successResponseBodyDecoders = BlobDecoder.noop,
errorResponseBodyDecoders = BlobDecoder.noop,
errorDiscriminator = _ => F.pure(HttpDiscriminator.Undetermined),
metadataEncoders = None,
metadataDecoders = None,
rawStringsAndBlobPayloads = false,
writeEmptyStructs = _ => false,
requestMediaType = "text/plain",
requestTransformation = F.pure(_),
responseTransformation = F.pure(_),
hostPrefixInjection = true
)
trait Builder[F[_], Request, Response] {
def withOperationPreprocessor(fk: PolyFunction5[OperationSchema, OperationSchema]): Builder[F, Request, Response]
def withBaseRequest(f: OperationSchema[_, _, _, _, _] => F[HttpRequest[Blob]]): Builder[F, Request, Response]
def withBodyEncoders(encoders: BlobEncoder.Compiler): Builder[F, Request, Response]
def withSuccessBodyDecoders(decoders: BlobDecoder.Compiler): Builder[F, Request, Response]
def withErrorBodyDecoders(decoders: BlobDecoder.Compiler): Builder[F, Request, Response]
def withErrorDiscriminator(f: HttpResponse[Blob] => F[HttpDiscriminator]): Builder[F, Request, Response]
def withMetadataEncoders(encoders: Metadata.Encoder.Compiler): Builder[F, Request, Response]
def withMetadataDecoders(decoders: Metadata.Decoder.Compiler): Builder[F, Request, Response]
def withRawStringsAndBlobsPayloads: Builder[F, Request, Response]
def withWriteEmptyStructs(cond: Schema[_] => Boolean): Builder[F, Request, Response]
def withRequestMediaType(mediaType: String): Builder[F, Request, Response]
def withRequestTransformation[Request1](f: Request => F[Request1]): Builder[F, Request1, Response]
def withResponseTransformation[Response0](f: Response0 => F[Response]): Builder[F, Request, Response0]
def withHostPrefixInjection(enabled: Boolean): Builder[F, Request, Response]
def build(): UnaryClientCodecs.Make[F, Request, Response]
}
private case class HttpUnaryClientCodecsBuilderImpl[F[_], Request, Response](
operationPreprocessor: PolyFunction5[OperationSchema, OperationSchema],
baseRequest: OperationSchema[_, _, _, _, _] => F[HttpRequest[Blob]],
requestBodyEncoders: BlobEncoder.Compiler,
successResponseBodyDecoders: BlobDecoder.Compiler,
errorResponseBodyDecoders: BlobDecoder.Compiler,
errorDiscriminator: HttpResponse[Blob] => F[HttpDiscriminator],
metadataEncoders: Option[Metadata.Encoder.Compiler],
metadataDecoders: Option[Metadata.Decoder.Compiler],
rawStringsAndBlobPayloads: Boolean,
writeEmptyStructs: Schema[_] => Boolean,
requestMediaType: String,
requestTransformation: HttpRequest[Blob] => F[Request],
responseTransformation: Response => F[HttpResponse[Blob]],
hostPrefixInjection: Boolean
)(implicit F: MonadThrowLike[F])
extends Builder[F, Request, Response] {
def withOperationPreprocessor(fk: PolyFunction5[OperationSchema, OperationSchema]): Builder[F, Request, Response] =
copy(operationPreprocessor = fk)
def withBaseRequest(f: OperationSchema[_, _, _, _, _] => F[HttpRequest[Blob]]): Builder[F, Request, Response] =
copy(baseRequest = f)
def withBodyEncoders(encoders: BlobEncoder.Compiler): Builder[F, Request, Response] =
copy(requestBodyEncoders = encoders)
def withSuccessBodyDecoders(decoders: BlobDecoder.Compiler): Builder[F, Request, Response] =
copy(successResponseBodyDecoders = decoders)
def withErrorBodyDecoders(decoders: BlobDecoder.Compiler): Builder[F, Request, Response] =
copy(errorResponseBodyDecoders = decoders)
def withErrorDiscriminator(f: HttpResponse[Blob] => F[HttpDiscriminator]): Builder[F, Request, Response] =
copy(errorDiscriminator = f)
def withMetadataEncoders(encoders: Metadata.Encoder.Compiler): Builder[F, Request, Response] =
copy(metadataEncoders = Some(encoders))
def withMetadataDecoders(decoders: Metadata.Decoder.Compiler): Builder[F, Request, Response] =
copy(metadataDecoders = Some(decoders))
def withRawStringsAndBlobsPayloads: Builder[F, Request, Response] =
copy(rawStringsAndBlobPayloads = true)
def withWriteEmptyStructs(cond: Schema[_] => Boolean): Builder[F, Request, Response] =
copy(writeEmptyStructs = cond)
def withRequestMediaType(mediaType: String): Builder[F, Request, Response] =
copy(requestMediaType = mediaType)
def withRequestTransformation[Request1](f: Request => F[Request1]): Builder[F, Request1, Response] =
copy(requestTransformation = requestTransformation.andThen(F.flatMap(_)(f)))
def withResponseTransformation[Response0](f: Response0 => F[Response]): Builder[F, Request, Response0] =
copy(responseTransformation = f.andThen(F.flatMap(_)(responseTransformation)))
def withHostPrefixInjection(enabled: Boolean): Builder[F, Request, Response] = copy(hostPrefixInjection = enabled)
def build(): UnaryClientCodecs.Make[F, Request, Response] = {
val setBody: HttpRequest.Writer[Blob, Blob] = Writer.lift((req, blob) => req.copy(body = blob))
val setBodyK = smithy4s.codecs.Encoder.pipeToWriterK[HttpRequest[Blob], Blob](setBody)
val mediaTypeWriters = new CachedSchemaCompiler.Uncached[HttpRequest.Writer[Blob, *]] {
def fromSchema[A](schema: Schema[A]): HttpRequest.Writer[Blob, A] = {
val maybeRawMediaType = HttpMediaType.fromSchema(schema).map(_.value)
maybeRawMediaType match {
case Some(mt) =>
new HttpRequest.Writer[Blob, A] {
def write(request: HttpRequest[Blob], value: A): HttpRequest[Blob] =
request.withContentType(mt)
}
case None =>
new HttpRequest.Writer[Blob, A] {
def write(request: HttpRequest[Blob], value: A): HttpRequest[Blob] =
if (request.body.isEmpty) request
else request.withContentType(requestMediaType)
}
}
}
}
val httpBodyWriters: CachedSchemaCompiler[HttpRequest.Writer[Blob, *]] = if (rawStringsAndBlobPayloads) {
val finalBodyEncoders = CachedSchemaCompiler
.getOrElse(smithy4s.codecs.StringAndBlobCodecs.encoders, requestBodyEncoders)
finalBodyEncoders.mapK(setBodyK)
} else requestBodyEncoders.mapK(setBodyK)
val httpMediaWriter: CachedSchemaCompiler[HttpRequest.Writer[Blob, *]] =
Writer.combineCompilers(httpBodyWriters, mediaTypeWriters)
def responseDecoders(blobDecoders: BlobDecoder.Compiler) = {
val httpBodyDecoders: CachedSchemaCompiler[Decoder[F, Blob, *]] = {
val decoders: BlobDecoder.Compiler = if (rawStringsAndBlobPayloads) {
CachedSchemaCompiler.getOrElse(smithy4s.codecs.StringAndBlobCodecs.decoders, blobDecoders)
} else blobDecoders
decoders.mapK(
Decoder
.of[Blob]
.liftPolyFunction(
MonadThrowLike
.liftEitherK[F, PayloadError]
.andThen(HttpContractError.fromPayloadErrorK[F])
)
)
}
metadataDecoders match {
case Some(mDecoders) => HttpResponse.Decoder.restSchemaCompiler(mDecoders, httpBodyDecoders, None)
case None => httpBodyDecoders.mapK(HttpResponse.extractBody[F, Blob])
}
}
val inputEncoders = metadataEncoders match {
case Some(mEncoders) =>
HttpRequest.Writer.restSchemaCompiler(mEncoders, httpMediaWriter, writeEmptyStructs)
case None => httpMediaWriter
}
val outputDecoders = responseDecoders(successResponseBodyDecoders)
val errorDecoders = responseDecoders(errorResponseBodyDecoders)
new UnaryClientCodecs.Make[F, Request, Response] {
private val inputEncoderCache: inputEncoders.Cache = inputEncoders.createCache()
private val outputDecoderCache: outputDecoders.Cache = outputDecoders.createCache()
def apply[I, E, O, SI, SO](
originalEndpoint: OperationSchema[I, E, O, SI, SO]
): UnaryClientCodecs[F, Request, Response, I, E, O] = {
val endpoint = operationPreprocessor(originalEndpoint)
val inputWriter: HttpRequest.Writer[Blob, I] =
HttpEndpoint.cast(endpoint).toOption match {
case Some(httpEndpoint) => {
val httpInputEncoder =
HttpRequest.Writer.fromHttpEndpoint[Blob, I](httpEndpoint)
val requestEncoder =
inputEncoders.fromSchema(endpoint.input, inputEncoderCache)
httpInputEncoder.combine(requestEncoder)
}
case None => inputEncoders.fromSchema(endpoint.input, inputEncoderCache)
}
val prefixedInputWriter: HttpRequest.Writer[Blob, I] =
if (hostPrefixInjection) inputWriter.combine(HttpRequest.Writer.hostPrefix(endpoint)) else inputWriter
val inputEncoder = (i: I) => F.map(baseRequest(endpoint))(prefixedInputWriter.write(_, i))
val outputDecoder: HttpResponse.Decoder[F, Blob, O] =
outputDecoders.fromSchema(endpoint.output, outputDecoderCache)
def toStrict(blob: Blob): F[(Blob, Blob)] = F.pure((blob, blob))
val errorDecoder: HttpResponse.Decoder[F, Blob, Throwable] =
HttpResponse.Decoder.forErrorAsThrowable(
endpoint.error,
errorDecoders,
errorDiscriminator,
toStrict
)
new UnaryClientCodecs(inputEncoder, errorDecoder.decode, outputDecoder.decode)
.transformRequest[Request](requestTransformation)
.transformResponse[Response](responseTransformation)
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy