2023-10-06 17:32:58 +00:00
|
|
|
//! Pattern matching.
|
|
|
|
//!
|
|
|
|
//! Note that this module is called "match" but is escaped as "r#match" because
|
|
|
|
//! "match" is a keyword in Rust.
|
|
|
|
|
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
use tree_sitter::Node;
|
|
|
|
|
2023-12-06 18:48:38 +00:00
|
|
|
use crate::{AbstractTree, Error, Expression, Map, Result, Statement, Type, Value};
|
2023-10-06 17:32:58 +00:00
|
|
|
|
2023-12-06 18:48:38 +00:00
|
|
|
/// Abstract representation of a match statement.
|
2023-10-06 17:32:58 +00:00
|
|
|
#[derive(Debug, Clone, Serialize, Deserialize, Eq, PartialEq, PartialOrd, Ord)]
|
2023-12-06 18:48:38 +00:00
|
|
|
pub struct Match {
|
|
|
|
matcher: Expression,
|
|
|
|
options: Vec<(Expression, Statement)>,
|
|
|
|
}
|
2023-10-06 17:32:58 +00:00
|
|
|
|
|
|
|
impl AbstractTree for Match {
|
2023-12-06 18:48:38 +00:00
|
|
|
fn from_syntax_node(source: &str, node: Node, context: &Map) -> Result<Self> {
|
|
|
|
Error::expect_syntax_node(source, "match", node)?;
|
|
|
|
|
|
|
|
let matcher_node = node.child(1).unwrap();
|
|
|
|
let matcher = Expression::from_syntax_node(source, matcher_node, context)?;
|
|
|
|
|
|
|
|
let mut options = Vec::new();
|
|
|
|
let mut previous_expression = None;
|
|
|
|
|
|
|
|
for index in 2..node.child_count() {
|
|
|
|
let child = node.child(index).unwrap();
|
|
|
|
|
|
|
|
if child.kind() == "expression" {
|
|
|
|
previous_expression = Some(Expression::from_syntax_node(source, child, context)?);
|
|
|
|
}
|
|
|
|
|
|
|
|
if child.kind() == "statement" {
|
|
|
|
if let Some(expression) = &previous_expression {
|
|
|
|
let statement = Statement::from_syntax_node(source, child, context)?;
|
|
|
|
|
|
|
|
options.push((expression.clone(), statement));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(Match { matcher, options })
|
2023-10-06 17:32:58 +00:00
|
|
|
}
|
|
|
|
|
2023-12-06 18:48:38 +00:00
|
|
|
fn run(&self, source: &str, context: &Map) -> Result<Value> {
|
|
|
|
let matcher_value = self.matcher.run(source, context)?;
|
|
|
|
|
|
|
|
for (expression, statement) in &self.options {
|
|
|
|
let option_value = expression.run(source, context)?;
|
|
|
|
|
|
|
|
if matcher_value == option_value {
|
|
|
|
return statement.run(source, context);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(Value::Empty)
|
2023-11-30 00:23:42 +00:00
|
|
|
}
|
|
|
|
|
2023-12-05 22:08:22 +00:00
|
|
|
fn expected_type(&self, _context: &Map) -> Result<Type> {
|
2023-10-06 17:32:58 +00:00
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
}
|
2023-12-06 18:48:38 +00:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use crate::{evaluate, Value};
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn evaluate_match() {
|
|
|
|
let test = evaluate(
|
|
|
|
"
|
|
|
|
match 1 {
|
|
|
|
3 => false
|
|
|
|
2 => { false }
|
|
|
|
1 => true
|
|
|
|
}
|
|
|
|
",
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(Value::Boolean(true), test);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn evaluate_match_assignment() {
|
|
|
|
let test = evaluate(
|
|
|
|
"
|
|
|
|
x = match 1 {
|
|
|
|
3 => false
|
|
|
|
2 => { false }
|
|
|
|
1 => true
|
|
|
|
}
|
|
|
|
x
|
|
|
|
",
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(Value::Boolean(true), test);
|
|
|
|
}
|
|
|
|
}
|