rust-crafting-interpreters-.../src/ast.rs

99 lines
2.6 KiB
Rust

use crate::tokens::Token;
/* --------- *
* AST nodes *
* --------- */
/// The AST node for the program
#[derive(Default, Debug, Clone)]
pub struct ProgramNode(pub Vec<StmtNode>);
/// An AST node that represents a statement.
#[derive(Debug, Clone)]
pub enum StmtNode {
/// A variable declaration
VarDecl(String, Option<ExprNode>),
/// An single expression
Expression(ExprNode),
/// The print statement
Print(ExprNode),
}
/// An AST node that represents an expression.
#[derive(Debug, Clone)]
pub enum ExprNode {
/// Binary expression.
Binary {
left: Box<ExprNode>,
operator: Token,
right: Box<ExprNode>,
},
/// Unary expression.
Unary {
operator: Token,
right: Box<ExprNode>,
},
/// Grouping expression, containing a sub-expression.
Grouping { expression: Box<ExprNode> },
/// A litteral value, represented by the corresponding token.
Litteral { value: Token },
/// A reference to a variable.
Variable { name: Token },
}
/* -------------------------------- *
* Dumper trait and implementations *
* -------------------------------- */
/// This trait should be implemented by nodes to allow AST dumps.
pub trait AstDumper {
/// Dump the node as a string.
fn dump(&self) -> String;
}
impl AstDumper for ProgramNode {
fn dump(&self) -> String {
self.0
.iter()
.map(|node| node.dump())
.collect::<Vec<String>>()
.join(" ")
}
}
impl AstDumper for StmtNode {
fn dump(&self) -> String {
match self {
Self::VarDecl(name, Some(expr)) => format!("( var {} {} )", name, expr.dump()),
Self::VarDecl(name, None) => format!("( var {} nil )", name),
Self::Expression(expr) => format!("( {} )", expr.dump()),
Self::Print(expr) => format!("(print {})", expr.dump()),
}
}
}
impl AstDumper for ExprNode {
fn dump(&self) -> String {
match self {
Self::Binary {
left,
operator,
right,
} => format!("( {} {} {} )", operator.lexeme, left.dump(), right.dump()),
Self::Unary { operator, right } => format!("( {} {} )", operator.lexeme, right.dump()),
Self::Grouping { expression } => format!("( {} )", expression.dump()),
Self::Variable { name } => name.lexeme.clone(),
Self::Litteral { value } => {
if value.is_litteral() {
value.lexeme.clone()
} else {
panic!("Unexpected token type for token {:#?}", value)
}
}
}
}
}