Complete reimplementation
This commit is contained in:
parent
ed73f779e9
commit
7896253fb9
@ -2,10 +2,10 @@
|
||||
name = "evalexpr"
|
||||
version = "0.5.0"
|
||||
description = "Expression evaluator"
|
||||
keywords = ["expression", "evaluate", "evaluator", "expr", "template"]
|
||||
authors = ["fengcen <fengcen.love@gmail.com>", "isibboi <isibboi@gmail.com>"]
|
||||
repository = "https://github.com/ISibboI/eval.git"
|
||||
homepage = "https://github.com/ISibboI/eval"
|
||||
keywords = ["expression", "evaluate", "evaluator", "expr", "template", "arithmetic", "boolean"]
|
||||
authors = ["isibboi <isibboi@gmail.com>"]
|
||||
repository = "https://github.com/ISibboI/evalexpr.git"
|
||||
homepage = "https://github.com/ISibboI/evalexpr"
|
||||
documentation = "https://docs.rs/evalexpr"
|
||||
readme = "README.md"
|
||||
license = "MIT"
|
||||
|
692
src/lib-old.rs
692
src/lib-old.rs
@ -1,692 +0,0 @@
|
||||
//! Eval is a powerful expression evaluator.
|
||||
//!
|
||||
//! Supported operators: `!` `!=` `""` `''` `()` `[]` `.` `,` `>` `<` `>=` `<=`
|
||||
//! `==` `+` unary/binary `-` `*` `/` `%` `&&` `||` `n..m`.
|
||||
//!
|
||||
//! Built-in functions: `min()` `max()` `len()` `is_empty()` `array()` `converge()`.
|
||||
//!
|
||||
//! ## Examples
|
||||
//!
|
||||
//! You can do mathematical calculations with supported operators:
|
||||
//!
|
||||
//! ```
|
||||
//! use evalexpr::{eval, to_value};
|
||||
//!
|
||||
//! assert_eq!(eval("1 + 2 + 3"), Ok(to_value(6)));
|
||||
//! assert_eq!(eval("2 * 2 + 3"), Ok(to_value(7)));
|
||||
//! assert_eq!(eval("2 / 2 + 3"), Ok(to_value(4.0)));
|
||||
//! assert_eq!(eval("2 / 2 + 3 / 3"), Ok(to_value(2.0)));
|
||||
//! ```
|
||||
//!
|
||||
//! You can eval with context:
|
||||
//!
|
||||
//! ```
|
||||
//! use evalexpr::{Expr, to_value};
|
||||
//!
|
||||
//! assert_eq!(Expr::new("foo == bar")
|
||||
//! .value("foo", true)
|
||||
//! .value("bar", true)
|
||||
//! .exec(),
|
||||
//! Ok(to_value(true)));
|
||||
//! ```
|
||||
//!
|
||||
//! You can access data like javascript by using `.` and `[]`. `[]` supports expression.
|
||||
//!
|
||||
//! ```
|
||||
//! use evalexpr::{Expr, to_value};
|
||||
//! use std::collections::HashMap;
|
||||
//!
|
||||
//! let mut object = HashMap::new();
|
||||
//! object.insert("foos", vec!["Hello", "world", "!"]);
|
||||
//!
|
||||
//! assert_eq!(Expr::new("object.foos[2-1] == 'world'") // Access field `foos` and index `2-1`
|
||||
//! .value("object", object)
|
||||
//! .exec(),
|
||||
//! Ok(to_value(true)));
|
||||
//! ```
|
||||
//!
|
||||
//! You can eval with function:
|
||||
//!
|
||||
//! ```
|
||||
//! use evalexpr::{Expr, to_value};
|
||||
//!
|
||||
//! assert_eq!(Expr::new("say_hello()")
|
||||
//! .function("say_hello", |_| Ok(to_value("Hello world!")))
|
||||
//! .exec(),
|
||||
//! Ok(to_value("Hello world!")));
|
||||
//! ```
|
||||
//!
|
||||
//! You can create an array with `array()`:
|
||||
//!
|
||||
//! ```
|
||||
//! use evalexpr::{eval, to_value};
|
||||
//!
|
||||
//! assert_eq!(eval("array(1, 2, 3, 4, 5)"), Ok(to_value(vec![1, 2, 3, 4, 5])));
|
||||
//! ```
|
||||
//!
|
||||
//! You can create an integer array with `n..m`:
|
||||
//!
|
||||
//! ```
|
||||
//! use evalexpr::{eval, to_value};
|
||||
//!
|
||||
//! assert_eq!(eval("0..5"), Ok(to_value(vec![0, 1, 2, 3, 4])));
|
||||
//! ```
|
||||
//!
|
||||
//! ## Built-in functions
|
||||
//!
|
||||
//! ### min()
|
||||
//! Accept multiple arguments and return the minimum value.
|
||||
//!
|
||||
//! ### max()
|
||||
//! Accept multiple arguments and return the maximum value.
|
||||
//!
|
||||
//! ### len()
|
||||
//! Accept single arguments and return the length of value. Only accept String, Array, Object and Null.
|
||||
//!
|
||||
//! ### is_empty()
|
||||
//! Accept single arguments and return a boolean. Check whether the value is empty or not.
|
||||
//!
|
||||
//! ### array()
|
||||
//! Accept multiple arguments and return an array.
|
||||
//!
|
||||
//! ### converge()
|
||||
//! See `builtin::create_converge_function()`.
|
||||
//!
|
||||
//!
|
||||
#![recursion_limit="100"]
|
||||
#![deny(missing_docs)]
|
||||
#![cfg_attr(all(feature = "unstable", test), feature(test))]
|
||||
|
||||
#[macro_use]
|
||||
extern crate quick_error;
|
||||
extern crate serde;
|
||||
extern crate serde_json;
|
||||
|
||||
mod math;
|
||||
mod function;
|
||||
mod operator;
|
||||
mod node;
|
||||
mod tree;
|
||||
mod error;
|
||||
mod builtin;
|
||||
mod expr;
|
||||
|
||||
mod reboot;
|
||||
|
||||
pub use expr::ExecOptions;
|
||||
pub use serde_json::Value;
|
||||
pub use error::Error;
|
||||
pub use function::Function;
|
||||
pub use expr::Expr;
|
||||
|
||||
use std::collections::HashMap;
|
||||
use serde_json::to_value as json_to_value;
|
||||
use serde::Serialize;
|
||||
|
||||
/// Convert variable to `serde_json::Value`
|
||||
pub fn to_value<S: Serialize>(v: S) -> Value {
|
||||
json_to_value(v).unwrap()
|
||||
}
|
||||
|
||||
/// Custom context.
|
||||
pub type Context = HashMap<String, Value>;
|
||||
/// Custom contexts. The value of the last context is searched first.
|
||||
pub type Contexts = Vec<Context>;
|
||||
/// Custom functions.
|
||||
pub type Functions = HashMap<String, Function>;
|
||||
|
||||
/// Evaluates the value of an expression.
|
||||
pub fn eval(expr: &str) -> Result<Value, Error> {
|
||||
Expr::new(expr).compile()?.exec()
|
||||
}
|
||||
|
||||
|
||||
type Compiled = Box<Fn(&[Context], &Functions) -> Result<Value, Error>>;
|
||||
|
||||
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use to_value;
|
||||
use error::Error;
|
||||
use Expr;
|
||||
use tree::Tree;
|
||||
use Value;
|
||||
use eval;
|
||||
use std::collections::HashMap;
|
||||
|
||||
#[test]
|
||||
fn test_add() {
|
||||
assert_eq!(eval("2 + 3"), Ok(to_value(5)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_brackets_add() {
|
||||
assert_eq!(eval("(2 + 3) + (3 + 5)"), Ok(to_value(13)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_brackets_float_add() {
|
||||
assert_eq!(eval("(2 + 3.2) + (3 + 5)"), Ok(to_value(13.2)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_brackets_float_mul() {
|
||||
assert_eq!(eval("(2 + 3.2) * 5"), Ok(to_value(26.0)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_brackets_sub() {
|
||||
assert_eq!(eval("(4 - 3) * 5"), Ok(to_value(5)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_useless_brackets() {
|
||||
assert_eq!(eval("2 + 3 + (5)"), Ok(to_value(10)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_error_brackets_not_with_function() {
|
||||
assert_eq!(eval("5 + ()"), Err(Error::BracketNotWithFunction));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_deep_brackets() {
|
||||
assert_eq!(eval("(2 + (3 + 4) + (6 + (6 + 7)) + 5)"), Ok(to_value(33)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_brackets_div() {
|
||||
assert_eq!(eval("(4 / (2 + 2)) * 5"), Ok(to_value(5.0)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_min() {
|
||||
assert_eq!(eval("min(30, 5, 245, 20)"), Ok(to_value(5)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_min_brackets() {
|
||||
assert_eq!(
|
||||
eval("(min(30, 5, 245, 20) * 10 + (5 + 5) * 5)"),
|
||||
Ok(to_value(100))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_min_and_mul() {
|
||||
assert_eq!(eval("min(30, 5, 245, 20) * 10"), Ok(to_value(50)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_max() {
|
||||
assert_eq!(eval("max(30, 5, 245, 20)"), Ok(to_value(245)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_max_brackets() {
|
||||
assert_eq!(
|
||||
eval("(max(30, 5, 245, 20) * 10 + (5 + 5) * 5)"),
|
||||
Ok(to_value(2500))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_max_and_mul() {
|
||||
assert_eq!(eval("max(30, 5, 245, 20) * 10"), Ok(to_value(2450)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_len_array() {
|
||||
assert_eq!(eval("len(array(2, 3, 4, 5, 6))"), Ok(to_value(5)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_null_and_number() {
|
||||
assert_eq!(eval("hos != 0"), Ok(to_value(true)));
|
||||
assert_eq!(eval("hos > 0"), Ok(to_value(false)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_len_string() {
|
||||
assert_eq!(eval("len('Hello world!')"), Ok(to_value(12)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_len_object() {
|
||||
let mut object = HashMap::new();
|
||||
object.insert("field1", "value1");
|
||||
object.insert("field2", "value2");
|
||||
object.insert("field3", "value3");
|
||||
assert_eq!(
|
||||
Expr::new("len(object)").value("object", object).exec(),
|
||||
Ok(to_value(3))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_brackets_1() {
|
||||
assert_eq!(eval("(5) + (min(3, 4, 5)) + 20"), Ok(to_value(28)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_brackets_2() {
|
||||
assert_eq!(eval("(((5) / 5))"), Ok(to_value(1.0)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_string_add() {
|
||||
assert_eq!(eval(r#""Hello"+", world!""#), Ok(to_value("Hello, world!")));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_equal() {
|
||||
assert_eq!(eval("1 == 1"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_not_equal() {
|
||||
assert_eq!(eval("1 != 2"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_multiple_equal() {
|
||||
assert_eq!(eval("(1 == 2) == (2 == 3)"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_multiple_not_equal() {
|
||||
assert_eq!(eval("(1 != 2) == (2 != 3)"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_greater_than() {
|
||||
assert_eq!(eval("1 > 2"), Ok(to_value(false)));
|
||||
assert_eq!(eval("2 > 1"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_less_than() {
|
||||
assert_eq!(eval("2 < 1"), Ok(to_value(false)));
|
||||
assert_eq!(eval("1 < 2"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_greater_and_less() {
|
||||
assert_eq!(eval("(2 > 1) == (1 < 2)"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_ge() {
|
||||
assert_eq!(eval("2 >= 1"), Ok(to_value(true)));
|
||||
assert_eq!(eval("2 >= 2"), Ok(to_value(true)));
|
||||
assert_eq!(eval("2 >= 3"), Ok(to_value(false)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_le() {
|
||||
assert_eq!(eval("2 <= 1"), Ok(to_value(false)));
|
||||
assert_eq!(eval("2 <= 2"), Ok(to_value(true)));
|
||||
assert_eq!(eval("2 <= 3"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_quotes() {
|
||||
assert_eq!(eval(r#""1><2" + "3<>4""#), Ok(to_value("1><23<>4")));
|
||||
assert_eq!(eval(r#""1==2" + "3--4""#), Ok(to_value("1==23--4")));
|
||||
assert_eq!(eval(r#""1!=2" + "3>>4""#), Ok(to_value("1!=23>>4")));
|
||||
assert_eq!(eval(r#""><1!=2" + "3>>4""#), Ok(to_value("><1!=23>>4")));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_single_quote() {
|
||||
assert_eq!(eval(r#"'1><2' + '3<>4'"#), Ok(to_value("1><23<>4")));
|
||||
assert_eq!(eval(r#"'1==2' + '3--4'"#), Ok(to_value("1==23--4")));
|
||||
assert_eq!(eval(r#"'1!=2' + '3>>4'"#), Ok(to_value("1!=23>>4")));
|
||||
assert_eq!(eval(r#"'!=1<>2' + '3>>4'"#), Ok(to_value("!=1<>23>>4")));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_single_and_double_quote() {
|
||||
assert_eq!(
|
||||
eval(r#"' """" ' + ' """" '"#),
|
||||
Ok(to_value(r#" """" """" "#))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_double_and_single_quote() {
|
||||
assert_eq!(
|
||||
eval(r#"" '''' " + " '''' ""#),
|
||||
Ok(to_value(r#" '''' '''' "#))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_array() {
|
||||
assert_eq!(eval("array(1, 2, 3, 4)"), Ok(to_value(vec![1, 2, 3, 4])));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_range() {
|
||||
assert_eq!(eval("0..5"), Ok(to_value(vec![0, 1, 2, 3, 4])));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_range_and_min() {
|
||||
assert_eq!(eval("min(0..5)"), Ok(to_value(0)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_rem_1() {
|
||||
assert_eq!(eval("2 % 2"), Ok(to_value(0)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_rem_2() {
|
||||
assert_eq!(eval("5 % 56 % 5"), Ok(to_value(0)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_rem_3() {
|
||||
assert_eq!(eval("5.5 % 23"), Ok(to_value(5.5)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_rem_4() {
|
||||
assert_eq!(eval("23 % 5.5"), Ok(to_value(1.0)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_and_1() {
|
||||
assert_eq!(eval("3 > 2 && 2 > 1"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_and_2() {
|
||||
assert_eq!(eval("3 == 2 && 2 == 1"), Ok(to_value(false)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_and_3() {
|
||||
assert_eq!(eval("3 > 2 && 2 == 1"), Ok(to_value(false)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_or_1() {
|
||||
assert_eq!(eval("3 > 2 || 2 > 1"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_or_2() {
|
||||
assert_eq!(eval("3 < 2 || 2 < 1"), Ok(to_value(false)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_or_3() {
|
||||
assert_eq!(eval("3 > 2 || 2 < 1"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_or_4() {
|
||||
assert_eq!(eval("3 < 2 || 2 > 1"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_not() {
|
||||
assert_eq!(eval("!false"), Ok(to_value(true)));
|
||||
assert_eq!(eval("!true"), Ok(to_value(false)));
|
||||
assert_eq!(eval("!(1 != 2)"), Ok(to_value(false)));
|
||||
assert_eq!(eval("!(1 == 2)"), Ok(to_value(true)));
|
||||
assert_eq!(eval("!(1 == 2) == true"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_not_and_brackets() {
|
||||
assert_eq!(eval("(!(1 == 2)) == true"), Ok(to_value(true)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_object_access() {
|
||||
let mut object = HashMap::new();
|
||||
object.insert("foo", "Foo, hello world!");
|
||||
object.insert("bar", "Bar, hello world!");
|
||||
assert_eq!(
|
||||
Expr::new("object.foo == 'Foo, hello world!'")
|
||||
.value("object", object)
|
||||
.exec(),
|
||||
Ok(to_value(true))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_object_dynamic_access() {
|
||||
let mut object = HashMap::new();
|
||||
object.insert("foo", "Foo, hello world!");
|
||||
object.insert("bar", "Bar, hello world!");
|
||||
assert_eq!(
|
||||
Expr::new("object['foo'] == 'Foo, hello world!'")
|
||||
.value("object", object)
|
||||
.exec(),
|
||||
Ok(to_value(true))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_object_dynamic_access_2() {
|
||||
let mut object = HashMap::new();
|
||||
object.insert("foo", "Foo, hello world!");
|
||||
object.insert("bar", "Bar, hello world!");
|
||||
assert_eq!(
|
||||
Expr::new("object[foo] == 'Foo, hello world!'")
|
||||
.value("object", object)
|
||||
.value("foo", "foo")
|
||||
.exec(),
|
||||
Ok(to_value(true))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_path() {
|
||||
assert_eq!(Expr::new("array[2-2].foo[2-2]").exec(), Ok(Value::Null));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_array_access() {
|
||||
let array = vec!["hello", "world", "!"];
|
||||
assert_eq!(
|
||||
Expr::new(
|
||||
"array[1-1] == 'hello' && array[1] == 'world' && array[2] == '!'",
|
||||
).value("array", array)
|
||||
.exec(),
|
||||
Ok(to_value(true))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_builtin_is_empty() {
|
||||
assert_eq!(
|
||||
Expr::new("is_empty(array)")
|
||||
.value("array", Vec::<String>::new())
|
||||
.exec(),
|
||||
Ok(to_value(true))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_builtin_min() {
|
||||
assert_eq!(
|
||||
Expr::new("min(array)")
|
||||
.value("array", vec![23, 34, 45, 2])
|
||||
.exec(),
|
||||
Ok(to_value(2))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_custom_function() {
|
||||
assert_eq!(
|
||||
Expr::new("output()")
|
||||
.function(
|
||||
"output",
|
||||
|_| Ok(to_value("This is custom function's output")),
|
||||
)
|
||||
.exec(),
|
||||
Ok(to_value("This is custom function's output"))
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_error_start_with_non_value_operator() {
|
||||
let mut tree = Tree {
|
||||
raw: "+ + 5".to_owned(),
|
||||
..Default::default()
|
||||
};
|
||||
|
||||
tree.parse_pos().unwrap();
|
||||
tree.parse_operators().unwrap();
|
||||
|
||||
assert_eq!(tree.parse_node(), Err(Error::StartWithNonValueOperator));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_error_duplicate_operator() {
|
||||
let mut tree = Tree {
|
||||
raw: "5 + + 5".to_owned(),
|
||||
..Default::default()
|
||||
};
|
||||
|
||||
tree.parse_pos().unwrap();
|
||||
tree.parse_operators().unwrap();
|
||||
|
||||
assert_eq!(tree.parse_node(), Err(Error::DuplicateOperatorNode));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_error_duplicate_value() {
|
||||
let mut tree = Tree {
|
||||
raw: "2 + 6 5".to_owned(),
|
||||
..Default::default()
|
||||
};
|
||||
|
||||
tree.parse_pos().unwrap();
|
||||
tree.parse_operators().unwrap();
|
||||
|
||||
assert_eq!(tree.parse_node(), Err(Error::DuplicateValueNode));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_error_unpaired_brackets() {
|
||||
let mut tree = Tree {
|
||||
raw: "(2 + 3)) * 5".to_owned(),
|
||||
..Default::default()
|
||||
};
|
||||
|
||||
tree.parse_pos().unwrap();
|
||||
|
||||
assert_eq!(tree.parse_operators(), Err(Error::UnpairedBrackets));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_error_comma() {
|
||||
let mut tree = Tree {
|
||||
raw: ", 2 + 5".to_owned(),
|
||||
..Default::default()
|
||||
};
|
||||
|
||||
tree.parse_pos().unwrap();
|
||||
tree.parse_operators().unwrap();
|
||||
|
||||
assert_eq!(tree.parse_node(), Err(Error::CommaNotWithFunction));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_eval_issue_2() {
|
||||
assert_eq!(eval("2 * (4 + 0) + 4"), Ok(to_value(12)));
|
||||
assert_eq!(eval("2 * (2 + 2) + (1 + 3)"), Ok(to_value(12)));
|
||||
assert_eq!(eval("2 * (4) + (4)"), Ok(to_value(12)));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_unary_minus() {
|
||||
assert_eq!(eval("3 + -4"), Ok(to_value(-1)));
|
||||
assert_eq!(eval("3+-4"), Ok(to_value(-1)));
|
||||
assert_eq!(eval("true && !false"), Ok(to_value(true)));
|
||||
assert_eq!(eval("!true || !true"), Ok(to_value(false)));
|
||||
assert_eq!(eval("-5 + -4"), Ok(to_value(-9)));
|
||||
assert_eq!(eval("-3-4"), Ok(to_value(-7)));
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(all(feature = "unstable", test))]
|
||||
mod benches {
|
||||
extern crate test;
|
||||
use eval;
|
||||
use tree::Tree;
|
||||
use Expr;
|
||||
|
||||
#[bench]
|
||||
fn bench_deep_brackets(b: &mut test::Bencher) {
|
||||
b.iter(|| eval("(2 + (3 + 4) + (6 + (6 + 7)) + 5)"));
|
||||
}
|
||||
|
||||
#[bench]
|
||||
fn bench_parse_pos(b: &mut test::Bencher) {
|
||||
let mut tree = Tree {
|
||||
raw: "(2 + (3 + 4) + (6 + (6 + 7)) + 5)".to_owned(),
|
||||
..Default::default()
|
||||
};
|
||||
|
||||
b.iter(|| tree.parse_pos().unwrap());
|
||||
}
|
||||
|
||||
#[bench]
|
||||
fn bench_parse_operators(b: &mut test::Bencher) {
|
||||
let mut tree = Tree {
|
||||
raw: "(2 + (3 + 4) + (6 + (6 + 7)) + 5)".to_owned(),
|
||||
..Default::default()
|
||||
};
|
||||
|
||||
tree.parse_pos().unwrap();
|
||||
b.iter(|| tree.parse_operators().unwrap());
|
||||
}
|
||||
|
||||
#[bench]
|
||||
fn bench_parse_nodes(b: &mut test::Bencher) {
|
||||
let mut tree = Tree {
|
||||
raw: "(2 + (3 + 4) + (6 + (6 + 7)) + 5)".to_owned(),
|
||||
..Default::default()
|
||||
};
|
||||
|
||||
tree.parse_pos().unwrap();
|
||||
tree.parse_operators().unwrap();
|
||||
b.iter(|| tree.parse_node().unwrap());
|
||||
}
|
||||
|
||||
#[bench]
|
||||
fn bench_compile(b: &mut test::Bencher) {
|
||||
b.iter(|| {
|
||||
let mut tree = Tree {
|
||||
raw: "(2 + (3 + 4) + (6 + (6 + 7)) + 5)".to_owned(),
|
||||
..Default::default()
|
||||
};
|
||||
tree.parse_pos().unwrap();
|
||||
tree.parse_operators().unwrap();
|
||||
tree.parse_node().unwrap();
|
||||
tree.compile().unwrap();
|
||||
});
|
||||
}
|
||||
|
||||
#[bench]
|
||||
fn bench_exec(b: &mut test::Bencher) {
|
||||
let expr = Expr::new("(2 + (3 + 4) + (6 + (6 + 7)) + 5)")
|
||||
.compile()
|
||||
.unwrap();
|
||||
b.iter(|| expr.exec().unwrap())
|
||||
}
|
||||
|
||||
#[bench]
|
||||
fn bench_eval(b: &mut test::Bencher) {
|
||||
b.iter(|| eval("(2 + (3 + 4) + (6 + (6 + 7)) + 5)"));
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user