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

smithy4s.http.HttpUnaryServerRouter.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.kinds._
import smithy4s.server.UnaryServerCodecs
import smithy4s.capability.MonadThrowLike
import scala.annotation.nowarn

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

  /**
    * Creates a router in the form isomorphic to `Kleisli[OptionT[F, *], Request, Response]`, which
    * is useful for integration with frameworks such as Http4s.
    */
  def apply[Alg[_[_, _, _, _, _]], F[_], Request, Response](
      service: smithy4s.Service[Alg]
  )(
      impl: service.Impl[F],
      makeServerCodecs: UnaryServerCodecs.Make[F, Request, Response],
      endpointMiddleware: Endpoint.Middleware[Request => F[Response]],
      getMethod: Request => HttpMethod,
      getUri: Request => HttpUri,
      addDecodedPathParams: (Request, PathParams) => Request
  )(implicit F: MonadThrowLike[F]): Request => Option[F[Response]] = {
    new KleisliRouter[Alg, service.Operation, F, Request, Response](
      service,
      service.toPolyFunction[smithy4s.kinds.Kind1[F]#toKind5](impl),
      makeServerCodecs,
      endpointMiddleware,
      getMethod,
      getUri,
      addDecodedPathParams
    )
  }

  /**
    * Creates a router in the form of a partial function, which is useful for integration with frameworks
    * such as Play.
    */
  def partialFunction[Alg[_[_, _, _, _, _]], F[_], RequestHead, Request, Response](
      service: smithy4s.Service[Alg]
  )(
      impl: service.Impl[F],
      makeServerCodecs: UnaryServerCodecs.Make[F, Request, Response],
      endpointMiddleware: Endpoint.Middleware[Request => F[Response]],
      getMethod: RequestHead => HttpMethod,
      getUri: RequestHead => HttpUri,
      addDecodedPathParams: (Request, PathParams) => Request
  )(implicit F: MonadThrowLike[F]): PartialFunction[RequestHead, Request => F[Response]] = {
    new PartialFunctionRouter[Alg, service.Operation, F, RequestHead, Request, Response](
      service,
      service.toPolyFunction[smithy4s.kinds.Kind1[F]#toKind5](impl),
      makeServerCodecs,
      endpointMiddleware,
      getMethod,
      getUri,
      addDecodedPathParams
    )
  }

  private class KleisliRouter[Alg[_[_, _, _, _, _]], Op[_, _, _, _, _], F[_], Request, Response](
      service: smithy4s.Service.Aux[Alg, Op],
      impl: FunctorInterpreter[Op, F],
      makeServerCodecs: UnaryServerCodecs.Make[F, Request, Response],
      endpointMiddleware: Endpoint.Middleware[Request => F[Response]],
      getMethod: Request => HttpMethod,
      getUri: Request => HttpUri,
      addDecodedPathParams: (Request, PathParams) => Request
  )(implicit F: MonadThrowLike[F])
      extends (Request => Option[F[Response]]) {

    @nowarn
    private final case class HttpEndpointHandler(
        httpEndpoint: HttpEndpoint[_],
        handler: Request => F[Response]
    )

    def apply(request: Request): Option[F[Response]] = {
      val method = getMethod(request)
      val path = getUri(request).path
      perMethodEndpoint.get(method) match {
        case Some(httpUnaryEndpoints) =>
          val maybeMatched =
            httpUnaryEndpoints.iterator
              .map(ep => (ep.handler, ep.httpEndpoint.matches(path)))
              .find(_._2.isDefined)
          maybeMatched.flatMap {
            case (handler, Some(pathParams)) =>
              val amendedRequest = addDecodedPathParams(request, pathParams)
              Some(handler(amendedRequest))
            case (_, None) => None
          }

        case None => None
      }

    }

    private def makeHttpEndpointHandler[I, E, O, SI, SO](
        endpoint: service.Endpoint[I, E, O, SI, SO]
    ): Either[HttpEndpoint.HttpEndpointError, HttpEndpointHandler] = {
      HttpEndpoint.cast(endpoint.schema).map { httpEndpoint =>
        val handler = smithy4s.server.UnaryServerEndpoint(
          impl,
          endpoint,
          makeServerCodecs(endpoint.schema),
          endpointMiddleware.prepare(service)(endpoint)
        )
        HttpEndpointHandler(httpEndpoint, handler)
      }
    }

    private val httpEndpointHandlers: List[HttpEndpointHandler] =
      service.endpoints.toList
        .map { makeHttpEndpointHandler(_) }
        .collect { case Right(endpointWrapper) => endpointWrapper }

    private val perMethodEndpoint: Map[HttpMethod, List[HttpEndpointHandler]] =
      httpEndpointHandlers.groupBy(_.httpEndpoint.method)

  }

  // scalafmt: {maxColumn = 120}
  private class PartialFunctionRouter[Alg[_[_, _, _, _, _]], Op[_, _, _, _, _], F[_], RequestHead, Request, Response](
      service: smithy4s.Service.Aux[Alg, Op],
      impl: FunctorInterpreter[Op, F],
      makeServerCodecs: UnaryServerCodecs.Make[F, Request, Response],
      endpointMiddleware: Endpoint.Middleware[Request => F[Response]],
      getMethod: RequestHead => HttpMethod,
      getUri: RequestHead => HttpUri,
      addDecodedPathParams: (Request, PathParams) => Request
  )(implicit F: MonadThrowLike[F])
      extends PartialFunction[RequestHead, Request => F[Response]] {

    @nowarn
    private final case class HttpEndpointHandler(
        httpEndpoint: HttpEndpoint[_],
        handler: Request => F[Response]
    )

    def isDefinedAt(requestHead: RequestHead): Boolean = {
      val method = getMethod(requestHead)
      val pathSegments = getUri(requestHead).path
      perMethodEndpoint.get(method).exists { httpUnaryEndpoints =>
        httpUnaryEndpoints.iterator.exists(_.httpEndpoint.matches(pathSegments).isDefined)
      }
    }

    def apply(requestHead: RequestHead): Request => F[Response] = {
      val method = getMethod(requestHead)
      val pathSegments = getUri(requestHead).path
      val httpUnaryEndpoints = perMethodEndpoint(method)
      httpUnaryEndpoints.iterator
        .map(ep => (ep.handler, ep.httpEndpoint.matches(pathSegments)))
        .collectFirst { case (handler, Some(pathParams)) =>
          (request: Request) => handler(addDecodedPathParams(request, pathParams))
        }
        .get
    }

    private def makeHttpEndpointHandler[I, E, O, SI, SO](
        endpoint: service.Endpoint[I, E, O, SI, SO]
    ): Either[HttpEndpoint.HttpEndpointError, HttpEndpointHandler] = {
      HttpEndpoint.cast(endpoint.schema).map { httpEndpoint =>
        val handler = smithy4s.server.UnaryServerEndpoint(
          impl,
          endpoint,
          makeServerCodecs(endpoint.schema),
          endpointMiddleware.prepare(service)(endpoint)
        )
        HttpEndpointHandler(httpEndpoint, handler)
      }
    }

    private val httpEndpointHandlers: List[HttpEndpointHandler] =
      service.endpoints.toList
        .map { makeHttpEndpointHandler(_) }
        .collect { case Right(endpointWrapper) => endpointWrapper }

    private val perMethodEndpoint: Map[HttpMethod, List[HttpEndpointHandler]] =
      httpEndpointHandlers.groupBy(_.httpEndpoint.method)

  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy