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

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

There is a newer version: 1.11.11
Show newest version
package sttp.tapir.internal

import sttp.tapir.Schema.annotations.format
import sttp.tapir.SchemaAnnotations

import scala.quoted.*

private[tapir] object SchemaAnnotationsMacro {
  def derived[T: Type](using q: Quotes): Expr[SchemaAnnotations[T]] = {
    import q.reflect.*

    val EnumerationValue = TypeTree.of[scala.Enumeration#Value].tpe
    val DescriptionAnn = TypeTree.of[sttp.tapir.Schema.annotations.description].tpe
    val EncodedExampleAnn = TypeTree.of[sttp.tapir.Schema.annotations.encodedExample].tpe
    val DefaultAnn = TypeTree.of[sttp.tapir.Schema.annotations.default[_]].tpe
    val FormatAnn = TypeTree.of[sttp.tapir.Schema.annotations.format].tpe
    val DeprecatedAnn = TypeTree.of[sttp.tapir.Schema.annotations.deprecated].tpe
    val HiddenAnn = TypeTree.of[sttp.tapir.Schema.annotations.hidden].tpe
    val EncodedNameAnn = TypeTree.of[sttp.tapir.Schema.annotations.encodedName].tpe
    val ValidateAnn = TypeTree.of[sttp.tapir.Schema.annotations.validate[_]].tpe
    val ValidateEachAnn = TypeTree.of[sttp.tapir.Schema.annotations.validateEach[_]].tpe

    val tpe = TypeRepr.of[T]

    // if derivation is for Enumeration.Value then we lookup annotations on parent object that extend Enumeration
    val annotations = if (tpe <:< EnumerationValue) {
      val enumerationPath = tpe.show.split("\\.").dropRight(1).mkString(".")
      Symbol.requiredModule(enumerationPath).annotations
    } else tpe.typeSymbol.annotations

    def firstAnnArg(tpe: TypeRepr): Option[Tree] = {
      annotations
        .collectFirst {
          case ann if ann.tpe <:< tpe =>
            ann match {
              case Apply(_, List(tree)) => tree
            }
        }
    }

    def allAnnArg(tpe: TypeRepr): List[Tree] = {
      annotations
        .collect {
          case ann if ann.tpe <:< tpe =>
            ann match {
              case Apply(_, List(tree)) => tree
            }
        }
    }

    def firstTwoAnnArgs(tpe: TypeRepr): Option[(Tree, Tree)] = {
      annotations
        .collectFirst {
          case ann if ann.tpe <:< tpe =>
            ann match {
              case Apply(_, List(t1, t2)) => (t1, t2)
            }
        }
    }

    val transformations: List[Expr[SchemaAnnotations[T]] => Expr[SchemaAnnotations[T]]] =
      List(
        sa => firstAnnArg(DescriptionAnn).map(arg => '{ ${ sa }.copy(description = Some(${ arg.asExprOf[String] })) }).getOrElse(sa),
        sa => firstAnnArg(EncodedExampleAnn).map(arg => '{ ${ sa }.copy(encodedExample = Some(${ arg.asExprOf[Any] })) }).getOrElse(sa),
        sa =>
          firstTwoAnnArgs(DefaultAnn)
            .map(args => '{ ${ sa }.copy(default = Some((${ args._1.asExprOf[T] }, ${ args._2.asExprOf[Option[Any]] }))) })
            .getOrElse(sa),
        sa => firstAnnArg(FormatAnn).map(arg => '{ ${ sa }.copy(format = Some(${ arg.asExprOf[String] })) }).getOrElse(sa),
        sa => annotations.find { _.tpe <:< DeprecatedAnn }.map(_ => '{ ${ sa }.copy(deprecated = Some(true)) }).getOrElse(sa),
        sa => annotations.find { _.tpe <:< HiddenAnn }.map(_ => '{ ${ sa }.copy(hidden = Some(true)) }).getOrElse(sa),
        sa => firstAnnArg(EncodedNameAnn).map(arg => '{ ${ sa }.copy(encodedName = Some(${ arg.asExprOf[String] })) }).getOrElse(sa),
        sa => '{ ${ sa }.copy(validate = ${ Expr.ofList(allAnnArg(ValidateAnn).map(_.asExprOf[sttp.tapir.Validator[T]])) }) },
        sa => '{ ${ sa }.copy(validateEach = ${ Expr.ofList(allAnnArg(ValidateEachAnn).map(_.asExprOf[sttp.tapir.Validator[Any]])) }) }
      )

    transformations.foldLeft('{ SchemaAnnotations.empty[T] })((sa, t) => t(sa))
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy