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

parsley.errors.DefaultErrorBuilder.scala Maven / Gradle / Ivy

There is a newer version: 5.0.0-M10
Show newest version
/*
 * Copyright 2020 Parsley Contributors 
 *
 * SPDX-License-Identifier: BSD-3-Clause
 */
package parsley.errors

import org.typelevel.scalaccompat.annotation.unused

// Turn coverage off, because the tests have their own error builder
// We might want to test this on its own though
// $COVERAGE-OFF$
/** This class us used to build Parsley's default error messages.
  *
  * While it compiles with the `ErrorBuilder` typeclass, it should not
  * be considered a stable contract: the formatting can be changed at any
  * time and without notice. The API, however, will remain stable.
  *
  * @since 3.0.0
  * @note this class is abstract as it does not implement `unexpectedToken`: when creating an instance mix-in an appropriate
  *       token extractor from `parsley.errors.tokenextractors`.
  * @group formatting
  */
abstract class DefaultErrorBuilder extends ErrorBuilder[String] {
    /** @inheritdoc */
    override def build(pos: Position, source: Source, lines: ErrorInfoLines): String = DefaultErrorBuilder.build(pos, source, lines)

    //override def build(pos: Position, source: Source, ctxs: NestedContexts, lines: ErrorInfoLines): String = {
    //    DefaultErrorBuilder.blockError(header = s"${DefaultErrorBuilder.mergeScopes(source, ctxs)}$pos", lines, indent = 2)"
    //}

    /** @inheritdoc */
    type Position = String
    /** @inheritdoc */
    type Source = Option[String]
    //type Context = Option[String]
    /** @inheritdoc */
    override def pos(line: Int, col: Int): Position = DefaultErrorBuilder.pos(line, col)
    /** @inheritdoc */
    override def source(sourceName: Option[String]): Source = DefaultErrorBuilder.source(sourceName)
    //override def contexualScope(context: String): Context = ???

    //type NestedContexts = Option[String]
    /*override def nestContexts(contexts: List[Context]): NestedContexts = ???*/

    /** @inheritdoc */
    type ErrorInfoLines = Seq[String]
    /** @inheritdoc */
    override def vanillaError(unexpected: UnexpectedLine, expected: ExpectedLine, reasons: Messages, lines: LineInfo): ErrorInfoLines = {
        DefaultErrorBuilder.vanillaError(unexpected, expected, reasons, lines)
    }
    /** @inheritdoc */
    override def specializedError(msgs: Messages, lines: LineInfo): ErrorInfoLines = DefaultErrorBuilder.specializedError(msgs, lines)

    /** @inheritdoc */
    type ExpectedItems = Option[String]
    /** @inheritdoc */
    type Messages = Seq[Message]
    /** @inheritdoc */
    override def combineExpectedItems(alts: Set[Item]): ExpectedItems = DefaultErrorBuilder.disjunct(alts)
    /** @inheritdoc */
    override def combineMessages(alts: Seq[Message]): Messages = DefaultErrorBuilder.combineMessages(alts)

    /** @inheritdoc */
    type UnexpectedLine = Option[String]
    /** @inheritdoc */
    type ExpectedLine = Option[String]
    /** @inheritdoc */
    type Message = String
    /** @inheritdoc */
    type LineInfo = Seq[String]
    /** @inheritdoc */
    override def unexpected(item: Option[Item]): UnexpectedLine = DefaultErrorBuilder.unexpected(item)
    /** @inheritdoc */
    override def expected(alts: ExpectedItems): ExpectedLine = DefaultErrorBuilder.expected(alts)
    /** @inheritdoc */
    override def reason(reason: String): Message = DefaultErrorBuilder.reason(reason)
    /** @inheritdoc */
    override def message(msg: String): Message = DefaultErrorBuilder.message(msg)

    /** @inheritdoc */
    override val numLinesBefore = DefaultErrorBuilder.NumLinesBefore
    /** @inheritdoc */
    override val numLinesAfter = DefaultErrorBuilder.NumLinesAfter
    /** @inheritdoc */
    override def lineInfo(line: String, linesBefore: Seq[String], linesAfter: Seq[String], lineNum: Int, errorPointsAt: Int, errorWidth: Int): LineInfo = {
        DefaultErrorBuilder.lineInfo(line, linesBefore, linesAfter, lineNum, errorPointsAt, errorWidth)
    }

    /** @inheritdoc */
    type Item = String
    /** @inheritdoc */
    type Raw = String
    /** @inheritdoc */
    type Named = String
    /** @inheritdoc */
    type EndOfInput = String
    /** @inheritdoc */
    override def raw(item: String): Raw = DefaultErrorBuilder.raw(item)
    /** @inheritdoc */
    override def named(item: String): Named = DefaultErrorBuilder.named(item)
    /** @inheritdoc */
    override val endOfInput: EndOfInput = DefaultErrorBuilder.EndOfInput
}
/** Helper functions used to build the `DefaultErrorBuilder` error messages.
  *
  * @since 4.3.0
  * @group formatting
  */
object DefaultErrorBuilder {
    final val Unknown = "unknown parse error"
    final val EndOfInput = "end of input"
    final val ErrorLineStart = ">"
    final val NumLinesBefore = 1
    final val NumLinesAfter = 1

    /** Forms an error message with `blockError`, with two spaces of indentation and
      * incorporating the source file and position into the header.
      *
      * @since 4.3.0
      */
    def build(pos: String, source: Option[String], lines: Seq[String]): String = {
        blockError(header = s"${source.fold("")(name => s"In $name ")}$pos", lines, indent = 2)
    }
    /** If the `sourceName` exists, wraps it in quotes and adds `file` onto the front.
      *
      * @since 4.3.0
      */
    def source(sourceName: Option[String]): Option[String] = sourceName.map(name => s"file '$name'")
    /** Forms a vanilla error by combining all the components in sequence, if there is no information
      * other than the `lines`, [[Unknown `Unknown`]] is used instead.
      *
      * @since 4.3.0
      */
    def vanillaError(unexpected: Option[String], expected: Option[String], reasons: Iterable[String], lines: Seq[String]): Seq[String] = {
        DefaultErrorBuilder.combineInfoWithLines(Seq.concat(unexpected, expected, reasons), lines)
    }
    /** Forms a specialized error by combining all components in sequence, if there are no `msgs`, then
      * [[Unknown `Unknown`]] is used instead.
      *
      * @since 4.3.0
      */
    def specializedError(msgs: Seq[String], lines: Seq[String]): Seq[String] = DefaultErrorBuilder.combineInfoWithLines(msgs, lines)

    /** Forms an error with the given `header` followed by a colon, a newline, then the remainder of the lines indented.
      *
      * @since 4.3.0
      */
    def blockError(header: String, lines: Iterable[String], indent: Int): String = s"$header:\n${indentAndUnlines(lines, indent)}"
    /** Indents and concatenates the given lines by the given depth.
      *
      * @since 4.3.0
      */
    def indentAndUnlines(lines: Iterable[String], indent: Int): String = lines.mkString(" " * indent, "\n" + " " * indent, "")

    /** Pairs the line and column up in the form `(line m, column n)`.
      *
      * @since 4.3.0
      */
    def pos(line: Int, col: Int): String = s"(line ${Integer.toUnsignedString(line)}, column ${Integer.toUnsignedString(col)})"

