-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
17 changed files
with
576 additions
and
300 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,127 +1,158 @@ | ||
use core::panic; | ||
use std::rc::Rc; | ||
use ordered_hash_map::OrderedHashMap; | ||
use crate::parser::{ArithmeticOp, Expression, Member, RelationOp}; | ||
use crate::{value::Value, Context}; | ||
|
||
pub trait Bag { | ||
fn unpack(self) -> Value; | ||
} | ||
|
||
impl Bag for Value { | ||
fn unpack(self) -> Value { | ||
self | ||
} | ||
} | ||
use crate::parser::{Atom, Expression, RelationOp}; | ||
use crate::value::value::{Val, Value}; | ||
use crate::Context; | ||
|
||
#[derive(Default)] | ||
pub struct Eval {} | ||
|
||
impl Eval { | ||
// fn eval_member(&self, expr: Expression, member: Member, ctx: &mut Context) -> impl Bag { | ||
// let v = self.eval(expr, ctx).unpack(); | ||
// match member { | ||
// crate::parser::Member::Attribute(attr) => { | ||
// if let Value::Map(v) = v { | ||
// let value = v.get(&Value::String(attr)).expect("TODO: unknown map key"); | ||
// return value.to_owned() | ||
// } | ||
// if let Some(val) = ctx.resolve(&attr) { | ||
// return val | ||
// } | ||
// panic!("unknown attribute {}", attr) | ||
// }, | ||
// crate::parser::Member::FunctionCall(name, mut rargs) => { | ||
// let mut args = Vec::with_capacity(rargs.len()); | ||
// rargs.reverse(); | ||
// for arg in rargs { | ||
// args.push(self.eval(arg, ctx).unpack()); | ||
// } | ||
|
||
fn eval_member(&self, expr: Expression, member: Member, ctx: &mut Context) -> impl Bag { | ||
let v = self.eval(expr, ctx).unpack(); | ||
match member { | ||
crate::parser::Member::Attribute(attr) => { | ||
if let Value::Map(v) = v { | ||
let value = v.get(&Value::String(attr)).expect("TODO: unknown map key"); | ||
return value.to_owned() | ||
} | ||
if let Some(val) = ctx.resolve(&attr) { | ||
return val | ||
} | ||
panic!("unknown attribute {}", attr) | ||
}, | ||
crate::parser::Member::FunctionCall(name, mut rargs) => { | ||
let mut args = Vec::with_capacity(rargs.len()); | ||
rargs.reverse(); | ||
for arg in rargs { | ||
args.push(self.eval(arg, ctx).unpack()); | ||
} | ||
|
||
if let Some(val) = ctx.resolve(&name) { | ||
args.push(v.clone()); | ||
args.reverse(); | ||
if let Value::Function(f) = val { | ||
return (f.overloads.first().unwrap().func)(args) | ||
} | ||
} | ||
|
||
panic!("is not a func") | ||
}, | ||
crate::parser::Member::Index(i) => { | ||
let i = self.eval(*i, ctx).unpack(); | ||
if let Value::Map(v) = v { | ||
let value = v.get(&i).expect("TODO: unknown map key"); | ||
return value.to_owned() | ||
} | ||
Value::Null | ||
}, | ||
crate::parser::Member::Fields(_) => todo!("Fields"), | ||
} | ||
} | ||
fn eval_map(&self, entries: Vec<(Expression, Expression)>, ctx: &mut Context) -> Value { | ||
let mut map = OrderedHashMap::with_capacity(entries.len()); | ||
for (kexpr, vexpr) in entries { | ||
let k = self.eval(kexpr, ctx).unpack(); | ||
let v = self.eval(vexpr, ctx).unpack(); | ||
map.insert(k, v); | ||
} | ||
Value::Map(Rc::new(map)) | ||
} | ||
// if let Some(val) = ctx.resolve(&name) { | ||
// args.push(v.clone()); | ||
// args.reverse(); | ||
// if let Value::Function(f) = val { | ||
// return (f.overloads.first().unwrap().func)(args) | ||
// } | ||
// } | ||
|
||
fn eval_list(&self, elems: Vec<Expression>, ctx: &mut Context) -> Value { | ||
let mut list = Vec::with_capacity(elems.len()); | ||
for expr in elems { | ||
let v = self.eval(expr, ctx).unpack(); | ||
list.push(v); | ||
} | ||
Value::List(Rc::new(list)) | ||
} | ||
// panic!("is not a func") | ||
// }, | ||
// crate::parser::Member::Index(i) => { | ||
// let i = self.eval(*i, ctx).unpack(); | ||
// if let Value::Map(v) = v { | ||
// let value = v.get(&i).expect("TODO: unknown map key"); | ||
// return value.to_owned() | ||
// } | ||
// Value::Null | ||
// }, | ||
// crate::parser::Member::Fields(_) => todo!("Fields"), | ||
// } | ||
// } | ||
// fn eval_map(&self, entries: Vec<(Expression, Expression)>, ctx: &mut Context) -> Value { | ||
// let mut map = OrderedHashMap::with_capacity(entries.len()); | ||
// for (kexpr, vexpr) in entries { | ||
// let k = self.eval(kexpr, ctx).unpack(); | ||
// let v = self.eval(vexpr, ctx).unpack(); | ||
// map.insert(k, v); | ||
// } | ||
// Value::Map(Rc::new(map)) | ||
// } | ||
|
||
pub fn eval(&self, expr: Expression, ctx: &mut Context) -> impl Bag { | ||
// fn eval_list(&self, elems: Vec<Expression>, ctx: &mut Context) -> Value { | ||
// let mut list = Vec::with_capacity(elems.len()); | ||
// for expr in elems { | ||
// let v = self.eval(expr, ctx).unpack(); | ||
// list.push(v); | ||
// } | ||
// Value::List(Rc::new(list)) | ||
// } | ||
|
||
pub fn eval(&self, expr: Expression, ctx: &mut Context) -> Val { | ||
match expr { | ||
Expression::Atom(atom) => Value::from(atom), | ||
Expression::Arithmetic(_, _, _) => todo!(), | ||
Expression::Relation(left, op, right) => { | ||
let left = self.eval(*left, ctx).unpack(); | ||
let right = self.eval(*right, ctx).unpack(); | ||
let result = match op { | ||
RelationOp::Equals => left.eq(&right), | ||
let l = self.eval(*left, ctx); | ||
let r = self.eval(*right, ctx); | ||
Val::new_bool(match op { | ||
RelationOp::Equals => l.eq(&r), | ||
RelationOp::LessThan => todo!("lt"), | ||
RelationOp::LessThanEq => todo!("lte"), | ||
RelationOp::GreaterThan => todo!("gt"), | ||
RelationOp::GreaterThanEq => todo!("gte"), | ||
RelationOp::NotEquals => todo!("ne"), | ||
RelationOp::In => todo!("in"), | ||
}; | ||
Value::Bool(result) | ||
} | ||
Expression::Arithmetic(left, op, right) => { | ||
let left = self.eval(*left, ctx).unpack(); | ||
let right = self.eval(*right, ctx).unpack(); | ||
match op { | ||
ArithmeticOp::Add => left + right, | ||
ArithmeticOp::Subtract => left - right, | ||
ArithmeticOp::Divide => left / right, | ||
ArithmeticOp::Multiply => left * right, | ||
ArithmeticOp::Modulus => todo!("modulus"), | ||
} | ||
}) | ||
} | ||
Expression::Ternary(_, _, _) => todo!(), | ||
Expression::Or(_, _) => todo!(), | ||
Expression::And(_, _) => todo!(), | ||
Expression::Unary(_, _) => todo!(), | ||
Expression::Member(expr, member) => self.eval_member(*expr, *member, ctx).unpack(), | ||
Expression::List(elems) => self.eval_list(elems, ctx), | ||
Expression::Map(entries) => self.eval_map(entries, ctx), | ||
Expression::Ident(r) => { | ||
let val = ctx.resolve(&r); | ||
if let Some(val) = val { | ||
return val | ||
} | ||
panic!("unknown attribute {}", &r) | ||
}, | ||
Expression::Member(_, _) => todo!(), | ||
Expression::FunctionCall(_) => todo!(), | ||
Expression::List(_) => todo!(), | ||
Expression::Map(_) => todo!(), | ||
Expression::Atom(atom) => self.eval_atom(atom, ctx), | ||
Expression::Ident(ident) => ctx | ||
.resolve(&ident) | ||
.unwrap_or(&Val::new_error(format!("unknown variable {}", ident))) | ||
.to_owned(), | ||
} | ||
} | ||
|
||
pub fn eval_atom(&self, atom: Atom, ctx: &mut Context) -> Val { | ||
match atom { | ||
Atom::Int(i) => Val::new_int(i), | ||
Atom::UInt(u) => Val::new_uint(u), | ||
Atom::Float(f) => Val::new_double(f), | ||
Atom::String(s) => Val::new_string(s), | ||
Atom::Bytes(b) => Val::new_bytes(b), | ||
Atom::Bool(b) => Val::new_bool(b), | ||
Atom::Null => Val::new_null(), | ||
} | ||
} | ||
|
||
// pub fn eval(&self, expr: Expression, ctx: &mut Context) -> impl Bag { | ||
// match expr { | ||
// Expression::Atom(atom) => Value::from(atom), | ||
// Expression::Relation(left, op, right) => { | ||
// let left = self.eval(*left, ctx).unpack(); | ||
// let right = self.eval(*right, ctx).unpack(); | ||
// let result = match op { | ||
// RelationOp::Equals => left.eq(&right), | ||
// RelationOp::LessThan => todo!("lt"), | ||
// RelationOp::LessThanEq => todo!("lte"), | ||
// RelationOp::GreaterThan => todo!("gt"), | ||
// RelationOp::GreaterThanEq => todo!("gte"), | ||
// RelationOp::NotEquals => todo!("ne"), | ||
// RelationOp::In => todo!("in"), | ||
// }; | ||
// Value::Bool(result) | ||
// } | ||
// Expression::Arithmetic(left, op, right) => { | ||
// let left = self.eval(*left, ctx).unpack(); | ||
// let right = self.eval(*right, ctx).unpack(); | ||
// match op { | ||
// ArithmeticOp::Add => left + right, | ||
// ArithmeticOp::Subtract => left - right, | ||
// ArithmeticOp::Divide => left / right, | ||
// ArithmeticOp::Multiply => left * right, | ||
// ArithmeticOp::Modulus => todo!("modulus"), | ||
// } | ||
// } | ||
// Expression::Ternary(_, _, _) => todo!(), | ||
// Expression::Or(_, _) => todo!(), | ||
// Expression::And(_, _) => todo!(), | ||
// Expression::Unary(_, _) => todo!(), | ||
// Expression::Member(expr, member) => self.eval_member(*expr, *member, ctx).unpack(), | ||
// Expression::List(elems) => self.eval_list(elems, ctx), | ||
// Expression::Map(entries) => self.eval_map(entries, ctx), | ||
// Expression::Ident(r) => { | ||
// let val = ctx.resolve(&r); | ||
// if let Some(val) = val { | ||
// return val | ||
// } | ||
// panic!("unknown attribute {}", &r) | ||
// }, | ||
// Expression::FunctionCall(_) => todo!(), | ||
// } | ||
// } | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,8 +1,9 @@ | ||
pub mod context; | ||
pub mod program; | ||
pub use crate::program::Program; | ||
pub use crate::context::Context; | ||
pub mod value; | ||
pub use crate::value::Value; | ||
mod context; | ||
mod value; | ||
mod program; | ||
mod eval; | ||
mod parser; | ||
mod parser; | ||
|
||
// public api | ||
pub use crate::program::Program; | ||
pub use crate::context::Context; |
Oops, something went wrong.