src/Pure/Isar/parse.scala
author wenzelm
Tue, 08 Apr 2014 13:24:08 +0200
changeset 56464 555f4be59be6
parent 56209 3c89e21d9be2
child 56801 8dd9df88f647
permissions -rw-r--r--
more precise token positions;
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
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    10
import scala.util.parsing.combinator.Parsers
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
36948
d2cdad45fd14 renamed Outer_Parse to Parse (in Scala);
wenzelm
parents: 34311
diff changeset
    14
object Parse
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    15
{
34161
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    16
  /* parsing tokens */
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    17
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    18
  trait Parser extends Parsers
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    19
  {
36956
21be4832c362 renamed class Outer_Lex to Token and Token_Kind to Token.Kind;
wenzelm
parents: 36948
diff changeset
    20
    type Elem = Token
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    21
48599
5e64b7770f35 tuned signature;
wenzelm
parents: 48484
diff changeset
    22
    def filter_proper: Boolean = true
34266
bfe8d6998734 added filter_proper parameter;
wenzelm
parents: 34168
diff changeset
    23
48599
5e64b7770f35 tuned signature;
wenzelm
parents: 48484
diff changeset
    24
    @tailrec private def proper(in: Input): Input =
5e64b7770f35 tuned signature;
wenzelm
parents: 48484
diff changeset
    25
      if (!filter_proper || in.atEnd || in.first.is_proper) in
34161
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    26
      else proper(in.rest)
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    27
56464
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    28
    def token(s: String, pred: Elem => Boolean): Parser[(Elem, Token.Pos)] =
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    29
      new Parser[(Elem, Token.Pos)] {
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    30
        def apply(raw_input: Input) =
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    31
        {
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    32
          val in = proper(raw_input)
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    33
          if (in.atEnd) Failure(s + " expected,\nbut end-of-input was found", in)
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    34
          else {
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    35
            val pos =
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    36
              in.pos match {
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    37
                case pos: Token.Pos => pos
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    38
                case _ => Token.Pos.none
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    39
              }
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    40
            val token = in.first
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    41
            if (pred(token)) Success((token, pos), proper(in.rest))
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    42
            else
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    43
              token.text match {
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    44
                case (txt, "") =>
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    45
                  Failure(s + " expected,\nbut " + txt + " was found", in)
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    46
                case (txt1, txt2) =>
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    47
                  Failure(s + " expected,\nbut " + txt1 + " was found:\n" + txt2, in)
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    48
              }
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    49
          }
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    50
        }
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    51
      }
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    52
34168
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    53
    def atom(s: String, pred: Elem => Boolean): Parser[String] =
56464
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    54
      token(s, pred) ^^ { case (tok, _) => tok.content }
34168
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    55
56464
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    56
    def command(name: String): Parser[Position.T] =
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    57
      token("command " + quote(name), tok => tok.is_command && tok.source == name) ^^
555f4be59be6 more precise token positions;
wenzelm
parents: 56209
diff changeset
    58
        { case (_, pos) => pos.position }
48718
73e6c22e2d94 more structural parsing for minor modes;
wenzelm
parents: 48600
diff changeset
    59
34168
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    60
    def keyword(name: String): Parser[String] =
48718
73e6c22e2d94 more structural parsing for minor modes;
wenzelm
parents: 48600
diff changeset
    61
      atom("keyword " + quote(name), tok => tok.is_keyword && tok.source == name)
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    62
46943
ac1c41ea856d clarified syntax of prospective keywords;
wenzelm
parents: 44181
diff changeset
    63
    def string: Parser[String] = atom("string", _.is_string)
48349
a78e5d399599 support Session.Queue with ordering and dependencies;
wenzelm
parents: 46943
diff changeset
    64
    def nat: Parser[Int] = atom("natural number", _.is_nat) ^^ (s => Integer.parseInt(s))
34168
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    65
    def name: Parser[String] = atom("name declaration", _.is_name)
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    66
    def xname: Parser[String] = atom("name reference", _.is_xname)
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    67
    def text: Parser[String] = atom("text", _.is_text)
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    68
    def ML_source: Parser[String] = atom("ML source", _.is_text)
51627
589daaf48dba tuned signature -- agree with markup terminology;
wenzelm
parents: 48912
diff changeset
    69
    def document_source: Parser[String] = atom("document source", _.is_text)
48484
70898d016538 more explicit checks during parsing;
wenzelm
parents: 48349
diff changeset
    70
    def path: Parser[String] =
55879
ac979f750c1a clarified path checks: avoid crash of rendering due to spurious errors;
wenzelm
parents: 55618
diff changeset
    71
      atom("file name/path specification", tok => tok.is_name && Path.is_wellformed(tok.content))
48484
70898d016538 more explicit checks during parsing;
wenzelm
parents: 48349
diff changeset
    72
    def theory_name: Parser[String] =
56209
3c89e21d9be2 simplified (despite 70898d016538);
wenzelm
parents: 56208
diff changeset
    73
      atom("theory name", tok => tok.is_name && Path.is_wellformed(tok.content))
34168
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    74
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    75
    private def tag_name: Parser[String] =
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    76
      atom("tag name", tok =>
36956
21be4832c362 renamed class Outer_Lex to Token and Token_Kind to Token.Kind;
wenzelm
parents: 36948
diff changeset
    77
          tok.kind == Token.Kind.IDENT ||
21be4832c362 renamed class Outer_Lex to Token and Token_Kind to Token.Kind;
wenzelm
parents: 36948
diff changeset
    78
          tok.kind == Token.Kind.STRING)
34168
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    79
18843829c7f2 clarified atom parser: return content;
wenzelm
parents: 34161
diff changeset
    80
    def tags: Parser[List[String]] = rep(keyword("%") ~> tag_name)
34161
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    81
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    82
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    83
    /* wrappers */
4c845a8f1357 consider proper input only;
wenzelm
parents: 34159
diff changeset
    84
36956
21be4832c362 renamed class Outer_Lex to Token and Token_Kind to Token.Kind;
wenzelm
parents: 36948
diff changeset
    85
    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
    86
48600
305ebcd9018a proper treatment of eof wrt. proper_input -- allow input of spaces/comments only;
wenzelm
parents: 48599
diff changeset
    87
    def parse_all[T](p: Parser[T], in: Token.Reader): ParseResult[T] =
48912
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
    88
    {
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
    89
      val result = parse(p, in)
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
    90
      val rest = proper(result.next)
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
    91
      if (result.successful && !rest.atEnd) Error("bad input", rest)
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
    92
      else result
ffdb37019b2f improved errors of parser combinators;
wenzelm
parents: 48911
diff changeset
    93
    }
34159
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    94
  }
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    95
}
903092d61519 Generic parsers for Isabelle/Isar outer syntax -- Scala version.
wenzelm
parents:
diff changeset
    96