    /** Combines the alternatives, separated by commas/semicolons, with the final two separated
      * by "or". An ''Oxford comma'' is added if there are more than two elements, as this
      * helps prevent ambiguity in the list. If the elements contain a comma, then semicolon
      * is used as the list separator.
      *
      * @since 4.3.0
      */
    def disjunct(alts: Iterable[String]): Option[String] = disjunct(alts, oxfordComma = true)
    /** Combines the alternatives, separated by commas/semicolons, with the final two separated
      * by "or". If the elements contain a comma, then semicolon
      * is used as the list separator.
      *
      * @param oxfordComma decides whether or not to employ an ''Oxford comma'' when there
      *                    more than two elements to join: this helps prevent ambiguity in the list.
      * @since 4.3.0
      */
    def disjunct(alts: Iterable[String], oxfordComma: Boolean): Option[String] = helpers.disjunct(alts.toList.filter(_.nonEmpty), oxfordComma)

    /** Filters out any empty messages and returns the rest.
      *
      * @since 4.3.0
      */
    def combineMessages(alts: Seq[String]): Seq[String] = alts.filter(_.nonEmpty)

    /** Joins together the given sequences: if the first is empty, then [[Unknown `Unknown`]]
      * is prepended onto `lines` instead.
      *
      * @since 4.3.0
      */
    def combineInfoWithLines(info: Seq[String], lines: Seq[String]): Seq[String] = {
        if (info.isEmpty) Unknown +: lines
        else info ++: lines
    }

    /** Adds "unexpected " before the given item should it exist.
      *
      * @since 4.3.0
      */
    def unexpected(item: Option[String]): Option[String] = item.map("unexpected " + _)
    /** Adds "expected " before the given alternatives should they exist.
      *
      * @since 4.3.0
      */
    def expected(alts: Option[String]): Option[String] = alts.map("expected " + _)
    /** Returns the given reason unchanged.
      *
      * @since 4.3.0
      */
    def reason(reason: String): String = reason
    /** Returns the given message unchanged.
      *
      * @since 4.3.0
      */
    def message(msg: String): String = msg

    /** If the given item is either a whitespace character or is otherwise "unprintable",
      * a special name is given to it, otherwise the item is enclosed in double-quotes.
      *
      * @since 4.3.0
      */
    def raw(item: String): String = helpers.renderRawString(item)
    /** Returns the given item unchanged.
      *
      * @since 4.3.0
      */
    def named(item: String): String = item

    /** Constructs error context by concatenating them together with a "caret line" underneath the
      * focus line, `line`, where the error occurs.
      *
      * @since 4.3.0
      */
    def lineInfo(line: String, linesBefore: Seq[String], linesAfter: Seq[String], @unused lineNum: Int, errorPointsAt: Int, errorWidth: Int): Seq[String] = {
        Seq.concat(linesBefore.map(inputLine), Seq(inputLine(line), caretLine(errorPointsAt, errorWidth)), linesAfter.map(inputLine))
    }

    /** Adds the [[ErrorLineStart `ErrorLineStart`]] character to the front of the given line.
      *
      * @since 4.3.0
      */
    def inputLine(line: String): String = s"$ErrorLineStart$line"
    /** Generates a line of `^` characters as wide as specified starting as seen in as the given
      * position, accounting for the length of the [[ErrorLineStart `ErrorLineStart`]] too.
      *
      * @since 4.3.0
      */
    def caretLine(caretAt: Int, caretWidth: Int): String = s"${" " * (ErrorLineStart.length + caretAt)}${"^" * caretWidth}"

    /*def mergeScopes(source: Option[String], ctxs: Option[String]): String = (source, ctxs) match {
        case (None, None) => ""
        case (Some(name), None) => s"In $name "
        case (None, Some(ctxs)) => s"In $ctxs "
        case (Some(name), Some(ctxs)) => s"In $name, $ctxs "
    }*/

    /*def nestContexts(contexts: List[Option[String]]): Option[String] = {
        val nonEmptyContexts = contexts.flatten
        if (nonEmptyContexts.nonEmpty) Some(nonEmptyContexts.mkString(", "))
        else None
    }*/
    /*def contexualScope(context: String): Option[String] = Some(context)*/
}
// $COVERAGE-ON$




© 2015 - 2025 Weber Informatics LLC | Privacy Policy