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

parsley.token.errors.ErrorConfig.scala Maven / Gradle / Ivy

/*
 * Copyright 2020 Parsley Contributors 
 *
 * SPDX-License-Identifier: BSD-3-Clause
 */
package parsley.token.errors

import org.typelevel.scalaccompat.annotation.unused

/** This class is used to specify how errors should be produced by the
  * [[parsley.token.Lexer `Lexer`]] class.
  *
  * The [[parsley.token.Lexer `Lexer`]] is set up to produce a variety of different
  * errors via `label`-ing, `explain`-ing, and `filter`-ing, and some applications of
  * the ''Verified'' and ''Preventative'' error patterns. The exact content of those
  * errors can be configured here. Errors can be suppressed or specified with different
  * levels of detail, or even switching between ''vanilla'' or ''specialised'' errors.
  *
  * This class should be used by extending it and overriding the relevant parts: all
  * methods here are non-abstract and their default is documented inside. Not configuring
  * something does not mean it will not appear in the message, but will mean it uses the
  * underlying base errors.
  *
  * @since 4.1.0
  * @group errconfig
  *
  * @groupprio numeric 0
  * @groupname numeric Numeric Errors
  * @groupdesc numeric These control the errors generated with the `numeric` component of the `Lexer`.
  *
  * @groupprio text 0
  * @groupname text Text Errors
  * @groupdesc text These control the errors generated with the `text` component of the `Lexer`.
  *
  * @groupprio names 0
  * @groupname names Name Errors
  * @groupdesc names These control the errors generated with the `names` component of the `Lexer`.
  *
  * @groupprio symbol 0
  * @groupname symbol Symbol Errors
  * @groupdesc symbol These control the errors generated with the `symbol` component of the `Lexer`.
  *
  * @groupprio space 0
  * @groupname space Space Errors
  * @groupdesc space These control the errors generated with the `space` component of the `Lexer`.
  */
class ErrorConfig {
    // numeric
    /** How a numeric break character should (like `_`) be referred to or explained within an error.
      * @since 4.1.0
      * @group numeric
      */
    def labelNumericBreakChar: LabelWithExplainConfig = NotConfigured

    /** How unsigned decimal integers should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to `labelIntegerUnsignedNumber`
      * @group numeric
      */
    def labelIntegerUnsignedDecimal: LabelWithExplainConfig = labelIntegerUnsignedNumber
    /** How unsigned hexadecimal integers should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to `labelIntegerUnsignedNumber`
      * @group numeric
      */
    def labelIntegerUnsignedHexadecimal: LabelWithExplainConfig = labelIntegerUnsignedNumber
    /** How unsigned octal integers should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to `labelIntegerUnsignedNumber`
      * @group numeric
      */
    def labelIntegerUnsignedOctal: LabelWithExplainConfig = labelIntegerUnsignedNumber
    /** How unsigned binary integers should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to `labelIntegerUnsignedNumber`
      * @group numeric
      */
    def labelIntegerUnsignedBinary: LabelWithExplainConfig = labelIntegerUnsignedNumber
    /** How generic unsigned integers should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group numeric
      */
    def labelIntegerUnsignedNumber: LabelWithExplainConfig = NotConfigured
    /** How unsigned decimal integers should of a given bit-width be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[parsley.token.errors.ErrorConfig.labelIntegerUnsignedNumber(bits:Int):* `labelIntegerUnsignedNumber`]]
      * @group numeric
      */
    def labelIntegerUnsignedDecimal(@unused bits: Int): LabelWithExplainConfig = labelIntegerUnsignedDecimal
    /** How unsigned hexadecimal integers should of a given bit-width be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[parsley.token.errors.ErrorConfig.labelIntegerUnsignedNumber(bits:Int):* `labelIntegerUnsignedNumber`]]
      * @group numeric
      */
    def labelIntegerUnsignedHexadecimal(@unused bits: Int): LabelWithExplainConfig = labelIntegerUnsignedHexadecimal
    /** How unsigned octal integers should of a given bit-width be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[parsley.token.errors.ErrorConfig.labelIntegerUnsignedNumber(bits:Int):* `labelIntegerUnsignedNumber`]]
      * @group numeric
      */
    def labelIntegerUnsignedOctal(@unused bits: Int): LabelWithExplainConfig = labelIntegerUnsignedOctal
    /** How unsigned binary integers should of a given bit-width be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[parsley.token.errors.ErrorConfig.labelIntegerUnsignedNumber(bits:Int):* `labelIntegerUnsignedNumber`]]
      * @group numeric
      */
    def labelIntegerUnsignedBinary(@unused bits: Int): LabelWithExplainConfig = labelIntegerUnsignedBinary
    /** How generic unsigned integers should of a given bit-width be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group numeric
      */
    def labelIntegerUnsignedNumber(@unused bits: Int): LabelWithExplainConfig = labelIntegerUnsignedNumber

    /** How signed decimal integers should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to `labelIntegerSignedNumber`
      * @group numeric
      */
    def labelIntegerSignedDecimal: LabelWithExplainConfig = labelIntegerSignedNumber
    /** How signed hexadecimal integers should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to `labelIntegerSignedNumber`
      * @group numeric
      */
    def labelIntegerSignedHexadecimal: LabelWithExplainConfig = labelIntegerSignedNumber
    /** How signed octal integers should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to `labelIntegerSignedNumber`
      * @group numeric
      */
    def labelIntegerSignedOctal: LabelWithExplainConfig = labelIntegerSignedNumber
    /** How signed binary integers should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to `labelIntegerSignedNumber`
      * @group numeric
      */
    def labelIntegerSignedBinary: LabelWithExplainConfig = labelIntegerSignedNumber
    /** How generic signed integers should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group numeric
      */
    def labelIntegerSignedNumber: LabelWithExplainConfig = NotConfigured
    /** How signed decimal integers should of a given bit-width be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[parsley.token.errors.ErrorConfig.labelIntegerSignedDecimal(bits:Int):* `labelIntegerSignedDecimal`]]
      * @group numeric
      */
    def labelIntegerSignedDecimal(@unused bits: Int): LabelWithExplainConfig = labelIntegerSignedDecimal
    /** How signed hexadecimal integers should of a given bit-width be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[parsley.token.errors.ErrorConfig.labelIntegerSignedHexadecimal(bits:Int):* `labelIntegerSignedHexadecimal`]]
      * @group numeric
      */
    def labelIntegerSignedHexadecimal(@unused bits: Int): LabelWithExplainConfig = labelIntegerSignedHexadecimal
    /** How signed octal integers should of a given bit-width be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[parsley.token.errors.ErrorConfig.labelIntegerSignedOctal(bits:Int):* `labelIntegerSignedOctal`]]
      * @group numeric
      */
    def labelIntegerSignedOctal(@unused bits: Int): LabelWithExplainConfig = labelIntegerSignedOctal
    /** How signed binary integers should of a given bit-width be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[parsley.token.errors.ErrorConfig.labelIntegerSignedBinary(bits:Int):* `labelIntegerSignedBinary`]]
      * @group numeric
      */
    def labelIntegerSignedBinary(@unused bits: Int): LabelWithExplainConfig = labelIntegerSignedBinary
    /** How generic signed integers should of a given bit-width be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group numeric
      */
    def labelIntegerSignedNumber(@unused bits: Int): LabelWithExplainConfig = labelIntegerSignedNumber

    /** How the fact that the end of a decimal integer literal is expected should be referred to within an error.
      * @since 4.1.0
      * @note defaults to [[labelIntegerNumberEnd `labelIntegerNumberEnd`]]
      * @group numeric
      */
    def labelIntegerDecimalEnd: LabelConfig = labelIntegerNumberEnd
    /** How the fact that the end of a hexadecimal integer literal is expected should be referred to within an error.
      * @since 4.1.0
      * @note defaults to [[labelIntegerNumberEnd `labelIntegerNumberEnd`]]
      * @group numeric
      */
    def labelIntegerHexadecimalEnd: LabelConfig = labelIntegerNumberEnd
    /** How the fact that the end of an octal integer literal is expected should be referred to within an error.
      * @since 4.1.0
      * @note defaults to [[labelIntegerNumberEnd `labelIntegerNumberEnd`]]
      * @group numeric
      */
    def labelIntegerOctalEnd: LabelConfig = labelIntegerNumberEnd
    /** How the fact that the end of a binary integer literal is expected should be referred to within an error.
      * @since 4.1.0
      * @note defaults to [[labelIntegerNumberEnd `labelIntegerNumberEnd`]]
      * @group numeric
      */
    def labelIntegerBinaryEnd: LabelConfig = labelIntegerNumberEnd
    /** How the fact that the end of a generic integer literal is expected should be referred to within an error.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group numeric
      */
    def labelIntegerNumberEnd: LabelConfig = NotConfigured

    /** How decimal reals should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealNumber `labelRealNumber`]]
      * @group numeric
      */
    def labelRealDecimal: LabelWithExplainConfig = labelRealNumber
    /** How hexadecimal reals should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealNumber `labelRealNumber`]]
      * @group numeric
      */
    def labelRealHexadecimal: LabelWithExplainConfig = labelRealNumber
    /** How octal reals should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealNumber `labelRealNumber`]]
      * @group numeric
      */
    def labelRealOctal: LabelWithExplainConfig = labelRealNumber
    /** How binary reals should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealNumber `labelRealNumber`]]
      * @group numeric
      */
    def labelRealBinary: LabelWithExplainConfig = labelRealNumber
    /** How generic reals should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group numeric
      */
    def labelRealNumber: LabelWithExplainConfig = NotConfigured
    /** How decimal floats should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealDecimal `labelRealDecimal`]]
      * @group numeric
      */
    def labelRealFloatDecimal: LabelWithExplainConfig = labelRealDecimal
    /** How hexadecimal floats should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealHexadecimal `labelRealHexadecimal`]]
      * @group numeric
      */
    def labelRealFloatHexadecimal: LabelWithExplainConfig = labelRealHexadecimal
    /** How octal floats should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealOctal `labelRealOctal`]]
      * @group numeric
      */
    def labelRealFloatOctal: LabelWithExplainConfig = labelRealOctal
    /** How binary floats should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealBinary `labelRealBinary`]]
      * @group numeric
      */
    def labelRealFloatBinary: LabelWithExplainConfig = labelRealBinary
    /** How generic floats should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealNumber `labelRealNumber`]]
      * @group numeric
      */
    def labelRealFloatNumber: LabelWithExplainConfig = labelRealNumber
    /** How decimal doubles should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealDecimal `labelRealDecimal`]]
      * @group numeric
      */
    def labelRealDoubleDecimal: LabelWithExplainConfig = labelRealDecimal
    /** How hexadecimal doubles should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealHexadecimal `labelRealHexadecimal`]]
      * @group numeric
      */
    def labelRealDoubleHexadecimal: LabelWithExplainConfig = labelRealHexadecimal
    /** How octal doubles should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealOctal `labelRealOctal`]]
      * @group numeric
      */
    def labelRealDoubleOctal: LabelWithExplainConfig = labelRealOctal
    /** How binary doubles should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealBinary `labelRealBinary`]]
      * @group numeric
      */
    def labelRealDoubleBinary: LabelWithExplainConfig = labelRealBinary
    /** How generic doubles should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealNumber `labelRealNumber`]]
      * @group numeric
      */
    def labelRealDoubleNumber: LabelWithExplainConfig = labelRealNumber

    /** How the fact that the end of a decimal real literal is expected should be referred to within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealNumberEnd `labelRealNumberEnd`]]
      * @group numeric
      */
    def labelRealDecimalEnd: LabelConfig = labelRealNumberEnd
    /** How the fact that the end of a hexadecimal real literal is expected should be referred to within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealNumberEnd `labelRealNumberEnd`]]
      * @group numeric
      */
    def labelRealHexadecimalEnd: LabelConfig = labelRealNumberEnd
    /** How the fact that the end of an octal real literal is expected should be referred to within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealNumberEnd `labelRealNumberEnd`]]
      * @group numeric
      */
    def labelRealOctalEnd: LabelConfig = labelRealNumberEnd
    /** How the fact that the end of a binary real literal is expected should be referred to within an error.
      * @since 4.1.0
      * @note defaults to [[labelRealNumberEnd `labelRealNumberEnd`]]
      * @group numeric
      */
    def labelRealBinaryEnd: LabelConfig = labelRealNumberEnd
    /** How the fact that the end of a generic real literal is expected should be referred to within an error.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group numeric
      */
    def labelRealNumberEnd: LabelConfig = NotConfigured

    /** How the "dot" that separates the integer and fractional part of a real number should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group numeric
      */
    def labelRealDot: LabelWithExplainConfig = NotConfigured
    /** How the trailing exponents of a real number should be referred to or explained within an error.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group numeric
      */
    def labelRealExponent: LabelWithExplainConfig = NotConfigured
    /** How the fact that the end of an exponent part of a real literal is expected should be referred to within an error.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group numeric
      */
    def labelRealExponentEnd: LabelConfig = NotConfigured

    /** Even if leading and trailing zeros can be dropped, `.` is not a valid real number: this
      * method specifies how to report that to the user.
      * @since 4.1.0
      * @note defaults to a ''vanilla'' explain: "a real number cannot drop both a leading and trailing zero"
      * @group numeric
      */
    def preventRealDoubleDroppedZero: PreventDotIsZeroConfig = ZeroDotReason("a real number cannot drop both a leading and trailing zero")

    /** This method describes the content of the error when an integer literal is parsed and it is not within the
      * required bit-width.
      * @param min the smallest value the integer could have taken
      * @param max the largest value the integer could have taken
      * @param nativeRadix the radix that the integer was parsed using
      * @since 4.1.0
      * @note defaults to a ''specialised'' error describing what the min and max bounds are.
      * @group numeric
      */
    def filterIntegerOutOfBounds(min: BigInt, max: BigInt, nativeRadix: Int): FilterConfig[BigInt] = new SpecializedMessage[BigInt] {
        def message(n: BigInt) = Seq(s"literal is not within the range ${min.toString(nativeRadix)} to ${max.toString(nativeRadix)}")
    }

    /** This method describes the content of the error when a real literal is parsed and it is not representable exactly as the required precision.
      * @param name the name of the required precision (one of `doubleName` or `floatName`)
      * @since 4.1.0
      * @note defaults to a ''specialised'' error stating that the literal is not exactly representable.
      * @group numeric
      */
    def filterRealNotExact(name: String): FilterConfig[BigDecimal] = new SpecializedMessage[BigDecimal] {
        def message(n: BigDecimal) = Seq(s"literal cannot be represented exactly as an $name")
    }

    /** This method describes the content of the error when a real literal is parsed and it is not within the bounds perscribed by the required precision.
      * @param name the name of the required precision (one of `doubleName` or `floatName`)
      * @param min the smallest value the real could have taken
      * @param max the largest value the real could have taken
      * @since 4.1.0
      * @note defaults to a ''specialised'' error describing what the min and max bounds are.
      * @group numeric
      */
    def filterRealOutOfBounds(name: String, min: BigDecimal, max: BigDecimal): FilterConfig[BigDecimal] =
        new SpecializedMessage[BigDecimal] {
            def message(n: BigDecimal) = Seq(s"literal is not within the range $min to $max and is not an $name")
        }

    // expensive ;)
    // this is not as effective as it may seem, because reasons cannot be hints
    // it's possible a preventative error could be more effective?
    /*def verifiedIntegerBadCharsUsedInLiteral: Option[(predicate.CharPredicate, Int => String)] =
        None*/

    /** The name given to doubles.
      * @since 4.1.0
      * @note defaults to "IEEE 754 double-precision float"
      * @group numeric
      */
    def doubleName: String = "IEEE 754 double-precision float"
    /** The name given to floats.
      * @since 4.1.0
      * @note defaults to "IEEE 754 single-precision float"
      * @group numeric
      */
    def floatName: String = "IEEE 754 single-precision float"

    private [token] final def labelDecimal(bits: Int, signed: Boolean): LabelWithExplainConfig = {
        if (signed) labelIntegerSignedDecimal(bits) else labelIntegerUnsignedDecimal(bits)
    }
    private [token] final def labelHexadecimal(bits: Int, signed: Boolean): LabelWithExplainConfig = {
        if (signed) labelIntegerSignedHexadecimal(bits) else labelIntegerUnsignedHexadecimal(bits)
    }
    private [token] final def labelOctal(bits: Int, signed: Boolean): LabelWithExplainConfig = {
        if (signed) labelIntegerSignedOctal(bits) else labelIntegerUnsignedOctal(bits)
    }
    private [token] final def labelBinary(bits: Int, signed: Boolean): LabelWithExplainConfig = {
        if (signed) labelIntegerSignedBinary(bits) else labelIntegerUnsignedBinary(bits)
    }
    private [token] final def labelNumber(bits: Int, signed: Boolean): LabelWithExplainConfig = {
        if (signed) labelIntegerSignedNumber(bits) else labelIntegerUnsignedNumber(bits)
    }

    /** How an identifier should be referred to in an error message.
      * @since 4.1.0
      * @note defaults to "identifier"
      * @group names
      */
    def labelNameIdentifier: String = "identifier"
    /** How a user-defined operator should be referred to in an error message.
      * @since 4.1.0
      * @note defaults to "operator"
      * @group names
      */
    def labelNameOperator: String = "operator"
    /** How an illegally parsed hard keyword should be referred to as an unexpected component.
      * @param v the illegal identifier
      * @since 4.1.0
      * @note defaults to "keyword v"
      * @group names
      */
    def unexpectedNameIllegalIdentifier(v: String): String = s"keyword $v"
    /** How an illegally parsed hard operator should be referred to as an unexpected component.
      * @since 4.1.0
      * @note defaults to "reserved operator v"
      * @group names
      */
    def unexpectedNameIllegalOperator(v: String): String = s"reserved operator $v"
    /** When parsing identifiers that are required to have specific start characters, how bad identifiers should be reported.
      * @since 4.1.0
      * @note defaults to unexpected "identifier v"
      * @group names
      */
    def filterNameIllFormedIdentifier: FilterConfig[String] = new Unexpected[String] {
        def unexpected(v: String) = s"identifer $v"
    }
    /** When parsing operators that are required to have specific start/end characters, how bad operators should be reported.
      * @since 4.1.0
      * @note defaults to unexpected "operator v"
      * @group names
      */
    def filterNameIllFormedOperator: FilterConfig[String] = new Unexpected[String] {
        def unexpected(v: String) = s"operator $v"
    }

    /** How a ASCII character literal should be referred to or explained in error messages.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelCharAscii: LabelWithExplainConfig = NotConfigured
    /** How a Latin1 (extended ASCII) character literal should be referred to or explained in error messages.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelCharLatin1: LabelWithExplainConfig = NotConfigured
    /** How a BMP (Basic Multilingual Plane) character literal should be referred to or explained in error messages.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelCharBasicMultilingualPlane: LabelWithExplainConfig = NotConfigured
    /** How a UTF-16 character literal should be referred to or explained in error messages.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelCharUtf16: LabelWithExplainConfig = NotConfigured

    /** How the closing quote of an ASCII character literal should be referred to in error messages.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelCharAsciiEnd: LabelConfig = NotConfigured
    /** How the closing quote of a Latin1 character literal should be referred to in error messages.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelCharLatin1End: LabelConfig = NotConfigured
    /** How the closing quote of a BMP character literal should be referred to in error messages.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelCharBasicMultilingualPlaneEnd: LabelConfig = NotConfigured
    /** How the closing quote of a UTF-16 character literal should be referred to in error messages.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelCharUtf16End: LabelConfig = NotConfigured

    /** How a ASCII-only string literal should be referred to or explained in error messages.
      * @since 4.1.0
      * @param multi whether this is for multi-line strings
      * @param raw whether this is for raw strings
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelStringAscii(@unused multi: Boolean, @unused raw: Boolean): LabelWithExplainConfig = NotConfigured
    /** How a Latin1-only string literal should be referred to or explained in error messages.
      * @since 4.1.0
      * @param multi whether this is for multi-line strings
      * @param raw whether this is for raw strings
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelStringLatin1(@unused multi: Boolean, @unused raw: Boolean): LabelWithExplainConfig = NotConfigured
    /** How a UTF-16-only string should literal be referred to or explained in error messages.
      * @since 4.1.0
      * @param multi whether this is for multi-line strings
      * @param raw whether this is for raw strings
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelStringUtf16(@unused multi: Boolean, @unused raw: Boolean): LabelWithExplainConfig = NotConfigured

    /** How the closing quote(s) of an ASCII string literal should be referred to in error messages.
      * @since 4.1.0
      * @param multi whether this is for multi-line strings
      * @param raw whether this is for raw strings
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelStringAsciiEnd(@unused multi: Boolean, @unused raw: Boolean): LabelConfig = NotConfigured
    /** How the closing quote(s) of a Latin1 string literal should be referred to in error messages.
      * @since 4.1.0
      * @param multi whether this is for multi-line strings
      * @param raw whether this is for raw strings
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelStringLatin1End(@unused multi: Boolean, @unused raw: Boolean): LabelConfig = NotConfigured
    /**  How the closing quote(s) of a UTF-16 string literal should be referred to in error messages.
      * @since 4.1.0
      * @param multi whether this is for multi-line strings
      * @param raw whether this is for raw strings
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelStringUtf16End(@unused multi: Boolean, @unused raw: Boolean): LabelConfig = NotConfigured

    /** How general string characters should be referred to in error messages.
      * @since 4.1.0
      * @note defaults to label of "string character"
      * @note this superscedes [[labelGraphicCharacter `labelGraphicCharacter`]] and [[labelEscapeSequence `labelEscapeSequence`]] within string literals.
      * @group text
      */
    def labelStringCharacter: LabelConfig = Label("string character")
    /** How a graphic character (a regular character in the literal) should be referred to or explained in error messages.
      * @since 4.1.0
      * @note defaults to a label of "graphic character"
      * @note explains for graphic characters do not work in string literals.
      * @group text
      */
    def labelGraphicCharacter: LabelWithExplainConfig = Label("graphic character")
    /** How an escape sequence should be referred to or explained in error messages.
      * @since 4.1.0
      * @note defaults to label of "escape sequence"
      * @note explains for escape characters do not work in string literals.
      * @see [[labelEscapeEnd `labelEscapeEnd`]] for how to explain what valid escape sequences may be when the lead character has been parsed.
      * @group text
      */
    def labelEscapeSequence: LabelWithExplainConfig = Label("escape sequence") //different to "invalid escape sequence"!
    /** How a numeric escape sequence (after the opening character) should be referred to or explained in error messages.
      * @since 4.1.0
      * @param radix the radix this specific configuration applies to
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelEscapeNumeric(@unused radix: Int): LabelWithExplainConfig = NotConfigured
    /** How the end of a numeric escape sequence (after a prefix) should be referred to or explained in error messages.
      * @since 4.1.0
      * @param radix the radix this specific configuration applies to
      * @param prefix the character that started this sequence
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelEscapeNumericEnd(@unused prefix: Char, @unused radix: Int): LabelWithExplainConfig = NotConfigured
    /** How the end of an escape sequence (anything past the opening character) should be referred to or explained within an error message.
      * @since 4.1.0
      * @note defaults to label of "end of escape sequence" with a reason of "invalid escape sequence"
      * @group text
      */
    def labelEscapeEnd: LabelWithExplainConfig = LabelAndReason(reason = "invalid escape sequence", label = "end of escape sequence")
    /** How zero-width escape characters should be referred to within error messages.
      * @since 4.1.0
      * @note defaults to [[NotConfigured `NotConfigured`]]
      * @group text
      */
    def labelStringEscapeEmpty: LabelConfig = NotConfigured
    /** How string gaps should be referred to within error messages.
      * @since 4.1.0
      * @note defaults to label of "string gap"
      * @group text
      */
    def labelStringEscapeGap: LabelConfig = Label("string gap")
    /** How the end of a string gap (the closing slash) should be referred to within error messages.
      * @since 4.1.0
      * @note defaults to label of "end of string gap"
      * @group text
      */
    def labelStringEscapeGapEnd: LabelConfig = Label("end of string gap")

    /** When a non-BMP character is found in a BMP-only character literal, specifies how this should be reported.
      * @since 4.1.0
      * @note defaults to a filter generating the reason "non-BMP character"
      * @group text
      */
    def filterCharNonBasicMultilingualPlane: VanillaFilterConfig[Int] = new Because[Int] {
        def reason(@unused x: Int) = "non-BMP character"
    }
    /** When a non-ASCII character is found in a ASCII-only character literal, specifies how this should be reported.
      * @since 4.1.0
      * @note defaults to a filter generating the reason "non-ascii character"
      * @group text
      */
    def filterCharNonAscii: VanillaFilterConfig[Int] = new Because[Int] {
        def reason(@unused x: Int) = "non-ascii character"
    }
    /** When a non-Latin1 character is found in a Latin1-only character literal, specifies how this should be reported.
      * @since 4.1.0
      * @note defaults to a filter generating the reason "non-latin1 character"
      * @group text
      */
    def filterCharNonLatin1: VanillaFilterConfig[Int] = new Because[Int] {
        def reason(@unused x: Int) = "non-latin1 character"
    }

    /** When a non-ASCII character is found in a ASCII-only string literal, specifies how this should be reported.
      * @since 4.1.0
      * @note defaults to a filter generating a ''specialised'' message of "non-ascii characters in string literal, this is not allowed"
      * @group text
      */
    def filterStringNonAscii: SpecializedFilterConfig[StringBuilder] = new SpecializedMessage[StringBuilder] {
        def message(@unused s: StringBuilder) = Seq("non-ascii characters in string literal, this is not allowed")
    }

    /** When a non-Latin1 character is found in a Latin1-only string literal, specifies how this should be reported.
      * @since 4.1.0
      * @note defaults to a filter generating a ''specialised'' message of "non-latin1 characters in string literal, this is not allowed"
      * @group text
      */
    def filterStringNonLatin1: SpecializedFilterConfig[StringBuilder] = new SpecializedMessage[StringBuilder] {
        def message(@unused s: StringBuilder) = Seq("non-latin1 characters in string literal, this is not allowed")
    }

    /** When a numeric escape sequence requires a specific number of digits but this was not successfully parsed, this describes how to
      * report that error given the number of successfully parsed digits up this point.
      * @since 4.1.0
      * @param radix the radix used for this numeric escape sequence
      * @param needed the possible numbers of digits required
      * @note defaults to a ''specialised'' message describing how many digits are required but how many were present.
      * @group text
      */
    def filterEscapeCharRequiresExactDigits(@unused radix: Int, needed: Seq[Int]): SpecializedFilterConfig[Int] =
        new SpecializedMessage[Int] {
            def message(got: Int) = {
                assume(needed.nonEmpty, "cannot be empty!")
                val Some(formatted) = parsley.errors.helpers.disjunct(needed.toList.map(_.toString), oxfordComma = true): @unchecked
                Seq(s"numeric escape requires $formatted digits, but only got $got")
            }
        }

    /** When a numeric escape sequence is not legal, this describes how to report that error, given the original illegal character.
      * @since 4.1.0
      * @param maxEscape the largest legal escape character
      * @param radix the radix used for this numeric escape sequence
      * @note defaults to a ''specialised'' message stating if the character is larger than the given maximum, or just an illegal codepoint otherwise.
      * @group text
      */
    def filterEscapeCharNumericSequenceIllegal(maxEscape: Int, radix: Int): SpecializedFilterConfig[BigInt] =
        new SpecializedMessage[BigInt] {
            def message(escapeChar: BigInt) = Seq(
                if (escapeChar > BigInt(maxEscape)) {
                    s"${escapeChar.toString(radix)} is greater than the maximum character value of ${BigInt(maxEscape).toString(radix)}"
                }
                else s"illegal unicode codepoint: ${escapeChar.toString(radix)}"
            )
        }

    /** Character literals parse either graphic characters or escape characters. This configuration allows for individual errors when a character ''not'' part
      * of either graphic characters or escape characters is encountered.
      * @since 4.1.0
      * @note defaults to [[Unverified `Unverified`]]
      * @group text
      */
    def verifiedCharBadCharsUsedInLiteral: VerifiedBadChars = Unverified
    /** String literals parse either graphic characters or escape characters. This configuration allows for individual errors when a character ''not'' part
      * of either graphic characters or escape characters is encountered.
      * @since 4.1.0
      * @note defaults to [[Unverified `Unverified`]]
      * @group text
      */
    def verifiedStringBadCharsUsedInLiteral: VerifiedBadChars = Unverified

    /** Gives names and/or reasons to symbols.
      *
      * Symbols that do not appear in the map are assumed to be `NotConfigured`.
      *
      * @since 5.0.0
      * @note defaults to the empty map
      * @group symbol
      */
    def labelSymbol: Map[String, LabelWithExplainConfig] = Map.empty
    // To unify, or not to unify
    private [parsley] def defaultSymbolKeyword: Labeller = Label
    private [parsley] def defaultSymbolOperator: Labeller = Label
    // Other?
    private [parsley] def defaultSymbolPunctuation: Labeller = NotConfigured
    /** How the required end of a given keyword should be specified in an error.
      * @since 4.1.0
      * @note defaults to "end of symbol"
      * @group symbol
      */
    def labelSymbolEndOfKeyword(symbol: String): String = s"end of $symbol"
    /** How the required end of a given operator should be specified in an error.
      * @since 4.1.0
      * @note defaults to "end of symbol"
      * @group symbol
      */
    def labelSymbolEndOfOperator(symbol: String): String = s"end of $symbol"

    /** How the end of a single-line comment should be described or explained.
      * @since 4.1.0
      * @note defaults to "end of comment"
      * @group space
      */
    def labelSpaceEndOfLineComment: LabelWithExplainConfig = Label("end of comment")
    /** How the end of a multi-line comment should be described or explained.
      * @since 4.1.0
      * @note defaults to "end of comment"
      * @group space
      */
    def labelSpaceEndOfMultiComment: LabelWithExplainConfig = Label("end of comment")
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy