2024-03-08 19:01:05 +00:00
|
|
|
use crate::{
|
|
|
|
context::Context,
|
|
|
|
error::{RuntimeError, ValidationError},
|
|
|
|
};
|
|
|
|
|
2024-03-17 04:49:01 +00:00
|
|
|
use super::{AbstractTree, Action, Block, Expression, Positioned, Type};
|
2024-03-08 19:01:05 +00:00
|
|
|
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord)]
|
2024-03-08 21:14:47 +00:00
|
|
|
pub struct IfElse {
|
2024-03-17 04:49:01 +00:00
|
|
|
if_expression: Positioned<Expression>,
|
|
|
|
if_block: Positioned<Block>,
|
|
|
|
else_block: Option<Positioned<Block>>,
|
2024-03-08 19:01:05 +00:00
|
|
|
}
|
|
|
|
|
2024-03-08 21:14:47 +00:00
|
|
|
impl IfElse {
|
2024-03-17 04:49:01 +00:00
|
|
|
pub fn new(
|
|
|
|
if_expression: Positioned<Expression>,
|
|
|
|
if_block: Positioned<Block>,
|
|
|
|
else_block: Option<Positioned<Block>>,
|
|
|
|
) -> Self {
|
2024-03-08 19:01:05 +00:00
|
|
|
Self {
|
|
|
|
if_expression,
|
2024-03-09 20:17:19 +00:00
|
|
|
if_block,
|
|
|
|
else_block,
|
2024-03-08 19:01:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-08 21:14:47 +00:00
|
|
|
impl AbstractTree for IfElse {
|
2024-03-08 19:01:05 +00:00
|
|
|
fn expected_type(&self, _context: &Context) -> Result<Type, ValidationError> {
|
2024-03-17 04:49:01 +00:00
|
|
|
self.if_block.node.expected_type(_context)
|
2024-03-08 19:01:05 +00:00
|
|
|
}
|
|
|
|
|
2024-03-10 18:48:53 +00:00
|
|
|
fn validate(&self, context: &Context) -> Result<(), ValidationError> {
|
2024-03-17 04:49:01 +00:00
|
|
|
if let Type::Boolean = self.if_expression.node.expected_type(context)? {
|
2024-03-10 18:48:53 +00:00
|
|
|
if let Some(else_block) = &self.else_block {
|
2024-03-17 04:49:01 +00:00
|
|
|
let expected = self.if_block.node.expected_type(context)?;
|
|
|
|
let actual = else_block.node.expected_type(context)?;
|
2024-03-10 18:48:53 +00:00
|
|
|
|
2024-03-17 04:49:01 +00:00
|
|
|
expected
|
|
|
|
.check(&actual)
|
|
|
|
.map_err(|conflict| ValidationError::TypeCheck {
|
|
|
|
conflict,
|
|
|
|
actual_position: self.if_block.position,
|
|
|
|
expected_position: self.if_expression.position,
|
|
|
|
})?;
|
2024-03-10 18:48:53 +00:00
|
|
|
}
|
|
|
|
|
2024-03-08 19:29:53 +00:00
|
|
|
Ok(())
|
|
|
|
} else {
|
|
|
|
Err(ValidationError::ExpectedBoolean)
|
|
|
|
}
|
2024-03-08 19:01:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn run(self, _context: &Context) -> Result<Action, RuntimeError> {
|
|
|
|
let if_boolean = self
|
|
|
|
.if_expression
|
2024-03-17 04:49:01 +00:00
|
|
|
.node
|
2024-03-08 19:01:05 +00:00
|
|
|
.run(_context)?
|
|
|
|
.as_return_value()?
|
|
|
|
.as_boolean()?;
|
|
|
|
|
|
|
|
if if_boolean {
|
2024-03-17 04:49:01 +00:00
|
|
|
self.if_block.node.run(_context)
|
2024-03-09 20:17:19 +00:00
|
|
|
} else if let Some(else_statement) = self.else_block {
|
2024-03-17 04:49:01 +00:00
|
|
|
else_statement.node.run(_context)
|
2024-03-08 19:01:05 +00:00
|
|
|
} else {
|
|
|
|
Ok(Action::None)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use crate::{
|
2024-03-16 19:01:45 +00:00
|
|
|
abstract_tree::{Statement, ValueNode},
|
2024-03-08 19:01:05 +00:00
|
|
|
Value,
|
|
|
|
};
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn simple_if() {
|
|
|
|
assert_eq!(
|
|
|
|
IfElse::new(
|
2024-03-17 06:51:33 +00:00
|
|
|
Expression::Value(ValueNode::Boolean(true)).positioned((0..0).into()),
|
|
|
|
Block::new(vec![Statement::Expression(
|
|
|
|
Expression::Value(ValueNode::String("foo".to_string()))
|
|
|
|
.positioned((0..0).into())
|
2024-03-17 10:26:12 +00:00
|
|
|
)
|
|
|
|
.positioned((0..0).into())])
|
2024-03-17 06:51:33 +00:00
|
|
|
.positioned((0..0).into()),
|
2024-03-08 19:01:05 +00:00
|
|
|
None
|
|
|
|
)
|
|
|
|
.run(&Context::new()),
|
2024-03-08 21:22:24 +00:00
|
|
|
Ok(Action::Return(Value::string("foo".to_string())))
|
2024-03-08 19:01:05 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn simple_if_else() {
|
|
|
|
assert_eq!(
|
|
|
|
IfElse::new(
|
2024-03-17 06:51:33 +00:00
|
|
|
Expression::Value(ValueNode::Boolean(false)).positioned((0..0).into()),
|
|
|
|
Block::new(vec![Statement::Expression(
|
|
|
|
Expression::Value(ValueNode::String("foo".to_string()))
|
|
|
|
.positioned((0..0).into())
|
2024-03-17 10:26:12 +00:00
|
|
|
)
|
|
|
|
.positioned((0..0).into())])
|
2024-03-17 06:51:33 +00:00
|
|
|
.positioned((0..0).into()),
|
2024-03-17 04:49:01 +00:00
|
|
|
Some(
|
2024-03-17 06:51:33 +00:00
|
|
|
Block::new(vec![Statement::Expression(
|
|
|
|
Expression::Value(ValueNode::String("bar".to_string()))
|
|
|
|
.positioned((0..0).into())
|
2024-03-17 10:26:12 +00:00
|
|
|
)
|
|
|
|
.positioned((0..0).into())])
|
2024-03-17 06:51:33 +00:00
|
|
|
.positioned((0..0).into())
|
2024-03-17 04:49:01 +00:00
|
|
|
)
|
2024-03-08 19:01:05 +00:00
|
|
|
)
|
|
|
|
.run(&Context::new()),
|
2024-03-08 21:22:24 +00:00
|
|
|
Ok(Action::Return(Value::string("bar".to_string())))
|
2024-03-08 19:01:05 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|