2024-03-17 17:36:31 +00:00
|
|
|
use std::cmp::Ordering;
|
|
|
|
|
2024-02-29 02:04:38 +00:00
|
|
|
use crate::{
|
|
|
|
context::Context,
|
|
|
|
error::{RuntimeError, ValidationError},
|
|
|
|
};
|
2024-02-25 18:49:26 +00:00
|
|
|
|
2024-03-17 11:31:45 +00:00
|
|
|
use super::{AbstractTree, Action, Statement, Type, WithPosition};
|
2024-02-25 18:49:26 +00:00
|
|
|
|
2024-03-17 17:36:31 +00:00
|
|
|
#[derive(Clone, Debug)]
|
2024-03-08 21:14:47 +00:00
|
|
|
pub struct Loop {
|
2024-03-17 11:31:45 +00:00
|
|
|
statements: Vec<WithPosition<Statement>>,
|
2024-03-02 01:17:55 +00:00
|
|
|
}
|
|
|
|
|
2024-03-08 21:14:47 +00:00
|
|
|
impl Loop {
|
2024-03-17 11:31:45 +00:00
|
|
|
pub fn new(statements: Vec<WithPosition<Statement>>) -> Self {
|
2024-03-02 01:17:55 +00:00
|
|
|
Self { statements }
|
|
|
|
}
|
2024-02-25 18:49:26 +00:00
|
|
|
}
|
|
|
|
|
2024-03-08 21:14:47 +00:00
|
|
|
impl AbstractTree for Loop {
|
2024-02-29 02:04:38 +00:00
|
|
|
fn expected_type(&self, _context: &Context) -> Result<Type, ValidationError> {
|
2024-03-08 17:24:11 +00:00
|
|
|
Ok(Type::None)
|
2024-02-29 02:04:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn validate(&self, _context: &Context) -> Result<(), ValidationError> {
|
2024-03-08 17:24:11 +00:00
|
|
|
for statement in &self.statements {
|
2024-03-17 10:26:12 +00:00
|
|
|
statement.node.validate(_context)?;
|
2024-03-08 17:24:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn run(self, _context: &Context) -> Result<Action, RuntimeError> {
|
|
|
|
loop {
|
2024-03-17 17:36:31 +00:00
|
|
|
for statement in &self.statements {
|
|
|
|
let action = statement.node.clone().run(_context)?;
|
2024-03-08 17:24:11 +00:00
|
|
|
|
2024-03-17 17:36:31 +00:00
|
|
|
match action {
|
|
|
|
Action::Return(_) => {}
|
|
|
|
Action::None => {}
|
|
|
|
Action::Break => return Ok(Action::Break),
|
|
|
|
}
|
2024-03-08 17:24:11 +00:00
|
|
|
}
|
|
|
|
}
|
2024-02-29 02:04:38 +00:00
|
|
|
}
|
2024-03-08 17:24:11 +00:00
|
|
|
}
|
|
|
|
|
2024-03-17 17:36:31 +00:00
|
|
|
impl Eq for Loop {}
|
|
|
|
|
|
|
|
impl PartialEq for Loop {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
self.statements.eq(&other.statements)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialOrd for Loop {
|
|
|
|
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
|
|
|
|
Some(self.cmp(other))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Ord for Loop {
|
|
|
|
fn cmp(&self, other: &Self) -> Ordering {
|
|
|
|
self.statements.cmp(&other.statements)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-08 17:24:11 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2024-03-17 17:36:31 +00:00
|
|
|
use crate::{
|
|
|
|
abstract_tree::{
|
|
|
|
Assignment, AssignmentOperator, Block, Expression, Identifier, IfElse, Logic, ValueNode,
|
|
|
|
},
|
|
|
|
Value,
|
|
|
|
};
|
|
|
|
|
2024-03-08 17:24:11 +00:00
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn basic_loop() {
|
2024-03-17 17:36:31 +00:00
|
|
|
let result = Loop::new(vec![Statement::Break.with_position((0, 0))]).run(&Context::new());
|
2024-02-29 02:04:38 +00:00
|
|
|
|
2024-03-11 21:44:52 +00:00
|
|
|
assert_eq!(result, Ok(Action::Break))
|
2024-02-25 18:49:26 +00:00
|
|
|
}
|
2024-03-17 17:36:31 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn complex_loop() {
|
|
|
|
let result = Block::new(vec![
|
|
|
|
Statement::Assignment(Assignment::new(
|
|
|
|
Identifier::new("i"),
|
|
|
|
None,
|
|
|
|
AssignmentOperator::Assign,
|
|
|
|
Statement::Expression(Expression::Value(ValueNode::Integer(1)))
|
|
|
|
.with_position((0, 0)),
|
|
|
|
))
|
|
|
|
.with_position((0, 0)),
|
|
|
|
Statement::Loop(Loop::new(vec![Statement::IfElse(IfElse::new(
|
|
|
|
Expression::Logic(Box::new(Logic::Greater(
|
|
|
|
Expression::Identifier(Identifier::new("i")).with_position((10, 11)),
|
|
|
|
Expression::Value(ValueNode::Integer(2)).with_position((14, 15)),
|
|
|
|
)))
|
|
|
|
.with_position((10, 15)),
|
|
|
|
Block::new(vec![Statement::Break.with_position((18, 24))]),
|
|
|
|
Some(Block::new(vec![Statement::Assignment(Assignment::new(
|
|
|
|
Identifier::new("i"),
|
|
|
|
None,
|
|
|
|
AssignmentOperator::AddAssign,
|
|
|
|
Statement::Expression(Expression::Value(ValueNode::Integer(1)))
|
|
|
|
.with_position((38, 39)),
|
|
|
|
))
|
|
|
|
.with_position((33, 39))])),
|
|
|
|
))
|
|
|
|
.with_position((0, 0))]))
|
|
|
|
.with_position((0, 0)),
|
|
|
|
Statement::Expression(Expression::Identifier(Identifier::new("i")))
|
|
|
|
.with_position((0, 0)),
|
|
|
|
])
|
|
|
|
.run(&Context::new());
|
|
|
|
|
|
|
|
assert_eq!(result, Ok(Action::Return(Value::integer(3))))
|
|
|
|
}
|
2024-02-25 18:49:26 +00:00
|
|
|
}
|