1
0

Make one report for each error

This commit is contained in:
Jeff 2024-03-06 22:15:35 -05:00
parent 37a88df613
commit fdf6983ab2
6 changed files with 148 additions and 49 deletions

@ -21,11 +21,42 @@ pub enum Logic<'src> {
impl<'src> AbstractTree for Logic<'src> { impl<'src> AbstractTree for Logic<'src> {
fn expected_type(&self, _context: &Context) -> Result<Type, ValidationError> { fn expected_type(&self, _context: &Context) -> Result<Type, ValidationError> {
todo!() Ok(Type::Boolean)
} }
fn validate(&self, _context: &Context) -> Result<(), ValidationError> { fn validate(&self, context: &Context) -> Result<(), ValidationError> {
todo!() match self {
Logic::Equal(left, right)
| Logic::NotEqual(left, right)
| Logic::Greater(left, right)
| Logic::Less(left, right)
| Logic::GreaterOrEqual(left, right)
| Logic::LessOrEqual(left, right) => {
let left = left.expected_type(context)?;
let right = right.expected_type(context)?;
left.check(&right)?;
Ok(())
}
Logic::And(left, right) | Logic::Or(left, right) => {
let left = left.expected_type(context)?;
let right = right.expected_type(context)?;
if let (Type::Boolean, Type::Boolean) = (left, right) {
Ok(())
} else {
Err(ValidationError::ExpectedBoolean)
}
}
Logic::Not(expression) => {
if let Type::Boolean = expression.expected_type(context)? {
Ok(())
} else {
Err(ValidationError::ExpectedBoolean)
}
}
}
} }
fn run(self, _context: &Context) -> Result<Value, RuntimeError> { fn run(self, _context: &Context) -> Result<Value, RuntimeError> {

@ -1,3 +1,5 @@
use std::fmt::{self, Display, Formatter};
use crate::{ use crate::{
abstract_tree::Identifier, abstract_tree::Identifier,
context::Context, context::Context,
@ -103,6 +105,37 @@ impl AbstractTree for Type {
} }
} }
impl Display for Type {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
match self {
Type::Any => write!(f, "any"),
Type::Boolean => write!(f, "boolean"),
Type::Custom(name) => write!(f, "{name}"),
Type::Float => write!(f, "float"),
Type::Integer => write!(f, "integer"),
Type::List => write!(f, "list"),
Type::ListOf(item_type) => write!(f, "list of {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}")?;
} else {
write!(f, "{item_type}, ")?;
}
}
write!(f, "]")
}
Type::Map => write!(f, "map"),
Type::None => write!(f, "none"),
Type::Range => write!(f, "range"),
Type::String => write!(f, "string"),
}
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;

@ -1,6 +1,6 @@
use std::sync::PoisonError; use std::sync::PoisonError;
use ariadne::{Label, Report, ReportKind}; use ariadne::{Color, Label, Report, ReportKind};
use chumsky::{prelude::Rich, span::SimpleSpan}; use chumsky::{prelude::Rich, span::SimpleSpan};
use crate::{ use crate::{
@ -25,6 +25,68 @@ pub enum Error {
}, },
} }
impl Error {
pub fn report(&self) -> Report {
match self {
Error::Parse { expected, span } => Report::build(
ReportKind::Custom("Parsing Error", Color::White),
(),
span.start,
)
.with_label(
Label::new(span.start..span.end).with_message(format!("Expected {expected}.")),
)
.finish(),
Error::Lex { expected, span } => {
let expected = match expected.as_str() {
"" => "something else",
expected => expected,
};
Report::build(
ReportKind::Custom("Lexing Error", Color::White),
(),
span.start,
)
.with_label(
Label::new(span.start..span.end).with_message(format!("Expected {expected}.")),
)
.finish()
}
Error::Runtime(_) => todo!(),
Error::Validation { error, span } => {
let mut report = Report::build(
ReportKind::Custom("Lexing Error", Color::White),
(),
span.start,
);
match error {
ValidationError::ExpectedBoolean => {
report = report.with_label(
Label::new(span.start..span.end).with_message("Expected boolean."),
);
}
ValidationError::RwLockPoison(_) => todo!(),
ValidationError::TypeCheck(TypeCheckError { actual, expected }) => {
report = report.with_label(Label::new(span.start..span.end).with_message(
format!("Type error. Expected {expected} but got {actual}."),
));
}
ValidationError::VariableNotFound(identifier) => {
report = report
.with_label(Label::new(span.start..span.end).with_message(format!(
"The variable {identifier} does not exist."
)));
}
}
report.finish()
}
}
}
}
impl From<Rich<'_, char>> for Error { impl From<Rich<'_, char>> for Error {
fn from(error: Rich<'_, char>) -> Self { fn from(error: Rich<'_, char>) -> Self {
Error::Lex { Error::Lex {
@ -101,42 +163,3 @@ pub struct TypeCheckError {
pub actual: Type, pub actual: Type,
pub expected: Type, pub expected: Type,
} }
pub fn create_report<'a>(errors: &'a [Error]) -> Report<'a> {
let mut report = Report::build(ReportKind::Error, (), 0);
for error in errors {
match &error {
Error::Parse { expected, span } => {
report = report.with_label(
Label::new(span.start..span.end).with_message(format!("Expected {expected}.")),
);
}
Error::Lex { expected, span } => {
let expected = match expected.as_str() {
"" => "something else",
expected => expected,
};
report = report.with_label(
Label::new(span.start..span.end).with_message(format!("Expected {expected}.")),
);
}
Error::Runtime(_) => todo!(),
Error::Validation { error, span } => match error {
ValidationError::ExpectedBoolean => todo!(),
ValidationError::RwLockPoison(_) => todo!(),
ValidationError::TypeCheck(_) => todo!(),
ValidationError::VariableNotFound(identifier) => {
report =
report
.with_label(Label::new(span.start..span.end).with_message(format!(
"The variable {identifier} does not exist."
)));
}
},
}
}
report.finish()
}

@ -6,7 +6,7 @@ use colored::Colorize;
use std::{fs::read_to_string, io::Write}; use std::{fs::read_to_string, io::Write};
use dust_lang::{context::Context, error::create_report, Interpreter}; use dust_lang::{context::Context, Interpreter};
/// Command-line arguments to be parsed. /// Command-line arguments to be parsed.
#[derive(Parser, Debug)] #[derive(Parser, Debug)]
@ -52,8 +52,10 @@ fn main() {
println!("{value}") println!("{value}")
} }
} }
Err(errors) => create_report(&errors) Err(errors) => {
.eprint(Source::from(&source)) for error in errors {
.unwrap(), error.report().eprint(Source::from(&source)).unwrap();
}
}
} }
} }

@ -107,10 +107,10 @@ pub fn parser<'src>() -> DustParser<'src> {
infix(left(1), just(Token::Operator("<=")), |left, right| { infix(left(1), just(Token::Operator("<=")), |left, right| {
Expression::Logic(Box::new(Logic::LessOrEqual(left, right))) Expression::Logic(Box::new(Logic::LessOrEqual(left, right)))
}), }),
infix(left(1), just(Token::Operator("&&")), |left, right| { infix(right(1), just(Token::Operator("&&")), |left, right| {
Expression::Logic(Box::new(Logic::And(left, right))) Expression::Logic(Box::new(Logic::And(left, right)))
}), }),
infix(left(1), just(Token::Operator("||")), |left, right| { infix(right(1), just(Token::Operator("||")), |left, right| {
Expression::Logic(Box::new(Logic::Or(left, right))) Expression::Logic(Box::new(Logic::Or(left, right)))
}), }),
)) ))

10
tests/expressions.rs Normal file

@ -0,0 +1,10 @@
use dust_lang::*;
#[test]
fn logic() {
assert_eq!(interpret("1 == 1"), Ok(Value::boolean(true)));
assert_eq!(
interpret("('42' == '42') && (42 != 0)"),
Ok(Value::boolean(true))
);
}