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

parsley.Result.scala Maven / Gradle / Ivy

There is a newer version: 5.0.0-M6
Show newest version
/* SPDX-FileCopyrightText: © 2020 Parsley Contributors 
 * SPDX-License-Identifier: BSD-3-Clause
 */
package parsley

import scala.annotation.nowarn
import scala.util.{Failure => TFailure, Success => TSuccess, Try}
import scala.util.hashing.MurmurHash3

/** This trait represents the result of a parser.
  *
  * Either a `Success[A]` or a `Failure`.
  *
  * @tparam A the type of expected success result.
  */
sealed abstract class Result[+Err, +A] {
    /** Returns the result of applying `ferr` to this result's error if this is a `Failure` or `fa` to the result stored in the `Success` otherwise.
      *
      * @param ferr the function to apply if this is a `Failure`.
      * @param fa the function to apply if this is a `Success`.
      * @return the results of applying the function
      * @since 1.7.0
      */
    def fold[B](ferr: Err => B, fa: A => B): B = this match {
        case Success(x)   => fa(x)
        case Failure(msg) => ferr(msg)
    }

    /** Executes the procedure `f` if this is a `Success`. Otherwise, do nothing.
      *
      * This is equivalent to:
      * {{{
      * result match {
      *   case Success(x) => f(x)
      *   case _          => ()
      * }
      * }}}
      *
      * @param f The side-effecting function to execute.
      * @since 1.7.0
      */
    def foreach[U](f: A => U): Unit = this match {
        case Success(x) => f(x): @nowarn
        case _          => ()
    }

    /** Returns the successful value within the result.
      *
      * This is equivalent to:
      * {{{
      * result match {
      *   case Success(x) => x
      *   case _          => throw new Exception
      * }
      * }}}
      *
      * @note the result must not be a failure.
      * @throws java.util.NoSuchElementException if the result is a failure.
      * @since 1.7.0
      */
    def get: A

    /** Returns the value from this `Success` or the result of evaluating `default` if this is a `Failure`.
      *
      * @since 1.7.0
      */
    def getOrElse[B >: A](default: =>B): B = orElse(Success(default)).get

    /** Returns this result if it is a `Success`, otherwise return the result of evaluating `alternative`.
      *
      * @since 1.7.0
      */
    def orElse[B >: A, Errʹ >: Err](alternative: =>Result[Errʹ, B]): Result[Errʹ, B] = this match {
        case Success(_) => this
        case _          => alternative
    }

    /** Returns `true` if this result is a `Success` and its value is equal to `elem` (as determined by `==`),
      * returns `false` otherwise.
      *
      * @param elem    the element to test.
      * @return `true` if this is a `Success` value equal to `elem`.
      * @since 1.7.0
      */
    def contains[B >: A](elem: B): Boolean = exists(_ == elem)

    /** Returns `true` if this result is a `Failure` or returns the result of the application of
      * the given predicate to the `Success` value.
      *
      * @since 1.7.0
      */
    def forall(f: A => Boolean): Boolean = this match {
        case Success(x) => f(x)
        case _          => true
    }

    /** Returns `false` if `Failure` or returns the result of the application of
      * the given predicate to the `Success` value.
      *
      * @since 1.7.0
      */
    def exists(p: A => Boolean): Boolean = this match {
        case Success(x) => p(x)
        case _          => false
    }

    /** Returns the result of applying `f` to this result if it is a success. Returns
      * a failure if this result is a failure. Differs from `map` as `f` returns a result
      * instead of just a value.
      *
      * @since 1.7.0
      */
    def flatMap[B, Errʹ >: Err](f: A => Result[Errʹ, B]): Result[Errʹ, B] = this match {
        case Success(x) => f(x)
        case _          => this.asInstanceOf[Result[Err, B]]
    }

    /** Returns the nested result if this result is a success, otherwise return this failure.
      *
      * Equivalent to `flatMap(identity[Result[Errʹ, B]])`.
      *
      * @since 1.7.0
      */
    def flatten[B, Errʹ >: Err](implicit ev: A <:< Result[Errʹ, B]): Result[Errʹ, B] = flatMap(ev)

    /** Returns a `Success` containing the result of applying `f` to this result's value if
      * this is a success. Otherwise, returns a failure.
      *
      * @since 1.7.0
      */
    def map[B](f: A => B): Result[Err, B] = this match {
        case Success(x) => Success(f(x))
        case _          => this.asInstanceOf[Result[Err, B]]
    }

    /** Returns `Success` with the existing value of `Success` if this is a `Success`
      * and the given predicate `p` holds for the right value,
      * or `Failure(msg)` if this is a `Success` and the given predicate `p` does not hold for the right value,
      * or `Failure` with the existing value of `Failure` if this is a `Failure`.
      *
      * @since 1.7.0
      */
    def filterOrElse[Errʹ >: Err](p: A => Boolean, msg: =>Errʹ): Result[Errʹ, A] = this match {
        case Success(x) if !p(x) => Failure(msg)
        case _                   => this
    }

    /** Returns a `Seq` containing the `Success` value if it exists or an empty `Seq` if this is a `Failure`.
      *
      * @since 1.7.0
      */
    def toSeq: Seq[A] = this match {
        case Success(x) => Seq(x)
        case _          => Seq.empty
    }

    /** Returns a `Some` containing the `Success` value if it exists or a `None` if this is a `Failure`.
      *
      * @since 1.7.0
      */
    def toOption: Option[A] = this match {
        case Success(x) => Some(x)
        case _          => None
    }

    /** Converts the `Result` into a `Try` where `Failure` maps to a plain `Exception`.
      *
      * @since 1.7.0
      */
    def toTry: Try[A] = this match {
        case Success(x)   => TSuccess(x)
        case Failure(msg) => TFailure(new Exception(s"ParseError: $msg"))
    }

    /** Converts the `Result` into a `Either` where `Failure` maps to a `Left[Err]`.
      *
      * @since 1.7.0
      */
    def toEither: Either[Err, A] = this match {
        case Success(x)   => Right(x)
        case Failure(msg) => Left(msg)
    }

    /** Returns `true` if this is a `Success`, `false` otherwise.
      *
      * @since 1.7.0
      */
    def isSuccess: Boolean

    /** Returns `true` if this is a `Failure`, `false` otherwise.
      *
      * @since 1.7.0
      */
    def isFailure: Boolean
}

/** This class is used for when a parser succeeds, and contains its result.
  *
  * @param x the result value of the successful parse.
  * @tparam A the type of expected success result.
  */
case class Success[A] private [parsley] (x: A) extends Result[Nothing, A] {
    /** @inheritdoc */
    override def isSuccess: Boolean = true
    /** @inheritdoc */
    override def isFailure: Boolean = false
    /** @inheritdoc */
    override def get: A = x
}

/** This class is used for a parser failure, and contains the error message.
  *
  * @tparam Err the type of the error message generated by the failing parse.
  * @param _msg the error message reported by the parser (passed lazily).
  */
class Failure[Err] private [parsley] (_msg: =>Err) extends Result[Err, Nothing] with Product with Serializable {
    lazy val msg: Err = _msg
    /** @inheritdoc */
    override def isSuccess: Boolean = false
    /** @inheritdoc */
    override def isFailure: Boolean = true
    /** @inheritdoc */
    override def get: Nothing = throw new NoSuchElementException("get called on Failure") // scalastyle:ignore throw
    // We are normally given everything below, but ideally we want to make error generation lazy
    // $COVERAGE-OFF$
    override def toString: String = s"Failure($msg)"
    override def hashCode: Int = MurmurHash3.productHash(this)
    override def canEqual(x: Any): Boolean = x.isInstanceOf[Failure[_]]
    override def productPrefix: String = "Failure"
    override def productArity: Int = 1
    override def productElement(idx: Int): Any = {
        if (idx != 0) throw new IndexOutOfBoundsException("Failure only has arity 1") else msg // scalastyle:ignore throw
    }
    override def equals(x: Any): Boolean = x != null && (x match {
        case x: Failure[_] => x.msg == msg
        case _ => false
    })
    def copy(msg: =>Err = this.msg): Failure[Err] = new Failure(msg)
    // $COVERAGE-ON$
}
// $COVERAGE-OFF$
object Failure {
    def apply[Err](msg: =>Err): Failure[Err] = new Failure(msg)
    def unapply[Err](x: Failure[Err]): Some[Err] = Some(x.msg)
    def andThen[A, Err](f: Failure[Err] => A): Err => A = msg => f(Failure(msg))
    def compose[A, Err](f: A => Err): A => Failure[Err] = msg => Failure(f(msg))
}
// $COVERAGE-ON$




© 2015 - 2025 Weber Informatics LLC | Privacy Policy