Skip to content

Commit

Permalink
Handle and report parse errors that cause trouble during evaluation.
Browse files Browse the repository at this point in the history
  • Loading branch information
torhovland committed Aug 26, 2022
1 parent 3972471 commit 5286c47
Show file tree
Hide file tree
Showing 8 changed files with 22 additions and 14 deletions.
3 changes: 3 additions & 0 deletions src/error.rs
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,8 @@ pub enum EvalError {
/* position of the original unevaluated expression */ TermPos,
/* evaluated expression */ RichTerm,
),
/// A wrapped parse error that causes trouble during evaluation.
ParseError(/* parse error */ ParseError),
/// A term which is not a function has been applied to an argument.
NotAFunc(
/* term */ RichTerm,
Expand Down Expand Up @@ -1036,6 +1038,7 @@ impl ToDiagnostic<FileId> for EvalError {
.with_labels(labels)
.with_notes(vec![msg.clone()])]
}
EvalError::ParseError(parse_error) => parse_error.to_diagnostic(files, contract_id),
EvalError::NotAFunc(t, arg, pos_opt) => vec![Diagnostic::error()
.with_message("not a function")
.with_labels(vec![
Expand Down
2 changes: 1 addition & 1 deletion src/eval/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -696,7 +696,7 @@ pub fn subst(rt: RichTerm, global_env: &Environment, env: &Environment) -> RichT
})
.unwrap_or_else(|| RichTerm::new(Term::Var(id), pos)),
v @ Term::Null
| v @ Term::ParseError
| v @ Term::ParseError(_)
| v @ Term::Bool(_)
| v @ Term::Num(_)
| v @ Term::Str(_)
Expand Down
2 changes: 2 additions & 0 deletions src/eval/operation.rs
Original file line number Diff line number Diff line change
Expand Up @@ -465,6 +465,8 @@ fn process_unary_operation(
pos_op,
)), //TODO include the position of operators on the stack
}
} else if let Term::ParseError(parse_error) = &*t {
Err(EvalError::ParseError(parse_error.clone()))
} else {
Err(EvalError::TypeError(
String::from("Record"),
Expand Down
6 changes: 4 additions & 2 deletions src/parser/grammar.lalrpop
Original file line number Diff line number Diff line change
Expand Up @@ -219,9 +219,11 @@ UniTerm: UniTerm = {
UniTerm::from(mk_app!(Term::Op1(UnaryOp::Ite(), cond), t1, t2)),
<l: @L> <t: !> <r: @R> => {
let pos = mk_pos(src_id, l, r);
errors.push(t);
errors.push(t.clone());

UniTerm::from(RichTerm::new(Term::ParseError, pos))
UniTerm::from(RichTerm::new(Term::ParseError(
crate::error::ParseError::from_lalrpop(t.error, src_id)
), pos))
},
};

Expand Down
2 changes: 1 addition & 1 deletion src/pretty.rs
Original file line number Diff line number Diff line change
Expand Up @@ -684,7 +684,7 @@ where
.append(allocator.space())
.append(allocator.as_string(f.to_string_lossy()).double_quotes()),
ResolvedImport(id) => allocator.text(format!("import <file_id: {:?}>", id)),
ParseError => allocator
ParseError(_) => allocator
.text("#<PARSE ERROR!>")
.append(allocator.hardline()),
}
Expand Down
15 changes: 8 additions & 7 deletions src/term.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
//! the term level, and together with [crate::eval::merge], they allow for flexible and modular
//! definitions of contracts, record and metadata all together.
use crate::destruct::Destruct;
use crate::error::ParseError;
use crate::identifier::Ident;
use crate::label::Label;
use crate::match_sharedterm;
Expand Down Expand Up @@ -157,7 +158,7 @@ pub enum Term {
#[serde(skip)]
ResolvedImport(FileId),
#[serde(skip)]
ParseError,
ParseError(ParseError),
}

pub type SealingKey = i32;
Expand Down Expand Up @@ -361,7 +362,7 @@ impl Term {
{
use self::Term::*;
match self {
Null | ParseError => (),
Null | ParseError(_) => (),
Switch(ref mut t, ref mut cases, ref mut def) => {
cases.iter_mut().for_each(|c| {
let (_, t) = c;
Expand Down Expand Up @@ -448,7 +449,7 @@ impl Term {
| Term::Import(_)
| Term::ResolvedImport(_)
| Term::StrChunks(_)
| Term::ParseError => None,
| Term::ParseError(_) => None,
}
.map(String::from)
}
Expand Down Expand Up @@ -512,7 +513,7 @@ impl Term {
format!("<{}{}={}>", content, value_label, value)
}
Term::Var(id) => id.to_string(),
Term::ParseError => String::from("<parse error>"),
Term::ParseError(_) => String::from("<parse error>"),
Term::Let(..)
| Term::LetPattern(..)
| Term::App(_, _)
Expand Down Expand Up @@ -580,7 +581,7 @@ impl Term {
| Term::ResolvedImport(_)
| Term::StrChunks(_)
| Term::RecRecord(..)
| Term::ParseError => false,
| Term::ParseError(_) => false,
}
}

Expand Down Expand Up @@ -620,7 +621,7 @@ impl Term {
| Term::ResolvedImport(_)
| Term::StrChunks(_)
| Term::RecRecord(..)
| Term::ParseError => false,
| Term::ParseError(_) => false,
}
}

Expand Down Expand Up @@ -652,7 +653,7 @@ impl Term {
| Term::MetaValue(..)
| Term::Import(..)
| Term::ResolvedImport(..)
| Term::ParseError => false,
| Term::ParseError(_) => false,
}
}
}
Expand Down
2 changes: 1 addition & 1 deletion src/transform/free_vars.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ fn collect_free_vars(rt: &mut RichTerm, free_vars: &mut HashSet<Ident>) {
Term::Var(id) => {
free_vars.insert(id.clone());
}
Term::ParseError
Term::ParseError(_)
| Term::Null
| Term::Bool(_)
| Term::Num(_)
Expand Down
4 changes: 2 additions & 2 deletions src/typecheck/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -290,7 +290,7 @@ fn walk<L: Linearizer>(
);

match t.as_ref() {
Term::ParseError
Term::ParseError(_)
| Term::Null
| Term::Bool(_)
| Term::Num(_)
Expand Down Expand Up @@ -528,7 +528,7 @@ fn type_check_<L: Linearizer>(
linearizer.add_term(lin, t, *pos, ty.clone());

match t.as_ref() {
Term::ParseError => Ok(()),
Term::ParseError(_) => Ok(()),
// null is inferred to be of type Dyn
Term::Null => unify(state, ty, mk_typewrapper::dynamic())
.map_err(|err| err.into_typecheck_err(state, rt.pos)),
Expand Down

0 comments on commit 5286c47

Please sign in to comment.