feat: 3 new error cases and more generic pretty printing

This commit is contained in:
rvcas
2022-07-15 23:26:57 -04:00
committed by Kasey White
parent 598c5364fe
commit f332dfeb38
10 changed files with 118 additions and 55 deletions

View File

@@ -1,3 +1,5 @@
use std::fmt::Display;
use thiserror::Error;
use crate::ast::{NamedDeBruijn, Term};
@@ -18,4 +20,31 @@ pub enum Error {
NonPolymorphicInstantiation(Value),
#[error("Attempted to apply a non-function: {0:#?}")]
NonFunctionalApplication(Value),
#[error("Type mismatch expected '{0}' got '{1}'")]
TypeMismatch(Type, Type),
#[error("A builtin received a term argument when something else was expected:\n\n{}\n\nYou probably forgot to wrap the builtin with a force.", .0.to_pretty())]
UnexpectedBuiltinTermArgument(Term<NamedDeBruijn>),
#[error("A builtin expected a term argument, but something else was received:\n\n{}\n\nYou probably have an extra force wrapped around a builtin", .0.to_pretty())]
BuiltinTermArgumentExpected(Term<NamedDeBruijn>),
}
#[derive(Debug, Clone)]
pub enum Type {
Bool,
Integer,
String,
ByteString,
Unit,
}
impl Display for Type {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
Type::Bool => write!(f, "bool"),
Type::Integer => write!(f, "integer"),
Type::String => write!(f, "string"),
Type::ByteString => write!(f, "bytestring"),
Type::Unit => write!(f, "unit"),
}
}
}

View File

@@ -2,9 +2,8 @@ use crate::{ast::Constant, builtins::DefaultFunction};
use super::{
cost_model::{BuiltinCosts, ExBudget},
// cost_model::{CostingFun, ExBudget},
Error,
Value,
error::Type,
Error, Value,
};
//#[derive(std::cmp::PartialEq)]
@@ -13,12 +12,6 @@ use super::{
// Deferred,
//}
// pub struct BuiltinRuntimeOptions<T, G> {
// immediate_eval: T,
// deferred_eval: T,
// budget: fn(CostingFun<G>) -> fn(Vec<u32>) -> ExBudget,
// }
#[derive(Clone, Debug)]
pub struct BuiltinRuntime {
args: Vec<Value>,
@@ -35,16 +28,6 @@ impl BuiltinRuntime {
}
}
// fn from_builtin_runtime_options<G>(
// eval_mode: EvalMode,
// cost: CostingFun<G>,
// runtime_options: BuiltinRuntimeOptions<T, G>,
// ) -> BuiltinRuntime {
// Self {
// budget: (runtime_options.budget)(cost),
// }
// }
pub fn is_arrow(&self) -> bool {
self.args.len() != self.fun.arity()
}
@@ -209,7 +192,7 @@ impl DefaultFunction {
if arg.is_integer() {
Ok(())
} else {
todo!("type error")
Err(Error::TypeMismatch(Type::Integer, arg.into()))
}
}
DefaultFunction::SubtractInteger => todo!(),
@@ -240,15 +223,11 @@ impl DefaultFunction {
DefaultFunction::EncodeUtf8 => todo!(),
DefaultFunction::DecodeUtf8 => todo!(),
DefaultFunction::IfThenElse => {
if args.is_empty() {
if arg.is_bool() {
return Ok(());
} else {
todo!("type error")
}
if args.is_empty() && !arg.is_bool() {
Err(Error::TypeMismatch(Type::Bool, arg.into()))
} else {
Ok(())
}
Ok(())
}
DefaultFunction::ChooseUnit => todo!(),
DefaultFunction::Trace => todo!(),
@@ -278,18 +257,19 @@ impl DefaultFunction {
}
}
// This should be safe because we've already checked
// the types of the args as they were pushed. Although
// the unreachables look ugly, it's the reality of the situation.
pub fn call(&self, args: &[Value]) -> Result<Value, Error> {
match self {
DefaultFunction::AddInteger => {
assert_eq!(args.len(), self.arity());
let args = (&args[0], &args[1]);
match args {
(Value::Con(Constant::Integer(arg1)), Value::Con(Constant::Integer(arg2))) => {
Ok(Value::Con(Constant::Integer(arg1 + arg2)))
}
_ => todo!("handle error"),
_ => unreachable!(),
}
}
DefaultFunction::SubtractInteger => todo!(),
@@ -327,7 +307,7 @@ impl DefaultFunction {
Ok(args[2].clone())
}
}
_ => todo!("handle error"),
_ => unreachable!(),
},
DefaultFunction::ChooseUnit => todo!(),
DefaultFunction::Trace => todo!(),