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

sttp.tapir.Mapping.scala Maven / Gradle / Ivy

package sttp.tapir

import sttp.tapir.DecodeResult.Value

import scala.util.{Failure, Success, Try}

/** A bi-directional mapping between values of type `L` and values of type `H`.
  *
  * Low-level values of type `L` can be **decoded** to a higher-level value of type `H`. The decoding can fail; this is represented by a
  * result of type [[DecodeResult.Failure]]. Failures might occur due to format errors, wrong arity, exceptions, or validation errors.
  * Validators can be added through the `validate` method.
  *
  * High-level values of type `H` can be **encoded** as a low-level value of type `L`.
  *
  * Mappings can be chained using one of the `map` functions.
  *
  * @tparam L
  *   The type of the low-level value.
  * @tparam H
  *   The type of the high-level value.
  */
trait Mapping[L, H] { outer =>
  def rawDecode(l: L): DecodeResult[H]
  def encode(h: H): L

  /**   - calls `rawDecode`
    *   - catches any exceptions that might occur, converting them to decode failures
    *   - validates the result
    */
  def decode(l: L): DecodeResult[H] = Mapping.decode(l, rawDecode, validator.apply)

  def validator: Validator[H]

  def map[HH](codec: Mapping[H, HH]): Mapping[L, HH] =
    new Mapping[L, HH] {
      override def rawDecode(l: L): DecodeResult[HH] = outer.rawDecode(l).flatMap(codec.rawDecode)
      override def encode(hh: HH): L = outer.encode(codec.encode(hh))
      override def validator: Validator[HH] = outer.validator.contramap(codec.encode).and(codec.validator)
    }

  def validate(v: Validator[H]): Mapping[L, H] =
    new Mapping[L, H] {
      override def rawDecode(l: L): DecodeResult[H] = outer.decode(l)
      override def encode(h: H): L = outer.encode(h)
      override def validator: Validator[H] = Mapping.addEncodeToEnumValidator(v, encode).and(outer.validator)
    }
}

object Mapping {
  def id[L]: Mapping[L, L] =
    new Mapping[L, L] {
      override def rawDecode(l: L): DecodeResult[L] = DecodeResult.Value(l)
      override def encode(h: L): L = h
      override def validator: Validator[L] = Validator.pass
    }
  def from[L, H](f: L => H)(g: H => L): Mapping[L, H] = fromDecode(f.andThen(Value(_)))(g)
  def fromDecode[L, H](f: L => DecodeResult[H])(g: H => L): Mapping[L, H] =
    new Mapping[L, H] {
      override def rawDecode(l: L): DecodeResult[H] = f(l)
      override def encode(h: H): L = g(h)
      override def validator: Validator[H] = Validator.pass
    }

  /** A mapping which, during encoding, adds the given `prefix`. When decoding, the prefix is removed (case insensitive,if present),
    * otherwise an error is reported.
    */
  def stringPrefixCaseInsensitive(prefix: String): Mapping[String, String] = {
    Mapping.fromDecode(cropPrefix(_, prefix))(v => s"$prefix$v")
  }

  def stringPrefixCaseInsensitiveForList(prefix: String): Mapping[List[String], List[String]] = {
    def removePrefix(v: List[String]): DecodeResult[List[String]] = {
      DecodeResult
        .sequence(v.map { s => cropPrefix(s, prefix) })
        .map(_.toList)
    }

    Mapping.fromDecode[List[String], List[String]](removePrefix)(v => v.map(d => s"$prefix$d"))
  }

  private def cropPrefix(s: String, prefix: String) = {
    val prefixLength = prefix.length
    val prefixLower = prefix.toLowerCase
    if (s.toLowerCase.startsWith(prefixLower)) DecodeResult.Value(s.substring(prefixLength))
    else DecodeResult.Error(s, new IllegalArgumentException(s"The given value doesn't start with $prefix"))
  }

  private[tapir] def decode[L, H](
      l: L,
      rawDecode: L => DecodeResult[H],
      applyValidation: H => List[ValidationError[_]]
  ): DecodeResult[H] = {
    def tryRawDecode(l: L): DecodeResult[H] = {
      Try(rawDecode(l)) match {
        case Success(r) => r
        case Failure(e) => DecodeResult.Error(l.toString, e)
      }
    }

    def validate(r: DecodeResult[H]): DecodeResult[H] = {
      r match {
        case DecodeResult.Value(v) =>
          val validationErrors = applyValidation(v)
          if (validationErrors.isEmpty) {
            DecodeResult.Value(v)
          } else {
            DecodeResult.InvalidValue(validationErrors)
          }
        case r => r
      }
    }

    validate(tryRawDecode(l))
  }

  private[tapir] def addEncodeToEnumValidator[T](v: Validator[T], encode: T => Any): Validator[T] = v match {
    case v @ Validator.Enumeration(_, None, _) => v.encode(encode)
    case _                                     => v
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy