src/Pure/Isar/parse.scala
author paulson <lp15@cam.ac.uk>
Wed, 24 Apr 2024 09:21:44 +0100
changeset 80148 b156869b826a
parent 76614 ac08b6e3b9e3
permissions -rw-r--r--
Another Nominal example
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
43283
446e6621762d updated headers;
wenzelm
parents: 36956
diff changeset
     1
/*  Title:      Pure/Isar/parse.scala
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
     2
    Author:     Makarius
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
     3
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
     4
Generic parsers for Isabelle/Isar outer syntax.
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
     5
*/
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
     6
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
     7
package isabelle
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
     8
55618
995162143ef4 tuned imports;
wenzelm
parents: 51627
diff changeset
     9
75405
b13ab7d11b90 clarified signature: avoid ambiguity in scala3;
wenzelm
parents: 75393
diff changeset
    10
import scala.util.parsing.combinator
48599
5e64b7770f35 tuned signature;
wenzelm
parents: 48484
diff changeset
    11
import scala.annotation.tailrec
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    12
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    13
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 74887
diff changeset
    14
object Parse {
34161
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    15
  /* parsing tokens */
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    16
75405
b13ab7d11b90 clarified signature: avoid ambiguity in scala3;
wenzelm
parents: 75393
diff changeset
    17
  trait Parsers extends combinator.Parsers {
36956
21be4832c362 renamed class Outer_Lex to Token and Token_Kind to Token.Kind;
wenzelm
parents: 36948
diff changeset
    18
    type Elem = Token
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    19
48599
5e64b7770f35 tuned signature;
wenzelm
parents: 48484
diff changeset
    20
    def filter_proper: Boolean = true
34266
bfe8d6998734 added filter_proper parameter;
wenzelm
parents: 34168
diff changeset
    21
48599
5e64b7770f35 tuned signature;
wenzelm
parents: 48484
diff changeset
    22
    @tailrec private def proper(in: Input): Input =
5e64b7770f35 tuned signature;
wenzelm
parents: 48484
diff changeset
    23
      if (!filter_proper || in.atEnd || in.first.is_proper) in
34161
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    24
      else proper(in.rest)
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    25
59692
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    26
    private def proper_position: Parser[Position.T] =
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    27
      new Parser[Position.T] {
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 74887
diff changeset
    28
        def apply(raw_input: Input) = {
59692
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    29
          val in = proper(raw_input)
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    30
          val pos =
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    31
            in.pos match {
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    32
              case pos: Token.Pos => pos
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    33
              case _ => Token.Pos.none
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    34
            }
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    35
          Success(if (in.atEnd) pos.position() else pos.position(in.first), in)
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    36
        }
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    37
      }
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    38
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    39
    def position[A](parser: Parser[A]): Parser[(A, Position.T)] =
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    40
      proper_position ~ parser ^^ { case x ~ y => (y, x) }
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    41
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    42
    def token(s: String, pred: Elem => Boolean): Parser[Elem] =
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    43
      new Parser[Elem] {
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 74887
diff changeset
    44
        def apply(raw_input: Input) = {
56464
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    45
          val in = proper(raw_input)
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    46
          if (in.atEnd) Failure(s + " expected,\nbut end-of-input was found", in)
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    47
          else {
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    48
            val token = in.first
59692
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    49
            if (pred(token)) Success(token, proper(in.rest))
58861
5ff61774df11 command-line terminator ";" is no longer accepted;
wenzelm
parents: 56801
diff changeset
    50
            else Failure(s + " expected,\nbut " + token.kind + " was found:\n" + token.source, in)
56464
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    51
          }
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    52
        }
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    53
      }
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    54
34168
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    55
    def atom(s: String, pred: Elem => Boolean): Parser[String] =
59692
03aa1b63af10 position parser as in ML;
wenzelm
parents: 59671
diff changeset
    56
      token(s, pred) ^^ (_.content)
34168
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    57
63446
19162a9ef7e3 tunes signature;
wenzelm
parents: 62969
diff changeset
    58
    def command(name: String): Parser[String] = atom("command " + quote(name), _.is_command(name))
19162a9ef7e3 tunes signature;
wenzelm
parents: 62969
diff changeset
    59
    def $$$(name: String): Parser[String] = atom("keyword " + quote(name), _.is_keyword(name))
46943
ac1c41ea856d clarified syntax of prospective keywords;
wenzelm
parents: 44181
diff changeset
    60
    def string: Parser[String] = atom("string", _.is_string)
48349
a78e5d399599 support Session.Queue with ordering and dependencies;
wenzelm
parents: 46943
diff changeset
    61
    def nat: Parser[Int] = atom("natural number", _.is_nat) ^^ (s => Integer.parseInt(s))
62969
9f394a16c557 eliminated "xname" and variants;
wenzelm
parents: 62965
diff changeset
    62
    def name: Parser[String] = atom("name", _.is_name)
64471
c40c2975fb02 more uniform path syntax, as in ML (see 5a7c919a4ada);
wenzelm
parents: 63446
diff changeset
    63
    def embedded: Parser[String] = atom("embedded content", _.is_embedded)
74887
56247fdb8bbb discontinued old-style {* verbatim *} tokens;
wenzelm
parents: 70668
diff changeset
    64
    def text: Parser[String] = atom("text", _.is_embedded)
56247fdb8bbb discontinued old-style {* verbatim *} tokens;
wenzelm
parents: 70668
diff changeset
    65
    def ML_source: Parser[String] = atom("ML source", _.is_embedded)
56247fdb8bbb discontinued old-style {* verbatim *} tokens;
wenzelm
parents: 70668
diff changeset
    66
    def document_source: Parser[String] = atom("document source", _.is_embedded)
59693
d96cb03caf9e tunes signature -- more uniform ML vs. Scala;
wenzelm
parents: 59692
diff changeset
    67
70668
9cac4dec0da9 support for explicit session directories;
wenzelm
parents: 69891
diff changeset
    68
    def opt_keyword(s: String): Parser[Boolean] =
76613
wenzelm
parents: 75987
diff changeset
    69
      ($$$("(") ~! $$$(s) ~ $$$(")")) ^^ (_ => true) | success(false)
70668
9cac4dec0da9 support for explicit session directories;
wenzelm
parents: 69891
diff changeset
    70
48484
70898d016538 more explicit checks during parsing;
wenzelm
parents: 48349
diff changeset
    71
    def path: Parser[String] =
64471
c40c2975fb02 more uniform path syntax, as in ML (see 5a7c919a4ada);
wenzelm
parents: 63446
diff changeset
    72
      atom("file name/path specification", tok => tok.is_embedded && Path.is_wellformed(tok.content))
59693
d96cb03caf9e tunes signature -- more uniform ML vs. Scala;
wenzelm
parents: 59692
diff changeset
    73
76614
ac08b6e3b9e3 clarified signature;
wenzelm
parents: 76613
diff changeset
    74
    def in_path(default: String): Parser[String] =
ac08b6e3b9e3 clarified signature;
wenzelm
parents: 76613
diff changeset
    75
      $$$("in") ~! path ^^ { case _ ~ x => x } | success(default)
ac08b6e3b9e3 clarified signature;
wenzelm
parents: 76613
diff changeset
    76
ac08b6e3b9e3 clarified signature;
wenzelm
parents: 76613
diff changeset
    77
    def in_path_parens(default: String): Parser[String] =
ac08b6e3b9e3 clarified signature;
wenzelm
parents: 76613
diff changeset
    78
      $$$("(") ~! ($$$("in") ~ path ~ $$$(")")) ^^ { case _ ~ (_ ~ x ~ _) => x } | success(default)
ac08b6e3b9e3 clarified signature;
wenzelm
parents: 76613
diff changeset
    79
75987
ff2e67d73592 more robust: proper system_name;
wenzelm
parents: 75405
diff changeset
    80
    def chapter_name: Parser[String] = atom("chapter name", _.is_system_name)
66914
fb3f13a9c756 uniform system name;
wenzelm
parents: 64471
diff changeset
    81
    def session_name: Parser[String] = atom("session name", _.is_system_name)
fb3f13a9c756 uniform system name;
wenzelm
parents: 64471
diff changeset
    82
    def theory_name: Parser[String] = atom("theory name", _.is_system_name)
34168
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    83
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    84
    private def tag_name: Parser[String] =
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    85
      atom("tag name", tok =>
36956
21be4832c362 renamed class Outer_Lex to Token and Token_Kind to Token.Kind;
wenzelm
parents: 36948
diff changeset
    86
          tok.kind == Token.Kind.IDENT ||
21be4832c362 renamed class Outer_Lex to Token and Token_Kind to Token.Kind;
wenzelm
parents: 36948
diff changeset
    87
          tok.kind == Token.Kind.STRING)
34168
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    88
69887
b9985133805d added semantic document markers;
wenzelm
parents: 66914
diff changeset
    89
    def tag: Parser[String] = $$$("%") ~> tag_name
b9985133805d added semantic document markers;
wenzelm
parents: 66914
diff changeset
    90
    def tags: Parser[List[String]] = rep(tag)
b9985133805d added semantic document markers;
wenzelm
parents: 66914
diff changeset
    91
69891
def3ec9cdb7e document markers are formal comments, and may thus occur anywhere in the command-span;
wenzelm
parents: 69887
diff changeset
    92
    def marker: Parser[String] = token("marker", _.is_marker) ^^ (_.content)
69887
b9985133805d added semantic document markers;
wenzelm
parents: 66914
diff changeset
    93
76613
wenzelm
parents: 75987
diff changeset
    94
    def annotation: Parser[Unit] = rep(tag | marker) ^^ (_ => ())
34161
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    95
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    96
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    97
    /* wrappers */
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    98
36956
21be4832c362 renamed class Outer_Lex to Token and Token_Kind to Token.Kind;
wenzelm
parents: 36948
diff changeset
    99
    def parse[T](p: Parser[T], in: Token.Reader): ParseResult[T] = p(in)
48912
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
   100
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 74887
diff changeset
   101
    def parse_all[T](p: Parser[T], in: Token.Reader): ParseResult[T] = {
48912
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
   102
      val result = parse(p, in)
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
   103
      val rest = proper(result.next)
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
   104
      if (result.successful && !rest.atEnd) Error("bad input", rest)
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
   105
      else result
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
   106
    }
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
   107
  }
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
   108
}