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

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

package sttp.tapir

sealed trait DecodeResult[+T] {
  def map[TT](f: T => TT): DecodeResult[TT]
  def flatMap[U](f: T => DecodeResult[U]): DecodeResult[U]
}
object DecodeResult {
  sealed trait Failure extends DecodeResult[Nothing] {
    override def map[TT](f: Nothing => TT): DecodeResult[TT] = this
    override def flatMap[U](f: Nothing => DecodeResult[U]): DecodeResult[U] = this
  }

  case class Value[T](v: T) extends DecodeResult[T] {
    override def map[TT](f: T => TT): DecodeResult[TT] = Value(f(v))
    override def flatMap[U](f: T => DecodeResult[U]): DecodeResult[U] = f(v)
  }
  case object Missing extends Failure
  case class Multiple[R](vs: Seq[R]) extends Failure

  /** Any error that occurred while decoding the `original` value. */
  case class Error(original: String, error: Throwable) extends Failure
  object Error {
    case class JsonDecodeException(errors: List[JsonError], underlying: Throwable)
        extends Exception(underlying.getMessage, underlying, true, false) {
      // a toString isn't generated by for the case class, as Exception overrides it
      override def toString: String = s"JsonDecodeException($errors,$underlying)"
    }
    case class JsonError(msg: String, path: List[FieldName])

    case class MultipartDecodeException(partFailures: List[(String, DecodeResult.Failure)])
        extends Exception(partFailures.toString, MultipartDecodeException.causeOrNull(partFailures), true, false)
    object MultipartDecodeException {
      def causeOrNull(partFailures: List[(String, DecodeResult.Failure)]): Throwable =
        partFailures.collectFirst { case (_, DecodeResult.Error(_, error)) => error }.orNull
    }
  }
  case class Mismatch(expected: String, actual: String) extends Failure

  /** A validation error that occurred when decoding the value, that is, when some `Validator` failed. */
  case class InvalidValue(errors: List[ValidationError[_]]) extends Failure

  def sequence[T](results: Seq[DecodeResult[T]]): DecodeResult[Seq[T]] = {
    results.foldRight(Value(List.empty[T]): DecodeResult[Seq[T]]) { case (result, acc) =>
      (result, acc) match {
        case (Value(v), Value(vs)) => Value(v +: vs)
        case (Value(_), r)         => r
        case (df: Failure, _)      => df
      }
    }
  }

  def fromOption[T](o: Option[T]): DecodeResult[T] = o.map(Value(_)).getOrElse(Missing)
  def fromEitherString[T](original: String, o: Either[String, T]): DecodeResult[T] = o match {
    case Left(e)      => DecodeResult.Error(original, new RuntimeException(e))
    case Right(value) => DecodeResult.Value(value)
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy