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

parsley.internal.deepembedding.frontend.Visitors.scala Maven / Gradle / Ivy

/*
 * Copyright 2020 Parsley Contributors 
 *
 * SPDX-License-Identifier: BSD-3-Clause
 */
package parsley.internal.deepembedding.frontend

import scala.collection.Factory

import parsley.debug.{Breakpoint, Profiler}
import parsley.errors.ErrorBuilder
import parsley.state.Ref
import parsley.token.descriptions.SpaceDesc
import parsley.token.descriptions.PlusSignPresence
import parsley.token.errors.{ErrorConfig, LabelConfig, LabelWithExplainConfig, SpecializedFilterConfig}
import parsley.token.CharPred

// scalastyle:off underscore.import
import parsley.internal.collection.immutable.Trie
import parsley.internal.deepembedding.Sign.SignType
import parsley.internal.deepembedding.singletons._
import parsley.internal.deepembedding.singletons.token._
import parsley.internal.errors.CaretWidth
// scalastyle:on underscore.import

/** Visitor class template for the processing of parsers without fully explicit exhaustive pattern
  * matching.
  *
  * This particular visitor is indexed on its return type to allow the preservation of relevant type
  * information, such as setting it to [[LazyParsley]] to tell the type system that you want to
  * produce new parsers using the visitor.
  *
  * This base visitor type within Parsley only implements methods for visiting internal Parsley
  * parsers that are native to Parsley itself. If you are making a library that extends Parsley with
  * more internal parsers (that will not be merged into Parsley itself), it is suggested that you
  * do not change this class, but rather make an abstract sub-class of this base visitor class that
  * is specific to your library's additional parsers.
  *
  * @tparam T Context type for holding processing information as the visitor visits parsers.
  * @tparam U Return value wrapper for the results of visiting the parsers.
  */
private [parsley] abstract class LazyParsleyIVisitor[-T, +U[+_]] { // scalastyle:ignore number.of.methods
    // Singleton parser visitors.
    def visit[A](self: Pure[A], context: T)(x: A): U[A]
    def visit[A](self: Fresh[A], context: T)(x: =>A): U[A]
    def visit(self: Satisfy, context: T)(f: Char => Boolean, expected: LabelConfig): U[Char]
    def visit(self: Line.type, context: T): U[Int]
    def visit(self: Col.type, context: T): U[Int]
    def visit(self: Offset.type, context: T): U[Int]
    def visit[S](self: Get[S], context: T)(ref: Ref[S]): U[S]
    def visit(self: WhiteSpace, context: T)(ws: Char => Boolean, desc: SpaceDesc, errorConfig: ErrorConfig): U[Unit]
    def visit(self: SkipComments, context: T)(desc: SpaceDesc, errorConfig: ErrorConfig): U[Unit]
    def visit(self: Comment, context: T)(desc: SpaceDesc, errorConfig: ErrorConfig): U[Unit]
    def visit[A](self: Sign[A], context: T)(ty: SignType, signPresence: PlusSignPresence): U[A => A]
    def visit(self: NonSpecific, context: T)(name: String,
                                             ue: String => String,
                                             start: Char => Boolean,
                                             letter: Char => Boolean,
                                             illegal: String => Boolean): U[String]
    def visit[A](self: CharTok[A], context: T)(c: Char, x: A, exp: LabelConfig): U[A]
    def visit[A](self: SupplementaryCharTok[A], context: T)(codepoint: Int, x: A, exp: LabelConfig): U[A]
    def visit[A](self: StringTok[A], context: T)(s: String, x: A, exp: LabelConfig): U[A]
    def visit(self: Eof.type, context: T): U[Unit]
    def visit(self: UniSatisfy, context: T)(f: Int => Boolean, exp: LabelConfig): U[Int]
    def visit[S](self: Modify[S], context: T)(ref: Ref[S], f: S => S): U[Unit]
    def visit(self: Empty, context: T)(width: Int): U[Nothing]
    def visit(self: Fail, context: T)(width: CaretWidth, msgs: Seq[String]): U[Nothing]
    def visit(self: Unexpected, context: T)(msg: String, width: CaretWidth): U[Nothing]
    def visit[A](self: VanillaGen[A], context: T)(gen: parsley.errors.VanillaGen[A]): U[((A, Int)) => Nothing]
    def visit[A](self: SpecializedGen[A], context: T)(gen: parsley.errors.SpecializedGen[A]): U[((A, Int)) => Nothing]
    def visit(self: EscapeMapped, context: T)(escTrie: Trie[Int], escs: Set[String]): U[Int]
    def visit(self: EscapeAtMost, context: T)(n: Int, radix: Int): U[BigInt]
    def visit(self: EscapeOneOfExactly, context: T)(radix: Int, ns: List[Int], ie: SpecializedFilterConfig[Int]): U[BigInt]
    def visit(self: SoftKeyword, context: T)(specific: String,
                                             letter: CharPred,
                                             caseSensitive: Boolean,
                                             expected: LabelWithExplainConfig,
                                             expectedEnd: String): U[Unit]
    def visit(self: SoftOperator, context: T)(specific: String,
                                              letter: CharPred,
                                              ops: Trie[Unit],
                                              expected: LabelWithExplainConfig,
                                              expectedEnd: String): U[Unit]

    // Primitive parser visitors.
    def visit[A](self: Atomic[A], context: T)(p: LazyParsley[A]): U[A]
    def visit[A](self: Look[A], context: T)(p: LazyParsley[A]): U[A]
    def visit[A](self: NotFollowedBy[A], context: T)(p: LazyParsley[A]): U[Unit]
    def visit[S](self: Put[S], context: T)(ref: Ref[S], p: LazyParsley[S]): U[Unit]
    def visit[S, A](self: NewReg[S, A], context: T)(ref: Ref[S], init: LazyParsley[S], body: =>LazyParsley[A]): U[A]
    def visit(self: Span, context: T)(p: LazyParsley[_]): U[String]
    def visit[A](self: Debug[A], context: T)(p: LazyParsley[A], name: String, ascii: Boolean, break: Breakpoint, watchedRefs: Seq[(Ref[_], String)]): U[A]
    def visit[A](self: DebugError[A], context: T)(p: LazyParsley[A], name: String, ascii: Boolean, errBuilder: ErrorBuilder[_]): U[A]
    def visit[A](self: Profile[A], context: T)(p: LazyParsley[A], name: String, profiler: Profiler): U[A]

    // Selective parser visitors.
    def visit[A, B, C](self: Branch[A, B, C], context: T)(b: LazyParsley[Either[A, B]], p: =>LazyParsley[A => C], q: =>LazyParsley[B => C]): U[C]
    def visit[A](self: If[A], context: T)(b: LazyParsley[Boolean], p: =>LazyParsley[A], q: =>LazyParsley[A]): U[A]
    def visit[A](self: Filter[A], context: T)(p: LazyParsley[A], pred: A => Boolean, err: =>LazyParsley[((A, Int)) => Nothing]): U[A]
    def visit[A, B](self: MapFilter[A, B], context: T)(p: LazyParsley[A], pred: A => Option[B], err: =>LazyParsley[((A, Int)) => Nothing]): U[B]

    // Alternative parser visitors.
    def visit[A](self: <|>[A], context: T)(p: LazyParsley[A], q: LazyParsley[A]): U[A]

    // Intrinsic parser visitors.
    def visit[A, B, C](self: Lift2[A, B, C], context: T)(f: (A, B) => C, p: LazyParsley[A], q: =>LazyParsley[B]): U[C]
    def visit[A, B, C, D](self: Lift3[A, B, C, D], context: T)(f: (A, B, C) => D, p: LazyParsley[A], q: =>LazyParsley[B], r: =>LazyParsley[C]): U[D]
    def visit[S, A](self: Local[S, A], context: T)(ref: Ref[S], p: LazyParsley[S], q: =>LazyParsley[A]): U[A]

    // Sequence parser visitors.
    def visit[A, B](self: A <*> B, context: T)(pf: LazyParsley[A => B], px: =>LazyParsley[A]): U[B]
    def visit[A, B](self: A >>= B, context: T)(p: LazyParsley[A], f: A => LazyParsley[B]): U[B]
    def visit[A](self: *>[A], context: T)(p: LazyParsley[_], _q: =>LazyParsley[A]): U[A]
    def visit[A](self: <*[A], context: T)(p: LazyParsley[A], _q: =>LazyParsley[_]): U[A]

    // Iterative parser visitors.
    def visit[A, C](self: Many[A, C], context: T)(p: LazyParsley[A], factory: Factory[A, C]): U[C]
    def visit[A](self: ChainPost[A], context: T)(p: LazyParsley[A], _op: =>LazyParsley[A => A]): U[A]
    def visit[A](self: ChainPre[A], context: T)(p: LazyParsley[A], op: =>LazyParsley[A => A]): U[A]
    def visit[A, B](self: Chainl[A, B], context: T)(init: LazyParsley[B], p: =>LazyParsley[A], op: =>LazyParsley[(B, A) => B]): U[B]
    def visit[A, B](self: Chainr[A, B], context: T)(p: LazyParsley[A], op: =>LazyParsley[(A, B) => B], wrap: A => B): U[B]
    def visit[A, C](self: SepEndBy1[A, C], context: T)(p: LazyParsley[A], sep: =>LazyParsley[_], factory: Factory[A, C]): U[C]
    def visit[A, C](self: ManyUntil[A, C], context: T)(body: LazyParsley[Any], factory: Factory[A, C]): U[C]
    def visit(self: SkipManyUntil, context: T)(body: LazyParsley[Any]): U[Unit]

    // Error parser visitors.
    def visit[A](self: ErrorLabel[A], context: T)(p: LazyParsley[A], label: String, labels: Seq[String]): U[A]
    def visit[A](self: ErrorHide[A], context: T)(p: LazyParsley[A]): U[A]
    def visit[A](self: ErrorExplain[A], context: T)(p: LazyParsley[A], reason: String): U[A]
    def visit[A](self: ErrorAmend[A], context: T)(p: LazyParsley[A], partial: Boolean): U[A]
    def visit[A](self: ErrorEntrench[A], context: T)(p: LazyParsley[A]): U[A]
    def visit[A](self: ErrorDislodge[A], context: T)(n: Int, p: LazyParsley[A]): U[A]
    def visit[A](self: ErrorLexical[A], context: T)(p: LazyParsley[A]): U[A]

    // Fallback case for any unknown parser types defined outside Parsley's main project.
    /** Specifically handles [[parsley.internal.deepembedding.frontend.Unary]],
      * [[parsley.internal.deepembedding.frontend.Binary]], and
      * [[parsley.internal.deepembedding.frontend.Ternary]] when the case is ambiguous.
      */
    def visitGeneric[A](self: GenericLazyParsley[A], context: T): U[A]

    /** Handles all other parsers that are not of Parsley's main internal parser implementations nor
      * uses the three generic 1-ary, 2-ary and 3-ary bases that Parsley provides (though it is
      * suggested that this method call [[visitGeneric]] with the parser should it find that the
      * parser is a subclass of [[parsley.internal.deepembedding.frontend.GenericLazyParsley]]).
      *
      * Assume all parsers passed into this method are black boxes.
      *
      * @note Implementations must account for any possible implementation with some default logic.
      */
    // $COVERAGE-OFF$
    def visitUnknown[A](self: LazyParsley[A], context: T): U[A]
    // $COVERAGE-ON$
}

/** Generalised version of [[LazyParsleyIVisitor]] that allows you to define default implementations
  * for parser classes that live under a base trait (e.g. [[Unary]]).
  *
  * This visitor should only require implementations for [[Singleton]], [[Unary]], [[Binary]],
  * [[Ternary]], [[<|>]], [[ChainPre]], and [[LazyParsleyIVisitor.visitUnknown]].
  *
  * Unless a specific override is needed, all other visitor methods are implemented relative to
  * these six default implementations.
  */
private [frontend] abstract class GenericLazyParsleyIVisitor[-T, +U[+_]] extends LazyParsleyIVisitor[T, U] { // scalastyle:ignore number.of.methods
    // Default methods for the four base parser types.
    // XXX: These names are different as otherwise some visit methods recurse in an unwanted manner.
    def visitSingleton[A](self: Singleton[A], context: T): U[A]
    def visitUnary[A, B](self: Unary[A, B], context: T)(p: LazyParsley[A]): U[B]
    def visitBinary[A, B, C](self: Binary[A, B, C], context: T)(l: LazyParsley[A], r: =>LazyParsley[B]): U[C]
    def visitTernary[A, B, C, D](self: Ternary[A, B, C, D], context: T)(f: LazyParsley[A], s: =>LazyParsley[B], t: =>LazyParsley[C]): U[D]

    // Forward the generic parser to one of the defined methods.
    // $COVERAGE-OFF$
    override def visitGeneric[A](self: GenericLazyParsley[A], context: T): U[A] = self match {
        case u: Unary[_, A]         => visitUnary(u, context)(u.p)
        case b: Binary[_, _, A]     => visitBinary(b, context)(b.left, b.right)
        case t: Ternary[_, _, _, A] => visitTernary(t, context)(t.first, t.second, t.third)
    }
    // $COVERAGE-ON$

    // Singleton overrides.
    override def visit[A](self: Pure[A], context: T)(x: A): U[A] = visitSingleton(self, context)
    override def visit[A](self: Fresh[A], context: T)(x: =>A): U[A] = visitSingleton(self, context)
    override def visit(self: Satisfy, context: T)(f: Char => Boolean, expected: LabelConfig): U[Char] = visitSingleton(self, context)
    override def visit(self: Line.type, context: T): U[Int] = visitSingleton(self, context)
    override def visit(self: Col.type, context: T): U[Int] = visitSingleton(self, context)
    override def visit(self: Offset.type, context: T): U[Int] = visitSingleton(self, context)
    override def visit[S](self: Get[S], context: T)(ref: Ref[S]): U[S] = visitSingleton(self, context)
    override def visit(self: WhiteSpace, context: T)(ws: Char => Boolean, desc: SpaceDesc, errorConfig: ErrorConfig): U[Unit] = visitSingleton(self, context)
    override def visit(self: SkipComments, context: T)(desc: SpaceDesc, errorConfig: ErrorConfig): U[Unit] = visitSingleton(self, context)
    override def visit(self: Comment, context: T)(desc: SpaceDesc, errorConfig: ErrorConfig): U[Unit] = visitSingleton(self, context)
    override def visit[A](self: Sign[A], context: T)(ty: SignType, signPresence: PlusSignPresence): U[A => A] = visitSingleton(self, context)
    override def visit(self: NonSpecific, context: T)(name: String,
                                                      ue: String => String,
                                                      start: Char => Boolean,
                                                      letter: Char => Boolean,
                                                      illegal: String => Boolean): U[String] = visitSingleton(self, context)
    override def visit[A](self: CharTok[A], context: T)(c: Char, x: A, exp: LabelConfig): U[A] = visitSingleton(self, context)
    override def visit[A](self: SupplementaryCharTok[A], context: T)(codepoint: Int, x: A, exp: LabelConfig): U[A] = visitSingleton(self, context)
    override def visit[A](self: StringTok[A], context: T)(s: String, x: A, exp: LabelConfig): U[A] = visitSingleton(self, context)
    override def visit(self: Eof.type, context: T): U[Unit] = visitSingleton(self, context)
    override def visit(self: UniSatisfy, context: T)(f: Int => Boolean, exp: LabelConfig): U[Int] = visitSingleton(self, context)
    override def visit[S](self: Modify[S], context: T)(ref: Ref[S], f: S => S): U[Unit] = visitSingleton(self, context)
    override def visit(self: Empty, context: T)(width: Int): U[Nothing] = visitSingleton(self, context)
    override def visit(self: Fail, context: T)(width: CaretWidth, msgs: Seq[String]): U[Nothing] = visitSingleton(self, context)
    override def visit(self: Unexpected, context: T)(msg: String, width: CaretWidth): U[Nothing] = visitSingleton(self, context)
    override def visit[A](self: VanillaGen[A], context: T)(gen: parsley.errors.VanillaGen[A]): U[((A, Int)) => Nothing] = visitSingleton(self, context)
    override def visit[A](self: SpecializedGen[A], context: T)(gen: parsley.errors.SpecializedGen[A]): U[((A, Int)) => Nothing] = {
        visitSingleton(self, context)
    }
    override def visit(self: EscapeMapped, context: T)(escTrie: Trie[Int], escs: Set[String]): U[Int] = visitSingleton(self, context)
    override def visit(self: EscapeAtMost, context: T)(n: Int, radix: Int): U[BigInt] = visitSingleton(self, context)
    override def visit(self: EscapeOneOfExactly, context: T)(radix: Int, ns: List[Int], ie: SpecializedFilterConfig[Int]): U[BigInt] = {
        visitSingleton(self, context)
    }
    override def visit(self: SoftKeyword, context: T)(specific: String,
                                                      letter: CharPred,
                                                      caseSensitive: Boolean,
                                                      expected: LabelWithExplainConfig,
                                                      expectedEnd: String): U[Unit] = visitSingleton(self, context)
    override def visit(self: SoftOperator, context: T)(specific: String,
                                                       letter: CharPred,
                                                       ops: Trie[Unit],
                                                       expected: LabelWithExplainConfig,
                                                       expectedEnd: String): U[Unit] = visitSingleton(self, context)

    // Primitive overrides.
    override def visit[A](self: Atomic[A], context: T)(p: LazyParsley[A]): U[A] = visitUnary(self, context)(p)
    override def visit[A](self: Look[A], context: T)(p: LazyParsley[A]): U[A] = visitUnary(self, context)(p)
    override def visit[A](self: NotFollowedBy[A], context: T)(p: LazyParsley[A]): U[Unit] = visitUnary(self, context)(p)
    override def visit[S](self: Put[S], context: T)(ref: Ref[S], p: LazyParsley[S]): U[Unit] = visitUnary(self, context)(p)
    override def visit[S, A](self: NewReg[S, A], context: T)(ref: Ref[S], init: LazyParsley[S], body: =>LazyParsley[A]): U[A] = {
        visitBinary(self, context)(init, body)
    }
    override def visit(self: Span, context: T)(p: LazyParsley[_]): U[String] = visitUnary[Any, String](self, context)(p)
    override def visit[A](self: Debug[A], context: T)
                         (p: LazyParsley[A], name: String, ascii: Boolean, break: Breakpoint, watchedRefs: Seq[(Ref[_], String)]): U[A] = {
        visitUnary(self, context)(p)
    }
    override def visit[A](self: DebugError[A], context: T)(p: LazyParsley[A], name: String, ascii: Boolean, errBuilder: ErrorBuilder[_]): U[A] = {
        visitUnary(self, context)(p)
    }
    override def visit[A](self: Profile[A], context: T)(p: LazyParsley[A], name: String, profiler: Profiler): U[A] = visitUnary(self, context)(p)

    // Selective overrides.
    override def visit[A, B, C](self: Branch[A, B, C], context: T)(b: LazyParsley[Either[A, B]], p: =>LazyParsley[A => C], q: =>LazyParsley[B => C]): U[C] = {
        visitTernary(self, context)(b, p, q)
    }
    override def visit[A](self: If[A], context: T)(b: LazyParsley[Boolean], p: =>LazyParsley[A], q: =>LazyParsley[A]): U[A] = {
        visitTernary(self, context)(b, p, q)
    }
    override def visit[A](self: Filter[A], context: T)(p: LazyParsley[A], pred: A => Boolean, err: =>LazyParsley[((A, Int)) => Nothing]): U[A] = {
        visitBinary(self, context)(p, err)
    }
    override def visit[A, B](self: MapFilter[A, B], context: T)(p: LazyParsley[A], pred: A => Option[B], err: =>LazyParsley[((A, Int)) => Nothing]): U[B] = {
        visitBinary(self, context)(p, err)
    }

    // Intrinsic overrides.
    override def visit[A, B, C](self: Lift2[A, B, C], context: T)(f: (A, B) => C, p: LazyParsley[A], q: =>LazyParsley[B]): U[C] = {
        visitBinary(self, context)(p, q)
    }
    override def visit[A, B, C, D](self: Lift3[A, B, C, D], context: T)(f: (A, B, C) => D,
                                                                        p: LazyParsley[A],
                                                                        q: =>LazyParsley[B],
                                                                        r: =>LazyParsley[C]): U[D] = visitTernary(self, context)(p, q, r)
    override def visit[S, A](self: Local[S, A], context: T)(ref: Ref[S], p: LazyParsley[S], q: =>LazyParsley[A]): U[A] = visitBinary(self, context)(p, q)

    // Sequence overrides.
    override def visit[A, B](self: A <*> B, context: T)(pf: LazyParsley[A => B], px: =>LazyParsley[A]): U[B] = visitBinary(self, context)(pf, px)
    override def visit[A, B](self: A >>= B, context: T)(p: LazyParsley[A], f: A => LazyParsley[B]): U[B] = visitUnary(self, context)(p)
    override def visit[A](self: *>[A], context: T)(p: LazyParsley[_], _q: =>LazyParsley[A]): U[A] = visitBinary[Any, A, A](self, context)(p, _q)
    override def visit[A](self: <*[A], context: T)(p: LazyParsley[A], _q: =>LazyParsley[_]): U[A] = visitBinary[A, Any, A](self, context)(p, _q)

    // Iterative overrides.
    override def visit[A, C](self: Many[A, C], context: T)(p: LazyParsley[A], factory: Factory[A, C]): U[C] = visitUnary(self, context)(p)
    override def visit[A](self: ChainPost[A], context: T)(p: LazyParsley[A], _op: =>LazyParsley[A => A]): U[A] = visitBinary(self, context)(p, _op)
    override def visit[A, B](self: Chainl[A, B], context: T)(init: LazyParsley[B], p: =>LazyParsley[A], op: =>LazyParsley[(B, A) => B]): U[B] = {
        visitTernary(self, context)(init, p, op)
    }
    override def visit[A, B](self: Chainr[A, B], context: T)(p: LazyParsley[A], op: =>LazyParsley[(A, B) => B], wrap: A => B): U[B] = {
        visitBinary(self, context)(p, op)
    }
    override def visit[A, C](self: SepEndBy1[A, C], context: T)(p: LazyParsley[A], sep: =>LazyParsley[_], factory: Factory[A, C]): U[C] = {
        visitBinary[A, Any, C](self, context)(p, sep)
    }
    override def visit[A, C](self: ManyUntil[A, C], context: T)(body: LazyParsley[Any], factory: Factory[A, C]): U[C] = visitUnary[Any, C](self, context)(body)
    override def visit(self: SkipManyUntil, context: T)(body: LazyParsley[Any]): U[Unit] = visitUnary[Any, Unit](self, context)(body)

    // Error overrides.
    override def visit[A](self: ErrorLabel[A], context: T)(p: LazyParsley[A], label: String, labels: Seq[String]): U[A] = visitUnary(self, context)(p)
    override def visit[A](self: ErrorHide[A], context: T)(p: LazyParsley[A]): U[A] = visitUnary(self, context)(p)
    override def visit[A](self: ErrorExplain[A], context: T)(p: LazyParsley[A], reason: String): U[A] = visitUnary(self, context)(p)
    override def visit[A](self: ErrorAmend[A], context: T)(p: LazyParsley[A], partial: Boolean): U[A] = visitUnary(self, context)(p)
    override def visit[A](self: ErrorEntrench[A], context: T)(p: LazyParsley[A]): U[A] = visitUnary(self, context)(p)
    override def visit[A](self: ErrorDislodge[A], context: T)(n: Int, p: LazyParsley[A]): U[A] = visitUnary(self, context)(p)
    override def visit[A](self: ErrorLexical[A], context: T)(p: LazyParsley[A]): U[A] = visitUnary(self, context)(p)
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy