Build Abstract Syntax Trees and tree-walking models quickly in Rust.
This example creates an AST for simple math expressions, and an interpreter to evaluate the result:
```rust use astmaker::{ast, model};
pub enum BinOp { Add, Sub, Mul, Div, }
pub enum UnOp { Add, Sub, }
ast!{ location = ();
pub node Expression =
| BinOp -> Node
pub node BinaryOperation = {
lhs: Node
pub node UnaryOperation = {
op: UnOp,
expr: Node
pub node Number = { value: f64, } }
pub struct Interpreter;
model!{ for Interpreter -> f64 { where Expression => { match node.data.asmut() { Expression::BinOp(childnode) => context.visit(childnode), Expression::UnOp(childnode) => context.visit(childnode), Expression::Num(childnode) => context.visit(child_node), } }, where BinaryOperation => { let lhs = context.visit(&mut node.data.lhs); let rhs = context.visit(&mut node.data.rhs);
match node.data.op {
BinOp::Add => lhs + rhs,
BinOp::Sub => lhs - rhs,
BinOp::Mul => lhs * rhs,
BinOp::Div => lhs / rhs,
}
},
where UnaryOperation => {
let val = context.visit(&mut node.data.expr);
match node.data.op {
UnOp::Add => val,
UnOp::Sub => -val,
}
},
where Number => node.data.value,
} }
fn eval() { let mut tree = Node::new((), Expression::BinOp( Node::new((), BinaryOperation { lhs: Node::new((), Expression::Num( Node::new((), Number { value: 1.0 }) )), op: BinOp::Add, rhs: Node::new((), Expression::Num( Node::new((), Number { value: 2.0 }) )) }) ));
let mut interpreter = Interpreter;
assert_eq!(interpreter.visit(&mut tree), 3.0); } ```
This project is released under the terms of the MIT License.