2023-04-07 18:16:22 -07:00
|
|
|
// Many of these have been borrowed from insect.
|
|
|
|
use crate::parser;
|
2023-06-11 13:53:45 -07:00
|
|
|
use crate::evaluate::evaluate;
|
2023-06-14 14:36:58 -07:00
|
|
|
use crate::context::Context;
|
2023-04-07 18:16:22 -07:00
|
|
|
|
|
|
|
fn eval_to_str(s: &str) -> Result<String, ()> {
|
2023-06-14 16:15:51 -07:00
|
|
|
let g = match parser::parse_no_context(&String::from(s)) {
|
2023-04-07 18:16:22 -07:00
|
|
|
Ok(x) => x,
|
|
|
|
Err(_) => return Err(())
|
|
|
|
};
|
|
|
|
//let out_str = g.print();
|
|
|
|
|
2023-08-03 13:56:50 -07:00
|
|
|
return match evaluate(&g, &mut Context::new(), false) {
|
2023-04-10 14:02:01 -07:00
|
|
|
Ok(x) => Ok(x.to_string_outer()),
|
2023-04-07 18:16:22 -07:00
|
|
|
Err(_) => Err(())
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
fn good_expr(r: &str, s: &str) {
|
|
|
|
let out = eval_to_str(s).unwrap();
|
|
|
|
assert_eq!(r, out);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn bad_expr(s: &str) {
|
|
|
|
let out = eval_to_str(s);
|
|
|
|
|
|
|
|
match out {
|
|
|
|
Err(_) => { return },
|
|
|
|
_ => {}
|
|
|
|
};
|
|
|
|
|
|
|
|
panic!()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn basic_numbers() {
|
|
|
|
good_expr("1", "1");
|
|
|
|
good_expr("1", "1.0");
|
|
|
|
good_expr("1", "1.0000");
|
|
|
|
good_expr("1", "+1.0");
|
|
|
|
good_expr("1", "+1");
|
|
|
|
good_expr("3.5", "3.5");
|
|
|
|
good_expr("3.5", "3.50");
|
|
|
|
good_expr("3.5", "+3.50");
|
|
|
|
good_expr("0.2", "0.2");
|
|
|
|
good_expr("0.2", "+0.2 ");
|
|
|
|
good_expr("0.2", ".2");
|
|
|
|
good_expr("0.2", "+.2");
|
|
|
|
good_expr("-0.61", "-0.61");
|
|
|
|
good_expr("-0.61", "-.61");
|
|
|
|
good_expr("-0.61", "- .61");
|
|
|
|
good_expr("0.05", ".05");
|
|
|
|
good_expr("-123.45", "-123.45");
|
|
|
|
|
|
|
|
bad_expr("123..");
|
|
|
|
bad_expr("0..");
|
|
|
|
bad_expr(".0.");
|
|
|
|
bad_expr(".");
|
|
|
|
bad_expr(". 2");
|
|
|
|
bad_expr("..2");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn big_numbers() {
|
|
|
|
good_expr("1.2346e15", "1234567890000000");
|
|
|
|
good_expr("1.2346e15", "1234567890000000.0");
|
|
|
|
good_expr("1.2346e15", "+1234567890000000.0");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn signs() {
|
|
|
|
good_expr( "5", "+++++5");
|
|
|
|
good_expr( "5", "++++5");
|
|
|
|
good_expr( "5", "+++5");
|
|
|
|
good_expr( "5", "++5");
|
|
|
|
good_expr( "5", "+5");
|
|
|
|
good_expr("-5", "-5");
|
|
|
|
good_expr( "5", "--5");
|
|
|
|
good_expr("-5", "---5");
|
|
|
|
good_expr( "5", "----5");
|
|
|
|
good_expr("-5", "-----5");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn bad_expressions() {
|
|
|
|
bad_expr("2^");
|
|
|
|
bad_expr("^2");
|
|
|
|
bad_expr("5*");
|
|
|
|
bad_expr("5/");
|
|
|
|
bad_expr("5%");
|
|
|
|
bad_expr("%2");
|
|
|
|
bad_expr("3 + ");
|
2023-06-14 21:11:00 -07:00
|
|
|
//bad_expr("3 + @");
|
2023-04-07 18:16:22 -07:00
|
|
|
bad_expr("3 - ");
|
|
|
|
bad_expr("()");
|
|
|
|
bad_expr("3+2)");
|
2023-07-31 15:52:52 -07:00
|
|
|
bad_expr("-");
|
|
|
|
bad_expr("*");
|
|
|
|
bad_expr("+");
|
2023-04-07 18:16:22 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn implicit_multiply() {
|
|
|
|
good_expr("15", "5(3)");
|
|
|
|
good_expr("15", "(5)3");
|
|
|
|
good_expr("15", "(5)(3)");
|
|
|
|
bad_expr("5 2");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn scientific() {
|
|
|
|
good_expr("100", "1e2");
|
|
|
|
good_expr("0.01", "1e-2");
|
|
|
|
good_expr("1", "1e0");
|
|
|
|
|
|
|
|
// In these expressions, `e` is euler's number
|
|
|
|
// under implicit multiplication
|
|
|
|
good_expr("5.4366", "1e(2)");
|
|
|
|
good_expr("14.778", "e2e");
|
|
|
|
|
|
|
|
bad_expr("2 2e2");
|
|
|
|
bad_expr("1e1.2");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn operators() {
|
|
|
|
|
|
|
|
good_expr("125", "5^(+3)");
|
|
|
|
good_expr("125", "+5^3");
|
|
|
|
good_expr("0.2148", "3 ^ (-1.4)");
|
|
|
|
|
|
|
|
// Should parse as ((2^3)^4)^5
|
|
|
|
good_expr("1.1529e18", "2^3^4^5");
|
|
|
|
|
|
|
|
// Should parse as 1/(2pi)
|
|
|
|
good_expr("0.15915", "1/2pi");
|
|
|
|
// Should parse as (1/2)*pi
|
|
|
|
good_expr("1.5708", "1/2*pi");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
good_expr("15", "5*3");
|
|
|
|
good_expr("15", "5 * 3 ");
|
|
|
|
good_expr("15", "( 5 ) * ( 3 )");
|
|
|
|
good_expr("15", "( 5 ) ( 3 )");
|
|
|
|
good_expr("15", "( ( 5 ) * ( 3 ) )");
|
|
|
|
good_expr("15", "((5)*(3");
|
|
|
|
good_expr("15", "( 5 * 3 )");
|
|
|
|
good_expr("15", "5(+3)");
|
|
|
|
good_expr("15", "+5*3");
|
|
|
|
|
|
|
|
good_expr("-15", "5*(-3)");
|
|
|
|
good_expr("-15", "5 * (-3)");
|
|
|
|
good_expr("-15", "( 5 ) * ( -3 )");
|
|
|
|
good_expr("-15", "( ( 5 ) * (-( 3 )) )");
|
|
|
|
good_expr("-15", "( 5 * (-3) )");
|
|
|
|
good_expr("-15", "+5*(-3)");
|
|
|
|
|
|
|
|
good_expr("2", "6/3");
|
|
|
|
good_expr("2", "5%3");
|
|
|
|
good_expr("8", "5+3");
|
|
|
|
good_expr("64", "4^3");
|
|
|
|
good_expr("64", "4 ^ 3");
|
|
|
|
good_expr("64", "4**3");
|
|
|
|
good_expr("-81", "-3^4");
|
|
|
|
good_expr("-81", "-(3^4)");
|
|
|
|
good_expr("0.5", "2^-1");
|
|
|
|
good_expr("0.25", "2^-2");
|
|
|
|
|
|
|
|
good_expr("2", "rt 4");
|
|
|
|
good_expr("2", "sqrt 4");
|
|
|
|
good_expr("6", "2 rt 9");
|
|
|
|
|
|
|
|
good_expr("7", "3!+1");
|
|
|
|
good_expr("18", "3!3");
|
2023-06-07 15:23:29 -07:00
|
|
|
good_expr("9.3326e157", "1e2!");
|
|
|
|
|
2023-04-07 18:16:22 -07:00
|
|
|
bad_expr("3.1!");
|
2023-06-07 15:23:29 -07:00
|
|
|
bad_expr("1e10!");
|
|
|
|
bad_expr("1e5!");
|
2023-06-07 15:21:36 -07:00
|
|
|
bad_expr("0^(-1)");
|
2023-04-07 18:16:22 -07:00
|
|
|
bad_expr("pi!");
|
2023-04-10 12:48:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2023-05-15 10:05:46 -07:00
|
|
|
fn basic_units() {
|
2023-04-10 12:48:36 -07:00
|
|
|
//good_expr("4 m*s", "2 m * 2s");
|
2023-04-10 14:02:01 -07:00
|
|
|
good_expr("1 s⁻¹", "1/s");
|
|
|
|
good_expr("6 kg", "2 * 3kg");
|
2023-04-10 12:48:36 -07:00
|
|
|
good_expr("1 m/s", "2 m / 2s");
|
|
|
|
good_expr("10 m", "10 m");
|
|
|
|
good_expr("10 m", "10 * m");
|
|
|
|
good_expr("12 m", "(2 + 10) m");
|
|
|
|
|
2023-06-14 12:38:26 -07:00
|
|
|
good_expr("2 m²", "2m^2");
|
|
|
|
good_expr("4 m²", "(2m)^2");
|
|
|
|
good_expr("2 m²", "2m * m");
|
|
|
|
good_expr("1 m²", "m m");
|
2023-04-10 12:48:36 -07:00
|
|
|
|
2023-06-07 09:18:08 -07:00
|
|
|
good_expr("2 m", "rt (4m^2)");
|
2023-04-10 12:48:36 -07:00
|
|
|
|
2023-04-10 18:47:00 -07:00
|
|
|
bad_expr("m + s");
|
|
|
|
bad_expr("m ^ s");
|
2023-04-10 14:12:16 -07:00
|
|
|
//bad_expr("m ^ pi");
|
2023-05-15 10:05:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn complex_units() {
|
|
|
|
good_expr("0.62137 mi", "1km to mi");
|
2023-06-07 09:18:08 -07:00
|
|
|
good_expr("3280.8 ft", "1km to ft");
|
2023-06-11 15:05:17 -07:00
|
|
|
good_expr("62.137 mi/h", "100 km/h to mph");
|
|
|
|
good_expr("20 mi", "10 mph * 2 hours");
|
2023-06-11 16:44:44 -07:00
|
|
|
good_expr("120 m", "1 (m/s) * 2 min");
|
2023-07-28 09:21:59 -07:00
|
|
|
good_expr("120 m", "(2 min) * (1 m/s)");
|
2023-07-28 10:33:51 -07:00
|
|
|
good_expr("180", "1 Hz * 3 min");
|
|
|
|
good_expr("3600", "1 hour * 1 Hz");
|
2023-08-01 10:01:32 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn functions() {
|
2023-08-02 09:51:36 -07:00
|
|
|
good_expr("1", "sin(90 deg)");
|
|
|
|
good_expr("-1", "cos(180 deg)");
|
|
|
|
good_expr("0.70711", "sin(pi/4)");
|
|
|
|
good_expr("0.70711", "sin((pi/4) r)");
|
|
|
|
|
2023-08-01 10:01:32 -07:00
|
|
|
good_expr("2", "nounit(2 mm)");
|
|
|
|
good_expr("2", "nounit(2 meter * second)");
|
|
|
|
//good_expr("5000 m²·g/(s²·A²)", "tobase(5H)");
|
2023-04-07 18:16:22 -07:00
|
|
|
}
|