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

parsley.internal.machine.instructions.DebugInstrs.scala Maven / Gradle / Ivy

There is a newer version: 5.0.0-M6
Show newest version
/* SPDX-FileCopyrightText: © 2022 Parsley Contributors 
 * SPDX-License-Identifier: BSD-3-Clause
 */
// $COVERAGE-OFF$
package parsley.internal.machine.instructions

import scala.annotation.nowarn

import parsley.XAssert._
import parsley.XCompat.unused
import parsley.errors.ErrorBuilder

import parsley.internal.errors.{ExpectItem, FancyError, ParseError, TrivialError}
import parsley.internal.machine.Context
import parsley.internal.machine.XAssert._

import Indenter.indentAndUnlines
import InputSlicer.Pad
import PrettyPortal.{Direction, Enter, Exit}

private [instructions] trait Colours {
    val ascii: Boolean

    final val newline = green("↙")
    final val space = white("·")
    final val endOfInput = red("•")

    final private def colour(str: String, colour: String): String = {
        if (ascii || parsley.debug.renderAscii) str else s"$colour$str${Console.RESET}"
    }
    final protected def green(str: String): String = colour(str, Console.GREEN)
    final protected def blue(str: String): String = colour(str, Console.BLUE)
    final protected def red(str: String): String = colour(str, Console.RED)
    final protected def white(str: String): String = colour(str, Console.WHITE)
}

private [instructions] trait PrettyPortal {
    val name: String

    final protected def portal(dir: Direction, @unused ctx: Context) = s"${dir.render}$name${dir.render}"
}
private [instructions] object PrettyPortal {
    sealed abstract class Direction {
        def render: String
    }
    case object Enter extends Direction {
        override val render: String = ">"
    }
    case object Exit extends Direction {
        override val render: String = "<"
    }
}

private [instructions] object Indenter {
    final val IndentWidth: Int = 2
    private def indent(ctx: Context) = " " * (ctx.debuglvl * 2)
    def indentAndUnlines(ctx: Context, lines: String*): String = {
        lines.map(line => s"${indent(ctx)}$line").mkString("\n")
    }
}

private [instructions] trait InputSlicer { this: Colours =>
    private def start(ctx: Context): Int = Math.max(ctx.offset - Pad, 0)
    private def end(ctx: Context): Int = Math.min(ctx.offset + Pad + 1, ctx.inputsz)
    protected final def slice(ctx: Context): String = {
        val end = this.end(ctx)
        val s = ctx.input.mkString.substring(start(ctx), end).replace("\n", newline)
                                                             .replace(" ", space)
        if (end == ctx.inputsz) s"$s$endOfInput" else s
    }
    protected final def caret(ctx: Context): String = {
        " " * (ctx.offset - start(ctx)) + blue("^")
    }
}
private [instructions] object InputSlicer {
    final val Pad: Int = 5
}

private [instructions] trait Logger extends PrettyPortal with InputSlicer with Colours {
    final protected def preludeString(dir: Direction, ctx: Context, ends: String) = {
        val input = this.slice(ctx)
        val prelude = s"${portal(dir, ctx)} (${ctx.line}, ${ctx.col}): "
        val caret = (" " * prelude.length) + this.caret(ctx)
        indentAndUnlines(ctx, s"$prelude$input$ends", caret)
    }
    final protected def doBreak(ctx: Context): Unit = {
        print(indentAndUnlines(ctx,
                s"{stack: ${ctx.stack.mkString(", ")}}",
                s"{registers: ${ctx.regs.zipWithIndex.map{case (x, i) => s"r$i: $x"}.mkString("[", ", ", "])}")}}",
                "..."))
        Console.in.read(): @nowarn
    }
}

private [internal] final class LogBegin(var label: Int, override val name: String, override val ascii: Boolean, break: Boolean)
    extends InstrWithLabel with Logger {
    override def apply(ctx: Context): Unit = {
        ensureRegularInstruction(ctx)
        println(preludeString(Enter, ctx, ""))
        if (break) doBreak(ctx)
        ctx.debuglvl += 1
        ctx.pushHandler(label)
        ctx.inc()
    }
    override def toString: String = s"LogBegin($label, $name)"
}

private [internal] final class LogEnd(val name: String, override val ascii: Boolean, break: Boolean) extends Instr with Logger {
    override def apply(ctx: Context): Unit = {
        assert(ctx.running, "cannot wrap a Halt with a debug")
        ctx.debuglvl -= 1
        val end = " " + {
            if (ctx.good) {
                ctx.handlers = ctx.handlers.tail
                ctx.inc()
                green("Good")
            }
            else {
                ctx.fail()
                red("Fail")
            }
        }
        println(preludeString(Exit, ctx, end))
        if (break) doBreak(ctx)
    }
    override def toString: String = s"LogEnd($name)"
}

private [instructions] trait ErrLogger extends PrettyPortal with Colours {
    final protected def preludeString(dir: Direction, ctx: Context, ends: String) = {
        val prelude = s"${portal(dir, ctx)} (offset ${ctx.offset}, line ${ctx.line}, col ${ctx.col})"
        indentAndUnlines(ctx, s"$prelude$ends")
    }
    /*
    final protected def doBreak(ctx: Context): Unit = {
        print(indentAndUnlines(ctx,
                s"{stack: ${ctx.stack.mkString(", ")}}",
                s"{registers: ${ctx.regs.zipWithIndex
                .map{case (x, i) => s"r$i: $x"}.mkString("[", ", ", "])}")}}",
                "..."))
        Console.in.read()
    }
    */
}

private [instructions] final case class ErrLogData(hintsOffset: Int, hints: Set[ExpectItem]) {
    def newHints(newHintsOffset: Int, newHints: Set[ExpectItem]): Set[ExpectItem] = {
        if (stillValid(newHintsOffset)) newHints &~ hints
        else newHints
    }
    def stillValid(newHintsOffset: Int): Boolean = hintsOffset == newHintsOffset
}

private [internal] final class LogErrBegin(var label: Int, override val name: String, override val ascii: Boolean)(implicit errBuilder: ErrorBuilder[_])
    extends InstrWithLabel with ErrLogger {
    override def apply(ctx: Context): Unit = {
        ensureRegularInstruction(ctx)
        val inFlightHints = ctx.inFlightHints.toSet
        // This should print out a classic opening line, followed by the currently in-flight hints
        println(preludeString(Enter, ctx, s": current hints are ${inFlightHints.map(_.formatExpect)} (valid at offset ${ctx.currentHintsValidOffset})"))
        ctx.debuglvl += 1
        ctx.stack.push(ErrLogData(ctx.currentHintsValidOffset, inFlightHints))
        ctx.pushHandler(label)
        ctx.inc()
    }
    override def toString: String = s"LogErrBegin($label, $name)"
}

private [internal] final class LogErrEnd(override val name: String, override val ascii: Boolean)(implicit errBuilder: ErrorBuilder[_])
    extends Instr with ErrLogger {
    override def apply(ctx: Context): Unit = {
        assert(ctx.running, "cannot wrap a Halt with a debug")
        ctx.debuglvl -= 1
        @unused val currentHintsValidOffset = ctx.currentHintsValidOffset
        if (ctx.good) {
            // In this case, the currently in-flight hints should be reported
            val oldData = ctx.stack.pop[ErrLogData]()
            val inFlightHints = ctx.inFlightHints.toSet
            val formattedInFlight = inFlightHints.map(_.formatExpect)
            val msgInit = s": ${green("Good")}, current hints are $formattedInFlight with"
            ctx.handlers = ctx.handlers.tail
            if (!oldData.stillValid(ctx.currentHintsValidOffset)) {
                println(preludeString(Exit, ctx, s"$msgInit old hints discarded (valid at offset ${ctx.currentHintsValidOffset})"))
            }
            else {
                val newHints = oldData.newHints(ctx.currentHintsValidOffset, inFlightHints)
                if (newHints.size == inFlightHints.size) {
                    println(preludeString(Exit, ctx, s"$msgInit all added since entry to debug (valid at offset ${ctx.currentHintsValidOffset})"))
                }
                else {
                    val formattedNewHints = oldData.newHints(ctx.currentHintsValidOffset, inFlightHints).map(_.formatExpect)
                    val msg = s"$msgInit $formattedNewHints added since entry to debug (valid at offset ${ctx.currentHintsValidOffset})"
                    println(preludeString(Exit, ctx, msg))
                }
            }
            ctx.inc()
        }
        else {
            // In this case, the current top of stack error message is reported
            // For this, there needs to be a verbose mode that prints out the composite error
            // as opposed to the simplified error (post-merge)
            // there should be different levels of verbose flags, such that we can also display the expecteds _under_ a label
            @unused val oldData = ctx.stack.peek[ErrLogData]
            val defuncErr = ctx.inFlightError
            val err = defuncErr.asParseError(ctx.errorItemBuilder)
            println(preludeString(Exit, ctx, s": ${red("Fail")}"))
            println(Indenter.indentAndUnlines(ctx, LogErrEnd.format(err): _*))
            ctx.fail()
        }
    }
    override def toString: String = s"LogErrEnd($name)"
}
private [instructions] object LogErrEnd {
    // TODO: We want to mark errors that are behind the current context offsets as amended in some way
    // ideally we can provide meta-information about merges and so on too: this needs to be extracted by
    // analysing the DefuncError itself! (Note that StringTok does not amend its error via operation,
    // it does so by direct manipulation. It should be treated as an amend regardless here).
    // Surpressed labels can be presumably captured by calling asParseErrorSlow under a label node
    // again, requiring a deeper traversal of the tree
    def format(err: ParseError): Seq[String] = err match {
        case FancyError(offset, line, col, msgs, _, _) => s"generated specialised error (offset $offset, line $line, col $col) {" +: msgs :+ "}"
        case TrivialError(offset, line, col, unexpected, expecteds, reasons, lexicalError) =>
            Seq(s"generated vanilla error (offset $offset, line $line, col $col) {",
                s"  unexpected item = ${unexpected.fold(_ => "missing", _.formatUnexpect(lexicalError)._1.toString)}",
                s"  expected item(s) = ${expecteds.map(_.formatExpect)}",
                s"  reasons =${if (reasons.isEmpty) " no reasons given" else ""}") ++
                reasons.map("  " + _) :+
                "}"
    }
}
// $COVERAGE-ON$




© 2015 - 2025 Weber Informatics LLC | Privacy Policy