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

sttp.tapir.internal.AnnotationsMacros.scala Maven / Gradle / Ivy

The newest version!
package sttp.tapir.internal

import sttp.tapir._
import sttp.tapir.CodecFormat.TextPlain
import sttp.tapir.typelevel.ParamConcat
import sttp.model.{QueryParams, Header, StatusCode}
import sttp.model.headers.{Cookie, CookieWithMeta, CookieValueWithMeta}

import scala.collection.mutable
import scala.quoted.*

private[tapir] class AnnotationsMacros[T <: Product: Type](using q: Quotes) {
  import quotes.reflect.*

  private val caseClass = new CaseClass[q.type, T](using summon[Type[T]], q)

  def deriveEndpointInputImpl: Expr[EndpointInput[T]] = {
    // the path inputs must be defined in the order as they appear in the argument to @endpointInput
    val pathSegments = caseClass
      .extractOptStringArgFromAnnotation(endpointInputAnnotationSymbol)
      .flatten
      .map { case path =>
        val pathWithoutLeadingSlash = if (path.startsWith("/")) path.drop(1) else path
        val result = if (pathWithoutLeadingSlash.endsWith("/")) pathWithoutLeadingSlash.dropRight(1) else pathWithoutLeadingSlash
        if (result.length == 0) Nil else result.split("/").toList
      }
      .getOrElse(Nil)

    val fieldsWithIndex = caseClass.fields.zipWithIndex
    val inputIdxToFieldIdx = mutable.Map.empty[Int, Int]

    val pathInputs = pathSegments.map { segment =>
      if (segment.startsWith("{") && segment.endsWith("}")) {
        val fieldName = segment.drop(1).dropRight(1)
        fieldsWithIndex.find { case (f, _) => f.name == fieldName && f.annotated(pathAnnotationSymbol) } match {
          case Some((field, idx)) =>
            field.tpe.asType match
              case '[f] =>
                inputIdxToFieldIdx += (inputIdxToFieldIdx.size -> idx)
                wrapInput[f](field, makePathInput[f](field))
          case None =>
            report.errorAndAbort(s"${caseClass.name}.${fieldName} must have the @path annotation, as it is referenced from @endpointInput.")
        }
      } else {
        makeFixedPath(segment)
      }
    }

    val (pathFields, nonPathFields) = fieldsWithIndex.partition((f, _) => f.annotated(pathAnnotationSymbol))

    if (inputIdxToFieldIdx.size != pathFields.size) {
      report.errorAndAbort(s"Not all fields of ${caseClass.name} annotated with @path are captured in the path in @endpointInput.")
    }

    val nonPathInputs = nonPathFields.map { case (field, fieldIdx) =>
      field.tpe.asType match
        case '[f] =>
          val input: Expr[EndpointInput.Single[f]] = field
            .extractOptStringArgFromAnnotation(queryAnnotationSymbol)
            .map(makeQueryInput[f](field))
            .orElse(field.extractOptStringArgFromAnnotation(headerAnnotationSymbol).map(makeHeaderIO[f](field)))
            .orElse(field.extractOptStringArgFromAnnotation(cookieAnnotationSymbol).map(makeCookieInput[f](field)))
            .orElse(field.annotation(bodyAnnotationSymbol).map(makeBodyIO[f](field)))
            .orElse(if (field.annotated(paramsAnnotationSymbol)) Some(makeQueryParamsInput[f](field)) else None)
            .orElse(if (field.annotated(headersAnnotationSymbol)) Some(makeHeadersIO[f](field)) else None)
            .orElse(if (field.annotated(cookiesAnnotationSymbol)) Some(makeCookiesIO[f](field)) else None)
            .orElse(
              field
                .annotation(bearerAnnotationSymbol)
                .map(makeBearerAuthInput[f](field, field.annotation(securitySchemeNameAnnotationSymbol), _))
            )
            .orElse(
              field
                .annotation(basicAnnotationSymbol)
                .map(makeBasicAuthInput[f](field, field.annotation(securitySchemeNameAnnotationSymbol), _))
            )
            .getOrElse {
              report.errorAndAbort(
                s"All fields of ${caseClass.name} must be annotated with one of the annotations from sttp.tapir.annotations. No annotations for field: ${field.name}."
              )
            }
          inputIdxToFieldIdx += (inputIdxToFieldIdx.size -> fieldIdx)
          wrapInput[f](field, input)
    }

    val result = (pathInputs ::: nonPathInputs).map(_.asTerm).reduceLeft { (left, right) =>
      (left.tpe.asType, right.tpe.asType) match
        case ('[EndpointInput[l]], '[EndpointInput[r]]) =>
          // we need to summon explicitly to get the instance for the "real" l, r types
          val concat = Expr.summon[ParamConcat[l, r]].get
          concat.asTerm.tpe.asType match {
            case '[ParamConcat.Aux[`l`, `r`, lr]] =>
              '{
                ${ left.asExprOf[EndpointInput[l]] }
                  .and(${ right.asExprOf[EndpointInput[r]] })(using $concat.asInstanceOf[ParamConcat.Aux[l, r, lr]])
              }.asTerm
          }
    }

    result.tpe.asType match {
      case '[EndpointInput[r]] =>
        '{
          ${ result.asExprOf[EndpointInput[r]] }.map[T](${ mapToTargetFunc[r](inputIdxToFieldIdx) })(${
            mapFromTargetFunc[r](inputIdxToFieldIdx)
          })
        }
    }
  }

  def deriveEndpointOutputImpl: Expr[EndpointOutput[T]] = {
    val fieldsWithIndex = caseClass.fields.zipWithIndex

    val outputs = fieldsWithIndex.map { case (field, fieldIdx) =>
      field.tpe.asType match
        case '[f] =>
          val output: Expr[EndpointOutput.Single[f]] = field
            .extractOptStringArgFromAnnotation(headerAnnotationSymbol)
            .map(makeHeaderIO[f](field))
            .orElse(field.extractOptStringArgFromAnnotation(setCookieAnnotationSymbol).map(makeSetCookieOutput[f](field)))
            .orElse(field.annotation(bodyAnnotationSymbol).map(makeBodyIO[f](field)))
            .orElse(if (field.annotated(statusCodeAnnotationSymbol)) Some(makeStatusCodeOutput[f](field)) else None)
            .orElse(if (field.annotated(headersAnnotationSymbol)) Some(makeHeadersIO[f](field)) else None)
            .orElse(if (field.annotated(cookiesAnnotationSymbol)) Some(makeCookiesIO[f](field)) else None)
            .orElse(if (field.annotated(setCookiesAnnotationSymbol)) Some(makeSetCookiesOutput[f](field)) else None)
            .getOrElse {
              report.errorAndAbort(
                s"All fields of ${caseClass.name} must be annotated with one of the annotations from sttp.tapir.annotations. No annotations for field: ${field.name}."
              )
            }
          '{ ${ addMetadataFromAnnotations[f](field, output) }.asInstanceOf[EndpointOutput.Single[f]] }
    }

    val result = outputs.map(_.asTerm).reduceLeft { (left, right) =>
      (left.tpe.asType, right.tpe.asType) match
        case ('[EndpointOutput[l]], '[EndpointOutput[r]]) =>
          // we need to summon explicitly to get the instance for the "real" l, r types
          val concat = Expr.summon[ParamConcat[l, r]].get
          concat.asTerm.tpe.asType match {
            case '[ParamConcat.Aux[`l`, `r`, lr]] =>
              '{
                ${ left.asExprOf[EndpointOutput[l]] }
                  .and(${ right.asExprOf[EndpointOutput[r]] })(using $concat.asInstanceOf[ParamConcat.Aux[l, r, lr]])
              }.asTerm
          }
    }

    result.tpe.asType match {
      case '[EndpointOutput[r]] =>
        val inputIdxToFieldIdx = mutable.Map((0 until outputs.size).map(i => (i, i)): _*)
        '{
          ${ result.asExprOf[EndpointOutput[r]] }.map[T](${ mapToTargetFunc[r](inputIdxToFieldIdx) })(${
            mapFromTargetFunc[r](inputIdxToFieldIdx)
          })
        }
    }
  }

  // annotation symbols
  private val endpointInputAnnotationSymbol = TypeTree.of[EndpointIO.annotations.endpointInput].tpe.typeSymbol
  private val pathAnnotationSymbol = TypeTree.of[EndpointIO.annotations.path].tpe.typeSymbol
  private val queryAnnotationSymbol = TypeTree.of[EndpointIO.annotations.query].tpe.typeSymbol
  private val headerAnnotationSymbol = TypeTree.of[EndpointIO.annotations.header].tpe.typeSymbol
  private val headersAnnotationSymbol = TypeTree.of[EndpointIO.annotations.headers].tpe.typeSymbol
  private val cookieAnnotationSymbol = TypeTree.of[EndpointIO.annotations.cookie].tpe.typeSymbol
  private val cookiesAnnotationSymbol = TypeTree.of[EndpointIO.annotations.cookies].tpe.typeSymbol
  private val setCookieAnnotationSymbol = TypeTree.of[EndpointIO.annotations.setCookie].tpe.typeSymbol
  private val setCookiesAnnotationSymbol = TypeTree.of[EndpointIO.annotations.setCookies].tpe.typeSymbol
  private val paramsAnnotationSymbol = TypeTree.of[EndpointIO.annotations.params].tpe.typeSymbol
  private val bearerAnnotationSymbol = TypeTree.of[EndpointIO.annotations.bearer].tpe.typeSymbol
  private val basicAnnotationSymbol = TypeTree.of[EndpointIO.annotations.basic].tpe.typeSymbol
  private val apikeyAnnotationSymbol = TypeTree.of[EndpointIO.annotations.apikey].tpe.typeSymbol
  private val securitySchemeNameAnnotationSymbol = TypeTree.of[EndpointIO.annotations.securitySchemeName].tpe.typeSymbol
  private val bodyAnnotationSymbol = TypeTree.of[EndpointIO.annotations.body].tpe.typeSymbol
  private val statusCodeAnnotationSymbol = TypeTree.of[EndpointIO.annotations.statusCode].tpe.typeSymbol

  private val descriptionAnnotationSymbol = TypeTree.of[EndpointIO.annotations.description].tpe.typeSymbol
  private val exampleAnnotationSymbol = TypeTree.of[EndpointIO.annotations.example].tpe.typeSymbol
  private val customiseAnnotationSymbol = TypeTree.of[EndpointIO.annotations.customise].tpe.typeSymbol

  // schema symbols
  private val schemaDescriptionAnnotationSymbol = TypeTree.of[Schema.annotations.description].tpe.typeSymbol
  private val schemaEncodedExampleAnnotationSymbol = TypeTree.of[Schema.annotations.encodedExample].tpe.typeSymbol
  private val schemaDefaultAnnotationSymbol = TypeTree.of[Schema.annotations.default].tpe.typeSymbol
  private val schemaFormatAnnotationSymbol = TypeTree.of[Schema.annotations.format].tpe.typeSymbol
  private val schemaDeprecatedAnnotationSymbol = TypeTree.of[Schema.annotations.deprecated].tpe.typeSymbol
  private val schemaValidateAnnotationSymbol = TypeTree.of[Schema.annotations.validate].tpe.typeSymbol

  // util
  private def summonCodec[L: Type, H: Type, CF <: CodecFormat: Type](field: CaseClassField[q.type, T]): Expr[Codec[L, H, CF]] =
    Expr.summon[Codec[L, H, CF]].getOrElse {
      report.errorAndAbort(
        s"Cannot summon codec from: ${Type.show[L]}, to: ${Type.show[H]}, formatted as: ${Type.show[CF]}, for field: ${field.name}."
      )
    }

  private def summonMultipartCodec[H: Type](field: CaseClassField[q.type, T]): Expr[MultipartCodec[H]] =
    Expr.summon[MultipartCodec[H]].getOrElse {
      report.errorAndAbort(
        s"Cannot summon multipart codec for type: ${Type.show[H]}, for field: ${field.name}."
      )
    }

  // inputs & outputs
  private def makeQueryInput[f: Type](field: CaseClassField[q.type, T])(altName: Option[String]): Expr[EndpointInput.Basic[f]] = {
    val name = Expr(altName.getOrElse(field.name))
    '{ query[f]($name)(using ${ summonCodec[List[String], f, TextPlain](field) }) }
  }

  private def makeHeaderIO[f: Type](field: CaseClassField[q.type, T])(altName: Option[String]): Expr[EndpointIO.Basic[f]] = {
    val name = Expr(altName.getOrElse(field.name))
    '{ header[f]($name)(using ${ summonCodec[List[String], f, TextPlain](field) }) }
  }

  private def makeCookieInput[f: Type](field: CaseClassField[q.type, T])(altName: Option[String]): Expr[EndpointInput.Basic[f]] = {
    val name = Expr(altName.getOrElse(field.name))
    '{ cookie[f]($name)(using ${ summonCodec[Option[String], f, TextPlain](field) }) }
  }

  private def makeBodyIO[f: Type](field: CaseClassField[q.type, T])(ann: Term): Expr[EndpointIO.Basic[f]] = {
    val annExp = ann.asExprOf[EndpointIO.annotations.body[_, _]]

    ann.tpe.asType match {
      case '[EndpointIO.annotations.multipartBody] =>
        val codec = summonMultipartCodec[f](field)
        '{
          EndpointIO.Body[Seq[RawPart], f](
            $codec.rawBodyType,
            $codec.codec,
            EndpointIO.Info.empty
          )
        }
      case '[EndpointIO.annotations.body[bt, cf]] =>
        '{
          EndpointIO.Body[bt, f](
            $annExp.bodyType.asInstanceOf[RawBodyType[bt]],
            ${ summonCodec[bt, f, cf](field) }.asInstanceOf[Codec[bt, f, CodecFormat]],
            EndpointIO.Info.empty
          )
        }
    }
  }

  private def makeQueryParamsInput[f: Type](field: CaseClassField[q.type, T]): Expr[EndpointInput.Basic[f]] = {
    summon[Type[f]] match {
      case '[QueryParams] => '{ queryParams.asInstanceOf[EndpointInput.Basic[f]] }
      case _              => report.errorAndAbort(annotationErrorMsg[f, QueryParams]("@params", field))
    }
  }

  private def makeHeadersIO[f: Type](field: CaseClassField[q.type, T]): Expr[EndpointIO.Basic[f]] =
    summon[Type[f]] match {
      case '[List[Header]] => '{ headers.asInstanceOf[EndpointIO.Basic[f]] }
      case _               => report.errorAndAbort(annotationErrorMsg[f, List[Header]]("@headers", field))
    }

  private def makeCookiesIO[f: Type](field: CaseClassField[q.type, T]): Expr[EndpointIO.Basic[f]] =
    summon[Type[f]] match {
      case '[List[Cookie]] => '{ cookies.asInstanceOf[EndpointIO.Basic[f]] }
      case _               => report.errorAndAbort(annotationErrorMsg[f, List[Cookie]]("@cookies", field))
    }

  private def makePathInput[f: Type](field: CaseClassField[q.type, T]): Expr[EndpointInput.Basic[f]] = {
    val name = Expr(field.name)
    '{ path[f]($name)(using ${ summonCodec[String, f, TextPlain](field) }) }
  }

  private def makeFixedPath(segment: String): Expr[EndpointInput.Basic[Unit]] = '{ stringToPath(${ Expr(segment) }) }

  private def makeStatusCodeOutput[f: Type](field: CaseClassField[q.type, T]): Expr[EndpointOutput.Basic[f]] = {
    summon[Type[f]] match {
      case '[StatusCode] => '{ statusCode.asInstanceOf[EndpointOutput.Basic[f]] }
      case _             => report.errorAndAbort(annotationErrorMsg[f, StatusCode]("@statusCode", field))
    }
  }

  private def makeSetCookieOutput[f: Type](field: CaseClassField[q.type, T])(altName: Option[String]): Expr[EndpointOutput.Basic[f]] = {
    val name = Expr(altName.getOrElse(field.name))
    summon[Type[f]] match {
      case '[CookieValueWithMeta] => '{ setCookie($name).asInstanceOf[EndpointOutput.Basic[f]] }
      case _                      => report.errorAndAbort(annotationErrorMsg[f, CookieValueWithMeta]("@setCookie", field))
    }
  }

  private def makeSetCookiesOutput[f: Type](field: CaseClassField[q.type, T]): Expr[EndpointOutput.Basic[f]] = {
    summon[Type[f]] match {
      case '[List[CookieWithMeta]] => '{ setCookies.asInstanceOf[EndpointOutput.Basic[f]] }
      case _                       => report.errorAndAbort(annotationErrorMsg[f, List[CookieWithMeta]]("@setCookies", field))
    }
  }

  private def annotationErrorMsg[f: Type, e: Type](annName: String, field: CaseClassField[q.type, T]) =
    s"Annotation $annName on ${field.name} can be applied only to a field with type ${Type.show[e]}, but got: ${Type.show[f]}"

  // auth inputs
  private def makeBearerAuthInput[f: Type](
      field: CaseClassField[q.type, T],
      schemeName: Option[Term],
      auth: Term
  ): Expr[EndpointInput.Single[f]] =
    setSecuritySchemeName(
      '{
        TapirAuth.bearer(${ auth.asExprOf[EndpointIO.annotations.bearer] }.challenge)(${
          summonCodec[List[String], f, CodecFormat.TextPlain](field)
        })
      },
      schemeName
    )

  private def makeBasicAuthInput[f: Type](
      field: CaseClassField[q.type, T],
      schemeName: Option[Term],
      auth: Term
  ): Expr[EndpointInput.Single[f]] =
    setSecuritySchemeName(
      '{
        TapirAuth.basic(${ auth.asExprOf[EndpointIO.annotations.basic] }.challenge)(${
          summonCodec[List[String], f, CodecFormat.TextPlain](field)
        })
      },
      schemeName
    )

  // schema & auth wrappers
  private def wrapInput[f: Type](field: CaseClassField[q.type, T], input: Expr[EndpointInput.Single[f]]): Expr[EndpointInput.Single[f]] = {
    val input2 = '{ ${ addMetadataFromAnnotations[f](field, input) }.asInstanceOf[EndpointInput.Single[f]] }
    wrapWithApiKey(input2, field.annotation(apikeyAnnotationSymbol), field.annotation(securitySchemeNameAnnotationSymbol))
  }

  private def wrapWithApiKey[f: Type](
      input: Expr[EndpointInput.Single[f]],
      apikey: Option[Term],
      schemeName: Option[Term]
  ): Expr[EndpointInput.Single[f]] =
    apikey
      .map(ak =>
        setSecuritySchemeName(
          '{
            EndpointInput.Auth(
              $input,
              ${ ak.asExprOf[EndpointIO.annotations.apikey] }.challenge,
              EndpointInput.AuthType.ApiKey(),
              EndpointInput.AuthInfo.Empty
            )
          },
          schemeName
        )
      )
      .getOrElse(input)

  private def setSecuritySchemeName[f: Type](
      auth: Expr[EndpointInput.Auth[f, _]],
      schemeName: Option[Term]
  ): Expr[EndpointInput.Single[f]] =
    schemeName
      .map(s => '{ $auth.securitySchemeName(${ s.asExprOf[EndpointIO.annotations.securitySchemeName] }.name) })
      .getOrElse(auth)

  private def addMetadataFromAnnotations[f: Type](
      field: CaseClassField[q.type, T],
      transput: Expr[EndpointTransput[f]]
  ): Expr[EndpointTransput[f]] = {
    val transformations: List[Expr[EndpointTransput[f]] => Expr[EndpointTransput[f]]] = List(
      t =>
        field
          .extractStringArgFromAnnotation(descriptionAnnotationSymbol)
          .map(d => addMetadataToAtom(field, t, '{ i => i.description(${ Expr(d) }) }))
          .getOrElse(t),
      t =>
        field
          .extractTreeFromAnnotation(exampleAnnotationSymbol)
          .map(e => addMetadataToAtom(field, t, '{ i => i.example(${ e.asExprOf[f] }) }))
          .getOrElse(t),
      t =>
        field
          .extractStringArgFromAnnotation(schemaDescriptionAnnotationSymbol)
          .map(d => addMetadataToAtom(field, t, '{ i => i.schema(_.description(${ Expr(d) })) }))
          .getOrElse(t),
      t =>
        field
          .extractTreeFromAnnotation(schemaEncodedExampleAnnotationSymbol)
          .map(e => addMetadataToAtom(field, t, '{ i => i.schema(_.encodedExample(${ e.asExprOf[Any] })) }))
          .getOrElse(t),
      t =>
        field
          .extractFirstTreeArgFromAnnotation(schemaDefaultAnnotationSymbol)
          .map(d => addMetadataToAtom(field, t, '{ i => i.default(${ d.asExprOf[f] }) }))
          .getOrElse(t),
      t =>
        field
          .extractStringArgFromAnnotation(schemaFormatAnnotationSymbol)
          .map(format => addMetadataToAtom(field, t, '{ i => i.schema(_.format(${ Expr(format) })) }))
          .getOrElse(t),
      t =>
        if (field.annotated(schemaDeprecatedAnnotationSymbol)) then addMetadataToAtom(field, t, '{ i => i.deprecated() })
        else t,
      t =>
        field
          .extractTreeFromAnnotation(schemaValidateAnnotationSymbol)
          .map(v => addMetadataToAtom(field, t, '{ i => i.validate(${ v.asExprOf[Validator[f]] }) }))
          .getOrElse(t),
      t =>
        field
          .extractTreeFromAnnotation(customiseAnnotationSymbol)
          .map(f => '{ AnnotationsMacros.customise($t, ${ f.asExprOf[EndpointTransput[_] => EndpointTransput[_]] }) })
          .getOrElse(t)
    )

    transformations.foldLeft(transput)((t, f) => f(t))
  }

  private def addMetadataToAtom[f: Type](
      field: CaseClassField[q.type, T],
      transput: Expr[EndpointTransput[f]],
      f: Expr[EndpointTransput.Atom[f] => Any]
  ): Expr[EndpointTransput[f]] = {
    transput.asTerm.tpe.asType match {
      // TODO: also handle Auth inputs, by adding the description to the nested input
      case '[EndpointTransput.Atom[`f`]] =>
        '{ $f($transput.asInstanceOf[EndpointTransput.Atom[f]]).asInstanceOf[EndpointTransput.Atom[f]] }
      case t =>
        report.errorAndAbort(
          s"Schema metadata can only be added to basic inputs/outputs, but got: ${Type.show(using t)}, on field: ${field.name}"
        )
    }
  }

  // mapping functions
  // A - single arg or tuple, T - target type
  private def mapToTargetFunc[A: Type](inputIdxToFieldIdx: mutable.Map[Int, Int]): Expr[A => T] = {
    if (inputIdxToFieldIdx.size > 1) {
      val fieldIdxToInputIdx = inputIdxToFieldIdx.map(_.swap)
      def ctorArgs(tupleExpr: Expr[A]) = (0 until fieldIdxToInputIdx.size).map { idx =>
        Select.unique(tupleExpr.asTerm, s"_${fieldIdxToInputIdx(idx) + 1}").asExprOf[Any]
      }

      '{ (a: A) => ${ caseClass.instanceFromValues(ctorArgs('a)) } }
    } else {
      '{ (a: A) => ${ caseClass.instanceFromValues('{ List(a) }) } }
    }
  }

  private def mapFromTargetFunc[A: Type](inputIdxToFieldIdx: mutable.Map[Int, Int]): Expr[T => A] = {
    def tupleArgs(tExpr: Expr[T]): Seq[Expr[Any]] = (0 until inputIdxToFieldIdx.size).map { idx =>
      val field = caseClass.fields(inputIdxToFieldIdx(idx))
      Select.unique(tExpr.asTerm, field.name).asExprOf[Any]
    }

    if (inputIdxToFieldIdx.size == 0) {
      '{ (t: T) => ().asInstanceOf[A] }
    } else if (inputIdxToFieldIdx.size > 1) {
      '{ (t: T) => ${ Expr.ofTupleFromSeq(tupleArgs('t)) }.asInstanceOf[A] }
    } else {
      '{ (t: T) => ${ tupleArgs('t).head }.asInstanceOf[A] }
    }
  }
}

// TODO: make private[tapir] once Scala3 compilation is fixed
object AnnotationsMacros:
  // we assume that the customisation function doesn't return a value of a different type
  def customise[X <: EndpointTransput[_]](i: X, f: EndpointTransput[_] => EndpointTransput[_]): X = f(i).asInstanceOf[X]




© 2015 - 2025 Weber Informatics LLC | Privacy Policy