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

eaming.smithy4s.smithy4s-decline_3.0.18.13.source-code.Smithy4sCli.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.decline

import cats.implicits._
import cats.MonadThrow
import cats.effect.std.Console
import com.monovore.decline.Opts
import com.monovore.decline.Command
import smithy.api.Documentation
import smithy.api.ExternalDocumentation
import smithy.api.Http
import smithy4s.Service
import smithy4s.decline.core._
import smithy4s.decline.util.PrinterApi
import smithy4s.http.HttpEndpoint
import smithy4s.kinds._
import commons._

final case class Entrypoint[Alg[_[_, _, _, _, _]], F[_]](
    interpreter: FunctorAlgebra[Alg, F],
    printerApi: PrinterApi[F]
)

/** Main entrypoint to Smithy4s CLIs.
  * @param mainOpts
  *   Opts providing an interpreter to execute commands, and printer to use when displaying the
  *   input/output/errors. See [[smithy4s.decline.util.PrinterApi]] for default options
  * @param service
  *   The service to build a client call for
  */
class Smithy4sCli[Alg[_[_, _, _, _, _]], F[_]: MonadThrow](
    mainOpts: Opts[Entrypoint[Alg, F]]
)(implicit service: Service[Alg]) {

  private def protocolSpecificHelp(
      endpoint: service.Endpoint[_, _, _, _, _]
  ): List[String] =
    HttpEndpoint
      .cast(endpoint.schema)
      .toOption
      .map { httpEndpoint =>
        val path = endpoint.hints
          .get(Http)
          .map(_.uri.value)
          .getOrElse(
            throw new Exception(
              "HTTP hint missing in a HTTP endpoint! This is probably a bug in Smithy4s codegen."
            )
          )

        s"HTTP ${httpEndpoint.method.showUppercase} $path"
      }
      .toList

  private def makeHelpBlocks(
      endpoint: service.Endpoint[_, _, _, _, _]
  ): List[String] =
    protocolSpecificHelp(endpoint) ++
      endpoint.hints.get[Documentation].map(_.value) ++
      endpoint.hints
        .get[ExternalDocumentation]
        .map(
          _.value
            .map { case (url, description) => s"$url: $description" }
            .mkString("\n")
        )

  private def endpointSubcommand[I, E, O](
      endpoint: service.Endpoint[I, E, O, _, _]
  ): Opts[F[Unit]] = {

    def compileToOpts[A](schema: smithy4s.Schema[A]): Opts[A] =
      schema.compile[Opts](OptsVisitor)

    val inputOpts: Opts[I] = compileToOpts(endpoint.input)

    Opts
      .subcommand(
        name = toKebabCase(endpoint.name),
        help = makeHelpBlocks(endpoint).mkString_("\n\n")
      ) {
        (
          inputOpts,
          mainOpts
        ).mapN { (input, entrypoint) =>
          val printers = entrypoint.printerApi
          val printer = printers.printer(endpoint)
          val polyFunction =
            service.toPolyFunction[Kind1[F]#toKind5](entrypoint.interpreter)
          val FO = printer
            .printInput(input)
            .flatMap(_ => polyFunction(endpoint.wrap(input)))

          FO.flatMap(printer.printOutput)
            .onError {
              case e if endpoint.error.flatMap(_.liftError(e)).nonEmpty =>
                printer
                  .printError(e)
            }
        }
      }
  }

  def opts: Opts[F[Unit]] =
    service.endpoints.toVector.foldMapK(endpointSubcommand(_))

  def command: Command[F[Unit]] = {
    Command(
      toKebabCase(service.id.name),
      header = service.hints
        .get[Documentation]
        .map(_.value)
        .getOrElse(s"Command line interface for ${service.id.show}"),
      helpFlag = true
    )(opts)
  }

}

object Smithy4sCli {
  def standalone[Alg[_[_, _, _, _, _]], F[_]: Console: MonadThrow](
      impl: Opts[FunctorAlgebra[Alg, F]]
  )(implicit service: Service[Alg]) = {
    new Smithy4sCli[Alg, F](
      (
        impl,
        PrinterApi.opts.default[F]()
      ).mapN(Entrypoint.apply[Alg, F])
    )
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy