Abackus crate adds a layer on top of earlgrey crate to simplify writing a grammar. You can simply use an EBNF style String instead of manually adding rules.
You can describe your grammar like this: ```rust let grammar = r#" S := S '+' N | N ; N := '[0-9]' ; "#;
ParserBuilder::default()
.plugterminal("[0-9]", |n| "1234567890".contains(n))
.plugterminal("[+]", |c| c == "+")
.intoparser("S")
Instead of the more verbose:
rust
// Gramar: S -> S + N | N; N -> [0-9];
let g = earlgrey::GrammarBuilder::default()
.nonterm("S")
.nonterm("N")
.terminal("[+]", |c| c == "+")
.terminal("[0-9]", |n| "1234567890".contains(n))
.rule("S", &["S", "[+]", "N"])
.rule("S", &["N"])
.rule("N", &["[0-9]"])
.intogrammar("S")
.unwrap();
earlgrey::EarleyParser::new(g) ```
Underneath the covers an earlgrey::EarleyParser
is used to build a parser for EBNF grammar. (For details you can check earlgrey/ebnf.rs
). That parser is then used to build a final parser for the grammar provided by the user.
```rust // NOTE: extract from abackus/examples/ebnftree.rs
fn main() { let grammar = r#" expr := expr ('+'|'-') term | term ; term := term ('*'|'/') factor | factor ; factor := '-' factor | power ; power := ufact '^' factor | ufact ; ufact := ufact '!' | group ; group := num | '(' expr ')' ; "#;
// Build a parser for our grammar and while at it, plug in an // evaluator to extract the resulting tree as S-expressions. use std::str::FromStr; let trif = abackus::ParserBuilder::default() .plugterminal("num", |n| f64::fromstr(n).is_ok()) .sexprificator(&grammar, "expr");
// Read some input from command-line
let input = std::env::args().skip(1).
collect::
// Print resulting parse trees match trif(&mut tokenizer(input.chars())) { Ok(trees) => for t in trees { println!("{}", t.print()); }, Err(e) => println!("{:?}", e) } } ```