use rayon::prelude::*; use serde::{Deserialize, Serialize}; use tree_sitter::Node; use crate::{AbstractTree, Block, Error, Expression, Identifier, Map, Result, Type, Value}; /// Abstract representation of a for loop statement. #[derive(Debug, Clone, Serialize, Deserialize, Eq, PartialEq, PartialOrd, Ord)] pub struct For { is_async: bool, item_id: Identifier, collection: Expression, block: Block, } impl AbstractTree for For { fn from_syntax_node(source: &str, node: Node, context: &Map) -> Result { Error::expect_syntax_node(source, "for", node)?; let for_node = node.child(0).unwrap(); let is_async = match for_node.kind() { "for" => false, "async for" => true, _ => { return Err(Error::UnexpectedSyntaxNode { expected: "for or async for".to_string(), actual: for_node.kind().to_string(), location: for_node.start_position(), relevant_source: source[for_node.byte_range()].to_string(), }) } }; let identifier_node = node.child(1).unwrap(); let identifier = Identifier::from_syntax_node(source, identifier_node, context)?; let expression_node = node.child(3).unwrap(); let expression = Expression::from_syntax_node(source, expression_node, context)?; let item_node = node.child(4).unwrap(); let item = Block::from_syntax_node(source, item_node, context)?; Ok(For { is_async, item_id: identifier, collection: expression, block: item, }) } fn run(&self, source: &str, context: &Map) -> Result { let expression_run = self.collection.run(source, context)?; let values = expression_run.as_list()?.items(); let key = self.item_id.inner(); if self.is_async { values.par_iter().try_for_each(|value| { let iter_context = Map::clone_from(context)?; iter_context.set(key.clone(), value.clone(), None)?; self.block.run(source, &iter_context).map(|_value| ()) })?; } else { let loop_context = Map::clone_from(context)?; for value in values.iter() { loop_context.set(key.clone(), value.clone(), None)?; self.block.run(source, &loop_context)?; } } Ok(Value::none()) } fn expected_type(&self, _context: &Map) -> Result { Ok(Type::None) } }