1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86
use super::expression::Expression; use super::value_definition::ValueDefinition; use crate::types::Type; use std::collections::{HashMap, HashSet}; #[derive(Clone, Debug, PartialEq)] pub struct LetValues { definitions: Vec<ValueDefinition>, expression: Box<Expression>, } impl LetValues { pub fn new(definitions: Vec<ValueDefinition>, expression: impl Into<Expression>) -> Self { Self { definitions, expression: Box::new(expression.into()), } } pub fn definitions(&self) -> &[ValueDefinition] { &self.definitions } pub fn expression(&self) -> &Expression { &self.expression } pub(crate) fn rename_variables(&self, names: &HashMap<String, String>) -> Self { let mut names = names.clone(); let mut definitions = Vec::with_capacity(self.definitions.len()); for definition in &self.definitions { definitions.push(definition.rename_variables(&names)); names.remove(definition.name()); } Self::new(definitions, self.expression.rename_variables(&names)) } pub(crate) fn find_variables(&self, excluded_variables: &HashSet<String>) -> HashSet<String> { let mut excluded_variables = excluded_variables.clone(); let mut variables = HashSet::new(); for definition in &self.definitions { variables.extend(definition.find_variables(&excluded_variables)); excluded_variables.insert(definition.name().into()); } variables.extend(self.expression.find_variables(&excluded_variables)); variables } pub(crate) fn infer_environment( &self, variables: &HashMap<String, Type>, global_variables: &HashSet<String>, ) -> Self { let mut variables = variables.clone(); let mut definitions = vec![]; for value_definition in &self.definitions { definitions.push(value_definition.infer_environment(&variables, global_variables)); variables.insert( value_definition.name().into(), value_definition.type_().clone().into(), ); } Self::new( definitions, self.expression .infer_environment(&variables, global_variables), ) } pub(crate) fn convert_types(&self, convert: &impl Fn(&Type) -> Type) -> Self { Self::new( self.definitions .iter() .map(|definition| definition.convert_types(convert)) .collect(), self.expression.convert_types(convert), ) } }