use std::{ cmp::Ordering, collections::BTreeMap, fmt::{self, Display, Formatter}, io::stdin, ops::Range, sync::{Arc, OnceLock}, }; use rand::{thread_rng, Rng}; use stanza::{ renderer::{console::Console, Renderer}, style::{HAlign, MinWidth, Styles}, table::Table, }; use crate::{ abstract_tree::{AbstractTree, Action, Block, Expression, Identifier, Type, WithPosition}, context::Context, error::{RuntimeError, ValidationError}, }; #[derive(Clone, Debug, PartialEq)] pub struct Value(Arc); impl Value { pub fn inner(&self) -> &Arc { &self.0 } pub fn boolean(boolean: bool) -> Self { Value(Arc::new(ValueInner::Boolean(boolean))) } pub fn float(float: f64) -> Self { Value(Arc::new(ValueInner::Float(float))) } pub fn integer(integer: i64) -> Self { Value(Arc::new(ValueInner::Integer(integer))) } pub fn list(list: Vec) -> Self { Value(Arc::new(ValueInner::List(list))) } pub fn map(map: BTreeMap) -> Self { Value(Arc::new(ValueInner::Map(map))) } pub fn range(range: Range) -> Self { Value(Arc::new(ValueInner::Range(range))) } pub fn string(string: String) -> Self { Value(Arc::new(ValueInner::String(string))) } pub fn function( parameters: Vec<(Identifier, WithPosition)>, return_type: WithPosition, body: WithPosition, ) -> Self { Value(Arc::new(ValueInner::Function(Function::Parsed( ParsedFunction { parameters, return_type, body, }, )))) } pub fn built_in_function(function: BuiltInFunction) -> Self { Value(Arc::new(ValueInner::Function(Function::BuiltIn(function)))) } pub fn r#type(&self, context: &Context) -> Result { let r#type = match self.0.as_ref() { ValueInner::Boolean(_) => Type::Boolean, ValueInner::Float(_) => Type::Float, ValueInner::Integer(_) => Type::Integer, ValueInner::List(values) => { let mut types = Vec::with_capacity(values.len()); for value in values { types.push(value.r#type(context)?); } Type::ListExact(types) } ValueInner::Map(_) => Type::Map, ValueInner::Range(_) => Type::Range, ValueInner::String(_) => Type::String, ValueInner::Function(function) => match function { Function::Parsed(parsed_function) => Type::Function { parameter_types: parsed_function .parameters .iter() .map(|(_, r#type)| r#type.node.clone()) .collect(), return_type: Box::new(parsed_function.return_type.node.clone()), }, Function::BuiltIn(built_in_function) => built_in_function.r#type(), }, ValueInner::Structure { name, fields: expressions, } => { let mut fields = Vec::with_capacity(expressions.len()); for (identifier, expression) in expressions { let r#type = expression .node .expected_type(context)? .with_position(expression.position); fields.push((identifier.clone(), r#type)); } Type::Structure { name: name.clone(), fields, } } }; Ok(r#type) } pub fn as_boolean(&self) -> Option { if let ValueInner::Boolean(boolean) = self.0.as_ref() { Some(*boolean) } else { None } } pub fn as_list(&self) -> Option<&Vec> { if let ValueInner::List(list) = self.inner().as_ref() { Some(list) } else { None } } pub fn as_integer(&self) -> Option { if let ValueInner::Integer(integer) = self.inner().as_ref() { Some(*integer) } else { None } } } impl Display for Value { fn fmt(&self, f: &mut Formatter) -> fmt::Result { fn create_table() -> Table { Table::with_styles(Styles::default().with(HAlign::Centred).with(MinWidth(3))) } match self.inner().as_ref() { ValueInner::Boolean(boolean) => write!(f, "{boolean}"), ValueInner::Float(float) => write!(f, "{float}"), ValueInner::Integer(integer) => write!(f, "{integer}"), ValueInner::List(list) => { let mut table = create_table(); for value in list { table = table.with_row([value.to_string()]); } write!(f, "{}", Console::default().render(&table)) } ValueInner::Map(map) => { let mut table = create_table(); for (identifier, value) in map { table = table.with_row([identifier.as_str(), &value.to_string()]); } write!(f, "{}", Console::default().render(&table)) } ValueInner::Range(_) => todo!(), ValueInner::String(string) => write!(f, "{string}"), ValueInner::Function(Function::Parsed(ParsedFunction { parameters, return_type, body, })) => { write!(f, "(")?; for (identifier, r#type) in parameters { write!(f, "{identifier}: {}", r#type.node)?; } write!(f, "): {} {:?}", return_type.node, body.node) } ValueInner::Function(Function::BuiltIn(built_in_function)) => { write!(f, "{built_in_function}") } ValueInner::Structure { name, fields } => todo!(), } } } impl Eq for Value {} impl PartialOrd for Value { fn partial_cmp(&self, other: &Self) -> Option { Some(self.cmp(other)) } } impl Ord for Value { fn cmp(&self, other: &Self) -> std::cmp::Ordering { self.0.as_ref().cmp(other.0.as_ref()) } } #[derive(Clone, Debug, PartialEq)] pub enum ValueInner { Boolean(bool), Float(f64), Function(Function), Integer(i64), List(Vec), Map(BTreeMap), Range(Range), String(String), Structure { name: Identifier, fields: Vec<(Identifier, WithPosition)>, }, } impl Eq for ValueInner {} impl PartialOrd for ValueInner { fn partial_cmp(&self, other: &Self) -> Option { Some(self.cmp(other)) } } impl Ord for ValueInner { fn cmp(&self, other: &Self) -> Ordering { use ValueInner::*; match (self, other) { (Boolean(left), Boolean(right)) => left.cmp(right), (Boolean(_), _) => Ordering::Greater, (Float(left), Float(right)) => left.total_cmp(right), (Float(_), _) => Ordering::Greater, (Integer(left), Integer(right)) => left.cmp(right), (Integer(_), _) => Ordering::Greater, (List(left), List(right)) => left.cmp(right), (List(_), _) => Ordering::Greater, (Map(left), Map(right)) => left.cmp(right), (Map(_), _) => Ordering::Greater, (Range(left), Range(right)) => { let start_cmp = left.start.cmp(&right.start); if start_cmp.is_eq() { left.end.cmp(&right.end) } else { start_cmp } } (Range(_), _) => Ordering::Greater, (String(left), String(right)) => left.cmp(right), (String(_), _) => Ordering::Greater, (Function(left), Function(right)) => left.cmp(right), (Function(_), _) => Ordering::Greater, ( Structure { name: left_name, fields: left_fields, }, Structure { name: right_name, fields: right_fields, }, ) => { let name_cmp = left_name.cmp(right_name); if name_cmp.is_eq() { left_fields.cmp(right_fields) } else { name_cmp } } (Structure { .. }, _) => Ordering::Greater, } } } #[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord)] pub enum Function { Parsed(ParsedFunction), BuiltIn(BuiltInFunction), } impl Function { pub fn call(self, arguments: Vec, context: Context) -> Result { let action = match self { Function::Parsed(ParsedFunction { parameters, body, .. }) => { for ((identifier, _), value) in parameters.into_iter().zip(arguments.into_iter()) { context.set_value(identifier.clone(), value)?; } body.node.run(&context)? } Function::BuiltIn(built_in_function) => built_in_function.call(arguments, &context)?, }; Ok(action) } } #[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord)] pub struct ParsedFunction { parameters: Vec<(Identifier, WithPosition)>, return_type: WithPosition, body: WithPosition, } static INT_PARSE: OnceLock = OnceLock::new(); static INT_RANDOM_RANGE: OnceLock = OnceLock::new(); static READ_LINE: OnceLock = OnceLock::new(); static WRITE_LINE: OnceLock = OnceLock::new(); pub const BUILT_IN_FUNCTIONS: [BuiltInFunction; 4] = [ BuiltInFunction::IntParse, BuiltInFunction::IntRandomRange, BuiltInFunction::ReadLine, BuiltInFunction::WriteLine, ]; #[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord)] pub enum BuiltInFunction { IntParse, IntRandomRange, ReadLine, WriteLine, } impl BuiltInFunction { pub fn name(&self) -> &'static str { match self { BuiltInFunction::IntParse => "parse", BuiltInFunction::IntRandomRange => "random_range", BuiltInFunction::ReadLine => "read_line", BuiltInFunction::WriteLine => "write_line", } } pub fn value(&self) -> Value { match self { BuiltInFunction::IntParse => { INT_PARSE.get_or_init(|| Value::built_in_function(BuiltInFunction::IntParse)) } BuiltInFunction::IntRandomRange => INT_RANDOM_RANGE .get_or_init(|| Value::built_in_function(BuiltInFunction::IntRandomRange)), BuiltInFunction::ReadLine => { READ_LINE.get_or_init(|| Value::built_in_function(BuiltInFunction::ReadLine)) } BuiltInFunction::WriteLine => { WRITE_LINE.get_or_init(|| Value::built_in_function(BuiltInFunction::WriteLine)) } } .clone() } pub fn r#type(&self) -> Type { match self { BuiltInFunction::IntParse => Type::Function { parameter_types: vec![Type::String], return_type: Box::new(Type::Integer), }, BuiltInFunction::IntRandomRange => Type::Function { parameter_types: vec![Type::Range], return_type: Box::new(Type::Integer), }, BuiltInFunction::ReadLine => Type::Function { parameter_types: Vec::with_capacity(0), return_type: Box::new(Type::String), }, BuiltInFunction::WriteLine => Type::Function { parameter_types: vec![Type::Any], return_type: Box::new(Type::None), }, } } pub fn call(&self, arguments: Vec, context: &Context) -> Result { match self { BuiltInFunction::IntParse => { let string = arguments.get(0).unwrap(); if let ValueInner::String(string) = string.inner().as_ref() { // let integer = string.parse(); todo!() // Ok(Action::Return(Value::integer(integer))) } else { let mut actual = Vec::with_capacity(arguments.len()); for value in arguments { let r#type = value.r#type(context)?; actual.push(r#type); } Err(RuntimeError::ValidationFailure( ValidationError::WrongArguments { expected: vec![Type::String], actual, }, )) } } BuiltInFunction::IntRandomRange => { let range = arguments.get(0).unwrap(); if let ValueInner::Range(range) = range.inner().as_ref() { let random = thread_rng().gen_range(range.clone()); Ok(Action::Return(Value::integer(random))) } else { panic!("Built-in function cannot have a non-function type.") } } BuiltInFunction::ReadLine => { let mut input = String::new(); stdin().read_line(&mut input)?; Ok(Action::Return(Value::string(input))) } BuiltInFunction::WriteLine => { println!("{}", arguments[0]); Ok(Action::None) } } } } impl Display for BuiltInFunction { fn fmt(&self, f: &mut Formatter) -> fmt::Result { match self { BuiltInFunction::IntParse => write!(f, "(input : int) : str {{ *MAGIC* }}"), BuiltInFunction::IntRandomRange => write!(f, "(input: range) : int {{ *MAGIC* }}"), BuiltInFunction::ReadLine => write!(f, "() : str {{ *MAGIC* }}"), BuiltInFunction::WriteLine => write!(f, "(to_output : any) : none {{ *MAGIC* }}"), } } } static INT: OnceLock = OnceLock::new(); static IO: OnceLock = OnceLock::new(); pub const BUILT_IN_MODULES: [BuiltInModule; 2] = [BuiltInModule::Integer, BuiltInModule::Io]; pub enum BuiltInModule { Integer, Io, } impl BuiltInModule { pub fn name(&self) -> &'static str { match self { BuiltInModule::Integer => "int", BuiltInModule::Io => "io", } } pub fn value(self) -> Value { match self { BuiltInModule::Integer => { let mut properties = BTreeMap::new(); properties.insert( Identifier::new("parse"), Value::built_in_function(BuiltInFunction::IntParse), ); properties.insert( Identifier::new("random_range"), Value::built_in_function(BuiltInFunction::IntRandomRange), ); INT.get_or_init(|| Value::map(properties)).clone() } BuiltInModule::Io => { let mut properties = BTreeMap::new(); properties.insert( Identifier::new("read_line"), Value::built_in_function(BuiltInFunction::ReadLine), ); properties.insert( Identifier::new("write_line"), Value::built_in_function(BuiltInFunction::WriteLine), ); IO.get_or_init(|| Value::map(properties)).clone() } } } pub fn r#type(self) -> Type { Type::Map } }