2023-10-06 17:32:58 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
|
2024-01-10 20:03:52 +00:00
|
|
|
use crate::{
|
2024-01-31 18:51:48 +00:00
|
|
|
error::{RuntimeError, SyntaxError, ValidationError},
|
2024-02-01 00:35:27 +00:00
|
|
|
AbstractTree, Expression, Format, LogicOperator, Map, SyntaxNode, Type, Value,
|
2024-01-10 20:03:52 +00:00
|
|
|
};
|
2023-10-06 17:32:58 +00:00
|
|
|
|
2023-12-06 19:13:22 +00:00
|
|
|
/// Abstract representation of a logic expression.
|
2023-10-06 17:32:58 +00:00
|
|
|
#[derive(Debug, Clone, Serialize, Deserialize, Eq, PartialEq, PartialOrd, Ord)]
|
|
|
|
pub struct Logic {
|
|
|
|
left: Expression,
|
|
|
|
operator: LogicOperator,
|
|
|
|
right: Expression,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AbstractTree for Logic {
|
2024-01-31 18:51:48 +00:00
|
|
|
fn from_syntax(node: SyntaxNode, source: &str, context: &Map) -> Result<Self, SyntaxError> {
|
2024-02-01 00:35:27 +00:00
|
|
|
SyntaxError::expect_syntax_node(source, "logic", node)?;
|
2023-12-30 02:15:03 +00:00
|
|
|
|
2023-12-02 03:16:50 +00:00
|
|
|
let first_node = node.child(0).unwrap();
|
|
|
|
let (left_node, operator_node, right_node) = {
|
|
|
|
if first_node.is_named() {
|
2024-01-06 13:11:09 +00:00
|
|
|
(first_node, node.child(1).unwrap(), node.child(2).unwrap())
|
2023-12-02 03:16:50 +00:00
|
|
|
} else {
|
|
|
|
(
|
|
|
|
node.child(1).unwrap(),
|
2024-01-06 13:11:09 +00:00
|
|
|
node.child(2).unwrap(),
|
2023-12-02 03:16:50 +00:00
|
|
|
node.child(3).unwrap(),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
};
|
2024-01-10 20:03:52 +00:00
|
|
|
let left = Expression::from_syntax(left_node, source, context)?;
|
|
|
|
let operator = LogicOperator::from_syntax(operator_node, source, context)?;
|
|
|
|
let right = Expression::from_syntax(right_node, source, context)?;
|
2023-10-06 17:32:58 +00:00
|
|
|
|
|
|
|
Ok(Logic {
|
|
|
|
left,
|
|
|
|
operator,
|
|
|
|
right,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-01-31 18:51:48 +00:00
|
|
|
fn expected_type(&self, _context: &Map) -> Result<Type, ValidationError> {
|
|
|
|
Ok(Type::Boolean)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn check_type(&self, _source: &str, _context: &Map) -> Result<(), ValidationError> {
|
|
|
|
self.left.check_type(_source, _context)?;
|
|
|
|
self.right.check_type(_source, _context)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn run(&self, source: &str, context: &Map) -> Result<Value, RuntimeError> {
|
2023-10-10 17:29:11 +00:00
|
|
|
let left = self.left.run(source, context)?;
|
|
|
|
let right = self.right.run(source, context)?;
|
2023-10-09 19:54:47 +00:00
|
|
|
let result = match self.operator {
|
|
|
|
LogicOperator::Equal => {
|
|
|
|
if let (Ok(left_num), Ok(right_num)) = (left.as_number(), right.as_number()) {
|
|
|
|
left_num == right_num
|
|
|
|
} else {
|
|
|
|
left == right
|
|
|
|
}
|
|
|
|
}
|
2023-10-28 14:28:43 +00:00
|
|
|
LogicOperator::NotEqual => {
|
|
|
|
if let (Ok(left_num), Ok(right_num)) = (left.as_number(), right.as_number()) {
|
|
|
|
left_num != right_num
|
|
|
|
} else {
|
|
|
|
left != right
|
|
|
|
}
|
|
|
|
}
|
2023-10-09 19:54:47 +00:00
|
|
|
LogicOperator::And => left.as_boolean()? && right.as_boolean()?,
|
|
|
|
LogicOperator::Or => left.as_boolean()? || right.as_boolean()?,
|
|
|
|
LogicOperator::Greater => left > right,
|
|
|
|
LogicOperator::Less => left < right,
|
|
|
|
LogicOperator::GreaterOrEqual => left >= right,
|
2024-01-06 10:00:36 +00:00
|
|
|
LogicOperator::LessOrEqual => left <= right,
|
2023-10-06 17:32:58 +00:00
|
|
|
};
|
|
|
|
|
2023-10-09 19:54:47 +00:00
|
|
|
Ok(Value::Boolean(result))
|
2023-10-06 17:32:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-06 13:11:09 +00:00
|
|
|
impl Format for Logic {
|
|
|
|
fn format(&self, output: &mut String, indent_level: u8) {
|
|
|
|
self.left.format(output, indent_level);
|
|
|
|
output.push(' ');
|
|
|
|
self.operator.format(output, indent_level);
|
|
|
|
output.push(' ');
|
|
|
|
self.right.format(output, indent_level);
|
2024-01-06 10:00:36 +00:00
|
|
|
}
|
2023-10-06 17:32:58 +00:00
|
|
|
}
|