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

parsley.position.scala Maven / Gradle / Ivy

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

import parsley.syntax.zipped.zippedSyntax3

import parsley.internal.deepembedding.singletons

// TODO: position grouping?
/** This module contains parsers that provide a way to extract position information during a parse.
  *
  * Position parsers can be important
  * for when the final result of the parser needs to encode position information for later consumption:
  * this is particularly useful for abstract syntax trees. Offset is also exposed by this interface, which
  * may be useful for establishing a caret size in specialised error messages.
  *
  * @since 4.2.0
  */
object position extends position
private [parsley] trait position {
    /** This parser returns the current line number (starting at 1) of the input without having any other effect.
      *
      * When this combinator is ran, no input is required, nor consumed, and
      * the current line number will always be successfully returned. It has no other
      * effect on the state of the parser.
      *
      * @example {{{
      * scala> import parsley.position.line, parsley.character.char
      * scala> line.parse("")
      * val res0 = Success(1)
      * scala> (char('a') *> line).parse("a")
      * val res0 = Success(1)
      * scala> (char('\n') *> line).parse("\n")
      * val res0 = Success(2)
      * }}}
      *
      * @return a parser that returns the line number the parser is currently at.
      */
    final val line: Parsley[Int] = _line
    @inline private def _line = new Parsley(singletons.Line)
    /** This parser returns the current column number (starting at 1) of the input without having any other effect.
      *
      * When this combinator is ran, no input is required, nor consumed, and
      * the current column number will always be successfully returned. It has no other
      * effect on the state of the parser.
      *
      * @example {{{
      * scala> import parsley.position.col, parsley.character.char
      * scala> col.parse("")
      * val res0 = Success(1)
      * scala> (char('a') *> col).parse("a")
      * val res0 = Success(2)
      * scala> (char('\n') *> col).parse("\n")
      * val res0 = Success(1)
      * }}}
      *
      * @return a parser that returns the column number the parser is currently at.
      * @note in the presence of wide unicode characters, the value returned may be inaccurate.
      */
    final val col: Parsley[Int] = _col
    @inline private def _col = new Parsley(singletons.Col)
    /** This parser returns the current line and column numbers (starting at 1) of the input without having any other effect.
      *
      * When this combinator is ran, no input is required, nor consumed, and
      * the current line and column number will always be successfully returned. It has no other
      * effect on the state of the parser.
      *
      * @example {{{
      * scala> import parsley.position.pos, parsley.character.char
      * scala> pos.parse("")
      * val res0 = Success((1, 1))
      * scala> (char('a') *> pos).parse("a")
      * val res0 = Success((1, 2))
      * scala> (char('\n') *> pos).parse("\n")
      * val res0 = Success((2, 1))
      * }}}
      *
      * @return a parser that returns the line and column number the parser is currently at.
      * @note in the presence of wide unicode characters, the column value returned may be inaccurate.
      */
    final val pos: Parsley[(Int, Int)] = _pos.uo("pos")
    @inline private def _pos = _line.zip(_col)

    // this is subject to change at the slightest notice, do NOT expose
    private [parsley] final def internalOffset: Parsley[Int] = new Parsley(singletons.Offset)

    /** This parser returns the current offset into the input (starting at 0) without having any other effect.
      *
      * When this combinator is ran, no input is required, nor consumed, and
      * the current offset into the input will always be successfully returned. It has no other
      * effect on the state of the parser.
      *
      * @example {{{
      * scala> import parsley.position.offset, parsley.character.char
      * scala> offset.parse("")
      * val res0 = Success(0)
      * scala> (char('a') *> offset).parse("a")
      * val res0 = Success(1)
      * scala> (char('\n') *> offset).parse("\n")
      * val res0 = Success(1)
      * }}}
      *
      * @return a parser that returns the offset the parser is currently at.
      * @note offset does not take wide unicode codepoints into account.
      */
    final val offset: Parsley[Int] = internalOffset

    private [parsley] final def withSpan[A, S](end: Parsley[S])(p: Parsley[A]): Parsley[(S, A, S)] = (end, p, end).zipped

    /** This combinator returns the result of a given parser and the number of characters it consumed.
      *
      * First records the initial `offset` on entry to given parser `p`, then executes `p`. If `p` succeeds,
      * then the `offset` is taken again, and the two values are subtracted to give width `w`. The result of
      * `p`, `x` is returned along with `w` as `(x, w)`. If `p` fails, this combinator will also fail.
      *
      * @example {{{
      * scala> import parsley.position.withWidth, parsley.character.string
      * scala> withWidth(string("abc")).parse("abc")
      * val res0 = Success(("abc", 3))
      * }}}
      *
      * @param p the parser to compute the width for
      * @return a parser that pairs the result of the parser `p` with the number of characters it consumed
      * @note the value returned is the number of 16-bit ''characters'' consumed, not unicode codepoints.
      * @since 4.4.0
      */
    final def withWidth[A](p: Parsley[A]): Parsley[(A, Int)] = (internalOffset.ut(), p, internalOffset.ut()).zipped((s, x, e) => (x, e-s)).uo("withWidth")
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy