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