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

smithy4s.http.HttpUnaryServerCodecs.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.server.UnaryServerCodecs
import smithy4s.codecs.{BlobEncoder, BlobDecoder}
import smithy4s.capability.MonadThrowLike
import smithy4s.codecs.Writer
import smithy4s.schema.CachedSchemaCompiler
import smithy4s.schema.OperationSchema
import smithy4s.codecs.Decoder
import smithy4s.codecs.PayloadError

// scalafmt: {maxColumn = 120}
object HttpUnaryServerCodecs {

  def builder[F[_]](implicit F: MonadThrowLike[F]): Builder[F, HttpRequest[Blob], HttpResponse[Blob]] =
    HttpUnaryClientCodecsBuilderImpl[F, HttpRequest[Blob], HttpResponse[Blob]](
      baseResponse = _ => F.raiseError(new Exception("Undefined base response")),
      requestBodyDecoders = BlobDecoder.noop,
      successResponseBodyEncoders = BlobEncoder.noop,
      errorResponseBodyEncoders = BlobEncoder.noop,
      metadataEncoders = None,
      metadataDecoders = None,
      rawStringsAndBlobPayloads = false,
      writeEmptyStructs = _ => false,
      errorTypeHeaders = Nil,
      responseMediaType = "text/plain",
      requestTransformation = F.pure(_),
      responseTransformation = F.pure(_)
    )

  trait Builder[F[_], Request, Response] {
    def withBaseResponse(f: OperationSchema[_, _, _, _, _] => F[HttpResponse[Blob]]): Builder[F, Request, Response]
    def withBodyDecoders(decoders: BlobDecoder.Compiler): Builder[F, Request, Response]
    def withSuccessBodyEncoders(decoders: BlobEncoder.Compiler): Builder[F, Request, Response]
    def withErrorBodyEncoders(encoders: BlobEncoder.Compiler): 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 withResponseMediaType(mediaType: String): Builder[F, Request, Response]
    def withErrorTypeHeaders(headerNames: String*): Builder[F, Request, Response]
    def withRequestTransformation[Request0](f: Request0 => F[Request]): Builder[F, Request0, Response]
    def withResponseTransformation[Response1](f: Response => F[Response1]): Builder[F, Request, Response1]
    def build(): UnaryServerCodecs.Make[F, Request, Response]
  }

  private case class HttpUnaryClientCodecsBuilderImpl[F[_], Request, Response](
      baseResponse: OperationSchema[_, _, _, _, _] => F[HttpResponse[Blob]],
      requestBodyDecoders: BlobDecoder.Compiler,
      successResponseBodyEncoders: BlobEncoder.Compiler,
      errorResponseBodyEncoders: BlobEncoder.Compiler,
      metadataEncoders: Option[Metadata.Encoder.Compiler],
      metadataDecoders: Option[Metadata.Decoder.Compiler],
      rawStringsAndBlobPayloads: Boolean,
      writeEmptyStructs: Schema[_] => Boolean,
      responseMediaType: String,
      errorTypeHeaders: List[String],
      requestTransformation: Request => F[HttpRequest[Blob]],
      responseTransformation: HttpResponse[Blob] => F[Response]
  )(implicit F: MonadThrowLike[F])
      extends Builder[F, Request, Response] {

    def withBaseResponse(f: OperationSchema[_, _, _, _, _] => F[HttpResponse[Blob]]): Builder[F, Request, Response] =
      copy(baseResponse = f)
    def withBodyDecoders(decoders: BlobDecoder.Compiler): Builder[F, Request, Response] =
      copy(requestBodyDecoders = decoders)
    def withSuccessBodyEncoders(encoders: BlobEncoder.Compiler): Builder[F, Request, Response] =
      copy(successResponseBodyEncoders = encoders)
    def withErrorBodyEncoders(encoders: BlobEncoder.Compiler): Builder[F, Request, Response] =
      copy(errorResponseBodyEncoders = encoders)
    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 withResponseMediaType(mediaType: String): Builder[F, Request, Response] =
      copy(responseMediaType = mediaType)
    def withErrorTypeHeaders(headerNames: String*): Builder[F, Request, Response] =
      copy(errorTypeHeaders = headerNames.toList)

    def withRequestTransformation[Request0](f: Request0 => F[Request]): Builder[F, Request0, Response] =
      copy(requestTransformation = f.andThen(F.flatMap(_)(requestTransformation)))
    def withResponseTransformation[Response1](f: Response => F[Response1]): Builder[F, Request, Response1] =
      copy(responseTransformation = responseTransformation.andThen(F.flatMap(_)(f)))

    def build(): UnaryServerCodecs.Make[F, Request, Response] = {
      val setBody: HttpResponse.Writer[Blob, Blob] = Writer.lift((res, blob) => res.copy(body = blob))
      val setBodyK = smithy4s.codecs.Encoder.pipeToWriterK[HttpResponse[Blob], Blob](setBody)

      val mediaTypeWriters = new CachedSchemaCompiler.Uncached[HttpResponse.Writer[Blob, *]] {
        def fromSchema[A](schema: Schema[A]): HttpResponse.Writer[Blob, A] = {
          val mt = if (rawStringsAndBlobPayloads) {
            HttpMediaType.fromSchema(schema).map(_.value).getOrElse(responseMediaType)
          } else responseMediaType
          new HttpResponse.Writer[Blob, A] {
            def write(request: HttpResponse[Blob], value: A): HttpResponse[Blob] =
              if (request.body.isEmpty) request
              else request.withContentType(mt)
          }
        }
      }

      def responseEncoders(blobEncoders: BlobEncoder.Compiler) = {
        val httpBodyWriters: CachedSchemaCompiler[HttpResponse.Writer[Blob, *]] = if (rawStringsAndBlobPayloads) {
          val finalBodyEncoders = CachedSchemaCompiler
            .getOrElse(smithy4s.codecs.StringAndBlobCodecs.encoders, successResponseBodyEncoders)
          finalBodyEncoders.mapK(setBodyK)
        } else successResponseBodyEncoders.mapK(setBodyK)

        val httpMediaWriters: CachedSchemaCompiler[HttpResponse.Writer[Blob, *]] =
          Writer.combineCompilers(httpBodyWriters, mediaTypeWriters)

        metadataEncoders match {
          case Some(mEncoders) =>
            HttpResponse.Encoder.restSchemaCompiler(mEncoders, httpMediaWriters, writeEmptyStructs)
          case None => httpMediaWriters
        }
      }

      val inputDecoders: CachedSchemaCompiler[HttpRequest.Decoder[F, Blob, *]] = {
        val httpBodyDecoders: CachedSchemaCompiler[Decoder[F, Blob, *]] = {
          val decoders: BlobDecoder.Compiler = if (rawStringsAndBlobPayloads) {
            CachedSchemaCompiler
              .getOrElse(smithy4s.codecs.StringAndBlobCodecs.decoders, requestBodyDecoders)
          } else requestBodyDecoders
          decoders.mapK(
            Decoder
              .of[Blob]
              .liftPolyFunction(
                MonadThrowLike
                  .liftEitherK[F, PayloadError]
                  .andThen(HttpContractError.fromPayloadErrorK[F])
              )
          )
        }

        metadataDecoders match {
          case Some(mDecoders) => HttpRequest.Decoder.restSchemaCompiler(mDecoders, httpBodyDecoders, None)
          case None            => httpBodyDecoders.mapK(HttpRequest.extractBody[F, Blob])
        }
      }

      val outputEncoders = responseEncoders(successResponseBodyEncoders)
      val errorEncoders = responseEncoders(errorResponseBodyEncoders)
      val httpContractErrorWriters = errorEncoders.fromSchema(HttpContractError.schema)

      new UnaryServerCodecs.Make[F, Request, Response] {

        private val inputDecoderCache: inputDecoders.Cache = inputDecoders.createCache()
        private val outputEncoderCache: outputEncoders.Cache = outputEncoders.createCache()

        def apply[I, E, O, SI, SO](
            endpoint: OperationSchema[I, E, O, SI, SO]
        ): UnaryServerCodecs[F, Request, Response, I, E, O] = {
          val outputW = endpoint.hints.get(smithy.api.Http) match {
            case Some(http) =>
              val preProcess: HttpResponse[Blob] => HttpResponse[Blob] =
                _.withStatusCode(http.code)
              // TODO : add unit-tests for this
              val postProcessResponse: HttpResponse[Blob] => HttpResponse[Blob] =
                if (http.code == 204 || http.method.value.toLowerCase == "head")
                  _.withBody(Blob.empty)
                else identity
              outputEncoders
                .fromSchema(endpoint.output, outputEncoderCache)
                .compose(preProcess)
                .andThen(postProcessResponse)
            case None => outputEncoders.fromSchema(endpoint.output, outputEncoderCache)
          }
          val errorW = HttpResponse.Encoder.forError(errorTypeHeaders, endpoint.error, errorEncoders)
          val inputDecoder: HttpRequest.Decoder[F, Blob, I] =
            inputDecoders.fromSchema(endpoint.input, inputDecoderCache)
          val base = baseResponse(endpoint)

          def encodeOutput(o: O) = F.map(base)(outputW.write(_, o))
          def encodeError(e: E) = F.map(base)(errorW.write(_, e))
          def httpContractErrorEncoder(e: HttpContractError) =
            F.map(base)(httpContractErrorWriters.write(_, e).withStatusCode(400))
          def throwableEncoders(throwable: Throwable): F[HttpResponse[Blob]] =
            throwable match {
              case e: HttpContractError => httpContractErrorEncoder(e)
              case e                    => F.raiseError(e)
            }

          new UnaryServerCodecs(inputDecoder.decode, encodeError, throwableEncoders, encodeOutput)
            .transformRequest(requestTransformation)
            .transformResponse(responseTransformation)
        }

      }
    }
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy