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

sttp.tapir.macros.MultipartCodecMacros.scala Maven / Gradle / Ivy

The newest version!
package sttp.tapir.macros

import sttp.tapir.{Codec, CodecFormat, FileRange, MultipartCodec, PartCodec, RawBodyType, Schema}
import sttp.tapir.generic.Configuration
import sttp.tapir.internal.{CaseClass, CaseClassField}
import sttp.model.Part

import scala.annotation.tailrec
import scala.collection.immutable.ListMap
import scala.quoted.*
import java.nio.charset.StandardCharsets

trait MultipartCodecMacros {
  inline given multipartCaseClassCodec[T](using inline c: Configuration): MultipartCodec[T] =
    ${ MultipartCodecMacros.multipartCaseClassCodecImpl[T]('c) }
}

private[tapir] object MultipartCodecMacros {
  def multipartCaseClassCodecImpl[T: Type](conf: Expr[Configuration])(using q: Quotes): Expr[MultipartCodec[T]] = {
    import quotes.reflect.*
    val caseClass = new CaseClass[q.type, T](using summon[Type[T]], q)
    val encodedNameAnnotationSymbol = TypeTree.of[Schema.annotations.encodedName].tpe.typeSymbol

    def summonPartCodec[f: Type](field: CaseClassField[q.type, T]) = {
      val candidates = List(
        () =>
          Expr
            .summon[Codec[List[Part[String]], f, CodecFormat.TextPlain]]
            .map(c => '{ PartCodec(RawBodyType.StringBody(StandardCharsets.UTF_8), $c) }),
        () =>
          Expr
            .summon[Codec[List[Part[String]], f, _ <: CodecFormat]]
            .map(c => '{ PartCodec(RawBodyType.StringBody(StandardCharsets.UTF_8), $c) }),
        () => Expr.summon[Codec[List[Part[Array[Byte]]], f, _ <: CodecFormat]].map(c => '{ PartCodec(RawBodyType.ByteArrayBody, $c) }),
        () =>
          Expr
            .summon[Codec[List[Part[java.io.InputStream]], f, _ <: CodecFormat]]
            .map(c => '{ PartCodec(RawBodyType.InputStreamBody, $c) }),
        () =>
          Expr.summon[Codec[List[Part[java.nio.ByteBuffer]], f, _ <: CodecFormat]].map(c => '{ PartCodec(RawBodyType.ByteBufferBody, $c) }),
        () => Expr.summon[Codec[List[Part[FileRange]], f, _ <: CodecFormat]].map(c => '{ PartCodec(RawBodyType.FileBody, $c) })
      )

      @tailrec
      def firstNotEmpty(c: List[() => Option[Expr[PartCodec[_, _]]]]): Expr[PartCodec[_, _]] = c match {
        case Nil =>
          report.errorAndAbort(s"Cannot find a codec between a List[Part[T]] for some basic type T and: ${field.tpe}")
        case h :: t =>
          h() match {
            case None    => firstNotEmpty(t)
            case Some(e) => e
          }
      }

      firstNotEmpty(candidates)
    }

    def fieldTransformedName(field: CaseClassField[q.type, T]): Expr[String] = {
      val encodedName = field.extractStringArgFromAnnotation(encodedNameAnnotationSymbol)
      '{ ${ Expr(encodedName) }.getOrElse($conf.toEncodedName(${ Expr(field.name) })) }
    }

    val partCodecs: Expr[ListMap[String, PartCodec[_, _]]] = {
      val partCodecPairs = caseClass.fields.map { field =>
        val partCodec = field.tpe.asType match
          case '[f] => summonPartCodec[f](field)

        '{ ${ fieldTransformedName(field) } -> $partCodec }
      }

      '{ ListMap(${ Varargs(partCodecPairs) }: _*) }
    }

    def encodeDefBody(tTerm: Term): Term = {
      val fieldsEncode = caseClass.fields.map { field =>
        val fieldValue = field.tpe.asType match
          case '[f] =>
            tTerm.select(field.symbol).asExprOf[f]
        '{ ${ fieldTransformedName(field) } -> $fieldValue }
      }

      '{ ListMap(${ Varargs(fieldsEncode) }: _*) }.asTerm
    }

    val encodeDefSymbol =
      Symbol.newMethod(
        Symbol.spliceOwner,
        "encode",
        MethodType(List("t"))(_ => List(caseClass.tpe), _ => TypeRepr.of[ListMap[String, Any]])
      )
    val encodeDef = DefDef(
      encodeDefSymbol,
      { case List(List(tTerm: Term)) =>
        Some(encodeDefBody(tTerm).changeOwner(encodeDefSymbol))
      }
    )
    val encodeExpr = Block(List(encodeDef), Closure(Ref(encodeDefSymbol), None)).asExprOf[T => ListMap[String, Any]]

    //

    def decodeDefBody(partsTerm: Term): Term = {
      def fieldsDecode(partsMap: Expr[ListMap[String, Any]]) = caseClass.fields.map { field =>
        '{ $partsMap(${ fieldTransformedName(field) }) }
      }

      '{
        val partsMap: ListMap[String, Any] = ${ partsTerm.asExprOf[ListMap[String, Any]] }
        val values = List(${ Varargs(fieldsDecode('partsMap)) }: _*)
        ${ caseClass.instanceFromValues('{ values }) }
      }.asTerm
    }

    val decodeDefSymbol =
      Symbol.newMethod(
        Symbol.spliceOwner,
        "decode",
        MethodType(List("params"))(_ => List(TypeRepr.of[ListMap[String, Any]]), _ => TypeRepr.of[T])
      )
    val decodeDef = DefDef(
      decodeDefSymbol,
      { case List(List(paramsTerm: Term)) =>
        Some(decodeDefBody(paramsTerm).changeOwner(decodeDefSymbol))
      }
    )
    val decodeExpr = Block(List(decodeDef), Closure(Ref(decodeDefSymbol), None)).asExprOf[ListMap[String, Any] => T]

    '{
      Codec
        .multipart($partCodecs, None)
        .map($decodeExpr)($encodeExpr)
        .schema(${ Expr.summon[Schema[T]].getOrElse(report.errorAndAbort(s"Cannot find a given Schema[${Type.show[T]}].")) })
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy