Parse EBNF: Difference between revisions
Added Algol 68
(Added Algol 68) |
|||
(4 intermediate revisions by 2 users not shown) | |||
Line 11:
See [[Parse EBNF/Tests|the tests]].
<br><br>
=={{header|ALGOL 68}}==
The source of the Algol 68 sample is somewhat largish, so is on a separate page on Rosetta Code here: [[Parse EBNF/ALGOL 68]].
{{out}}
<pre>
Valid EBNF: a/z: 'a' { a = 'a1' ( 'a2' | 'a3' ) { 'a4' } [ 'a5' ] 'a6' ; } 'z'
a: seq
literal "a1"
oneof
seq
literal "a2"
seq
literal "a3"
list
literal "a4"
opt
literal "a5"
literal "a6"
"a1a3a4a4a5a6" is valid according to a
"a1 a2a6" is valid according to a
"a1 a3 a4 a6" is valid according to a
**** Syntax error near "a1 a4 a5 a6"
"a1 a4 a5 a6" is not valid according to a
**** Syntax error near "a5 a5 a6"
"a1 a2 a4 a5 a5 a6" is not valid according to a
**** Unexpected text: "a7" at the end of source
"a1 a2 a4 a5 a6 a7" is not valid according to a
**** Syntax error near "your ad here"
"your ad here" is not valid according to a
Valid EBNF: Arithmetic expressions:
'Arithmetic expressions'
{
expr = term { plus term } .
term = factor { times factor } .
factor = number | '(' expr ')' .
plus = '+' | '-' .
times = '*' | '/' .
number = digit { digit } .
digit = '0' | '1' | '2' | '3' | '4' | '5' | '6' | '7' | '8' | '9' .
}
expr: seq
rule term
list
rule plus
rule term
term: seq
rule factor
list
rule times
rule factor
factor: oneof
seq
rule number
seq
literal "("
rule expr
literal ")"
plus: oneof
seq
literal "+"
seq
literal "-"
times: oneof
seq
literal "*"
seq
literal "/"
number: seq
rule digit
list
rule digit
digit: oneof
seq
literal "0"
seq
literal "1"
seq
literal "2"
seq
literal "3"
seq
literal "4"
seq
literal "5"
seq
literal "6"
seq
literal "7"
seq
literal "8"
seq
literal "9"
"2" is valid according to Arithmetic expressions
"2*3 + 4/23 - 7" is valid according to Arithmetic expressions
"(3 + 4) * 6-2+(4*(4))" is valid according to Arithmetic expressions
**** Syntax error near "-2"
"-2" is not valid according to Arithmetic expressions
**** Unexpected text: "+" at the end of source
"3 +" is not valid according to Arithmetic expressions
**** Syntax error near " 3"
"(4 + 3" is not valid according to Arithmetic expressions
**** Expected "{", not "a"
Invalid EBNF: a = '1';
**** Expected "}", not "(eof)"
Invalid EBNF: { a = '1' ;
**** Expected "=", not "world"
Invalid EBNF: { hello world = '1'; }
**** Rule "bar" not defined
Invalid EBNF: { foo = bar . }
</pre>
=={{header|Go}}==
Line 16 ⟶ 149:
<br>
A more or less faithful translation except that indices are 0-based rather than 1-based and so 1 less than in the Phix results.
<
import (
Line 420 ⟶ 553:
fmt.Println()
}
}</
{{out}}
Line 503 ⟶ 636:
We use Parsec to generate Parsec.
<
import Control.Monad
import Data.Maybe
Line 675 ⟶ 808:
lc c = char c <* ws
ws = many $ oneOf " \n\t"</
=={{header|Julia}}==
Line 683 ⟶ 816:
Tested with Julia v1.7.2
<
struct Grammar
regex::Regex
Line 835 ⟶ 968:
end
</syntaxhighlight>
=={{header|Modula-2}}==
<
FROM ASCII IMPORT EOL;
Line 927 ⟶ 1,060:
Tabulate (T0);
Tabulate (T1);
END EBNF.</
And the source for the EBNF scanner. I hope you like nested procedures.
<
FROM ASCII IMPORT LF;
Line 1,096 ⟶ 1,229:
Ino := 0;
ch := ' '
END EBNFScanner.</
=={{header|Perl}}==
<
use strict; # http://www.rosettacode.org/wiki/Parse_EBNF
Line 1,273 ⟶ 1,406:
----------------------------------------------------------------------
{ foo = bar . } "undefined production check"
----------------------------------------------------------------------</
{{out}}
<pre>
Line 1,387 ⟶ 1,520:
=={{header|Phix}}==
<!--<
<span style="color: #008080;">with</span> <span style="color: #008080;">javascript_semantics</span>
<span style="color: #004080;">string</span> <span style="color: #000000;">src</span>
Line 1,704 ⟶ 1,837:
<span style="color: #008080;">end</span> <span style="color: #008080;">if</span>
<span style="color: #008080;">end</span> <span style="color: #008080;">for</span>
<!--</
In real use, I would be tempted to use numeric literals rather than string tags in such structures, but the latter certainly make things ten times easier to debug, plus I got an instantly legible syntax tree dump (the bit just after "===>" below) practically for free.
{{out}}
Line 1,771 ⟶ 1,904:
=={{header|PicoLisp}}==
<
"expr : term ( ( PLUS | MINUS ) term )* ;"
"term : factor ( ( MULT | DIV ) factor )* ;"
Line 1,780 ⟶ 1,913:
(unless (and (match '(@S : @E ;) (str E)) (not (cdr @S)))
(quit "Invalid EBNF" E) )
(put (car @S) 'ebnf @E) ) )</
<
(cond
((asoq Pat '((PLUS . +) (MINUS . -) (MULT . *) (DIV . /)))
Line 1,823 ⟶ 1,956:
(let *Lst (str Str "")
(catch NIL
(parseLst (get 'expr 'ebnf)) ) ) )</
Output:
<pre>: (parseEbnf "1 + 2 * -3 / 7 - 3 * 4")
Line 1,835 ⟶ 1,968:
It is implemented and exercised using the flavor of EBNF and test cases specified on the [[Parse EBNF/Tests|test page]].
<syntaxhighlight lang="raku" line># A Raku grammar to parse EBNF
rule TOP { ^ <title>? '{' [ <production> ]+ '}' <comment>? $ }
rule
rule
rule factor { <group> | <repeat> | <optional> | <identifier> | <literal> }
rule
token identifier { <-[\|\(\)\{\}\[\]\.\;\"\'\s]>+ } #"
token
token
}
class EBNF::Actions {
method TOP($/) {
say "Syntax Tree:\n", $/; # Dump the syntax tree to STDOUT
Line 1,861 ⟶ 1,993:
">]+\$\}\n " ~ $<production>>>.ast ~ "\}"
}
make 'token ' ~ $<name> ~ ' {' ~
$<expression>.ast ~ "}\n"
}
make $<literal> ?? $<literal> !!
$<group> ?? '[' ~ $<group>.ast ~ ']' !!
Line 1,874 ⟶ 2,006:
'<' ~ $<identifier> ~ '>'
}
{
ebnf =>
Line 1,937 ⟶ 2,069:
teststrings => ['foobar']
}
unless EBNF.parse($test<ebnf>) {
say "Parsing EBNF grammar:\n";
Line 1,972 ⟶ 2,104:
say '*' x 79, "\n";
unlink $fn;
}</syntaxhighlight>
Output:
Line 2,244 ⟶ 2,375:
{{in progress|lang=Ruby|day=12|month=May|year=2011}}
{{incomplete|Ruby|The tokenizer is here, but the parser is very incomplete.}}
<
# The tokenizer splits the input into Tokens like "identifier",
# ":", ")*" and so on. This design uses a StringScanner on each line of
Line 2,382 ⟶ 2,513:
parse
end
</syntaxhighlight>
=={{header|Tcl}}==
Line 2,388 ⟶ 2,519:
Demonstration lexer and parser. Note that this parser supports parenthesized expressions, making the grammar recursive.
<
# Utilities to make the coroutine easier to use
Line 2,522 ⟶ 2,653:
}
throw SYNTAX "\"$payload\" at position $index"
}</
<
puts [parse "1 - 2 - -3 * 4 + 5"]
puts [parse "1 - 2 - -3 * (4 + 5)"]</
Output:
<pre>
Line 2,536 ⟶ 2,667:
{{trans|Phix}}
{{libheader|Wren-fmt}}
{{libheader|Wren-
{{libheader|Wren-str}}
{{libheader|Wren-seq}}
Translated via the Go entry.
<
import "./
import "./str" for Char
import "./seq" for Lst
var src = ""
Line 2,907 ⟶ 3,038:
System.print()
i = i + 1
}</
{{out}}
|