Add type constructor
This commit is contained in:
parent
a0b754cc1c
commit
e448c9dd4c
@ -9,17 +9,22 @@ use crate::{
|
|||||||
Value,
|
Value,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, ExpectedType, Type, ValueExpression, WithPosition};
|
use super::{
|
||||||
|
AbstractNode, Evaluation, ExpectedType, Expression, Type, TypeConstructor, WithPosition,
|
||||||
|
};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct As {
|
pub struct As {
|
||||||
expression: ValueExpression,
|
expression: Expression,
|
||||||
r#type: WithPosition<Type>,
|
constructor: WithPosition<TypeConstructor>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl As {
|
impl As {
|
||||||
pub fn new(expression: ValueExpression, r#type: WithPosition<Type>) -> Self {
|
pub fn new(expression: Expression, constructor: WithPosition<TypeConstructor>) -> Self {
|
||||||
Self { expression, r#type }
|
Self {
|
||||||
|
expression,
|
||||||
|
constructor,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -29,28 +34,35 @@ impl AbstractNode for As {
|
|||||||
_context: &mut Context,
|
_context: &mut Context,
|
||||||
_manage_memory: bool,
|
_manage_memory: bool,
|
||||||
) -> Result<(), ValidationError> {
|
) -> Result<(), ValidationError> {
|
||||||
match self.r#type.node {
|
match self.constructor.node {
|
||||||
Type::Boolean | Type::Float | Type::Integer | Type::String => {}
|
TypeConstructor::Type(_) => {}
|
||||||
_ => todo!("Create an error for this occurence."),
|
_ => todo!("Create an error for this occurence."),
|
||||||
};
|
};
|
||||||
|
|
||||||
match self.expression.expected_type(_context)? {
|
match self.expression.expected_type(_context)? {
|
||||||
Type::Boolean | Type::Float | Type::Integer | Type::String => Ok(()),
|
Type::Boolean | Type::Float | Type::Integer | Type::String => {}
|
||||||
_ => todo!("Create an error for this occurence."),
|
_ => todo!("Create an error for this occurence."),
|
||||||
}
|
};
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, _context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
let expression_position = self.expression.position();
|
let expression_position = self.expression.position();
|
||||||
let action = self.expression.run(_context, _manage_memory)?;
|
let action = self.expression.evaluate(context, _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
ValidationError::InterpreterExpectedReturn(expression_position),
|
ValidationError::InterpreterExpectedReturn(expression_position),
|
||||||
));
|
));
|
||||||
};
|
};
|
||||||
let (from_value, to_type): (&ValueInner, Type) = (value.inner().borrow(), self.r#type.node);
|
let r#type = self.constructor.node.construct(&context)?;
|
||||||
|
let (from_value, to_type): (&ValueInner, Type) = (value.inner().borrow(), r#type);
|
||||||
|
|
||||||
let converted = match (from_value, to_type) {
|
let converted = match (from_value, to_type) {
|
||||||
(ValueInner::Boolean(boolean), Type::String) => Value::string(boolean.to_string()),
|
(ValueInner::Boolean(boolean), Type::String) => Value::string(boolean.to_string()),
|
||||||
@ -58,12 +70,12 @@ impl AbstractNode for As {
|
|||||||
_ => todo!("Create an error for this occurence."),
|
_ => todo!("Create an error for this occurence."),
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(Action::Return(converted))
|
Ok(Evaluation::Return(converted))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ExpectedType for As {
|
impl ExpectedType for As {
|
||||||
fn expected_type(&self, _: &mut Context) -> Result<Type, ValidationError> {
|
fn expected_type(&self, context: &mut Context) -> Result<Type, ValidationError> {
|
||||||
Ok(self.r#type.node.clone())
|
self.constructor.node.clone().construct(&context)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -7,12 +7,12 @@ use crate::{
|
|||||||
Context, Value,
|
Context, Value,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, ExpectedType, Statement, Type, WithPosition};
|
use super::{AbstractNode, Evaluation, ExpectedType, Statement, TypeConstructor, WithPosition};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct Assignment {
|
pub struct Assignment {
|
||||||
identifier: WithPosition<Identifier>,
|
identifier: WithPosition<Identifier>,
|
||||||
r#type: Option<WithPosition<Type>>,
|
constructor: Option<WithPosition<TypeConstructor>>,
|
||||||
operator: AssignmentOperator,
|
operator: AssignmentOperator,
|
||||||
statement: Box<Statement>,
|
statement: Box<Statement>,
|
||||||
}
|
}
|
||||||
@ -27,13 +27,13 @@ pub enum AssignmentOperator {
|
|||||||
impl Assignment {
|
impl Assignment {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
identifier: WithPosition<Identifier>,
|
identifier: WithPosition<Identifier>,
|
||||||
r#type: Option<WithPosition<Type>>,
|
constructor: Option<WithPosition<TypeConstructor>>,
|
||||||
operator: AssignmentOperator,
|
operator: AssignmentOperator,
|
||||||
statement: Statement,
|
statement: Statement,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Self {
|
Self {
|
||||||
identifier,
|
identifier,
|
||||||
r#type,
|
constructor,
|
||||||
operator,
|
operator,
|
||||||
statement: Box::new(statement),
|
statement: Box::new(statement),
|
||||||
}
|
}
|
||||||
@ -45,19 +45,21 @@ impl AbstractNode for Assignment {
|
|||||||
let statement_type = self.statement.expected_type(context)?;
|
let statement_type = self.statement.expected_type(context)?;
|
||||||
|
|
||||||
if let Some(WithPosition {
|
if let Some(WithPosition {
|
||||||
node: expected_type,
|
node: constructor,
|
||||||
position: expected_position,
|
position: expected_position,
|
||||||
}) = &self.r#type
|
}) = &self.constructor
|
||||||
{
|
{
|
||||||
expected_type.check(&statement_type).map_err(|conflict| {
|
let r#type = constructor.clone().construct(&context)?;
|
||||||
ValidationError::TypeCheck {
|
|
||||||
|
r#type
|
||||||
|
.check(&statement_type)
|
||||||
|
.map_err(|conflict| ValidationError::TypeCheck {
|
||||||
conflict,
|
conflict,
|
||||||
actual_position: self.statement.position(),
|
actual_position: self.statement.position(),
|
||||||
expected_position: expected_position.clone(),
|
expected_position: Some(expected_position.clone()),
|
||||||
}
|
|
||||||
})?;
|
})?;
|
||||||
|
|
||||||
context.set_type(self.identifier.node.clone(), expected_type.clone())?;
|
context.set_type(self.identifier.node.clone(), r#type.clone())?;
|
||||||
} else {
|
} else {
|
||||||
context.set_type(self.identifier.node.clone(), statement_type)?;
|
context.set_type(self.identifier.node.clone(), statement_type)?;
|
||||||
}
|
}
|
||||||
@ -67,10 +69,14 @@ impl AbstractNode for Assignment {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, context: &mut Context, manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
let action = self.statement.run(context, manage_memory)?;
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
|
let action = self.statement.evaluate(context, manage_memory)?;
|
||||||
let right = match action {
|
let right = match action {
|
||||||
Action::Return(value) => value,
|
Evaluation::Return(value) => value,
|
||||||
r#break => return Ok(r#break),
|
r#break => return Ok(r#break),
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -170,6 +176,6 @@ impl AbstractNode for Assignment {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(Action::None)
|
Ok(Evaluation::None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -8,7 +8,7 @@ use crate::{
|
|||||||
error::{RuntimeError, RwLockPoisonError, ValidationError},
|
error::{RuntimeError, RwLockPoisonError, ValidationError},
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, ExpectedType, Statement, Type};
|
use super::{AbstractNode, Evaluation, ExpectedType, Statement, Type};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct AsyncBlock {
|
pub struct AsyncBlock {
|
||||||
@ -30,15 +30,19 @@ impl AbstractNode for AsyncBlock {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, _context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
_context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
let statement_count = self.statements.len();
|
let statement_count = self.statements.len();
|
||||||
let final_result = RwLock::new(Ok(Action::None));
|
let final_result = RwLock::new(Ok(Evaluation::None));
|
||||||
|
|
||||||
self.statements
|
self.statements
|
||||||
.into_par_iter()
|
.into_par_iter()
|
||||||
.enumerate()
|
.enumerate()
|
||||||
.find_map_first(|(index, statement)| {
|
.find_map_first(|(index, statement)| {
|
||||||
let result = statement.run(&mut _context.clone(), false);
|
let result = statement.evaluate(&mut _context.clone(), false);
|
||||||
|
|
||||||
if index == statement_count - 1 {
|
if index == statement_count - 1 {
|
||||||
let get_write_lock = final_result.write();
|
let get_write_lock = final_result.write();
|
||||||
|
@ -5,7 +5,7 @@ use crate::{
|
|||||||
error::{RuntimeError, ValidationError},
|
error::{RuntimeError, ValidationError},
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, ExpectedType, Statement};
|
use super::{AbstractNode, Evaluation, ExpectedType, Statement, Type};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct Block {
|
pub struct Block {
|
||||||
@ -39,11 +39,15 @@ impl AbstractNode for Block {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, _context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
let mut previous = Action::None;
|
self,
|
||||||
|
_context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
|
let mut previous = Evaluation::None;
|
||||||
|
|
||||||
for statement in self.statements {
|
for statement in self.statements {
|
||||||
previous = statement.run(_context, _manage_memory)?;
|
previous = statement.evaluate(_context, _manage_memory)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(previous)
|
Ok(previous)
|
||||||
@ -51,7 +55,7 @@ impl AbstractNode for Block {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl ExpectedType for Block {
|
impl ExpectedType for Block {
|
||||||
fn expected_type(&self, _context: &mut Context) -> Result<super::Type, ValidationError> {
|
fn expected_type(&self, _context: &mut Context) -> Result<Type, ValidationError> {
|
||||||
self.last_statement().expected_type(_context)
|
self.last_statement().expected_type(_context)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -59,7 +63,7 @@ impl ExpectedType for Block {
|
|||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use crate::{
|
use crate::{
|
||||||
abstract_tree::{Type, ValueExpression, ValueNode, WithPos},
|
abstract_tree::{Expression, ValueNode, WithPos},
|
||||||
Value,
|
Value,
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -68,30 +72,30 @@ mod tests {
|
|||||||
#[test]
|
#[test]
|
||||||
fn run_returns_value_of_final_statement() {
|
fn run_returns_value_of_final_statement() {
|
||||||
let block = Block::new(vec![
|
let block = Block::new(vec![
|
||||||
Statement::ValueExpression(ValueExpression::Value(
|
Statement::ValueExpression(Expression::Value(
|
||||||
ValueNode::Integer(1).with_position((0, 0)),
|
ValueNode::Integer(1).with_position((0, 0)),
|
||||||
)),
|
)),
|
||||||
Statement::ValueExpression(ValueExpression::Value(
|
Statement::ValueExpression(Expression::Value(
|
||||||
ValueNode::Integer(2).with_position((0, 0)),
|
ValueNode::Integer(2).with_position((0, 0)),
|
||||||
)),
|
)),
|
||||||
Statement::ValueExpression(ValueExpression::Value(
|
Statement::ValueExpression(Expression::Value(
|
||||||
ValueNode::Integer(42).with_position((0, 0)),
|
ValueNode::Integer(42).with_position((0, 0)),
|
||||||
)),
|
)),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
block.run(&mut Context::new(None), true).unwrap(),
|
block.evaluate(&mut Context::new(None), true).unwrap(),
|
||||||
Action::Return(Value::integer(42))
|
Evaluation::Return(Value::integer(42))
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn expected_type_returns_type_of_final_statement() {
|
fn expected_type_returns_type_of_final_statement() {
|
||||||
let block = Block::new(vec![
|
let block = Block::new(vec![
|
||||||
Statement::ValueExpression(ValueExpression::Value(
|
Statement::ValueExpression(Expression::Value(
|
||||||
ValueNode::String("42".to_string()).with_position((0, 0)),
|
ValueNode::String("42".to_string()).with_position((0, 0)),
|
||||||
)),
|
)),
|
||||||
Statement::ValueExpression(ValueExpression::Value(
|
Statement::ValueExpression(Expression::Value(
|
||||||
ValueNode::Integer(42).with_position((0, 0)),
|
ValueNode::Integer(42).with_position((0, 0)),
|
||||||
)),
|
)),
|
||||||
]);
|
]);
|
||||||
|
@ -8,23 +8,22 @@ use std::{
|
|||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
abstract_tree::{Action, Type},
|
|
||||||
context::Context,
|
context::Context,
|
||||||
error::{RuntimeError, ValidationError},
|
error::{RuntimeError, ValidationError},
|
||||||
value::ValueInner,
|
value::ValueInner,
|
||||||
Value,
|
Value,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, ExpectedType, ValueExpression, WithPosition};
|
use super::{AbstractNode, Evaluation, ExpectedType, Expression, Type, TypeConstructor};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub enum BuiltInFunctionCall {
|
pub enum BuiltInFunctionCall {
|
||||||
JsonParse(WithPosition<Type>, ValueExpression),
|
JsonParse(TypeConstructor, Expression),
|
||||||
Length(ValueExpression),
|
Length(Expression),
|
||||||
ReadFile(ValueExpression),
|
ReadFile(Expression),
|
||||||
ReadLine,
|
ReadLine,
|
||||||
Sleep(ValueExpression),
|
Sleep(Expression),
|
||||||
WriteLine(ValueExpression),
|
WriteLine(Expression),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AbstractNode for BuiltInFunctionCall {
|
impl AbstractNode for BuiltInFunctionCall {
|
||||||
@ -51,11 +50,15 @@ impl AbstractNode for BuiltInFunctionCall {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
match self {
|
match self {
|
||||||
BuiltInFunctionCall::JsonParse(_type, expression) => {
|
BuiltInFunctionCall::JsonParse(_type, expression) => {
|
||||||
let action = expression.clone().run(context, _manage_memory)?;
|
let action = expression.clone().evaluate(context, _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -66,7 +69,7 @@ impl AbstractNode for BuiltInFunctionCall {
|
|||||||
if let ValueInner::String(string) = value.inner().as_ref() {
|
if let ValueInner::String(string) = value.inner().as_ref() {
|
||||||
let deserialized = serde_json::from_str(string)?;
|
let deserialized = serde_json::from_str(string)?;
|
||||||
|
|
||||||
Ok(Action::Return(deserialized))
|
Ok(Evaluation::Return(deserialized))
|
||||||
} else {
|
} else {
|
||||||
Err(RuntimeError::ValidationFailure(
|
Err(RuntimeError::ValidationFailure(
|
||||||
ValidationError::ExpectedString {
|
ValidationError::ExpectedString {
|
||||||
@ -77,8 +80,8 @@ impl AbstractNode for BuiltInFunctionCall {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
BuiltInFunctionCall::Length(expression) => {
|
BuiltInFunctionCall::Length(expression) => {
|
||||||
let action = expression.clone().run(context, _manage_memory)?;
|
let action = expression.clone().evaluate(context, _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -91,11 +94,11 @@ impl AbstractNode for BuiltInFunctionCall {
|
|||||||
0
|
0
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(Action::Return(Value::integer(length)))
|
Ok(Evaluation::Return(Value::integer(length)))
|
||||||
}
|
}
|
||||||
BuiltInFunctionCall::ReadFile(expression) => {
|
BuiltInFunctionCall::ReadFile(expression) => {
|
||||||
let action = expression.clone().run(context, _manage_memory)?;
|
let action = expression.clone().evaluate(context, _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -108,20 +111,20 @@ impl AbstractNode for BuiltInFunctionCall {
|
|||||||
String::with_capacity(0)
|
String::with_capacity(0)
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(Action::Return(Value::string(file_contents)))
|
Ok(Evaluation::Return(Value::string(file_contents)))
|
||||||
}
|
}
|
||||||
BuiltInFunctionCall::ReadLine => {
|
BuiltInFunctionCall::ReadLine => {
|
||||||
let mut buffer = String::new();
|
let mut buffer = String::new();
|
||||||
|
|
||||||
stdin().read_line(&mut buffer)?;
|
stdin().read_line(&mut buffer)?;
|
||||||
|
|
||||||
Ok(Action::Return(Value::string(
|
Ok(Evaluation::Return(Value::string(
|
||||||
buffer.strip_suffix('\n').unwrap_or(&buffer),
|
buffer.strip_suffix('\n').unwrap_or(&buffer),
|
||||||
)))
|
)))
|
||||||
}
|
}
|
||||||
BuiltInFunctionCall::Sleep(expression) => {
|
BuiltInFunctionCall::Sleep(expression) => {
|
||||||
let action = expression.clone().run(context, _manage_memory)?;
|
let action = expression.clone().evaluate(context, _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -133,11 +136,11 @@ impl AbstractNode for BuiltInFunctionCall {
|
|||||||
thread::sleep(Duration::from_millis(*milliseconds as u64));
|
thread::sleep(Duration::from_millis(*milliseconds as u64));
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(Action::None)
|
Ok(Evaluation::None)
|
||||||
}
|
}
|
||||||
BuiltInFunctionCall::WriteLine(expression) => {
|
BuiltInFunctionCall::WriteLine(expression) => {
|
||||||
let action = expression.clone().run(context, _manage_memory)?;
|
let action = expression.clone().evaluate(context, _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -153,16 +156,16 @@ impl AbstractNode for BuiltInFunctionCall {
|
|||||||
stdout.flush()?;
|
stdout.flush()?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(Action::None)
|
Ok(Evaluation::None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ExpectedType for BuiltInFunctionCall {
|
impl ExpectedType for BuiltInFunctionCall {
|
||||||
fn expected_type(&self, _context: &mut Context) -> Result<Type, ValidationError> {
|
fn expected_type(&self, context: &mut Context) -> Result<Type, ValidationError> {
|
||||||
match self {
|
match self {
|
||||||
BuiltInFunctionCall::JsonParse(r#type, _) => Ok(r#type.node.clone()),
|
BuiltInFunctionCall::JsonParse(r#type, _) => Ok(r#type.clone().construct(&context)?),
|
||||||
BuiltInFunctionCall::Length(_) => Ok(Type::Integer),
|
BuiltInFunctionCall::Length(_) => Ok(Type::Integer),
|
||||||
BuiltInFunctionCall::ReadFile(_) => Ok(Type::String),
|
BuiltInFunctionCall::ReadFile(_) => Ok(Type::String),
|
||||||
BuiltInFunctionCall::ReadLine => Ok(Type::String),
|
BuiltInFunctionCall::ReadLine => Ok(Type::String),
|
||||||
|
@ -6,25 +6,27 @@ use crate::{
|
|||||||
value::ValueInner,
|
value::ValueInner,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, ExpectedType, Type, ValueExpression, WithPosition};
|
use super::{
|
||||||
|
AbstractNode, Evaluation, ExpectedType, Expression, Type, TypeConstructor, WithPosition,
|
||||||
|
};
|
||||||
|
|
||||||
#[derive(Debug, Clone, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Debug, Clone, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct FunctionCall {
|
pub struct FunctionCall {
|
||||||
function: Box<ValueExpression>,
|
function: Box<Expression>,
|
||||||
type_arguments: Vec<WithPosition<Type>>,
|
type_arguments: Option<Vec<WithPosition<TypeConstructor>>>,
|
||||||
arguments: Vec<ValueExpression>,
|
value_arguments: Vec<Expression>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl FunctionCall {
|
impl FunctionCall {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
function: ValueExpression,
|
function: Expression,
|
||||||
type_arguments: Vec<WithPosition<Type>>,
|
type_arguments: Option<Vec<WithPosition<TypeConstructor>>>,
|
||||||
arguments: Vec<ValueExpression>,
|
value_arguments: Vec<Expression>,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
FunctionCall {
|
FunctionCall {
|
||||||
function: Box::new(function),
|
function: Box::new(function),
|
||||||
type_arguments,
|
type_arguments,
|
||||||
arguments,
|
value_arguments,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -33,48 +35,28 @@ impl AbstractNode for FunctionCall {
|
|||||||
fn validate(&self, context: &mut Context, manage_memory: bool) -> Result<(), ValidationError> {
|
fn validate(&self, context: &mut Context, manage_memory: bool) -> Result<(), ValidationError> {
|
||||||
self.function.validate(context, manage_memory)?;
|
self.function.validate(context, manage_memory)?;
|
||||||
|
|
||||||
for expression in &self.arguments {
|
for expression in &self.value_arguments {
|
||||||
expression.validate(context, manage_memory)?;
|
expression.validate(context, manage_memory)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
let function_node_type = self.function.expected_type(context)?;
|
let function_node_type = self.function.expected_type(context)?;
|
||||||
|
|
||||||
if let Type::Function {
|
if let Type::Function {
|
||||||
parameter_types,
|
type_parameters,
|
||||||
|
value_parameters: _,
|
||||||
return_type: _,
|
return_type: _,
|
||||||
} = function_node_type
|
} = function_node_type
|
||||||
{
|
{
|
||||||
for (type_parameter, type_argument) in
|
match (type_parameters, &self.type_arguments) {
|
||||||
parameter_types.iter().zip(self.type_arguments.iter())
|
(Some(type_parameters), Some(type_arguments)) => {
|
||||||
{
|
if type_parameters.len() != type_arguments.len() {
|
||||||
if let Type::Argument(_) = type_parameter.node {
|
return Err(ValidationError::WrongTypeArgumentCount {
|
||||||
continue;
|
actual: type_parameters.len(),
|
||||||
|
expected: type_arguments.len(),
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
type_parameter
|
|
||||||
.node
|
|
||||||
.check(&type_argument.node)
|
|
||||||
.map_err(|conflict| ValidationError::TypeCheck {
|
|
||||||
conflict,
|
|
||||||
actual_position: type_argument.position,
|
|
||||||
expected_position: type_parameter.position,
|
|
||||||
})?;
|
|
||||||
}
|
}
|
||||||
|
_ => {}
|
||||||
for (type_parameter, expression) in parameter_types.iter().zip(self.arguments.iter()) {
|
|
||||||
if let Type::Argument(_) = type_parameter.node {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
let actual = expression.expected_type(context)?;
|
|
||||||
|
|
||||||
type_parameter.node.check(&actual).map_err(|conflict| {
|
|
||||||
ValidationError::TypeCheck {
|
|
||||||
conflict,
|
|
||||||
actual_position: expression.position(),
|
|
||||||
expected_position: type_parameter.position,
|
|
||||||
}
|
|
||||||
})?;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -86,10 +68,14 @@ impl AbstractNode for FunctionCall {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, context: &mut Context, clear_variables: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
clear_variables: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
let function_position = self.function.position();
|
let function_position = self.function.position();
|
||||||
let action = self.function.run(context, clear_variables)?;
|
let action = self.function.evaluate(context, clear_variables)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -97,7 +83,7 @@ impl AbstractNode for FunctionCall {
|
|||||||
));
|
));
|
||||||
};
|
};
|
||||||
let function = if let ValueInner::Function(function) = value.inner().as_ref() {
|
let function = if let ValueInner::Function(function) = value.inner().as_ref() {
|
||||||
function
|
function.clone()
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
ValidationError::ExpectedFunction {
|
ValidationError::ExpectedFunction {
|
||||||
@ -106,12 +92,12 @@ impl AbstractNode for FunctionCall {
|
|||||||
},
|
},
|
||||||
));
|
));
|
||||||
};
|
};
|
||||||
let mut arguments = Vec::with_capacity(self.arguments.len());
|
let mut arguments = Vec::with_capacity(self.value_arguments.len());
|
||||||
|
|
||||||
for expression in self.arguments {
|
for expression in self.value_arguments {
|
||||||
let expression_position = expression.position();
|
let expression_position = expression.position();
|
||||||
let action = expression.run(context, clear_variables)?;
|
let action = expression.evaluate(context, clear_variables)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -124,16 +110,18 @@ impl AbstractNode for FunctionCall {
|
|||||||
|
|
||||||
let mut function_context = Context::new(Some(&context));
|
let mut function_context = Context::new(Some(&context));
|
||||||
|
|
||||||
for (type_parameter, type_argument) in function
|
match (function.type_parameters(), self.type_arguments) {
|
||||||
.type_parameters()
|
(Some(type_parameters), Some(type_arguments)) => {
|
||||||
.iter()
|
for (parameter, constructor) in
|
||||||
.map(|r#type| r#type.node.clone())
|
type_parameters.into_iter().zip(type_arguments.into_iter())
|
||||||
.zip(self.type_arguments.into_iter().map(|r#type| r#type.node))
|
|
||||||
{
|
{
|
||||||
if let Type::Argument(identifier) = type_parameter {
|
let r#type = constructor.node.construct(context)?;
|
||||||
function_context.set_type(identifier, type_argument)?;
|
|
||||||
|
function_context.set_type(parameter.clone(), r#type)?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
|
||||||
function
|
function
|
||||||
.clone()
|
.clone()
|
||||||
@ -146,7 +134,7 @@ impl ExpectedType for FunctionCall {
|
|||||||
let function_node_type = self.function.expected_type(_context)?;
|
let function_node_type = self.function.expected_type(_context)?;
|
||||||
|
|
||||||
if let Type::Function { return_type, .. } = function_node_type {
|
if let Type::Function { return_type, .. } = function_node_type {
|
||||||
Ok(return_type.node)
|
Ok(*return_type)
|
||||||
} else {
|
} else {
|
||||||
Err(ValidationError::ExpectedFunction {
|
Err(ValidationError::ExpectedFunction {
|
||||||
actual: function_node_type,
|
actual: function_node_type,
|
||||||
|
@ -6,21 +6,21 @@ use crate::{
|
|||||||
value::ValueInner,
|
value::ValueInner,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, Block, ExpectedType, Type, ValueExpression, WithPosition};
|
use super::{AbstractNode, Block, Evaluation, ExpectedType, Expression, Type, WithPosition};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct IfElse {
|
pub struct IfElse {
|
||||||
if_expression: ValueExpression,
|
if_expression: Expression,
|
||||||
if_block: WithPosition<Block>,
|
if_block: WithPosition<Block>,
|
||||||
else_ifs: Vec<(ValueExpression, WithPosition<Block>)>,
|
else_ifs: Vec<(Expression, WithPosition<Block>)>,
|
||||||
else_block: Option<WithPosition<Block>>,
|
else_block: Option<WithPosition<Block>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl IfElse {
|
impl IfElse {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
if_expression: ValueExpression,
|
if_expression: Expression,
|
||||||
if_block: WithPosition<Block>,
|
if_block: WithPosition<Block>,
|
||||||
else_ifs: Vec<(ValueExpression, WithPosition<Block>)>,
|
else_ifs: Vec<(Expression, WithPosition<Block>)>,
|
||||||
else_block: Option<WithPosition<Block>>,
|
else_block: Option<WithPosition<Block>>,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Self {
|
Self {
|
||||||
@ -51,7 +51,7 @@ impl AbstractNode for IfElse {
|
|||||||
.map_err(|conflict| ValidationError::TypeCheck {
|
.map_err(|conflict| ValidationError::TypeCheck {
|
||||||
conflict,
|
conflict,
|
||||||
actual_position: else_block.node.last_statement().position(),
|
actual_position: else_block.node.last_statement().position(),
|
||||||
expected_position: self.if_block.node.first_statement().position(),
|
expected_position: Some(self.if_block.node.first_statement().position()),
|
||||||
})?;
|
})?;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -74,7 +74,7 @@ impl AbstractNode for IfElse {
|
|||||||
.map_err(|conflict| ValidationError::TypeCheck {
|
.map_err(|conflict| ValidationError::TypeCheck {
|
||||||
conflict,
|
conflict,
|
||||||
actual_position: self.if_block.node.last_statement().position(),
|
actual_position: self.if_block.node.last_statement().position(),
|
||||||
expected_position: self.if_expression.position(),
|
expected_position: Some(self.if_expression.position()),
|
||||||
})?;
|
})?;
|
||||||
} else {
|
} else {
|
||||||
return Err(ValidationError::ExpectedBoolean {
|
return Err(ValidationError::ExpectedBoolean {
|
||||||
@ -87,10 +87,14 @@ impl AbstractNode for IfElse {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
let if_position = self.if_expression.position();
|
let if_position = self.if_expression.position();
|
||||||
let action = self.if_expression.run(context, _manage_memory)?;
|
let action = self.if_expression.evaluate(context, _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -100,12 +104,12 @@ impl AbstractNode for IfElse {
|
|||||||
|
|
||||||
if let ValueInner::Boolean(if_boolean) = value.inner().as_ref() {
|
if let ValueInner::Boolean(if_boolean) = value.inner().as_ref() {
|
||||||
if *if_boolean {
|
if *if_boolean {
|
||||||
self.if_block.node.run(context, _manage_memory)
|
self.if_block.node.evaluate(context, _manage_memory)
|
||||||
} else {
|
} else {
|
||||||
for (expression, block) in self.else_ifs {
|
for (expression, block) in self.else_ifs {
|
||||||
let expression_position = expression.position();
|
let expression_position = expression.position();
|
||||||
let action = expression.run(context, _manage_memory)?;
|
let action = expression.evaluate(context, _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -115,7 +119,7 @@ impl AbstractNode for IfElse {
|
|||||||
|
|
||||||
if let ValueInner::Boolean(else_if_boolean) = value.inner().as_ref() {
|
if let ValueInner::Boolean(else_if_boolean) = value.inner().as_ref() {
|
||||||
if *else_if_boolean {
|
if *else_if_boolean {
|
||||||
return block.node.run(context, _manage_memory);
|
return block.node.evaluate(context, _manage_memory);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -128,9 +132,9 @@ impl AbstractNode for IfElse {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if let Some(else_statement) = self.else_block {
|
if let Some(else_statement) = self.else_block {
|
||||||
else_statement.node.run(context, _manage_memory)
|
else_statement.node.evaluate(context, _manage_memory)
|
||||||
} else {
|
} else {
|
||||||
Ok(Action::None)
|
Ok(Evaluation::None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -163,17 +167,17 @@ mod tests {
|
|||||||
fn simple_if() {
|
fn simple_if() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
IfElse::new(
|
IfElse::new(
|
||||||
ValueExpression::Value(ValueNode::Boolean(true).with_position((0, 0))),
|
Expression::Value(ValueNode::Boolean(true).with_position((0, 0))),
|
||||||
Block::new(vec![Statement::ValueExpression(ValueExpression::Value(
|
Block::new(vec![Statement::ValueExpression(Expression::Value(
|
||||||
ValueNode::String("foo".to_string()).with_position((0, 0))
|
ValueNode::String("foo".to_string()).with_position((0, 0))
|
||||||
))])
|
))])
|
||||||
.with_position((0, 0)),
|
.with_position((0, 0)),
|
||||||
Vec::with_capacity(0),
|
Vec::with_capacity(0),
|
||||||
None
|
None
|
||||||
)
|
)
|
||||||
.run(&mut Context::new(None), true)
|
.evaluate(&mut Context::new(None), true)
|
||||||
.unwrap(),
|
.unwrap(),
|
||||||
Action::Return(Value::string("foo".to_string()))
|
Evaluation::Return(Value::string("foo".to_string()))
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,64 +5,72 @@ use crate::{
|
|||||||
error::{RuntimeError, ValidationError},
|
error::{RuntimeError, ValidationError},
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, ExpectedType, Type, ValueExpression, ValueNode, WithPosition};
|
use super::{AbstractNode, Evaluation, ExpectedType, Expression, Type, ValueNode, WithPosition};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct ListIndex {
|
pub struct ListIndex {
|
||||||
left: ValueExpression,
|
collection: Expression,
|
||||||
right: ValueExpression,
|
index: Expression,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ListIndex {
|
impl ListIndex {
|
||||||
pub fn new(left: ValueExpression, right: ValueExpression) -> Self {
|
pub fn new(left: Expression, right: Expression) -> Self {
|
||||||
Self { left, right }
|
Self {
|
||||||
|
collection: left,
|
||||||
|
index: right,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AbstractNode for ListIndex {
|
impl AbstractNode for ListIndex {
|
||||||
fn validate(&self, context: &mut Context, _manage_memory: bool) -> Result<(), ValidationError> {
|
fn validate(&self, context: &mut Context, _manage_memory: bool) -> Result<(), ValidationError> {
|
||||||
self.left.validate(context, _manage_memory)?;
|
self.collection.validate(context, _manage_memory)?;
|
||||||
self.right.validate(context, _manage_memory)?;
|
self.index.validate(context, _manage_memory)?;
|
||||||
|
|
||||||
let left_type = self.left.expected_type(context)?;
|
let collection_type = self.collection.expected_type(context)?;
|
||||||
|
let index_type = self.index.expected_type(context)?;
|
||||||
|
|
||||||
match left_type {
|
match collection_type {
|
||||||
Type::List => todo!(),
|
Type::List {
|
||||||
Type::ListOf(_) => todo!(),
|
length: _,
|
||||||
Type::ListExact(_) => {
|
item_type: _,
|
||||||
let right_type = self.right.expected_type(context)?;
|
} => {
|
||||||
|
if index_type == Type::Integer {
|
||||||
if let Type::Integer = right_type {
|
|
||||||
Ok(())
|
Ok(())
|
||||||
} else {
|
} else {
|
||||||
Err(ValidationError::CannotIndexWith {
|
Err(ValidationError::CannotIndexWith {
|
||||||
collection_type: left_type,
|
collection_type,
|
||||||
collection_position: self.left.position(),
|
collection_position: self.collection.position(),
|
||||||
index_type: right_type,
|
index_type,
|
||||||
index_position: self.right.position(),
|
index_position: self.index.position(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Type::ListOf(_) => todo!(),
|
||||||
_ => Err(ValidationError::CannotIndex {
|
_ => Err(ValidationError::CannotIndex {
|
||||||
r#type: left_type,
|
r#type: collection_type,
|
||||||
position: self.left.position(),
|
position: self.collection.position(),
|
||||||
}),
|
}),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, context: &mut Context, _clear_variables: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
let left_position = self.left.position();
|
self,
|
||||||
let left_action = self.left.run(context, _clear_variables)?;
|
context: &mut Context,
|
||||||
let left_value = if let Action::Return(value) = left_action {
|
_clear_variables: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
|
let left_position = self.collection.position();
|
||||||
|
let left_action = self.collection.evaluate(context, _clear_variables)?;
|
||||||
|
let left_value = if let Evaluation::Return(value) = left_action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
ValidationError::InterpreterExpectedReturn(left_position),
|
ValidationError::InterpreterExpectedReturn(left_position),
|
||||||
));
|
));
|
||||||
};
|
};
|
||||||
let right_position = self.right.position();
|
let right_position = self.index.position();
|
||||||
let right_action = self.right.run(context, _clear_variables)?;
|
let right_action = self.index.evaluate(context, _clear_variables)?;
|
||||||
let right_value = if let Action::Return(value) = right_action {
|
let right_value = if let Evaluation::Return(value) = right_action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -74,9 +82,9 @@ impl AbstractNode for ListIndex {
|
|||||||
let found_item = list.get(index as usize);
|
let found_item = list.get(index as usize);
|
||||||
|
|
||||||
if let Some(item) = found_item {
|
if let Some(item) = found_item {
|
||||||
Ok(Action::Return(item.node.clone()))
|
Ok(Evaluation::Return(item.node.clone()))
|
||||||
} else {
|
} else {
|
||||||
Ok(Action::None)
|
Ok(Evaluation::None)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Err(RuntimeError::ValidationFailure(
|
Err(RuntimeError::ValidationFailure(
|
||||||
@ -93,18 +101,18 @@ impl AbstractNode for ListIndex {
|
|||||||
|
|
||||||
impl ExpectedType for ListIndex {
|
impl ExpectedType for ListIndex {
|
||||||
fn expected_type(&self, _context: &mut Context) -> Result<Type, ValidationError> {
|
fn expected_type(&self, _context: &mut Context) -> Result<Type, ValidationError> {
|
||||||
let left_type = self.left.expected_type(_context)?;
|
let left_type = self.collection.expected_type(_context)?;
|
||||||
|
|
||||||
if let (
|
if let (
|
||||||
ValueExpression::Value(WithPosition {
|
Expression::Value(WithPosition {
|
||||||
node: ValueNode::List(expression_list),
|
node: ValueNode::List(expression_list),
|
||||||
..
|
..
|
||||||
}),
|
}),
|
||||||
ValueExpression::Value(WithPosition {
|
Expression::Value(WithPosition {
|
||||||
node: ValueNode::Integer(index),
|
node: ValueNode::Integer(index),
|
||||||
..
|
..
|
||||||
}),
|
}),
|
||||||
) = (&self.left, &self.right)
|
) = (&self.collection, &self.index)
|
||||||
{
|
{
|
||||||
let expression = if let Some(expression) = expression_list.get(*index as usize) {
|
let expression = if let Some(expression) = expression_list.get(*index as usize) {
|
||||||
expression
|
expression
|
||||||
@ -116,7 +124,7 @@ impl ExpectedType for ListIndex {
|
|||||||
} else {
|
} else {
|
||||||
Err(ValidationError::CannotIndex {
|
Err(ValidationError::CannotIndex {
|
||||||
r#type: left_type,
|
r#type: left_type,
|
||||||
position: self.left.position(),
|
position: self.collection.position(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -7,19 +7,19 @@ use crate::{
|
|||||||
Value,
|
Value,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, ExpectedType, Type, ValueExpression};
|
use super::{AbstractNode, Evaluation, ExpectedType, Expression, Type};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub enum Logic {
|
pub enum Logic {
|
||||||
Equal(ValueExpression, ValueExpression),
|
Equal(Expression, Expression),
|
||||||
NotEqual(ValueExpression, ValueExpression),
|
NotEqual(Expression, Expression),
|
||||||
Greater(ValueExpression, ValueExpression),
|
Greater(Expression, Expression),
|
||||||
Less(ValueExpression, ValueExpression),
|
Less(Expression, Expression),
|
||||||
GreaterOrEqual(ValueExpression, ValueExpression),
|
GreaterOrEqual(Expression, Expression),
|
||||||
LessOrEqual(ValueExpression, ValueExpression),
|
LessOrEqual(Expression, Expression),
|
||||||
And(ValueExpression, ValueExpression),
|
And(Expression, Expression),
|
||||||
Or(ValueExpression, ValueExpression),
|
Or(Expression, Expression),
|
||||||
Not(ValueExpression),
|
Not(Expression),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AbstractNode for Logic {
|
impl AbstractNode for Logic {
|
||||||
@ -42,7 +42,7 @@ impl AbstractNode for Logic {
|
|||||||
.map_err(|conflict| ValidationError::TypeCheck {
|
.map_err(|conflict| ValidationError::TypeCheck {
|
||||||
conflict,
|
conflict,
|
||||||
actual_position: left.position(),
|
actual_position: left.position(),
|
||||||
expected_position: right.position(),
|
expected_position: Some(right.position()),
|
||||||
})?;
|
})?;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -89,11 +89,15 @@ impl AbstractNode for Logic {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
let run_and_expect_value = |expression: ValueExpression| -> Result<Value, RuntimeError> {
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
|
let run_and_expect_value = |expression: Expression| -> Result<Value, RuntimeError> {
|
||||||
let expression_position = expression.position();
|
let expression_position = expression.position();
|
||||||
let action = expression.run(&mut context.clone(), _manage_memory)?;
|
let action = expression.evaluate(&mut context.clone(), _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -104,10 +108,10 @@ impl AbstractNode for Logic {
|
|||||||
Ok(value)
|
Ok(value)
|
||||||
};
|
};
|
||||||
|
|
||||||
let run_and_expect_boolean = |expression: ValueExpression| -> Result<bool, RuntimeError> {
|
let run_and_expect_boolean = |expression: Expression| -> Result<bool, RuntimeError> {
|
||||||
let expression_position = expression.position();
|
let expression_position = expression.position();
|
||||||
let action = expression.run(&mut context.clone(), _manage_memory)?;
|
let action = expression.evaluate(&mut context.clone(), _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -187,7 +191,7 @@ impl AbstractNode for Logic {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(Action::Return(Value::boolean(boolean)))
|
Ok(Evaluation::Return(Value::boolean(boolean)))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -207,11 +211,11 @@ mod tests {
|
|||||||
fn equal() {
|
fn equal() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Logic::Equal(
|
Logic::Equal(
|
||||||
ValueExpression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
Expression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
||||||
ValueExpression::Value(ValueNode::Integer(42).with_position((0, 0)))
|
Expression::Value(ValueNode::Integer(42).with_position((0, 0)))
|
||||||
)
|
)
|
||||||
.run(&mut Context::new(None), true),
|
.evaluate(&mut Context::new(None), true),
|
||||||
Ok(Action::Return(Value::boolean(true)))
|
Ok(Evaluation::Return(Value::boolean(true)))
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -219,11 +223,11 @@ mod tests {
|
|||||||
fn not_equal() {
|
fn not_equal() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Logic::NotEqual(
|
Logic::NotEqual(
|
||||||
ValueExpression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
Expression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
||||||
ValueExpression::Value(ValueNode::Integer(43).with_position((0, 0)))
|
Expression::Value(ValueNode::Integer(43).with_position((0, 0)))
|
||||||
)
|
)
|
||||||
.run(&mut Context::new(None), true),
|
.evaluate(&mut Context::new(None), true),
|
||||||
Ok(Action::Return(Value::boolean(true)))
|
Ok(Evaluation::Return(Value::boolean(true)))
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -231,11 +235,11 @@ mod tests {
|
|||||||
fn greater() {
|
fn greater() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Logic::Greater(
|
Logic::Greater(
|
||||||
ValueExpression::Value(ValueNode::Integer(43).with_position((0, 0))),
|
Expression::Value(ValueNode::Integer(43).with_position((0, 0))),
|
||||||
ValueExpression::Value(ValueNode::Integer(42).with_position((0, 0)))
|
Expression::Value(ValueNode::Integer(42).with_position((0, 0)))
|
||||||
)
|
)
|
||||||
.run(&mut Context::new(None), true),
|
.evaluate(&mut Context::new(None), true),
|
||||||
Ok(Action::Return(Value::boolean(true)))
|
Ok(Evaluation::Return(Value::boolean(true)))
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -243,11 +247,11 @@ mod tests {
|
|||||||
fn less() {
|
fn less() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Logic::Less(
|
Logic::Less(
|
||||||
ValueExpression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
Expression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
||||||
ValueExpression::Value(ValueNode::Integer(43).with_position((0, 0)))
|
Expression::Value(ValueNode::Integer(43).with_position((0, 0)))
|
||||||
)
|
)
|
||||||
.run(&mut Context::new(None), true),
|
.evaluate(&mut Context::new(None), true),
|
||||||
Ok(Action::Return(Value::boolean(true)))
|
Ok(Evaluation::Return(Value::boolean(true)))
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -255,20 +259,20 @@ mod tests {
|
|||||||
fn greater_or_equal() {
|
fn greater_or_equal() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Logic::GreaterOrEqual(
|
Logic::GreaterOrEqual(
|
||||||
ValueExpression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
Expression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
||||||
ValueExpression::Value(ValueNode::Integer(41).with_position((0, 0)))
|
Expression::Value(ValueNode::Integer(41).with_position((0, 0)))
|
||||||
)
|
)
|
||||||
.run(&mut Context::new(None), true),
|
.evaluate(&mut Context::new(None), true),
|
||||||
Ok(Action::Return(Value::boolean(true)))
|
Ok(Evaluation::Return(Value::boolean(true)))
|
||||||
);
|
);
|
||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Logic::GreaterOrEqual(
|
Logic::GreaterOrEqual(
|
||||||
ValueExpression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
Expression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
||||||
ValueExpression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
Expression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
||||||
)
|
)
|
||||||
.run(&mut Context::new(None), true),
|
.evaluate(&mut Context::new(None), true),
|
||||||
Ok(Action::Return(Value::boolean(true)))
|
Ok(Evaluation::Return(Value::boolean(true)))
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -276,20 +280,20 @@ mod tests {
|
|||||||
fn less_or_equal() {
|
fn less_or_equal() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Logic::LessOrEqual(
|
Logic::LessOrEqual(
|
||||||
ValueExpression::Value(ValueNode::Integer(41).with_position((0, 0))),
|
Expression::Value(ValueNode::Integer(41).with_position((0, 0))),
|
||||||
ValueExpression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
Expression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
||||||
)
|
)
|
||||||
.run(&mut Context::new(None), true),
|
.evaluate(&mut Context::new(None), true),
|
||||||
Ok(Action::Return(Value::boolean(true)))
|
Ok(Evaluation::Return(Value::boolean(true)))
|
||||||
);
|
);
|
||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Logic::LessOrEqual(
|
Logic::LessOrEqual(
|
||||||
ValueExpression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
Expression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
||||||
ValueExpression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
Expression::Value(ValueNode::Integer(42).with_position((0, 0))),
|
||||||
)
|
)
|
||||||
.run(&mut Context::new(None), true),
|
.evaluate(&mut Context::new(None), true),
|
||||||
Ok(Action::Return(Value::boolean(true)))
|
Ok(Evaluation::Return(Value::boolean(true)))
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -297,11 +301,11 @@ mod tests {
|
|||||||
fn and() {
|
fn and() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Logic::And(
|
Logic::And(
|
||||||
ValueExpression::Value(ValueNode::Boolean(true).with_position((0, 0))),
|
Expression::Value(ValueNode::Boolean(true).with_position((0, 0))),
|
||||||
ValueExpression::Value(ValueNode::Boolean(true).with_position((0, 0))),
|
Expression::Value(ValueNode::Boolean(true).with_position((0, 0))),
|
||||||
)
|
)
|
||||||
.run(&mut Context::new(None), true),
|
.evaluate(&mut Context::new(None), true),
|
||||||
Ok(Action::Return(Value::boolean(true)))
|
Ok(Evaluation::Return(Value::boolean(true)))
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -309,22 +313,22 @@ mod tests {
|
|||||||
fn or() {
|
fn or() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Logic::Or(
|
Logic::Or(
|
||||||
ValueExpression::Value(ValueNode::Boolean(true).with_position((0, 0))),
|
Expression::Value(ValueNode::Boolean(true).with_position((0, 0))),
|
||||||
ValueExpression::Value(ValueNode::Boolean(false).with_position((0, 0))),
|
Expression::Value(ValueNode::Boolean(false).with_position((0, 0))),
|
||||||
)
|
)
|
||||||
.run(&mut Context::new(None), true),
|
.evaluate(&mut Context::new(None), true),
|
||||||
Ok(Action::Return(Value::boolean(true)))
|
Ok(Evaluation::Return(Value::boolean(true)))
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn not() {
|
fn not() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Logic::Not(ValueExpression::Value(
|
Logic::Not(Expression::Value(
|
||||||
ValueNode::Boolean(false).with_position((0, 0))
|
ValueNode::Boolean(false).with_position((0, 0))
|
||||||
))
|
))
|
||||||
.run(&mut Context::new(None), true),
|
.evaluate(&mut Context::new(None), true),
|
||||||
Ok(Action::Return(Value::boolean(true)))
|
Ok(Evaluation::Return(Value::boolean(true)))
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,7 +5,7 @@ use crate::{
|
|||||||
error::{RuntimeError, ValidationError},
|
error::{RuntimeError, ValidationError},
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, Statement};
|
use super::{AbstractNode, Evaluation, Statement};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct Loop {
|
pub struct Loop {
|
||||||
@ -31,15 +31,19 @@ impl AbstractNode for Loop {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, _context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
_context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
loop {
|
loop {
|
||||||
for statement in &self.statements {
|
for statement in &self.statements {
|
||||||
let action = statement.clone().run(_context, false)?;
|
let action = statement.clone().evaluate(_context, false)?;
|
||||||
|
|
||||||
match action {
|
match action {
|
||||||
Action::Return(_) => {}
|
Evaluation::Return(_) => {}
|
||||||
Action::None => {}
|
Evaluation::None => {}
|
||||||
Action::Break => return Ok(Action::Break),
|
Evaluation::Break => return Ok(Evaluation::Break),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,16 +6,16 @@ use crate::{
|
|||||||
value::ValueInner,
|
value::ValueInner,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, ExpectedType, Type, ValueExpression, ValueNode, WithPosition};
|
use super::{AbstractNode, Evaluation, ExpectedType, Expression, Type, ValueNode, WithPosition};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct MapIndex {
|
pub struct MapIndex {
|
||||||
collection: ValueExpression,
|
collection: Expression,
|
||||||
index: ValueExpression,
|
index: Expression,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl MapIndex {
|
impl MapIndex {
|
||||||
pub fn new(left: ValueExpression, right: ValueExpression) -> Self {
|
pub fn new(left: Expression, right: Expression) -> Self {
|
||||||
Self {
|
Self {
|
||||||
collection: left,
|
collection: left,
|
||||||
index: right,
|
index: right,
|
||||||
@ -32,10 +32,14 @@ impl AbstractNode for MapIndex {
|
|||||||
self.collection.validate(_context, _manage_memory)
|
self.collection.validate(_context, _manage_memory)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
let collection_position = self.collection.position();
|
let collection_position = self.collection.position();
|
||||||
let action = self.collection.run(context, _manage_memory)?;
|
let action = self.collection.evaluate(context, _manage_memory)?;
|
||||||
let collection = if let Action::Return(value) = action {
|
let collection = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -43,13 +47,13 @@ impl AbstractNode for MapIndex {
|
|||||||
));
|
));
|
||||||
};
|
};
|
||||||
|
|
||||||
if let (ValueInner::Map(map), ValueExpression::Identifier(index)) =
|
if let (ValueInner::Map(map), Expression::Identifier(index)) =
|
||||||
(collection.inner().as_ref(), self.index)
|
(collection.inner().as_ref(), self.index)
|
||||||
{
|
{
|
||||||
let action = map
|
let action = map
|
||||||
.get(&index.node)
|
.get(&index.node)
|
||||||
.map(|value| Action::Return(value.clone()))
|
.map(|value| Evaluation::Return(value.clone()))
|
||||||
.unwrap_or(Action::None);
|
.unwrap_or(Evaluation::None);
|
||||||
|
|
||||||
Ok(action)
|
Ok(action)
|
||||||
} else {
|
} else {
|
||||||
@ -65,7 +69,7 @@ impl AbstractNode for MapIndex {
|
|||||||
|
|
||||||
impl ExpectedType for MapIndex {
|
impl ExpectedType for MapIndex {
|
||||||
fn expected_type(&self, context: &mut Context) -> Result<Type, ValidationError> {
|
fn expected_type(&self, context: &mut Context) -> Result<Type, ValidationError> {
|
||||||
if let (ValueExpression::Identifier(collection), ValueExpression::Identifier(index)) =
|
if let (Expression::Identifier(collection), Expression::Identifier(index)) =
|
||||||
(&self.collection, &self.index)
|
(&self.collection, &self.index)
|
||||||
{
|
{
|
||||||
let collection = if let Some(collection) = context.get_value(&collection.node)? {
|
let collection = if let Some(collection) = context.get_value(&collection.node)? {
|
||||||
@ -90,17 +94,19 @@ impl ExpectedType for MapIndex {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if let (
|
if let (
|
||||||
ValueExpression::Value(WithPosition {
|
Expression::Value(WithPosition {
|
||||||
node: ValueNode::Map(properties),
|
node: ValueNode::Map(properties),
|
||||||
..
|
..
|
||||||
}),
|
}),
|
||||||
ValueExpression::Identifier(index),
|
Expression::Identifier(index),
|
||||||
) = (&self.collection, &self.index)
|
) = (&self.collection, &self.index)
|
||||||
{
|
{
|
||||||
for (property, type_option, expression) in properties {
|
for (property, constructor_option, expression) in properties {
|
||||||
if property == &index.node {
|
if property == &index.node {
|
||||||
return if let Some(r#type) = type_option {
|
return if let Some(constructor) = constructor_option {
|
||||||
Ok(r#type.node.clone())
|
let r#type = constructor.node.clone().construct(&context)?;
|
||||||
|
|
||||||
|
Ok(r#type)
|
||||||
} else {
|
} else {
|
||||||
Ok(expression.expected_type(context)?)
|
Ok(expression.expected_type(context)?)
|
||||||
};
|
};
|
||||||
@ -111,15 +117,15 @@ impl ExpectedType for MapIndex {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if let (
|
if let (
|
||||||
ValueExpression::Value(WithPosition {
|
Expression::Value(WithPosition {
|
||||||
node: ValueNode::Structure { fields, .. },
|
node: ValueNode::Structure { fields, .. },
|
||||||
..
|
..
|
||||||
}),
|
}),
|
||||||
ValueExpression::Identifier(index),
|
Expression::Identifier(index),
|
||||||
) = (&self.collection, &self.index)
|
) = (&self.collection, &self.index)
|
||||||
{
|
{
|
||||||
return if let Some(type_result) = fields.iter().find_map(|(property, expression)| {
|
return if let Some(type_result) = fields.iter().find_map(|(property, expression)| {
|
||||||
if property == &index.node {
|
if property.node == index.node {
|
||||||
Some(expression.expected_type(context))
|
Some(expression.expected_type(context))
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
|
@ -7,15 +7,15 @@ use crate::{
|
|||||||
Value,
|
Value,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, ExpectedType, SourcePosition, Type, ValueExpression};
|
use super::{AbstractNode, Evaluation, ExpectedType, Expression, SourcePosition, Type};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub enum Math {
|
pub enum Math {
|
||||||
Add(ValueExpression, ValueExpression),
|
Add(Expression, Expression),
|
||||||
Subtract(ValueExpression, ValueExpression),
|
Subtract(Expression, Expression),
|
||||||
Multiply(ValueExpression, ValueExpression),
|
Multiply(Expression, Expression),
|
||||||
Divide(ValueExpression, ValueExpression),
|
Divide(Expression, Expression),
|
||||||
Modulo(ValueExpression, ValueExpression),
|
Modulo(Expression, Expression),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AbstractNode for Math {
|
impl AbstractNode for Math {
|
||||||
@ -67,12 +67,15 @@ impl AbstractNode for Math {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, _context: &mut Context, _clear_variables: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
let run_and_expect_value = |position: SourcePosition,
|
self,
|
||||||
expression: ValueExpression|
|
_context: &mut Context,
|
||||||
-> Result<Value, RuntimeError> {
|
_clear_variables: bool,
|
||||||
let action = expression.run(&mut _context.clone(), _clear_variables)?;
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
let value = if let Action::Return(value) = action {
|
let run_and_expect_value =
|
||||||
|
|position: SourcePosition, expression: Expression| -> Result<Value, RuntimeError> {
|
||||||
|
let action = expression.evaluate(&mut _context.clone(), _clear_variables)?;
|
||||||
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -288,7 +291,7 @@ impl AbstractNode for Math {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(Action::Return(value))
|
Ok(Evaluation::Return(value))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -14,6 +14,7 @@ pub mod statement;
|
|||||||
pub mod structure_definition;
|
pub mod structure_definition;
|
||||||
pub mod r#type;
|
pub mod r#type;
|
||||||
pub mod type_alias;
|
pub mod type_alias;
|
||||||
|
pub mod type_constructor;
|
||||||
pub mod value_expression;
|
pub mod value_expression;
|
||||||
pub mod value_node;
|
pub mod value_node;
|
||||||
pub mod r#while;
|
pub mod r#while;
|
||||||
@ -41,7 +42,8 @@ pub use self::{
|
|||||||
statement::Statement,
|
statement::Statement,
|
||||||
structure_definition::StructureDefinition,
|
structure_definition::StructureDefinition,
|
||||||
type_alias::TypeAssignment,
|
type_alias::TypeAssignment,
|
||||||
value_expression::ValueExpression,
|
type_constructor::TypeConstructor,
|
||||||
|
value_expression::Expression,
|
||||||
value_node::ValueNode,
|
value_node::ValueNode,
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -84,7 +86,7 @@ impl From<(usize, usize)> for SourcePosition {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Eq, PartialEq, PartialOrd, Ord)]
|
#[derive(Debug, Clone, Eq, PartialEq, PartialOrd, Ord)]
|
||||||
pub enum Action {
|
pub enum Evaluation {
|
||||||
Return(Value),
|
Return(Value),
|
||||||
Break,
|
Break,
|
||||||
None,
|
None,
|
||||||
@ -114,12 +116,12 @@ impl AbstractTree {
|
|||||||
|
|
||||||
for statement in valid_statements {
|
for statement in valid_statements {
|
||||||
let position = statement.position();
|
let position = statement.position();
|
||||||
let run = statement.run(context, manage_memory);
|
let run = statement.evaluate(context, manage_memory);
|
||||||
|
|
||||||
match run {
|
match run {
|
||||||
Ok(action) => match action {
|
Ok(action) => match action {
|
||||||
Action::Return(value) => previous_value = Some(value),
|
Evaluation::Return(value) => previous_value = Some(value),
|
||||||
Action::None => previous_value = None,
|
Evaluation::None => previous_value = None,
|
||||||
_ => {}
|
_ => {}
|
||||||
},
|
},
|
||||||
Err(runtime_error) => {
|
Err(runtime_error) => {
|
||||||
@ -153,7 +155,7 @@ impl AbstractTree {
|
|||||||
} else if errors.is_empty() {
|
} else if errors.is_empty() {
|
||||||
if let Statement::StructureDefinition(_) = statement {
|
if let Statement::StructureDefinition(_) = statement {
|
||||||
let position = statement.position();
|
let position = statement.position();
|
||||||
let run = statement.run(context, true);
|
let run = statement.evaluate(context, true);
|
||||||
|
|
||||||
if let Err(runtime_error) = run {
|
if let Err(runtime_error) = run {
|
||||||
errors.push(Error::Runtime {
|
errors.push(Error::Runtime {
|
||||||
@ -189,7 +191,11 @@ impl Index<usize> for AbstractTree {
|
|||||||
|
|
||||||
pub trait AbstractNode: Sized {
|
pub trait AbstractNode: Sized {
|
||||||
fn validate(&self, context: &mut Context, manage_memory: bool) -> Result<(), ValidationError>;
|
fn validate(&self, context: &mut Context, manage_memory: bool) -> Result<(), ValidationError>;
|
||||||
fn run(self, context: &mut Context, manage_memory: bool) -> Result<Action, RuntimeError>;
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError>;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait ExpectedType {
|
pub trait ExpectedType {
|
||||||
|
@ -6,9 +6,8 @@ use crate::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
AbstractNode, Action, Assignment, AsyncBlock, Block, ExpectedType, IfElse, Loop,
|
AbstractNode, Assignment, AsyncBlock, Block, Evaluation, ExpectedType, Expression, IfElse,
|
||||||
SourcePosition, StructureDefinition, Type, TypeAssignment, ValueExpression, While,
|
Loop, SourcePosition, StructureDefinition, Type, TypeAssignment, While, WithPosition,
|
||||||
WithPosition,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
@ -21,7 +20,7 @@ pub enum Statement {
|
|||||||
Loop(WithPosition<Loop>),
|
Loop(WithPosition<Loop>),
|
||||||
StructureDefinition(WithPosition<StructureDefinition>),
|
StructureDefinition(WithPosition<StructureDefinition>),
|
||||||
TypeAssignment(WithPosition<TypeAssignment>),
|
TypeAssignment(WithPosition<TypeAssignment>),
|
||||||
ValueExpression(ValueExpression),
|
ValueExpression(Expression),
|
||||||
While(WithPosition<While>),
|
While(WithPosition<While>),
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -68,20 +67,26 @@ impl AbstractNode for Statement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, context: &mut Context, manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
let result = match self {
|
let result = match self {
|
||||||
Statement::Assignment(assignment) => assignment.node.run(context, manage_memory),
|
Statement::Assignment(assignment) => assignment.node.evaluate(context, manage_memory),
|
||||||
Statement::AsyncBlock(async_block) => async_block.node.run(context, manage_memory),
|
Statement::AsyncBlock(async_block) => async_block.node.evaluate(context, manage_memory),
|
||||||
Statement::Block(block) => block.node.run(context, manage_memory),
|
Statement::Block(block) => block.node.evaluate(context, manage_memory),
|
||||||
Statement::Break(_) => Ok(Action::Break),
|
Statement::Break(_) => Ok(Evaluation::Break),
|
||||||
Statement::ValueExpression(expression) => expression.run(context, manage_memory),
|
Statement::ValueExpression(expression) => expression.evaluate(context, manage_memory),
|
||||||
Statement::IfElse(if_else) => if_else.node.run(context, manage_memory),
|
Statement::IfElse(if_else) => if_else.node.evaluate(context, manage_memory),
|
||||||
Statement::Loop(r#loop) => r#loop.node.run(context, manage_memory),
|
Statement::Loop(r#loop) => r#loop.node.evaluate(context, manage_memory),
|
||||||
Statement::StructureDefinition(structure_definition) => {
|
Statement::StructureDefinition(structure_definition) => {
|
||||||
structure_definition.node.run(context, manage_memory)
|
structure_definition.node.evaluate(context, manage_memory)
|
||||||
}
|
}
|
||||||
Statement::TypeAssignment(type_alias) => type_alias.node.run(context, manage_memory),
|
Statement::TypeAssignment(type_alias) => {
|
||||||
Statement::While(r#while) => r#while.node.run(context, manage_memory),
|
type_alias.node.evaluate(context, manage_memory)
|
||||||
|
}
|
||||||
|
Statement::While(r#while) => r#while.node.evaluate(context, manage_memory),
|
||||||
};
|
};
|
||||||
|
|
||||||
if manage_memory {
|
if manage_memory {
|
||||||
|
@ -6,16 +6,16 @@ use crate::{
|
|||||||
identifier::Identifier,
|
identifier::Identifier,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, Type, WithPosition};
|
use super::{AbstractNode, Evaluation, Type, TypeConstructor, WithPosition};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct StructureDefinition {
|
pub struct StructureDefinition {
|
||||||
name: Identifier,
|
name: Identifier,
|
||||||
fields: Vec<(Identifier, WithPosition<Type>)>,
|
fields: Vec<(Identifier, WithPosition<TypeConstructor>)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl StructureDefinition {
|
impl StructureDefinition {
|
||||||
pub fn new(name: Identifier, fields: Vec<(Identifier, WithPosition<Type>)>) -> Self {
|
pub fn new(name: Identifier, fields: Vec<(Identifier, WithPosition<TypeConstructor>)>) -> Self {
|
||||||
Self { name, fields }
|
Self { name, fields }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -29,14 +29,26 @@ impl AbstractNode for StructureDefinition {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
|
let mut fields = Vec::with_capacity(self.fields.len());
|
||||||
|
|
||||||
|
for (identifier, constructor) in self.fields {
|
||||||
|
let r#type = constructor.node.construct(&context)?;
|
||||||
|
|
||||||
|
fields.push((identifier, r#type));
|
||||||
|
}
|
||||||
|
|
||||||
let struct_type = Type::Structure {
|
let struct_type = Type::Structure {
|
||||||
name: self.name.clone(),
|
name: self.name.clone(),
|
||||||
fields: self.fields,
|
fields,
|
||||||
};
|
};
|
||||||
|
|
||||||
context.set_type(self.name, struct_type)?;
|
context.set_type(self.name, struct_type)?;
|
||||||
|
|
||||||
Ok(Action::None)
|
Ok(Evaluation::None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -9,29 +9,31 @@ use crate::{
|
|||||||
identifier::Identifier,
|
identifier::Identifier,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, WithPosition};
|
use super::{AbstractNode, Evaluation};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub enum Type {
|
pub enum Type {
|
||||||
Any,
|
Any,
|
||||||
Argument(Identifier),
|
|
||||||
Boolean,
|
Boolean,
|
||||||
Float,
|
Float,
|
||||||
Function {
|
Function {
|
||||||
parameter_types: Vec<WithPosition<Type>>,
|
type_parameters: Option<Vec<Identifier>>,
|
||||||
return_type: Box<WithPosition<Type>>,
|
value_parameters: Vec<(Identifier, Type)>,
|
||||||
|
return_type: Box<Type>,
|
||||||
},
|
},
|
||||||
Integer,
|
Integer,
|
||||||
List,
|
List {
|
||||||
ListOf(Box<WithPosition<Type>>),
|
length: usize,
|
||||||
ListExact(Vec<WithPosition<Type>>),
|
item_type: Box<Type>,
|
||||||
|
},
|
||||||
|
ListOf(Box<Type>),
|
||||||
Map,
|
Map,
|
||||||
None,
|
None,
|
||||||
Range,
|
Range,
|
||||||
String,
|
String,
|
||||||
Structure {
|
Structure {
|
||||||
name: Identifier,
|
name: Identifier,
|
||||||
fields: Vec<(Identifier, WithPosition<Type>)>,
|
fields: Vec<(Identifier, Type)>,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -43,46 +45,15 @@ impl Type {
|
|||||||
| (Type::Boolean, Type::Boolean)
|
| (Type::Boolean, Type::Boolean)
|
||||||
| (Type::Float, Type::Float)
|
| (Type::Float, Type::Float)
|
||||||
| (Type::Integer, Type::Integer)
|
| (Type::Integer, Type::Integer)
|
||||||
| (Type::List, Type::List)
|
|
||||||
| (Type::List, Type::ListOf(_))
|
|
||||||
| (Type::List, Type::ListExact(_))
|
|
||||||
| (Type::ListOf(_), Type::List)
|
|
||||||
| (Type::ListExact(_), Type::List)
|
|
||||||
| (Type::Map, Type::Map)
|
| (Type::Map, Type::Map)
|
||||||
| (Type::None, Type::None)
|
| (Type::None, Type::None)
|
||||||
| (Type::Range, Type::Range)
|
| (Type::Range, Type::Range)
|
||||||
| (Type::String, Type::String) => return Ok(()),
|
| (Type::String, Type::String) => return Ok(()),
|
||||||
(Type::Argument(left), Type::Argument(right)) => {
|
|
||||||
if left == right {
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
(Type::ListOf(left), Type::ListOf(right)) => {
|
(Type::ListOf(left), Type::ListOf(right)) => {
|
||||||
if let Ok(()) = left.node.check(&right.node) {
|
if let Ok(()) = left.check(&right) {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
(Type::ListOf(list_of), Type::ListExact(list_exact)) => {
|
|
||||||
for r#type in list_exact {
|
|
||||||
list_of.node.check(&r#type.node)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
(Type::ListExact(list_exact), Type::ListOf(list_of)) => {
|
|
||||||
for r#type in list_exact {
|
|
||||||
r#type.node.check(&list_of.node)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
(Type::ListExact(left), Type::ListExact(right)) => {
|
|
||||||
for (left, right) in left.iter().zip(right.iter()) {
|
|
||||||
left.node.check(&right.node)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
(
|
(
|
||||||
Type::Structure {
|
Type::Structure {
|
||||||
name: left_name,
|
name: left_name,
|
||||||
@ -97,8 +68,7 @@ impl Type {
|
|||||||
for ((left_field_name, left_type), (right_field_name, right_type)) in
|
for ((left_field_name, left_type), (right_field_name, right_type)) in
|
||||||
left_fields.iter().zip(right_fields.iter())
|
left_fields.iter().zip(right_fields.iter())
|
||||||
{
|
{
|
||||||
if left_field_name != right_field_name || left_type.node != right_type.node
|
if left_field_name != right_field_name || left_type != right_type {
|
||||||
{
|
|
||||||
return Err(TypeConflict {
|
return Err(TypeConflict {
|
||||||
actual: other.clone(),
|
actual: other.clone(),
|
||||||
expected: self.clone(),
|
expected: self.clone(),
|
||||||
@ -111,19 +81,33 @@ impl Type {
|
|||||||
}
|
}
|
||||||
(
|
(
|
||||||
Type::Function {
|
Type::Function {
|
||||||
parameter_types: left_parameters,
|
type_parameters: left_type_parameters,
|
||||||
|
value_parameters: left_value_parameters,
|
||||||
return_type: left_return,
|
return_type: left_return,
|
||||||
},
|
},
|
||||||
Type::Function {
|
Type::Function {
|
||||||
parameter_types: right_parameters,
|
type_parameters: right_type_parameters,
|
||||||
|
value_parameters: right_value_parameters,
|
||||||
return_type: right_return,
|
return_type: right_return,
|
||||||
},
|
},
|
||||||
) => {
|
) => {
|
||||||
if left_return.node == right_return.node {
|
if left_return == right_return {
|
||||||
for (left_parameter, right_parameter) in
|
for (left_parameter, right_parameter) in left_type_parameters
|
||||||
left_parameters.iter().zip(right_parameters.iter())
|
.iter()
|
||||||
|
.zip(right_type_parameters.iter())
|
||||||
{
|
{
|
||||||
if left_parameter.node != right_parameter.node {
|
if left_parameter != right_parameter {
|
||||||
|
return Err(TypeConflict {
|
||||||
|
actual: other.clone(),
|
||||||
|
expected: self.clone(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (left_parameter, right_parameter) in left_value_parameters
|
||||||
|
.iter()
|
||||||
|
.zip(right_value_parameters.iter())
|
||||||
|
{
|
||||||
|
if left_parameter != right_parameter {
|
||||||
return Err(TypeConflict {
|
return Err(TypeConflict {
|
||||||
actual: other.clone(),
|
actual: other.clone(),
|
||||||
expected: self.clone(),
|
expected: self.clone(),
|
||||||
@ -153,8 +137,12 @@ impl AbstractNode for Type {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, _context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
Ok(Action::None)
|
self,
|
||||||
|
_context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
|
Ok(Evaluation::None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -165,47 +153,40 @@ impl Display for Type {
|
|||||||
Type::Boolean => write!(f, "bool"),
|
Type::Boolean => write!(f, "bool"),
|
||||||
Type::Float => write!(f, "float"),
|
Type::Float => write!(f, "float"),
|
||||||
Type::Integer => write!(f, "int"),
|
Type::Integer => write!(f, "int"),
|
||||||
Type::List => write!(f, "list"),
|
Type::List { length, item_type } => write!(f, "[{length}; {}]", item_type),
|
||||||
Type::ListOf(item_type) => write!(f, "list({})", item_type.node),
|
Type::ListOf(item_type) => write!(f, "list({})", item_type),
|
||||||
Type::ListExact(item_types) => {
|
|
||||||
write!(f, "[")?;
|
|
||||||
|
|
||||||
for (index, item_type) in item_types.into_iter().enumerate() {
|
|
||||||
if index == item_types.len() - 1 {
|
|
||||||
write!(f, "{}", item_type.node)?;
|
|
||||||
} else {
|
|
||||||
write!(f, "{}, ", item_type.node)?;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
write!(f, "]")
|
|
||||||
}
|
|
||||||
Type::Map => write!(f, "map"),
|
Type::Map => write!(f, "map"),
|
||||||
Type::None => write!(f, "none"),
|
Type::None => write!(f, "none"),
|
||||||
Type::Range => write!(f, "range"),
|
Type::Range => write!(f, "range"),
|
||||||
Type::String => write!(f, "str"),
|
Type::String => write!(f, "str"),
|
||||||
Type::Function {
|
Type::Function {
|
||||||
parameter_types,
|
type_parameters,
|
||||||
|
value_parameters,
|
||||||
return_type,
|
return_type,
|
||||||
} => {
|
} => {
|
||||||
write!(f, "(")?;
|
write!(f, "(")?;
|
||||||
|
|
||||||
for r#type in parameter_types {
|
if let Some(type_parameters) = type_parameters {
|
||||||
write!(f, "{} ", r#type.node)?;
|
for identifier in type_parameters {
|
||||||
|
write!(f, "{} ", identifier)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
write!(f, ") : {}", return_type.node)
|
write!(f, ")(")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (identifier, r#type) in value_parameters {
|
||||||
|
write!(f, "{identifier}: {type}")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
write!(f, ") : {}", return_type)
|
||||||
}
|
}
|
||||||
Type::Structure { name, .. } => write!(f, "{name}"),
|
Type::Structure { name, .. } => write!(f, "{name}"),
|
||||||
Type::Argument(identifier) => write!(f, "{identifier}"),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use crate::abstract_tree::WithPos;
|
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@ -214,18 +195,22 @@ mod tests {
|
|||||||
assert_eq!(Type::Boolean.check(&Type::Boolean), Ok(()));
|
assert_eq!(Type::Boolean.check(&Type::Boolean), Ok(()));
|
||||||
assert_eq!(Type::Float.check(&Type::Float), Ok(()));
|
assert_eq!(Type::Float.check(&Type::Float), Ok(()));
|
||||||
assert_eq!(Type::Integer.check(&Type::Integer), Ok(()));
|
assert_eq!(Type::Integer.check(&Type::Integer), Ok(()));
|
||||||
assert_eq!(Type::List.check(&Type::List), Ok(()));
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Type::ListOf(Box::new(Type::Integer.with_position((0, 0))))
|
Type::List {
|
||||||
.check(&Type::ListOf(Box::new(Type::Integer.with_position((0, 0))))),
|
length: 4,
|
||||||
|
item_type: Box::new(Type::Boolean),
|
||||||
|
}
|
||||||
|
.check(&Type::List {
|
||||||
|
length: 4,
|
||||||
|
item_type: Box::new(Type::Boolean),
|
||||||
|
}),
|
||||||
|
Ok(())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
Type::ListOf(Box::new(Type::Integer)).check(&Type::ListOf(Box::new(Type::Integer))),
|
||||||
Ok(())
|
Ok(())
|
||||||
);
|
);
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
Type::ListExact(vec![Type::Float.with_position((0, 0))])
|
|
||||||
.check(&Type::ListExact(vec![Type::Float.with_position((0, 0))])),
|
|
||||||
Ok(())
|
|
||||||
);
|
|
||||||
assert_eq!(Type::Map.check(&Type::Map), Ok(()));
|
assert_eq!(Type::Map.check(&Type::Map), Ok(()));
|
||||||
assert_eq!(Type::None.check(&Type::None), Ok(()));
|
assert_eq!(Type::None.check(&Type::None), Ok(()));
|
||||||
assert_eq!(Type::Range.check(&Type::Range), Ok(()));
|
assert_eq!(Type::Range.check(&Type::Range), Ok(()));
|
||||||
@ -257,9 +242,11 @@ mod tests {
|
|||||||
Type::Boolean,
|
Type::Boolean,
|
||||||
Type::Float,
|
Type::Float,
|
||||||
Type::Integer,
|
Type::Integer,
|
||||||
Type::List,
|
Type::List {
|
||||||
Type::ListOf(Box::new(Type::Boolean.with_position((0, 0)))),
|
length: 10,
|
||||||
Type::ListExact(vec![Type::Integer.with_position((0, 0))]),
|
item_type: Box::new(Type::Integer),
|
||||||
|
},
|
||||||
|
Type::ListOf(Box::new(Type::Boolean)),
|
||||||
Type::Map,
|
Type::Map,
|
||||||
Type::None,
|
Type::None,
|
||||||
Type::Range,
|
Type::Range,
|
||||||
@ -283,18 +270,13 @@ mod tests {
|
|||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn check_list_types() {
|
fn check_list_types() {
|
||||||
let list = Type::List;
|
let list = Type::List {
|
||||||
let list_exact = Type::ListExact(vec![
|
length: 42,
|
||||||
Type::Integer.with_position((0, 0)),
|
item_type: Box::new(Type::Integer),
|
||||||
Type::Integer.with_position((0, 0)),
|
};
|
||||||
]);
|
let list_of = Type::ListOf(Box::new(Type::Integer));
|
||||||
let list_of = Type::ListOf(Box::new(Type::Integer.with_position((0, 0))));
|
|
||||||
|
|
||||||
assert_eq!(list.check(&list_exact), Ok(()));
|
|
||||||
assert_eq!(list.check(&list_of), Ok(()));
|
assert_eq!(list.check(&list_of), Ok(()));
|
||||||
assert_eq!(list_exact.check(&list), Ok(()));
|
|
||||||
assert_eq!(list_exact.check(&list_of), Ok(()));
|
|
||||||
assert_eq!(list_of.check(&list), Ok(()));
|
assert_eq!(list_of.check(&list), Ok(()));
|
||||||
assert_eq!(list_of.check(&list_exact), Ok(()));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,17 +6,23 @@ use crate::{
|
|||||||
identifier::Identifier,
|
identifier::Identifier,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, Type, WithPosition};
|
use super::{AbstractNode, Evaluation, TypeConstructor, WithPosition};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct TypeAssignment {
|
pub struct TypeAssignment {
|
||||||
identifier: WithPosition<Identifier>,
|
identifier: WithPosition<Identifier>,
|
||||||
r#type: WithPosition<Type>,
|
constructor: WithPosition<TypeConstructor>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TypeAssignment {
|
impl TypeAssignment {
|
||||||
pub fn new(identifier: WithPosition<Identifier>, r#type: WithPosition<Type>) -> Self {
|
pub fn new(
|
||||||
Self { identifier, r#type }
|
identifier: WithPosition<Identifier>,
|
||||||
|
constructor: WithPosition<TypeConstructor>,
|
||||||
|
) -> Self {
|
||||||
|
Self {
|
||||||
|
identifier,
|
||||||
|
constructor,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -29,9 +35,15 @@ impl AbstractNode for TypeAssignment {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
context.set_type(self.identifier.node, self.r#type.node)?;
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
|
let r#type = self.constructor.node.construct(&context)?;
|
||||||
|
|
||||||
Ok(Action::None)
|
context.set_type(self.identifier.node, r#type)?;
|
||||||
|
|
||||||
|
Ok(Evaluation::None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
78
dust-lang/src/abstract_tree/type_constructor.rs
Normal file
78
dust-lang/src/abstract_tree/type_constructor.rs
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
use std::fmt::{self, Display, Formatter};
|
||||||
|
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
use crate::{context::Context, error::ValidationError, identifier::Identifier};
|
||||||
|
|
||||||
|
use super::{ExpectedType, Type, WithPosition};
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
|
pub enum TypeConstructor {
|
||||||
|
Function {
|
||||||
|
type_parameters: Option<Vec<WithPosition<Identifier>>>,
|
||||||
|
value_parameters: Vec<(WithPosition<Identifier>, Box<WithPosition<TypeConstructor>>)>,
|
||||||
|
return_type: Box<WithPosition<TypeConstructor>>,
|
||||||
|
},
|
||||||
|
Identifier(WithPosition<Identifier>),
|
||||||
|
List {
|
||||||
|
length: usize,
|
||||||
|
item_type: Box<WithPosition<TypeConstructor>>,
|
||||||
|
},
|
||||||
|
ListOf(WithPosition<Box<TypeConstructor>>),
|
||||||
|
Type(Type),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TypeConstructor {
|
||||||
|
pub fn validate(
|
||||||
|
&self,
|
||||||
|
_context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<(), ValidationError> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn construct(self, context: &Context) -> Result<Type, ValidationError> {
|
||||||
|
match self {
|
||||||
|
TypeConstructor::Function {
|
||||||
|
type_parameters: _,
|
||||||
|
value_parameters: _,
|
||||||
|
return_type: _,
|
||||||
|
} => todo!(),
|
||||||
|
TypeConstructor::Identifier(WithPosition {
|
||||||
|
node: identifier,
|
||||||
|
position,
|
||||||
|
}) => {
|
||||||
|
if let Some(r#type) = context.get_type(&identifier)? {
|
||||||
|
Ok(r#type)
|
||||||
|
} else {
|
||||||
|
Err(ValidationError::VariableNotFound {
|
||||||
|
identifier,
|
||||||
|
position,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
TypeConstructor::List { length, item_type } => {
|
||||||
|
let constructed_type = item_type.node.construct(context)?;
|
||||||
|
|
||||||
|
Ok(Type::List {
|
||||||
|
length,
|
||||||
|
item_type: Box::new(constructed_type),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
TypeConstructor::Type(r#type) => Ok(r#type),
|
||||||
|
TypeConstructor::ListOf(_) => todo!(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ExpectedType for TypeConstructor {
|
||||||
|
fn expected_type(&self, _: &mut Context) -> Result<Type, ValidationError> {
|
||||||
|
Ok(Type::None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for TypeConstructor {
|
||||||
|
fn fmt(&self, _: &mut Formatter) -> fmt::Result {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
}
|
@ -7,12 +7,12 @@ use crate::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
AbstractNode, Action, As, BuiltInFunctionCall, ExpectedType, FunctionCall, ListIndex, Logic,
|
AbstractNode, As, BuiltInFunctionCall, Evaluation, ExpectedType, FunctionCall, ListIndex,
|
||||||
MapIndex, Math, SourcePosition, Type, ValueNode, WithPosition,
|
Logic, MapIndex, Math, SourcePosition, Type, ValueNode, WithPosition,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub enum ValueExpression {
|
pub enum Expression {
|
||||||
As(WithPosition<Box<As>>),
|
As(WithPosition<Box<As>>),
|
||||||
BuiltInFunctionCall(WithPosition<Box<BuiltInFunctionCall>>),
|
BuiltInFunctionCall(WithPosition<Box<BuiltInFunctionCall>>),
|
||||||
FunctionCall(WithPosition<FunctionCall>),
|
FunctionCall(WithPosition<FunctionCall>),
|
||||||
@ -24,30 +24,30 @@ pub enum ValueExpression {
|
|||||||
Value(WithPosition<ValueNode>),
|
Value(WithPosition<ValueNode>),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ValueExpression {
|
impl Expression {
|
||||||
pub fn position(&self) -> SourcePosition {
|
pub fn position(&self) -> SourcePosition {
|
||||||
match self {
|
match self {
|
||||||
ValueExpression::As(inner) => inner.position,
|
Expression::As(inner) => inner.position,
|
||||||
ValueExpression::FunctionCall(inner) => inner.position,
|
Expression::FunctionCall(inner) => inner.position,
|
||||||
ValueExpression::Identifier(inner) => inner.position,
|
Expression::Identifier(inner) => inner.position,
|
||||||
ValueExpression::MapIndex(inner) => inner.position,
|
Expression::MapIndex(inner) => inner.position,
|
||||||
ValueExpression::ListIndex(inner) => inner.position,
|
Expression::ListIndex(inner) => inner.position,
|
||||||
ValueExpression::Logic(inner) => inner.position,
|
Expression::Logic(inner) => inner.position,
|
||||||
ValueExpression::Math(inner) => inner.position,
|
Expression::Math(inner) => inner.position,
|
||||||
ValueExpression::Value(inner) => inner.position,
|
Expression::Value(inner) => inner.position,
|
||||||
ValueExpression::BuiltInFunctionCall(inner) => inner.position,
|
Expression::BuiltInFunctionCall(inner) => inner.position,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AbstractNode for ValueExpression {
|
impl AbstractNode for Expression {
|
||||||
fn validate(&self, context: &mut Context, manage_memory: bool) -> Result<(), ValidationError> {
|
fn validate(&self, context: &mut Context, manage_memory: bool) -> Result<(), ValidationError> {
|
||||||
match self {
|
match self {
|
||||||
ValueExpression::As(r#as) => r#as.node.validate(context, manage_memory),
|
Expression::As(r#as) => r#as.node.validate(context, manage_memory),
|
||||||
ValueExpression::FunctionCall(function_call) => {
|
Expression::FunctionCall(function_call) => {
|
||||||
function_call.node.validate(context, manage_memory)
|
function_call.node.validate(context, manage_memory)
|
||||||
}
|
}
|
||||||
ValueExpression::Identifier(identifier) => {
|
Expression::Identifier(identifier) => {
|
||||||
let found = if manage_memory {
|
let found = if manage_memory {
|
||||||
context.add_expected_use(&identifier.node)?
|
context.add_expected_use(&identifier.node)?
|
||||||
} else {
|
} else {
|
||||||
@ -63,26 +63,28 @@ impl AbstractNode for ValueExpression {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ValueExpression::MapIndex(map_index) => map_index.node.validate(context, manage_memory),
|
Expression::MapIndex(map_index) => map_index.node.validate(context, manage_memory),
|
||||||
ValueExpression::ListIndex(list_index) => {
|
Expression::ListIndex(list_index) => list_index.node.validate(context, manage_memory),
|
||||||
list_index.node.validate(context, manage_memory)
|
Expression::Logic(logic) => logic.node.validate(context, manage_memory),
|
||||||
}
|
Expression::Math(math) => math.node.validate(context, manage_memory),
|
||||||
ValueExpression::Logic(logic) => logic.node.validate(context, manage_memory),
|
Expression::Value(value_node) => value_node.node.validate(context, manage_memory),
|
||||||
ValueExpression::Math(math) => math.node.validate(context, manage_memory),
|
Expression::BuiltInFunctionCall(built_in_function_call) => {
|
||||||
ValueExpression::Value(value_node) => value_node.node.validate(context, manage_memory),
|
|
||||||
ValueExpression::BuiltInFunctionCall(built_in_function_call) => {
|
|
||||||
built_in_function_call.node.validate(context, manage_memory)
|
built_in_function_call.node.validate(context, manage_memory)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, context: &mut Context, manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
match self {
|
match self {
|
||||||
ValueExpression::As(r#as) => r#as.node.run(context, manage_memory),
|
Expression::As(r#as) => r#as.node.evaluate(context, manage_memory),
|
||||||
ValueExpression::FunctionCall(function_call) => {
|
Expression::FunctionCall(function_call) => {
|
||||||
function_call.node.run(context, manage_memory)
|
function_call.node.evaluate(context, manage_memory)
|
||||||
}
|
}
|
||||||
ValueExpression::Identifier(identifier) => {
|
Expression::Identifier(identifier) => {
|
||||||
let value_option = if manage_memory {
|
let value_option = if manage_memory {
|
||||||
context.use_value(&identifier.node)?
|
context.use_value(&identifier.node)?
|
||||||
} else {
|
} else {
|
||||||
@ -90,7 +92,7 @@ impl AbstractNode for ValueExpression {
|
|||||||
};
|
};
|
||||||
|
|
||||||
if let Some(value) = value_option {
|
if let Some(value) = value_option {
|
||||||
Ok(Action::Return(value))
|
Ok(Evaluation::Return(value))
|
||||||
} else {
|
} else {
|
||||||
Err(RuntimeError::ValidationFailure(
|
Err(RuntimeError::ValidationFailure(
|
||||||
ValidationError::VariableNotFound {
|
ValidationError::VariableNotFound {
|
||||||
@ -100,26 +102,24 @@ impl AbstractNode for ValueExpression {
|
|||||||
))
|
))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ValueExpression::MapIndex(map_index) => map_index.node.run(context, manage_memory),
|
Expression::MapIndex(map_index) => map_index.node.evaluate(context, manage_memory),
|
||||||
ValueExpression::ListIndex(list_index) => list_index.node.run(context, manage_memory),
|
Expression::ListIndex(list_index) => list_index.node.evaluate(context, manage_memory),
|
||||||
ValueExpression::Logic(logic) => logic.node.run(context, manage_memory),
|
Expression::Logic(logic) => logic.node.evaluate(context, manage_memory),
|
||||||
ValueExpression::Math(math) => math.node.run(context, manage_memory),
|
Expression::Math(math) => math.node.evaluate(context, manage_memory),
|
||||||
ValueExpression::Value(value_node) => value_node.node.run(context, manage_memory),
|
Expression::Value(value_node) => value_node.node.evaluate(context, manage_memory),
|
||||||
ValueExpression::BuiltInFunctionCall(built_in_function_call) => {
|
Expression::BuiltInFunctionCall(built_in_function_call) => {
|
||||||
built_in_function_call.node.run(context, manage_memory)
|
built_in_function_call.node.evaluate(context, manage_memory)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ExpectedType for ValueExpression {
|
impl ExpectedType for Expression {
|
||||||
fn expected_type(&self, _context: &mut Context) -> Result<Type, ValidationError> {
|
fn expected_type(&self, _context: &mut Context) -> Result<Type, ValidationError> {
|
||||||
match self {
|
match self {
|
||||||
ValueExpression::As(r#as) => r#as.node.expected_type(_context),
|
Expression::As(r#as) => r#as.node.expected_type(_context),
|
||||||
ValueExpression::FunctionCall(function_call) => {
|
Expression::FunctionCall(function_call) => function_call.node.expected_type(_context),
|
||||||
function_call.node.expected_type(_context)
|
Expression::Identifier(identifier) => {
|
||||||
}
|
|
||||||
ValueExpression::Identifier(identifier) => {
|
|
||||||
if let Some(r#type) = _context.get_type(&identifier.node)? {
|
if let Some(r#type) = _context.get_type(&identifier.node)? {
|
||||||
Ok(r#type)
|
Ok(r#type)
|
||||||
} else {
|
} else {
|
||||||
@ -129,12 +129,12 @@ impl ExpectedType for ValueExpression {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ValueExpression::MapIndex(map_index) => map_index.node.expected_type(_context),
|
Expression::MapIndex(map_index) => map_index.node.expected_type(_context),
|
||||||
ValueExpression::ListIndex(list_index) => list_index.node.expected_type(_context),
|
Expression::ListIndex(list_index) => list_index.node.expected_type(_context),
|
||||||
ValueExpression::Logic(logic) => logic.node.expected_type(_context),
|
Expression::Logic(logic) => logic.node.expected_type(_context),
|
||||||
ValueExpression::Math(math) => math.node.expected_type(_context),
|
Expression::Math(math) => math.node.expected_type(_context),
|
||||||
ValueExpression::Value(value_node) => value_node.node.expected_type(_context),
|
Expression::Value(value_node) => value_node.node.expected_type(_context),
|
||||||
ValueExpression::BuiltInFunctionCall(built_in_function_call) => {
|
Expression::BuiltInFunctionCall(built_in_function_call) => {
|
||||||
built_in_function_call.node.expected_type(_context)
|
built_in_function_call.node.expected_type(_context)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -10,7 +10,7 @@ use crate::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
AbstractNode, Action, Block, ExpectedType, Type, ValueExpression, WithPos, WithPosition,
|
AbstractNode, Block, Evaluation, ExpectedType, Expression, Type, TypeConstructor, WithPosition,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
|
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
|
||||||
@ -18,18 +18,24 @@ pub enum ValueNode {
|
|||||||
Boolean(bool),
|
Boolean(bool),
|
||||||
Float(f64),
|
Float(f64),
|
||||||
Integer(i64),
|
Integer(i64),
|
||||||
List(Vec<ValueExpression>),
|
List(Vec<Expression>),
|
||||||
Map(Vec<(Identifier, Option<WithPosition<Type>>, ValueExpression)>),
|
Map(
|
||||||
|
Vec<(
|
||||||
|
Identifier,
|
||||||
|
Option<WithPosition<TypeConstructor>>,
|
||||||
|
Expression,
|
||||||
|
)>,
|
||||||
|
),
|
||||||
Range(Range<i64>),
|
Range(Range<i64>),
|
||||||
String(String),
|
String(String),
|
||||||
Structure {
|
Structure {
|
||||||
name: WithPosition<Identifier>,
|
name: WithPosition<Identifier>,
|
||||||
fields: Vec<(Identifier, ValueExpression)>,
|
fields: Vec<(WithPosition<Identifier>, Expression)>,
|
||||||
},
|
},
|
||||||
ParsedFunction {
|
ParsedFunction {
|
||||||
type_arguments: Vec<WithPosition<Type>>,
|
type_parameters: Option<Vec<WithPosition<Identifier>>>,
|
||||||
parameters: Vec<(Identifier, WithPosition<Type>)>,
|
value_parameters: Vec<(Identifier, WithPosition<TypeConstructor>)>,
|
||||||
return_type: WithPosition<Type>,
|
return_type: WithPosition<TypeConstructor>,
|
||||||
body: WithPosition<Block>,
|
body: WithPosition<Block>,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@ -37,17 +43,18 @@ pub enum ValueNode {
|
|||||||
impl AbstractNode for ValueNode {
|
impl AbstractNode for ValueNode {
|
||||||
fn validate(&self, context: &mut Context, _manage_memory: bool) -> Result<(), ValidationError> {
|
fn validate(&self, context: &mut Context, _manage_memory: bool) -> Result<(), ValidationError> {
|
||||||
if let ValueNode::Map(map_assignments) = self {
|
if let ValueNode::Map(map_assignments) = self {
|
||||||
for (_identifier, r#type, expression) in map_assignments {
|
for (_identifier, constructor_option, expression) in map_assignments {
|
||||||
expression.validate(context, _manage_memory)?;
|
expression.validate(context, _manage_memory)?;
|
||||||
|
|
||||||
if let Some(expected_type) = r#type {
|
if let Some(constructor) = constructor_option {
|
||||||
let actual_type = expression.expected_type(context)?;
|
let actual_type = expression.expected_type(context)?;
|
||||||
|
let exprected_type = constructor.node.clone().construct(&context)?;
|
||||||
|
|
||||||
expected_type.node.check(&actual_type).map_err(|conflict| {
|
exprected_type.check(&actual_type).map_err(|conflict| {
|
||||||
ValidationError::TypeCheck {
|
ValidationError::TypeCheck {
|
||||||
conflict,
|
conflict,
|
||||||
actual_position: expression.position(),
|
actual_position: expression.position(),
|
||||||
expected_position: expected_type.position,
|
expected_position: Some(constructor.position),
|
||||||
}
|
}
|
||||||
})?;
|
})?;
|
||||||
}
|
}
|
||||||
@ -57,22 +64,18 @@ impl AbstractNode for ValueNode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if let ValueNode::ParsedFunction {
|
if let ValueNode::ParsedFunction {
|
||||||
type_arguments,
|
type_parameters: _,
|
||||||
parameters,
|
value_parameters,
|
||||||
return_type,
|
return_type,
|
||||||
body,
|
body,
|
||||||
} = self
|
} = self
|
||||||
{
|
{
|
||||||
let mut function_context = Context::new(Some(&context));
|
let mut function_context = Context::new(Some(&context));
|
||||||
|
|
||||||
for r#type in type_arguments {
|
for (identifier, type_constructor) in value_parameters {
|
||||||
if let Type::Argument(identifier) = &r#type.node {
|
let r#type = type_constructor.node.clone().construct(&function_context)?;
|
||||||
function_context.set_type(identifier.clone(), r#type.node.clone())?;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (identifier, r#type) in parameters {
|
function_context.set_type(identifier.clone(), r#type)?;
|
||||||
function_context.set_type(identifier.clone(), r#type.node.clone())?;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
body.node.validate(&mut function_context, _manage_memory)?;
|
body.node.validate(&mut function_context, _manage_memory)?;
|
||||||
@ -81,11 +84,13 @@ impl AbstractNode for ValueNode {
|
|||||||
|
|
||||||
return_type
|
return_type
|
||||||
.node
|
.node
|
||||||
|
.clone()
|
||||||
|
.construct(&function_context)?
|
||||||
.check(&actual_return_type)
|
.check(&actual_return_type)
|
||||||
.map_err(|conflict| ValidationError::TypeCheck {
|
.map_err(|conflict| ValidationError::TypeCheck {
|
||||||
conflict,
|
conflict,
|
||||||
actual_position: body.position,
|
actual_position: body.position,
|
||||||
expected_position: return_type.position,
|
expected_position: Some(return_type.position),
|
||||||
})?;
|
})?;
|
||||||
|
|
||||||
return Ok(());
|
return Ok(());
|
||||||
@ -111,11 +116,11 @@ impl AbstractNode for ValueNode {
|
|||||||
for ((_, expression), (_, expected_type)) in expressions.iter().zip(types.iter()) {
|
for ((_, expression), (_, expected_type)) in expressions.iter().zip(types.iter()) {
|
||||||
let actual_type = expression.expected_type(context)?;
|
let actual_type = expression.expected_type(context)?;
|
||||||
|
|
||||||
expected_type.node.check(&actual_type).map_err(|conflict| {
|
expected_type.check(&actual_type).map_err(|conflict| {
|
||||||
ValidationError::TypeCheck {
|
ValidationError::TypeCheck {
|
||||||
conflict,
|
conflict,
|
||||||
actual_position: expression.position(),
|
actual_position: expression.position(),
|
||||||
expected_position: expected_type.position,
|
expected_position: None,
|
||||||
}
|
}
|
||||||
})?
|
})?
|
||||||
}
|
}
|
||||||
@ -125,7 +130,11 @@ impl AbstractNode for ValueNode {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, _context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
let value = match self {
|
let value = match self {
|
||||||
ValueNode::Boolean(boolean) => Value::boolean(boolean),
|
ValueNode::Boolean(boolean) => Value::boolean(boolean),
|
||||||
ValueNode::Float(float) => Value::float(float),
|
ValueNode::Float(float) => Value::float(float),
|
||||||
@ -135,8 +144,8 @@ impl AbstractNode for ValueNode {
|
|||||||
|
|
||||||
for expression in expression_list {
|
for expression in expression_list {
|
||||||
let expression_position = expression.position();
|
let expression_position = expression.position();
|
||||||
let action = expression.run(_context, _manage_memory)?;
|
let action = expression.evaluate(context, _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
WithPosition {
|
WithPosition {
|
||||||
node: value,
|
node: value,
|
||||||
position: expression_position,
|
position: expression_position,
|
||||||
@ -157,8 +166,8 @@ impl AbstractNode for ValueNode {
|
|||||||
|
|
||||||
for (identifier, _type, expression) in property_list {
|
for (identifier, _type, expression) in property_list {
|
||||||
let expression_position = expression.position();
|
let expression_position = expression.position();
|
||||||
let action = expression.run(_context, _manage_memory)?;
|
let action = expression.evaluate(context, _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -174,11 +183,29 @@ impl AbstractNode for ValueNode {
|
|||||||
ValueNode::Range(range) => Value::range(range),
|
ValueNode::Range(range) => Value::range(range),
|
||||||
ValueNode::String(string) => Value::string(string),
|
ValueNode::String(string) => Value::string(string),
|
||||||
ValueNode::ParsedFunction {
|
ValueNode::ParsedFunction {
|
||||||
type_arguments,
|
type_parameters,
|
||||||
parameters,
|
value_parameters: constructors,
|
||||||
return_type,
|
return_type,
|
||||||
body,
|
body,
|
||||||
} => Value::function(type_arguments, parameters, return_type, body),
|
} => {
|
||||||
|
let type_parameters = type_parameters.map(|parameter_list| {
|
||||||
|
parameter_list
|
||||||
|
.into_iter()
|
||||||
|
.map(|parameter| parameter.node)
|
||||||
|
.collect()
|
||||||
|
});
|
||||||
|
let mut value_parameters = Vec::with_capacity(constructors.len());
|
||||||
|
|
||||||
|
for (identifier, constructor) in constructors {
|
||||||
|
let r#type = constructor.node.construct(&context)?;
|
||||||
|
|
||||||
|
value_parameters.push((identifier, r#type));
|
||||||
|
}
|
||||||
|
|
||||||
|
let return_type = return_type.node.construct(&context)?;
|
||||||
|
|
||||||
|
Value::function(type_parameters, value_parameters, return_type, body.node)
|
||||||
|
}
|
||||||
ValueNode::Structure {
|
ValueNode::Structure {
|
||||||
name,
|
name,
|
||||||
fields: expressions,
|
fields: expressions,
|
||||||
@ -187,8 +214,8 @@ impl AbstractNode for ValueNode {
|
|||||||
|
|
||||||
for (identifier, expression) in expressions {
|
for (identifier, expression) in expressions {
|
||||||
let expression_position = expression.position();
|
let expression_position = expression.position();
|
||||||
let action = expression.run(_context, _manage_memory)?;
|
let action = expression.evaluate(context, _manage_memory)?;
|
||||||
let value = if let Action::Return(value) = action {
|
let value = if let Evaluation::Return(value) = action {
|
||||||
value
|
value
|
||||||
} else {
|
} else {
|
||||||
return Err(RuntimeError::ValidationFailure(
|
return Err(RuntimeError::ValidationFailure(
|
||||||
@ -196,14 +223,14 @@ impl AbstractNode for ValueNode {
|
|||||||
));
|
));
|
||||||
};
|
};
|
||||||
|
|
||||||
fields.push((identifier, value));
|
fields.push((identifier.node, value));
|
||||||
}
|
}
|
||||||
|
|
||||||
Value::structure(name, fields)
|
Value::structure(name, fields)
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(Action::Return(value))
|
Ok(Evaluation::Return(value))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -244,14 +271,14 @@ impl Ord for ValueNode {
|
|||||||
(String(_), _) => Ordering::Greater,
|
(String(_), _) => Ordering::Greater,
|
||||||
(
|
(
|
||||||
ParsedFunction {
|
ParsedFunction {
|
||||||
type_arguments: left_type_arguments,
|
type_parameters: left_type_arguments,
|
||||||
parameters: left_parameters,
|
value_parameters: left_parameters,
|
||||||
return_type: left_return,
|
return_type: left_return,
|
||||||
body: left_body,
|
body: left_body,
|
||||||
},
|
},
|
||||||
ParsedFunction {
|
ParsedFunction {
|
||||||
type_arguments: right_type_arguments,
|
type_parameters: right_type_arguments,
|
||||||
parameters: right_parameters,
|
value_parameters: right_parameters,
|
||||||
return_type: right_return,
|
return_type: right_return,
|
||||||
body: right_body,
|
body: right_body,
|
||||||
},
|
},
|
||||||
@ -307,32 +334,44 @@ impl ExpectedType for ValueNode {
|
|||||||
ValueNode::Float(_) => Type::Float,
|
ValueNode::Float(_) => Type::Float,
|
||||||
ValueNode::Integer(_) => Type::Integer,
|
ValueNode::Integer(_) => Type::Integer,
|
||||||
ValueNode::List(items) => {
|
ValueNode::List(items) => {
|
||||||
let mut item_types = Vec::with_capacity(items.len());
|
let item_type = items.first().unwrap().expected_type(context)?;
|
||||||
|
|
||||||
for expression in items {
|
Type::List {
|
||||||
item_types.push(
|
length: items.len(),
|
||||||
expression
|
item_type: Box::new(item_type),
|
||||||
.expected_type(context)?
|
|
||||||
.with_position(expression.position()),
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Type::ListExact(item_types)
|
|
||||||
}
|
}
|
||||||
ValueNode::Map(_) => Type::Map,
|
ValueNode::Map(_) => Type::Map,
|
||||||
ValueNode::Range(_) => Type::Range,
|
ValueNode::Range(_) => Type::Range,
|
||||||
ValueNode::String(_) => Type::String,
|
ValueNode::String(_) => Type::String,
|
||||||
ValueNode::ParsedFunction {
|
ValueNode::ParsedFunction {
|
||||||
parameters,
|
type_parameters,
|
||||||
|
value_parameters,
|
||||||
return_type,
|
return_type,
|
||||||
..
|
..
|
||||||
} => Type::Function {
|
} => {
|
||||||
parameter_types: parameters
|
let mut value_parameter_types = Vec::with_capacity(value_parameters.len());
|
||||||
|
|
||||||
|
for (identifier, type_constructor) in value_parameters {
|
||||||
|
let r#type = type_constructor.node.clone().construct(&context)?;
|
||||||
|
|
||||||
|
value_parameter_types.push((identifier.clone(), r#type));
|
||||||
|
}
|
||||||
|
|
||||||
|
let type_parameters = type_parameters.clone().map(|parameters| {
|
||||||
|
parameters
|
||||||
.iter()
|
.iter()
|
||||||
.map(|(_, r#type)| r#type.clone())
|
.map(|identifier| identifier.node.clone())
|
||||||
.collect(),
|
.collect()
|
||||||
return_type: Box::new(return_type.clone()),
|
});
|
||||||
},
|
let return_type = return_type.node.clone().construct(&context)?;
|
||||||
|
|
||||||
|
Type::Function {
|
||||||
|
type_parameters,
|
||||||
|
value_parameters: value_parameter_types,
|
||||||
|
return_type: Box::new(return_type),
|
||||||
|
}
|
||||||
|
}
|
||||||
ValueNode::Structure {
|
ValueNode::Structure {
|
||||||
name,
|
name,
|
||||||
fields: expressions,
|
fields: expressions,
|
||||||
@ -353,7 +392,10 @@ impl ExpectedType for ValueNode {
|
|||||||
|
|
||||||
Type::Structure {
|
Type::Structure {
|
||||||
name: name.node.clone(),
|
name: name.node.clone(),
|
||||||
fields: types,
|
fields: types
|
||||||
|
.into_iter()
|
||||||
|
.map(|(identifier, r#type)| (identifier.node, r#type.node))
|
||||||
|
.collect(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -7,16 +7,16 @@ use crate::{
|
|||||||
Value,
|
Value,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{AbstractNode, Action, Statement, ValueExpression};
|
use super::{AbstractNode, Evaluation, Expression, Statement};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct While {
|
pub struct While {
|
||||||
expression: ValueExpression,
|
expression: Expression,
|
||||||
statements: Vec<Statement>,
|
statements: Vec<Statement>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl While {
|
impl While {
|
||||||
pub fn new(expression: ValueExpression, statements: Vec<Statement>) -> Self {
|
pub fn new(expression: Expression, statements: Vec<Statement>) -> Self {
|
||||||
Self {
|
Self {
|
||||||
expression,
|
expression,
|
||||||
statements,
|
statements,
|
||||||
@ -39,12 +39,19 @@ impl AbstractNode for While {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(self, _context: &mut Context, _manage_memory: bool) -> Result<Action, RuntimeError> {
|
fn evaluate(
|
||||||
|
self,
|
||||||
|
_context: &mut Context,
|
||||||
|
_manage_memory: bool,
|
||||||
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
let get_boolean = || -> Result<Value, RuntimeError> {
|
let get_boolean = || -> Result<Value, RuntimeError> {
|
||||||
let expression_position = self.expression.position();
|
let expression_position = self.expression.position();
|
||||||
let action = self.expression.clone().run(&mut _context.clone(), false)?;
|
let action = self
|
||||||
|
.expression
|
||||||
|
.clone()
|
||||||
|
.evaluate(&mut _context.clone(), false)?;
|
||||||
|
|
||||||
if let Action::Return(value) = action {
|
if let Evaluation::Return(value) = action {
|
||||||
Ok(value)
|
Ok(value)
|
||||||
} else {
|
} else {
|
||||||
Err(RuntimeError::ValidationFailure(
|
Err(RuntimeError::ValidationFailure(
|
||||||
@ -55,16 +62,16 @@ impl AbstractNode for While {
|
|||||||
|
|
||||||
while let ValueInner::Boolean(true) = get_boolean()?.inner().as_ref() {
|
while let ValueInner::Boolean(true) = get_boolean()?.inner().as_ref() {
|
||||||
for statement in &self.statements {
|
for statement in &self.statements {
|
||||||
let action = statement.clone().run(&mut _context.clone(), false)?;
|
let action = statement.clone().evaluate(&mut _context.clone(), false)?;
|
||||||
|
|
||||||
match action {
|
match action {
|
||||||
Action::Return(_) => {}
|
Evaluation::Return(_) => {}
|
||||||
Action::None => {}
|
Evaluation::None => {}
|
||||||
Action::Break => return Ok(Action::Break),
|
Evaluation::Break => return Ok(Evaluation::Break),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(Action::None)
|
Ok(Evaluation::None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -12,7 +12,7 @@ use crate::{
|
|||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct Context<'a> {
|
pub struct Context<'a> {
|
||||||
variables: Arc<RwLock<BTreeMap<Identifier, (ValueData, UsageData)>>>,
|
variables: Arc<RwLock<BTreeMap<Identifier, (VariableData, UsageData)>>>,
|
||||||
parent: Option<&'a Context<'a>>,
|
parent: Option<&'a Context<'a>>,
|
||||||
is_clean: Arc<RwLock<bool>>,
|
is_clean: Arc<RwLock<bool>>,
|
||||||
}
|
}
|
||||||
@ -28,7 +28,7 @@ impl<'a> Context<'a> {
|
|||||||
|
|
||||||
pub fn inner(
|
pub fn inner(
|
||||||
&self,
|
&self,
|
||||||
) -> Result<RwLockReadGuard<BTreeMap<Identifier, (ValueData, UsageData)>>, RwLockPoisonError>
|
) -> Result<RwLockReadGuard<BTreeMap<Identifier, (VariableData, UsageData)>>, RwLockPoisonError>
|
||||||
{
|
{
|
||||||
Ok(self.variables.read()?)
|
Ok(self.variables.read()?)
|
||||||
}
|
}
|
||||||
@ -50,8 +50,8 @@ impl<'a> Context<'a> {
|
|||||||
log::trace!("Getting {identifier}'s type.");
|
log::trace!("Getting {identifier}'s type.");
|
||||||
|
|
||||||
let r#type = match value_data {
|
let r#type = match value_data {
|
||||||
ValueData::Type(r#type) => r#type.clone(),
|
VariableData::Type(r#type) => r#type.clone(),
|
||||||
ValueData::Value(value) => value.r#type(self)?,
|
VariableData::Value(value) => value.r#type(self)?,
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(Some(r#type.clone()))
|
Ok(Some(r#type.clone()))
|
||||||
@ -63,7 +63,8 @@ impl<'a> Context<'a> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn use_value(&self, identifier: &Identifier) -> Result<Option<Value>, RwLockPoisonError> {
|
pub fn use_value(&self, identifier: &Identifier) -> Result<Option<Value>, RwLockPoisonError> {
|
||||||
if let Some((ValueData::Value(value), usage_data)) = self.variables.read()?.get(identifier)
|
if let Some((VariableData::Value(value), usage_data)) =
|
||||||
|
self.variables.read()?.get(identifier)
|
||||||
{
|
{
|
||||||
log::trace!("Using {identifier}'s value.");
|
log::trace!("Using {identifier}'s value.");
|
||||||
|
|
||||||
@ -79,7 +80,7 @@ impl<'a> Context<'a> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_value(&self, identifier: &Identifier) -> Result<Option<Value>, RwLockPoisonError> {
|
pub fn get_value(&self, identifier: &Identifier) -> Result<Option<Value>, RwLockPoisonError> {
|
||||||
if let Some((ValueData::Value(value), _)) = self.variables.read()?.get(identifier) {
|
if let Some((VariableData::Value(value), _)) = self.variables.read()?.get(identifier) {
|
||||||
log::trace!("Getting {identifier}'s value.");
|
log::trace!("Getting {identifier}'s value.");
|
||||||
|
|
||||||
Ok(Some(value.clone()))
|
Ok(Some(value.clone()))
|
||||||
@ -93,7 +94,7 @@ impl<'a> Context<'a> {
|
|||||||
pub fn get_data(
|
pub fn get_data(
|
||||||
&self,
|
&self,
|
||||||
identifier: &Identifier,
|
identifier: &Identifier,
|
||||||
) -> Result<Option<(ValueData, UsageData)>, RwLockPoisonError> {
|
) -> Result<Option<(VariableData, UsageData)>, RwLockPoisonError> {
|
||||||
if let Some(full_data) = self.variables.read()?.get(identifier) {
|
if let Some(full_data) = self.variables.read()?.get(identifier) {
|
||||||
log::trace!("Getting {identifier}'s value.");
|
log::trace!("Getting {identifier}'s value.");
|
||||||
|
|
||||||
@ -110,7 +111,7 @@ impl<'a> Context<'a> {
|
|||||||
|
|
||||||
self.variables
|
self.variables
|
||||||
.write()?
|
.write()?
|
||||||
.insert(identifier, (ValueData::Type(r#type), UsageData::new()));
|
.insert(identifier, (VariableData::Type(r#type), UsageData::new()));
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -128,9 +129,9 @@ impl<'a> Context<'a> {
|
|||||||
.map(|(_, usage_data)| usage_data);
|
.map(|(_, usage_data)| usage_data);
|
||||||
|
|
||||||
if let Some(usage_data) = old_usage_data {
|
if let Some(usage_data) = old_usage_data {
|
||||||
variables.insert(identifier, (ValueData::Value(value), usage_data));
|
variables.insert(identifier, (VariableData::Value(value), usage_data));
|
||||||
} else {
|
} else {
|
||||||
variables.insert(identifier, (ValueData::Value(value), UsageData::new()));
|
variables.insert(identifier, (VariableData::Value(value), UsageData::new()));
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -144,7 +145,7 @@ impl<'a> Context<'a> {
|
|||||||
self.variables
|
self.variables
|
||||||
.write()?
|
.write()?
|
||||||
.retain(|identifier, (value_data, usage_data)| {
|
.retain(|identifier, (value_data, usage_data)| {
|
||||||
if let ValueData::Value(_) = value_data {
|
if let VariableData::Value(_) = value_data {
|
||||||
let usage = usage_data.inner().read().unwrap();
|
let usage = usage_data.inner().read().unwrap();
|
||||||
|
|
||||||
if usage.actual < usage.expected {
|
if usage.actual < usage.expected {
|
||||||
@ -196,7 +197,7 @@ impl<'a> Context<'a> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq)]
|
#[derive(Clone, Debug, PartialEq)]
|
||||||
pub enum ValueData {
|
pub enum VariableData {
|
||||||
Type(Type),
|
Type(Type),
|
||||||
Value(Value),
|
Value(Value),
|
||||||
}
|
}
|
||||||
|
@ -3,7 +3,7 @@ use std::{io, sync::PoisonError};
|
|||||||
use chumsky::{prelude::Rich, span::Span};
|
use chumsky::{prelude::Rich, span::Span};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
abstract_tree::{SourcePosition, Type},
|
abstract_tree::{r#type::Type, SourcePosition, TypeConstructor},
|
||||||
identifier::Identifier,
|
identifier::Identifier,
|
||||||
lexer::Token,
|
lexer::Token,
|
||||||
};
|
};
|
||||||
@ -141,11 +141,15 @@ pub enum ValidationError {
|
|||||||
actual_position: SourcePosition,
|
actual_position: SourcePosition,
|
||||||
|
|
||||||
/// The position of the item that gave the "expected" type.
|
/// The position of the item that gave the "expected" type.
|
||||||
expected_position: SourcePosition,
|
expected_position: Option<SourcePosition>,
|
||||||
|
},
|
||||||
|
WrongTypeArgumentCount {
|
||||||
|
expected: usize,
|
||||||
|
actual: usize,
|
||||||
},
|
},
|
||||||
WrongArguments {
|
WrongArguments {
|
||||||
expected: Vec<Type>,
|
expected: Vec<TypeConstructor>,
|
||||||
actual: Vec<Type>,
|
actual: Vec<TypeConstructor>,
|
||||||
},
|
},
|
||||||
VariableNotFound {
|
VariableNotFound {
|
||||||
identifier: Identifier,
|
identifier: Identifier,
|
||||||
|
@ -325,21 +325,23 @@ impl InterpreterError {
|
|||||||
} => {
|
} => {
|
||||||
let TypeConflict { actual, expected } = conflict;
|
let TypeConflict { actual, expected } = conflict;
|
||||||
|
|
||||||
builder.add_labels([
|
if let Some(position) = expected_position {
|
||||||
Label::new((
|
builder.add_label(
|
||||||
self.source_id.clone(),
|
Label::new((self.source_id.clone(), position.0..position.1))
|
||||||
expected_position.0..expected_position.1,
|
|
||||||
))
|
|
||||||
.with_message(format!(
|
.with_message(format!(
|
||||||
"Type {} established here.",
|
"Type {} established here.",
|
||||||
expected.fg(type_color)
|
expected.fg(type_color)
|
||||||
)),
|
)),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
builder.add_label(
|
||||||
Label::new((
|
Label::new((
|
||||||
self.source_id.clone(),
|
self.source_id.clone(),
|
||||||
actual_position.0..actual_position.1,
|
actual_position.0..actual_position.1,
|
||||||
))
|
))
|
||||||
.with_message(format!("Got type {} here.", actual.fg(type_color))),
|
.with_message(format!("Got type {} here.", actual.fg(type_color))),
|
||||||
]);
|
);
|
||||||
}
|
}
|
||||||
ValidationError::VariableNotFound {
|
ValidationError::VariableNotFound {
|
||||||
identifier,
|
identifier,
|
||||||
@ -390,6 +392,7 @@ impl InterpreterError {
|
|||||||
ValidationError::ExpectedValue(_) => todo!(),
|
ValidationError::ExpectedValue(_) => todo!(),
|
||||||
ValidationError::PropertyNotFound { .. } => todo!(),
|
ValidationError::PropertyNotFound { .. } => todo!(),
|
||||||
ValidationError::WrongArguments { .. } => todo!(),
|
ValidationError::WrongArguments { .. } => todo!(),
|
||||||
|
ValidationError::WrongTypeArgumentCount { .. } => todo!(),
|
||||||
ValidationError::ExpectedIntegerFloatOrString { actual, position } => {
|
ValidationError::ExpectedIntegerFloatOrString { actual, position } => {
|
||||||
builder = builder.with_message(format!(
|
builder = builder.with_message(format!(
|
||||||
"Expected an {}, {} or {}.",
|
"Expected an {}, {} or {}.",
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -9,7 +9,7 @@ use std::{
|
|||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
abstract_tree::{AbstractNode, Action, Block, Type, WithPos, WithPosition},
|
abstract_tree::{AbstractNode, Block, Evaluation, Type, WithPosition},
|
||||||
context::Context,
|
context::Context,
|
||||||
error::{RuntimeError, ValidationError},
|
error::{RuntimeError, ValidationError},
|
||||||
identifier::Identifier,
|
identifier::Identifier,
|
||||||
@ -52,14 +52,14 @@ impl Value {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn function(
|
pub fn function(
|
||||||
type_arguments: Vec<WithPosition<Type>>,
|
type_parameters: Option<Vec<Identifier>>,
|
||||||
parameters: Vec<(Identifier, WithPosition<Type>)>,
|
value_parameters: Vec<(Identifier, Type)>,
|
||||||
return_type: WithPosition<Type>,
|
return_type: Type,
|
||||||
body: WithPosition<Block>,
|
body: Block,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Value(Arc::new(ValueInner::Function(Function {
|
Value(Arc::new(ValueInner::Function(Function {
|
||||||
type_parameters: type_arguments,
|
type_parameters,
|
||||||
parameters,
|
value_parameters,
|
||||||
return_type,
|
return_type,
|
||||||
body,
|
body,
|
||||||
})))
|
})))
|
||||||
@ -129,19 +129,19 @@ impl Display for Value {
|
|||||||
ValueInner::Range(_) => todo!(),
|
ValueInner::Range(_) => todo!(),
|
||||||
ValueInner::String(string) => write!(f, "{string}"),
|
ValueInner::String(string) => write!(f, "{string}"),
|
||||||
ValueInner::Function(Function {
|
ValueInner::Function(Function {
|
||||||
type_parameters: type_arguments,
|
type_parameters,
|
||||||
parameters,
|
value_parameters: parameters,
|
||||||
return_type,
|
return_type,
|
||||||
body,
|
body,
|
||||||
}) => {
|
}) => {
|
||||||
if !type_arguments.is_empty() {
|
if let Some(type_parameters) = type_parameters {
|
||||||
write!(f, "(")?;
|
write!(f, "(")?;
|
||||||
|
|
||||||
for (index, r#type) in type_arguments.into_iter().enumerate() {
|
for (index, r#type) in type_parameters.into_iter().enumerate() {
|
||||||
if index == type_arguments.len() - 1 {
|
if index == type_parameters.len() - 1 {
|
||||||
write!(f, "{}", r#type.node)?;
|
write!(f, "{}", r#type)?;
|
||||||
} else {
|
} else {
|
||||||
write!(f, "{} ", r#type.node)?;
|
write!(f, "{} ", r#type)?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -151,10 +151,10 @@ impl Display for Value {
|
|||||||
write!(f, "(")?;
|
write!(f, "(")?;
|
||||||
|
|
||||||
for (identifier, r#type) in parameters {
|
for (identifier, r#type) in parameters {
|
||||||
write!(f, "{identifier}: {}", r#type.node)?;
|
write!(f, "{identifier}: {}", r#type)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
write!(f, "): {} {:?}", return_type.node, body.node)
|
write!(f, "): {} {:?}", return_type, body)
|
||||||
}
|
}
|
||||||
ValueInner::Structure { name, fields } => {
|
ValueInner::Structure { name, fields } => {
|
||||||
write!(f, "{}\n{{", name.node)?;
|
write!(f, "{}\n{{", name.node)?;
|
||||||
@ -224,23 +224,19 @@ impl ValueInner {
|
|||||||
ValueInner::Float(_) => Type::Float,
|
ValueInner::Float(_) => Type::Float,
|
||||||
ValueInner::Integer(_) => Type::Integer,
|
ValueInner::Integer(_) => Type::Integer,
|
||||||
ValueInner::List(values) => {
|
ValueInner::List(values) => {
|
||||||
let mut types = Vec::with_capacity(values.len());
|
let item_type = values.first().unwrap().node.r#type(context)?;
|
||||||
|
|
||||||
for value in values {
|
Type::List {
|
||||||
types.push(value.node.r#type(context)?.with_position(value.position));
|
length: values.len(),
|
||||||
|
item_type: Box::new(item_type),
|
||||||
}
|
}
|
||||||
|
|
||||||
Type::ListExact(types)
|
|
||||||
}
|
}
|
||||||
ValueInner::Map(_) => Type::Map,
|
ValueInner::Map(_) => Type::Map,
|
||||||
ValueInner::Range(_) => Type::Range,
|
ValueInner::Range(_) => Type::Range,
|
||||||
ValueInner::String(_) => Type::String,
|
ValueInner::String(_) => Type::String,
|
||||||
ValueInner::Function(function) => Type::Function {
|
ValueInner::Function(function) => Type::Function {
|
||||||
parameter_types: function
|
type_parameters: None,
|
||||||
.parameters
|
value_parameters: function.value_parameters.clone(),
|
||||||
.iter()
|
|
||||||
.map(|(_, r#type)| r#type.clone())
|
|
||||||
.collect(),
|
|
||||||
return_type: Box::new(function.return_type.clone()),
|
return_type: Box::new(function.return_type.clone()),
|
||||||
},
|
},
|
||||||
ValueInner::Structure { name, .. } => {
|
ValueInner::Structure { name, .. } => {
|
||||||
@ -321,14 +317,14 @@ impl Ord for ValueInner {
|
|||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct Function {
|
pub struct Function {
|
||||||
type_parameters: Vec<WithPosition<Type>>,
|
type_parameters: Option<Vec<Identifier>>,
|
||||||
parameters: Vec<(Identifier, WithPosition<Type>)>,
|
value_parameters: Vec<(Identifier, Type)>,
|
||||||
return_type: WithPosition<Type>,
|
return_type: Type,
|
||||||
body: WithPosition<Block>,
|
body: Block,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Function {
|
impl Function {
|
||||||
pub fn type_parameters(&self) -> &Vec<WithPosition<Type>> {
|
pub fn type_parameters(&self) -> &Option<Vec<Identifier>> {
|
||||||
&self.type_parameters
|
&self.type_parameters
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -337,11 +333,12 @@ impl Function {
|
|||||||
arguments: Vec<Value>,
|
arguments: Vec<Value>,
|
||||||
context: &mut Context,
|
context: &mut Context,
|
||||||
clear_variables: bool,
|
clear_variables: bool,
|
||||||
) -> Result<Action, RuntimeError> {
|
) -> Result<Evaluation, RuntimeError> {
|
||||||
for ((identifier, _), value) in self.parameters.into_iter().zip(arguments.into_iter()) {
|
for ((identifier, _), value) in self.value_parameters.into_iter().zip(arguments.into_iter())
|
||||||
|
{
|
||||||
context.set_value(identifier.clone(), value)?;
|
context.set_value(identifier.clone(), value)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
self.body.node.run(context, clear_variables)
|
self.body.evaluate(context, clear_variables)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -56,7 +56,7 @@ fn field_type_error() {
|
|||||||
expected: Type::Integer
|
expected: Type::Integer
|
||||||
},
|
},
|
||||||
actual_position: (128, 134).into(),
|
actual_position: (128, 134).into(),
|
||||||
expected_position: (56, 59).into()
|
expected_position: Some((56, 59).into()),
|
||||||
},
|
},
|
||||||
position: (96, 153).into()
|
position: (96, 153).into()
|
||||||
}]
|
}]
|
||||||
|
@ -148,7 +148,7 @@ fn map_type_errors() {
|
|||||||
expected: Type::Boolean
|
expected: Type::Boolean
|
||||||
},
|
},
|
||||||
actual_position: (15, 20).into(),
|
actual_position: (15, 20).into(),
|
||||||
expected_position: (8, 12).into(),
|
expected_position: Some((8, 12).into()),
|
||||||
},
|
},
|
||||||
position: (0, 22).into()
|
position: (0, 22).into()
|
||||||
}]
|
}]
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
use dust_lang::{
|
use dust_lang::{
|
||||||
abstract_tree::{Block, Statement, Type, ValueExpression, WithPos},
|
abstract_tree::{Block, Expression, Statement, Type, WithPos},
|
||||||
error::{Error, TypeConflict, ValidationError},
|
error::{Error, TypeConflict, ValidationError},
|
||||||
identifier::Identifier,
|
identifier::Identifier,
|
||||||
*,
|
*,
|
||||||
@ -34,7 +34,7 @@ fn set_variable_with_type_error() {
|
|||||||
expected: Type::String
|
expected: Type::String
|
||||||
},
|
},
|
||||||
actual_position: (14, 18).into(),
|
actual_position: (14, 18).into(),
|
||||||
expected_position: (8, 11).into()
|
expected_position: Some((8, 11).into())
|
||||||
},
|
},
|
||||||
position: (0, 18).into()
|
position: (0, 18).into()
|
||||||
}]
|
}]
|
||||||
@ -46,13 +46,12 @@ fn function_variable() {
|
|||||||
assert_eq!(
|
assert_eq!(
|
||||||
interpret("test", "foobar = fn (x: int) int { x }; foobar"),
|
interpret("test", "foobar = fn (x: int) int { x }; foobar"),
|
||||||
Ok(Some(Value::function(
|
Ok(Some(Value::function(
|
||||||
Vec::with_capacity(0),
|
Some(Vec::with_capacity(0)),
|
||||||
vec![(Identifier::new("x"), Type::Integer.with_position((16, 19)))],
|
vec![(Identifier::new("x"), Type::Integer)],
|
||||||
Type::Integer.with_position((21, 24)),
|
Type::Integer,
|
||||||
Block::new(vec![Statement::ValueExpression(
|
Block::new(vec![Statement::ValueExpression(Expression::Identifier(
|
||||||
ValueExpression::Identifier(Identifier::new("x").with_position((27, 28)))
|
Identifier::new("x").with_position((27, 28))
|
||||||
)])
|
))])
|
||||||
.with_position((25, 30))
|
|
||||||
)))
|
)))
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user