parsley.internal.machine.errors.DefuncBuilders.scala Maven / Gradle / Ivy
The newest version!
/*
* Copyright 2020 Parsley Contributors
*
* SPDX-License-Identifier: BSD-3-Clause
*/
package parsley.internal.machine.errors
import scala.collection.mutable
import parsley.XAssert._
import parsley.internal.errors.{CaretWidth, EndOfInput, ExpectItem, FancyError, TrivialError, UnexpectDesc, UnexpectItem}
import TrivialErrorBuilder.{BuilderUnexpectItem, EmptyItem, NoItem, Other, Raw}
/** When building a true `TrivialError` from a `TrivialDefuncError`, this builder is used to keep track of the
* components that will make up the final error. This allows for more efficient construction of the object,
* as it can internally use mutable collections, before finally freezing them into an immutable form.
*
* @param offset the offset that the error being built occured at
* @param outOfRange whether or not this error occured at the end of the input or not
*/
private [errors] final class TrivialErrorBuilder(presentationOffset: Int, outOfRange: Boolean, lexicalError: Boolean) {
private var line: Int = _
private var col: Int = _
private val expecteds = mutable.Set.empty[ExpectItem]
private var unexpected = EmptyItem
private val reasons = mutable.Set.empty[String]
private var acceptingExpected = true
/** Updates the position of the error message.
*
* @param line the new line number
* @param col the new column number
*/
def pos_=(line: Int, col: Int): Unit = {
this.line = line
this.col = col
}
/** Updates the width of the raw unexpected token generated by the error: this may expand its width,
* but will not shrink it. If a raw unexpected token is not going to be generated, this has no
* effect.
*
* If this error is `outOfRange`, `EndOfInput` is generated instead.
*
* @param width
*/
def updateUnexpected(width: Int): Unit = {
assume(width != 0 || this.unexpected.pickHigher(EmptyItem) == this.unexpected, s"trying to pick empty item never does anything (${this.unexpected} becomes ${this.unexpected.pickHigher(EmptyItem)})")
//if (width == 0) updateUnexpected(EmptyItem)
if (width != 0) {
if (outOfRange) updateUnexpected(new Other(EndOfInput))
else updateUnexpected(new Raw(width))
}
}
/** Updates the unexpected token generated by the error, so long as this new item takes precedence
* over the old one.
*
* @param item
*/
def updateUnexpected(item: UnexpectDesc): Unit = updateUnexpected(new Other(item))
/** Updates the unexpected (but empty) token generated by the error, so long as no other error item
* has been used.
* @param width
*/
def updateEmptyUnexpected(width: Int): Unit = updateUnexpected(new NoItem(width))
private def updateUnexpected(item: BuilderUnexpectItem): Unit = this.unexpected = this.unexpected.pickHigher(item)
/** If this error is accepting new expected items, adds all the given items into the error
*
* @param expecteds the items to add
*/
def ++=(expecteds: Iterable[ExpectItem]): Unit = if (acceptingExpected) this.expecteds ++= expecteds
/** Adds the given reason onto the end of the errors reasons.
*
* @param reason the reason to add
*/
def +=(reason: String): Unit = this.reasons += reason
/** Builds the final error within the context of some builder for error items.
*
* @param itemBuilder a builder for constructing unexpected error items from the raw input of the parse
* @return the final error message
*/
def mkError(implicit itemBuilder: ErrorItemBuilder): TrivialError = {
new TrivialError(presentationOffset, line, col, unexpected.toErrorItem(presentationOffset), expecteds.toSet, reasons.toSet, lexicalError)
}
/** Performs some given action only when this builder is currently accepting new expected items
*
* @param action the action to perform
*/
def whenAcceptingExpected(action: =>Unit): Unit = if (acceptingExpected) action
/** For the duration of the given action, all expected items that are provided to the builder
* will be ignored.
*
* @param action the action to perform, during which no expected items can be accepted
*/
def ignoreExpected(action: =>Unit): Unit = {
val old = acceptingExpected
acceptingExpected = false
action
acceptingExpected = old
}
/** Makes a `HintCollector` for collecting any hints from a `DefuncHints`
* directly into this builder's expected error items.
*/
def makeHintCollector: HintCollector = new HintCollector(expecteds)
}
private [errors] object TrivialErrorBuilder {
final def EmptyItem: BuilderUnexpectItem = new NoItem(0)
private [TrivialErrorBuilder] sealed abstract class BuilderUnexpectItem {
def pickHigher(other: BuilderUnexpectItem): BuilderUnexpectItem
protected [TrivialErrorBuilder] def pickRaw(other: Raw): BuilderUnexpectItem
protected [TrivialErrorBuilder] def pickOther(other: Other): Other
protected [TrivialErrorBuilder] def pickNoItem(other: NoItem): BuilderUnexpectItem
def toErrorItem(offset: Int)(implicit builder: ErrorItemBuilder): Either[Int, UnexpectItem]
}
private [TrivialErrorBuilder] final class Raw(val size: Int) extends BuilderUnexpectItem {
assume(size > 0, "Raw should not try and represent 0-width tokens")
final def pickHigher(other: BuilderUnexpectItem): BuilderUnexpectItem = other.pickRaw(this)
final override def pickRaw(other: Raw): Raw = if (this.size > other.size) this else other
final override def pickOther(other: Other): Other = other.pickRaw(this)
final override def pickNoItem(other: NoItem): Raw = if (this.size >= other.width) this else new Raw(other.width)
def toErrorItem(offset: Int)(implicit builder: ErrorItemBuilder): Either[Int, UnexpectItem] = Right(builder(offset, size))
}
private [TrivialErrorBuilder] final class Other(val underlying: UnexpectItem) extends BuilderUnexpectItem {
final def pickHigher(other: BuilderUnexpectItem): BuilderUnexpectItem = other.pickOther(this)
final override def pickRaw(other: Raw): Other = {
if (underlying.isFlexible) new Other(underlying.widen(other.size))
else this
}
final override def pickOther(other: Other): Other = if (this.underlying.higherPriority(other.underlying)) this else other
final override def pickNoItem(other: NoItem): Other = {
if (underlying.isFlexible) new Other(underlying.widen(other.width))
else this
}
def toErrorItem(offset: Int)(implicit builder: ErrorItemBuilder): Either[Int, UnexpectItem] = Right(underlying)
}
private [TrivialErrorBuilder] class NoItem(val width: Int) extends BuilderUnexpectItem {
assume(width >= 0, "NoItem should not try and represent negative tokens")
final def pickHigher(other: BuilderUnexpectItem): BuilderUnexpectItem = other.pickNoItem(this)
final override def pickRaw(other: Raw): Raw = other.pickNoItem(this)
final override def pickOther(other: Other): Other = other.pickNoItem(this)
final override def pickNoItem(other: NoItem): NoItem = if (this.width > other.width) this else other
def toErrorItem(offset: Int)(implicit builder: ErrorItemBuilder): Either[Int, UnexpectItem] = Left(width)
}
}
/** When building a true `FancyError` from a `FancyDefuncError`, this builder is used to keep track of the
* components that will make up the final error. This allows for more efficient construction of the object,
* as it can internally use mutable collections, before finally freezing them into an immutable form.
*
* @param offset the offset that the error being built occured at
*/
private [errors] final class FancyErrorBuilder(presentationOffset: Int) {
private var line: Int = _
private var col: Int = _
private var caretWidth: Int = 0
private var flexibleCaret: Boolean = true
private val msgs = mutable.ListBuffer.empty[String]
/** Updates the position of the error message.
*
* @param line the new line number
* @param col the new column number
*/
def pos_=(line: Int, col: Int): Unit = {
this.line = line
this.col = col
}
def updateCaretWidth(width: Int): Unit = {
assume(flexibleCaret, "if we are updating direct from a TrivialError, we better be flexible!")
this.caretWidth = math.max(this.caretWidth, width)
}
def updateCaretWidth(width: CaretWidth): Unit = {
// if they match, just take the max
if (width.isFlexible == this.flexibleCaret) this.caretWidth = math.max(this.caretWidth, width.width)
// if they don't match and we are rigid, then we override the flexible caret, otherwise do nothing
else if (!width.isFlexible) {
this.caretWidth = width.width
this.flexibleCaret = false
}
}
/** Adds a collection of new error message lines into this error.
*
* @param msgs the messages to add
*/
def ++=(msgs: Seq[String]): Unit = this.msgs ++= msgs
/** Builds the final error. */
def mkError: FancyError = {
new FancyError(presentationOffset, line, col, msgs.toList.distinct, caretWidth)
}
}
/** This collects up hints generated by the ghosts of old error messages,
* which are usually piped directly into a builder's expected messages.
*
* @param hints the place to collect the items into
*/
private [errors] final class HintCollector(hints: mutable.Set[ExpectItem]) {
/** Creates a new collector that starts empty */
def this() = this(mutable.Set.empty)
private var width = Option.empty[Int]
/** Adds a new hint into the collector. */
def +=(hint: ExpectItem): Unit = this.hints += hint
/** Adds several hints into the collector. */
def ++=(hints: Iterable[ExpectItem]): Unit = this.hints ++= hints
/** Gets the width of the unexpected token that may have accompanied the original hints */
def unexpectWidth: Option[Int] = width
/** Updates the width of the unexpected token that may have accompanied these hints: this
* can only get wider.
*/
def updateWidth(sz: Int): Unit = width = Some(Math.max(width.getOrElse(0), sz))
/** Generates an immutable snapshot of this collector */
def mkSet: Set[ExpectItem] = this.hints.toSet
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy