summaryrefslogtreecommitdiffstats
path: root/crates/rebel-parse
diff options
context:
space:
mode:
authorMatthias Schiffer <mschiffer@universe-factory.net>2024-04-29 20:49:55 +0200
committerMatthias Schiffer <mschiffer@universe-factory.net>2024-04-29 21:15:42 +0200
commitbcec2e72909bc63bdc2cb88808789a04f3a13d72 (patch)
treedd6a0437ecc216398c324b3ce5ccea1559227831 /crates/rebel-parse
parent62abc9e8059a70b2f8106238d724428af9bda76b (diff)
downloadrebel-bcec2e72909bc63bdc2cb88808789a04f3a13d72.tar
rebel-bcec2e72909bc63bdc2cb88808789a04f3a13d72.zip
rebel-parse, rebel-lang: distinguish let statements and simple assignments
Diffstat (limited to 'crates/rebel-parse')
-rw-r--r--crates/rebel-parse/src/ast/mod.rs24
-rw-r--r--crates/rebel-parse/src/grammar/recipe.rs10
2 files changed, 29 insertions, 5 deletions
diff --git a/crates/rebel-parse/src/ast/mod.rs b/crates/rebel-parse/src/ast/mod.rs
index ebf7241..4fa8e63 100644
--- a/crates/rebel-parse/src/ast/mod.rs
+++ b/crates/rebel-parse/src/ast/mod.rs
@@ -53,8 +53,12 @@ impl<'a> Block<'a> {
#[derive(Debug, Clone, PartialEq, Eq)]
pub enum BlockStmt<'a> {
- Assign {
+ Let {
dest: Box<TypedPat<'a>>,
+ expr: Option<Box<Expr<'a>>>,
+ },
+ Assign {
+ dest: Box<Pat<'a>>,
expr: Box<Expr<'a>>,
},
Expr {
@@ -64,15 +68,22 @@ pub enum BlockStmt<'a> {
}
impl<'a> BlockStmt<'a> {
+ pub(crate) fn let_assign(dest: TypedPat<'a>, expr: Option<Expr<'a>>) -> Self {
+ BlockStmt::Let {
+ dest: Box::new(dest),
+ expr: expr.map(Box::new),
+ }
+ }
+
pub(crate) fn assign(
- dest: TypedPat<'a>,
+ dest: Pat<'a>,
op: Option<OpBinary>,
swapped: bool,
expr: Expr<'a>,
) -> Self {
let expr = match op {
Some(op) => {
- let dest_expr = match &dest.pat {
+ let dest_expr = match &dest {
Pat::Path(path) => Expr::Path(path.clone()),
};
if swapped {
@@ -91,6 +102,13 @@ impl<'a> BlockStmt<'a> {
pub fn validate(&self) -> Result<(), ValidationError> {
match self {
+ BlockStmt::Let { dest: _, expr } => {
+ // TODO: Destination validation
+ if let Some(expr) = expr {
+ expr.validate()?;
+ }
+ Ok(())
+ }
BlockStmt::Assign { dest: _, expr } => {
// TODO: Destination validation
expr.validate()
diff --git a/crates/rebel-parse/src/grammar/recipe.rs b/crates/rebel-parse/src/grammar/recipe.rs
index 6b96c85..c9fe1e2 100644
--- a/crates/rebel-parse/src/grammar/recipe.rs
+++ b/crates/rebel-parse/src/grammar/recipe.rs
@@ -34,10 +34,16 @@ peg::parser! {
= block:block_stmt() ++ p(';') { ast::Block(block) }
pub rule block_stmt() -> ast::BlockStmt<'a>
- = dest:typed_pat() op:assign_op() expr:expr() {
+ = [Token::Keyword(Keyword::Let)] dest:typed_pat() p('=') expr:expr() {
+ ast::BlockStmt::let_assign(dest, Some(expr))
+ }
+ / [Token::Keyword(Keyword::Let)] dest:typed_pat() {
+ ast::BlockStmt::let_assign(dest, None)
+ }
+ / dest:pat() op:assign_op() expr:expr() {
ast::BlockStmt::assign(dest, op, false, expr)
}
- / dest:typed_pat() p2('=', '+') expr:expr() {
+ / dest:pat() p2('=', '+') expr:expr() {
ast::BlockStmt::assign(dest, Some(Add), true, expr)
}
/ expr:expr() {