All Downloads are FREE. Search and download functionalities are using the official Maven repository.

smithy4s.http.HttpUnaryClientCodecs.scala Maven / Gradle / Ivy

There is a newer version: 0.19.0-41-91762fb
Show newest version
/*
 *  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