2020-06-13 20:06:21 +02:00
|
|
|
use crate::ast::{Expr, Stmt};
|
2020-06-13 19:01:33 +02:00
|
|
|
use crate::lexer::TokenKind;
|
2020-06-13 22:18:37 +02:00
|
|
|
use crate::parser::CalcError;
|
2020-06-14 19:23:02 +02:00
|
|
|
use crate::parser::DECL_UNIT;
|
2020-06-18 16:20:18 +02:00
|
|
|
use crate::prelude;
|
2020-06-13 20:06:21 +02:00
|
|
|
use crate::symbol_table::SymbolTable;
|
2020-12-10 23:40:29 +01:00
|
|
|
use lazy_static::lazy_static;
|
|
|
|
use std::collections::HashMap;
|
2020-06-13 19:01:33 +02:00
|
|
|
|
2020-12-10 23:40:29 +01:00
|
|
|
lazy_static! {
|
|
|
|
pub static ref INVERSE_UNARY_FUNCS: HashMap<&'static str, &'static str> = {
|
|
|
|
let mut m = HashMap::new();
|
|
|
|
m.insert("cos", "acos");
|
|
|
|
m.insert("cosec", "acosec");
|
|
|
|
m.insert("cosech", "cosech");
|
|
|
|
m.insert("cosh", "acosh");
|
|
|
|
m.insert("cot", "acot");
|
|
|
|
m.insert("coth", "acoth");
|
|
|
|
m.insert("sec", "asec");
|
|
|
|
m.insert("sech", "asech");
|
|
|
|
m.insert("sin", "asin");
|
|
|
|
m.insert("sinh", "asinh");
|
|
|
|
m.insert("tan", "atan");
|
|
|
|
m.insert("tanh", "atanh");
|
2020-06-18 16:20:18 +02:00
|
|
|
|
2020-12-10 23:40:29 +01:00
|
|
|
m.insert("acos", "cos");
|
|
|
|
m.insert("acosec", "cosec");
|
|
|
|
m.insert("acosech", "cosech");
|
|
|
|
m.insert("acosh", "cosh");
|
|
|
|
m.insert("acot", "cot");
|
|
|
|
m.insert("acoth", "coth");
|
|
|
|
m.insert("asec", "sec");
|
|
|
|
m.insert("asech", "sech");
|
|
|
|
m.insert("asin", "sin");
|
|
|
|
m.insert("asinh", "sinh");
|
|
|
|
m.insert("atan", "tan");
|
|
|
|
m.insert("atanh", "tanh");
|
|
|
|
m
|
|
|
|
};
|
|
|
|
}
|
2020-06-18 16:20:18 +02:00
|
|
|
|
2020-06-13 19:01:33 +02:00
|
|
|
impl Expr {
|
2020-06-13 22:18:37 +02:00
|
|
|
pub fn invert(&self, symbol_table: &mut SymbolTable) -> Result<Self, CalcError> {
|
2020-06-14 19:23:02 +02:00
|
|
|
let target_expr = Expr::Var(DECL_UNIT.into());
|
|
|
|
let result = invert(target_expr, symbol_table, self);
|
2020-06-14 21:35:56 +02:00
|
|
|
|
2020-06-14 19:23:02 +02:00
|
|
|
Ok(result?.0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn invert(
|
|
|
|
target_expr: Expr,
|
|
|
|
symbol_table: &mut SymbolTable,
|
|
|
|
expr: &Expr,
|
|
|
|
) -> Result<(Expr, Expr), CalcError> {
|
|
|
|
match expr {
|
|
|
|
Expr::Binary(left, op, right) => {
|
|
|
|
invert_binary(target_expr, symbol_table, &left, op, &right)
|
|
|
|
}
|
2020-06-14 21:35:56 +02:00
|
|
|
Expr::Unary(op, expr) => invert_unary(target_expr, op, &expr),
|
2020-06-14 19:23:02 +02:00
|
|
|
Expr::Unit(identifier, expr) => invert_unit(target_expr, &identifier, &expr),
|
2020-06-17 17:45:46 +02:00
|
|
|
Expr::Var(identifier) => invert_var(target_expr, symbol_table, identifier),
|
2020-06-14 19:23:02 +02:00
|
|
|
Expr::Group(expr) => Ok((target_expr, *expr.clone())),
|
|
|
|
Expr::FnCall(identifier, arguments) => {
|
|
|
|
invert_fn_call(target_expr, symbol_table, &identifier, arguments)
|
2020-06-13 19:01:33 +02:00
|
|
|
}
|
2020-06-14 19:23:02 +02:00
|
|
|
Expr::Literal(_) => Ok((target_expr, expr.clone())),
|
2020-06-13 19:01:33 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-13 20:06:21 +02:00
|
|
|
fn invert_binary(
|
2020-06-14 19:23:02 +02:00
|
|
|
target_expr: Expr,
|
2020-06-13 20:06:21 +02:00
|
|
|
symbol_table: &mut SymbolTable,
|
|
|
|
left: &Expr,
|
|
|
|
op: &TokenKind,
|
|
|
|
right: &Expr,
|
2020-06-14 19:23:02 +02:00
|
|
|
) -> Result<(Expr, Expr), CalcError> {
|
2020-06-13 19:01:33 +02:00
|
|
|
let op_inv = match op {
|
|
|
|
TokenKind::Plus => TokenKind::Minus,
|
2020-06-14 21:35:56 +02:00
|
|
|
TokenKind::Minus => {
|
2020-06-14 21:54:39 +02:00
|
|
|
// Eg. a-(b+c)
|
|
|
|
// Multiply "-1" into the group, resulting in it becoming a normal expression. Then invert it normally.
|
2020-06-14 21:35:56 +02:00
|
|
|
if let Expr::Group(inside_group) = right {
|
|
|
|
return invert_binary(
|
|
|
|
target_expr,
|
|
|
|
symbol_table,
|
|
|
|
left,
|
2020-06-17 17:45:46 +02:00
|
|
|
&TokenKind::Plus,
|
2020-12-13 15:52:22 +01:00
|
|
|
&multiply_into(&Expr::Literal(-1f64), inside_group)?,
|
2020-06-14 21:35:56 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
TokenKind::Plus
|
|
|
|
}
|
2020-06-14 19:23:02 +02:00
|
|
|
TokenKind::Star => {
|
2020-06-14 21:54:39 +02:00
|
|
|
// If the left expression is a group, multiply the right expression into it, dissolving the group.
|
|
|
|
// It can then be inverted normally.
|
2020-06-14 19:23:02 +02:00
|
|
|
if let Expr::Group(inside_group) = left {
|
|
|
|
return invert(
|
|
|
|
target_expr,
|
|
|
|
symbol_table,
|
2020-06-15 21:27:47 +02:00
|
|
|
&multiply_into(right, inside_group)?,
|
2020-06-14 19:23:02 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-06-14 21:54:39 +02:00
|
|
|
// Same as above but left/right switched.
|
2020-06-14 19:23:02 +02:00
|
|
|
if let Expr::Group(inside_group) = right {
|
2020-06-15 21:27:47 +02:00
|
|
|
return invert(
|
|
|
|
target_expr,
|
|
|
|
symbol_table,
|
|
|
|
&multiply_into(left, inside_group)?,
|
|
|
|
);
|
2020-06-14 19:23:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
TokenKind::Slash
|
|
|
|
}
|
|
|
|
TokenKind::Slash => {
|
2020-06-14 21:54:39 +02:00
|
|
|
// Eg. (a+b)/c
|
|
|
|
// Just dissolve the group. Nothing more needs to be done mathematically.
|
2020-06-14 19:23:02 +02:00
|
|
|
if let Expr::Group(inside_group) = left {
|
|
|
|
return invert(
|
|
|
|
target_expr,
|
|
|
|
symbol_table,
|
|
|
|
&Expr::Binary(inside_group.clone(), op.clone(), Box::new(right.clone())),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-06-14 21:54:39 +02:00
|
|
|
// Eg. a/(b+c)
|
|
|
|
// Same as above.
|
2020-06-14 19:23:02 +02:00
|
|
|
if let Expr::Group(inside_group) = right {
|
|
|
|
return invert(
|
|
|
|
target_expr,
|
|
|
|
symbol_table,
|
|
|
|
&Expr::Binary(Box::new(left.clone()), op.clone(), inside_group.clone()),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
TokenKind::Star
|
|
|
|
}
|
2020-06-13 19:01:33 +02:00
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
|
2020-06-14 21:54:39 +02:00
|
|
|
// If the left expression contains the unit, invert the right one instead,
|
|
|
|
// since the unit should not be moved.
|
2020-06-17 17:45:46 +02:00
|
|
|
if contains_the_unit(symbol_table, left) {
|
2020-06-17 21:28:54 +02:00
|
|
|
// But if the right expression *also* contains the unit,
|
|
|
|
// throw an error, since it can't handle this yet.
|
|
|
|
if contains_the_unit(symbol_table, right) {
|
2020-06-18 16:20:18 +02:00
|
|
|
return Err(CalcError::UnableToInvert(String::from(
|
|
|
|
"Expressions with several instances of an unknown variable (this might be supported in the future). Try simplifying the expression.",
|
2020-06-17 21:28:54 +02:00
|
|
|
)));
|
|
|
|
}
|
|
|
|
|
2020-06-14 19:23:02 +02:00
|
|
|
return Ok(invert(
|
|
|
|
Expr::Binary(Box::new(target_expr), op_inv, Box::new(right.clone())),
|
|
|
|
symbol_table,
|
|
|
|
left,
|
|
|
|
)?);
|
|
|
|
}
|
2020-06-13 19:01:33 +02:00
|
|
|
|
2020-06-14 21:54:39 +02:00
|
|
|
// Otherwise, invert the left side.
|
2020-06-14 21:35:56 +02:00
|
|
|
let final_target_expr = Expr::Binary(Box::new(target_expr), op_inv, Box::new(left.clone()));
|
2020-06-14 19:23:02 +02:00
|
|
|
Ok(invert(
|
2020-06-14 21:54:39 +02:00
|
|
|
// Eg. 2-a
|
|
|
|
// If the operator is minus (and the left expression is being inverted),
|
|
|
|
// make the target expression negative to keep balance.
|
|
|
|
if let TokenKind::Minus = op {
|
2020-06-14 21:35:56 +02:00
|
|
|
Expr::Unary(TokenKind::Minus, Box::new(final_target_expr))
|
|
|
|
} else {
|
|
|
|
final_target_expr
|
|
|
|
},
|
2020-06-14 19:23:02 +02:00
|
|
|
symbol_table,
|
2020-06-14 21:54:39 +02:00
|
|
|
right, // Then invert the right expression.
|
2020-06-14 19:23:02 +02:00
|
|
|
)?)
|
2020-06-13 19:01:33 +02:00
|
|
|
}
|
|
|
|
|
2020-06-14 21:35:56 +02:00
|
|
|
fn invert_unary(target_expr: Expr, op: &TokenKind, expr: &Expr) -> Result<(Expr, Expr), CalcError> {
|
|
|
|
match op {
|
|
|
|
TokenKind::Minus => Ok((
|
2020-06-15 21:27:47 +02:00
|
|
|
// Make the target expression negative
|
2020-06-14 21:35:56 +02:00
|
|
|
Expr::Unary(TokenKind::Minus, Box::new(target_expr)),
|
2020-06-15 21:27:47 +02:00
|
|
|
expr.clone(), // And then continue inverting the inner-expression.
|
2020-06-14 21:35:56 +02:00
|
|
|
)),
|
|
|
|
_ => unimplemented!(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-14 19:23:02 +02:00
|
|
|
fn invert_unit(
|
|
|
|
_target_expr: Expr,
|
|
|
|
_identifier: &str,
|
|
|
|
_expr: &Expr,
|
|
|
|
) -> Result<(Expr, Expr), CalcError> {
|
2020-06-18 16:20:18 +02:00
|
|
|
Err(CalcError::UnableToInvert(String::from(
|
|
|
|
"Expressions containing other units (this should be supported in the future).",
|
2020-06-17 21:28:54 +02:00
|
|
|
)))
|
2020-06-13 19:01:33 +02:00
|
|
|
}
|
|
|
|
|
2020-06-17 17:45:46 +02:00
|
|
|
fn invert_var(
|
|
|
|
target_expr: Expr,
|
|
|
|
symbol_table: &mut SymbolTable,
|
|
|
|
identifier: &str,
|
|
|
|
) -> Result<(Expr, Expr), CalcError> {
|
2020-06-18 16:58:01 +02:00
|
|
|
if identifier == DECL_UNIT {
|
|
|
|
Ok((target_expr, Expr::Var(identifier.into())))
|
|
|
|
} else if let Some(Stmt::VarDecl(_, var_expr)) = symbol_table.get_var(identifier).cloned() {
|
2020-06-17 17:45:46 +02:00
|
|
|
invert(target_expr, symbol_table, &var_expr)
|
|
|
|
} else {
|
|
|
|
Ok((target_expr, Expr::Var(identifier.into())))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-13 22:18:37 +02:00
|
|
|
fn invert_fn_call(
|
2020-06-14 19:23:02 +02:00
|
|
|
target_expr: Expr,
|
2020-06-13 22:18:37 +02:00
|
|
|
symbol_table: &mut SymbolTable,
|
|
|
|
identifier: &str,
|
|
|
|
arguments: &Vec<Expr>,
|
2020-06-14 19:23:02 +02:00
|
|
|
) -> Result<(Expr, Expr), CalcError> {
|
2020-06-18 16:20:18 +02:00
|
|
|
// If prelude function
|
|
|
|
match arguments.len() {
|
|
|
|
1 => {
|
|
|
|
if prelude::UNARY_FUNCS.contains_key(identifier) {
|
|
|
|
if let Some(fn_inv) = INVERSE_UNARY_FUNCS.get(identifier) {
|
|
|
|
return Ok((
|
|
|
|
Expr::FnCall(fn_inv.to_string(), vec![target_expr]),
|
|
|
|
arguments[0].clone(),
|
|
|
|
));
|
|
|
|
} else {
|
|
|
|
match identifier {
|
|
|
|
"sqrt" => {
|
|
|
|
return Ok((
|
|
|
|
Expr::Binary(
|
|
|
|
Box::new(target_expr),
|
|
|
|
TokenKind::Power,
|
2020-12-13 15:52:22 +01:00
|
|
|
Box::new(Expr::Literal(2f64)),
|
2020-06-18 16:20:18 +02:00
|
|
|
),
|
|
|
|
arguments[0].clone(),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
return Err(CalcError::UnableToInvert(format!(
|
|
|
|
"Function '{}'",
|
|
|
|
identifier
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
2 => {
|
|
|
|
if prelude::BINARY_FUNCS.contains_key(identifier) {
|
|
|
|
return Err(CalcError::UnableToInvert(format!(
|
|
|
|
"Function '{}'",
|
|
|
|
identifier
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => (),
|
|
|
|
}
|
|
|
|
|
2020-06-15 21:27:47 +02:00
|
|
|
// Get the function definition from the symbol table.
|
2020-06-13 20:06:21 +02:00
|
|
|
let (parameters, body) =
|
|
|
|
if let Some(Stmt::FnDecl(_, parameters, body)) = symbol_table.get_fn(identifier).cloned() {
|
|
|
|
(parameters, body)
|
|
|
|
} else {
|
2020-06-13 22:18:37 +02:00
|
|
|
return Err(CalcError::UndefinedFn(identifier.into()));
|
2020-06-13 20:06:21 +02:00
|
|
|
};
|
|
|
|
|
2020-06-17 17:45:46 +02:00
|
|
|
// Make sure the input is valid.
|
2020-06-13 22:18:37 +02:00
|
|
|
if parameters.len() != arguments.len() {
|
|
|
|
return Err(CalcError::IncorrectAmountOfArguments(
|
|
|
|
parameters.len(),
|
|
|
|
identifier.into(),
|
|
|
|
arguments.len(),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2020-06-15 21:27:47 +02:00
|
|
|
// Make the parameters usable as variables inside the function.
|
2020-06-13 20:06:21 +02:00
|
|
|
let mut parameters_iter = parameters.iter();
|
|
|
|
for argument in arguments {
|
|
|
|
symbol_table.insert(Stmt::VarDecl(
|
|
|
|
parameters_iter.next().unwrap().to_string(),
|
|
|
|
Box::new(argument.clone()),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2020-06-15 21:27:47 +02:00
|
|
|
// Invert everything in the function body.
|
2020-06-14 19:23:02 +02:00
|
|
|
invert(target_expr, symbol_table, &body)
|
2020-06-13 19:01:33 +02:00
|
|
|
}
|
|
|
|
|
2020-06-17 17:45:46 +02:00
|
|
|
fn contains_the_unit(symbol_table: &SymbolTable, expr: &Expr) -> bool {
|
2020-06-15 21:27:47 +02:00
|
|
|
// Recursively scan the expression for the unit.
|
2020-06-14 19:23:02 +02:00
|
|
|
match expr {
|
2020-06-17 17:45:46 +02:00
|
|
|
Expr::Binary(left, _, right) => {
|
|
|
|
contains_the_unit(symbol_table, left) || contains_the_unit(symbol_table, right)
|
|
|
|
}
|
|
|
|
Expr::Unary(_, expr) => contains_the_unit(symbol_table, expr),
|
|
|
|
Expr::Unit(_, expr) => contains_the_unit(symbol_table, expr),
|
|
|
|
Expr::Var(identifier) => {
|
|
|
|
identifier == DECL_UNIT
|
|
|
|
|| if let Some(Stmt::VarDecl(_, var_expr)) = symbol_table.get_var(identifier) {
|
|
|
|
contains_the_unit(symbol_table, var_expr)
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Expr::Group(expr) => contains_the_unit(symbol_table, expr),
|
2020-06-14 19:23:02 +02:00
|
|
|
Expr::FnCall(_, args) => {
|
|
|
|
for arg in args {
|
2020-06-17 17:45:46 +02:00
|
|
|
if contains_the_unit(symbol_table, arg) {
|
2020-06-14 19:23:02 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
false
|
|
|
|
}
|
|
|
|
Expr::Literal(_) => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-15 21:27:47 +02:00
|
|
|
/// Multiply an expression into a group.
|
|
|
|
fn multiply_into(expr: &Expr, base_expr: &Expr) -> Result<Expr, CalcError> {
|
2020-06-14 19:23:02 +02:00
|
|
|
match base_expr {
|
|
|
|
Expr::Binary(left, op, right) => match op {
|
2020-06-15 21:27:47 +02:00
|
|
|
// If + or -, multiply the expression with each term.
|
2020-06-14 19:23:02 +02:00
|
|
|
TokenKind::Plus | TokenKind::Minus => Ok(Expr::Binary(
|
2020-06-15 21:27:47 +02:00
|
|
|
Box::new(multiply_into(expr, &left)?),
|
2020-06-14 19:23:02 +02:00
|
|
|
op.clone(),
|
2020-06-15 21:27:47 +02:00
|
|
|
Box::new(multiply_into(expr, &right)?),
|
2020-06-14 19:23:02 +02:00
|
|
|
)),
|
2020-06-15 21:27:47 +02:00
|
|
|
// If * or /, only multiply with the first factor.
|
2020-06-14 19:23:02 +02:00
|
|
|
TokenKind::Star | TokenKind::Slash => Ok(Expr::Binary(
|
2020-06-15 21:27:47 +02:00
|
|
|
Box::new(multiply_into(expr, &left)?),
|
2020-06-14 19:23:02 +02:00
|
|
|
op.clone(),
|
|
|
|
right.clone(),
|
|
|
|
)),
|
|
|
|
_ => unimplemented!(),
|
|
|
|
},
|
2020-06-15 21:27:47 +02:00
|
|
|
// If it's a literal, just multiply them together.
|
2020-06-14 19:23:02 +02:00
|
|
|
Expr::Literal(_) | Expr::Var(_) => Ok(Expr::Binary(
|
|
|
|
Box::new(expr.clone()),
|
|
|
|
TokenKind::Star,
|
|
|
|
Box::new(base_expr.clone()),
|
|
|
|
)),
|
2020-06-18 16:20:18 +02:00
|
|
|
Expr::Group(_) => Err(CalcError::UnableToInvert(String::from(
|
|
|
|
"Parenthesis multiplied with parenthesis (this should be possible in the future).",
|
2020-06-17 21:28:54 +02:00
|
|
|
))),
|
2020-06-14 19:23:02 +02:00
|
|
|
_ => unimplemented!(),
|
|
|
|
}
|
2020-06-13 19:01:33 +02:00
|
|
|
}
|
2020-06-17 17:45:46 +02:00
|
|
|
|
|
|
|
#[allow(unused_imports, dead_code)] // Getting warnings for some reason
|
|
|
|
mod tests {
|
|
|
|
use crate::ast::Expr;
|
|
|
|
use crate::lexer::TokenKind::*;
|
|
|
|
use crate::symbol_table::SymbolTable;
|
|
|
|
use crate::test_helpers::*;
|
|
|
|
|
|
|
|
fn decl_unit() -> Box<Expr> {
|
|
|
|
Box::new(Expr::Var(crate::parser::DECL_UNIT.into()))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_binary() {
|
2020-12-13 15:52:22 +01:00
|
|
|
let ladd = binary(decl_unit(), Plus, literal(1f64));
|
|
|
|
let lsub = binary(decl_unit(), Minus, literal(1f64));
|
|
|
|
let lmul = binary(decl_unit(), Star, literal(1f64));
|
|
|
|
let ldiv = binary(decl_unit(), Slash, literal(1f64));
|
2020-06-17 17:45:46 +02:00
|
|
|
|
2020-12-13 15:52:22 +01:00
|
|
|
let radd = binary(literal(1f64), Plus, decl_unit());
|
|
|
|
let rsub = binary(literal(1f64), Minus, decl_unit());
|
|
|
|
let rmul = binary(literal(1f64), Star, decl_unit());
|
|
|
|
let rdiv = binary(literal(1f64), Slash, decl_unit());
|
2020-06-17 17:45:46 +02:00
|
|
|
|
|
|
|
let mut symbol_table = SymbolTable::new();
|
|
|
|
assert_eq!(
|
|
|
|
ladd.invert(&mut symbol_table).unwrap(),
|
2020-12-13 15:52:22 +01:00
|
|
|
*binary(decl_unit(), Minus, literal(1f64))
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
lsub.invert(&mut symbol_table).unwrap(),
|
2020-12-13 15:52:22 +01:00
|
|
|
*binary(decl_unit(), Plus, literal(1f64))
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
lmul.invert(&mut symbol_table).unwrap(),
|
2020-12-13 15:52:22 +01:00
|
|
|
*binary(decl_unit(), Slash, literal(1f64))
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
ldiv.invert(&mut symbol_table).unwrap(),
|
2020-12-13 15:52:22 +01:00
|
|
|
*binary(decl_unit(), Star, literal(1f64))
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
radd.invert(&mut symbol_table).unwrap(),
|
2020-12-13 15:52:22 +01:00
|
|
|
*binary(decl_unit(), Minus, literal(1f64))
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
rsub.invert(&mut symbol_table).unwrap(),
|
2020-12-13 15:52:22 +01:00
|
|
|
*unary(Minus, binary(decl_unit(), Plus, literal(1f64)))
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
rmul.invert(&mut symbol_table).unwrap(),
|
2020-12-13 15:52:22 +01:00
|
|
|
*binary(decl_unit(), Slash, literal(1f64))
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
rdiv.invert(&mut symbol_table).unwrap(),
|
2020-12-13 15:52:22 +01:00
|
|
|
*binary(decl_unit(), Star, literal(1f64))
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_unary() {
|
|
|
|
let neg = unary(Minus, decl_unit());
|
|
|
|
|
|
|
|
let mut symbol_table = SymbolTable::new();
|
|
|
|
assert_eq!(neg.invert(&mut symbol_table).unwrap(), *neg);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_fn_call() {
|
2020-12-13 15:52:22 +01:00
|
|
|
let call_with_literal = binary(fn_call("f", vec![*literal(2f64)]), Plus, decl_unit());
|
2020-06-17 17:45:46 +02:00
|
|
|
let call_with_decl_unit = fn_call("f", vec![*decl_unit()]);
|
|
|
|
let call_with_decl_unit_and_literal =
|
2020-12-13 15:52:22 +01:00
|
|
|
fn_call("f", vec![*binary(decl_unit(), Plus, literal(2f64))]);
|
2020-06-17 17:45:46 +02:00
|
|
|
let decl = fn_decl(
|
|
|
|
"f",
|
|
|
|
vec![String::from("x")],
|
2020-12-13 15:52:22 +01:00
|
|
|
binary(var("x"), Plus, literal(1f64)),
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
let mut symbol_table = SymbolTable::new();
|
|
|
|
symbol_table.insert(decl);
|
|
|
|
assert_eq!(
|
|
|
|
call_with_literal.invert(&mut symbol_table).unwrap(),
|
2020-12-13 15:52:22 +01:00
|
|
|
*binary(decl_unit(), Minus, fn_call("f", vec![*literal(2f64)])),
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
call_with_decl_unit.invert(&mut symbol_table).unwrap(),
|
2020-12-13 15:52:22 +01:00
|
|
|
*binary(decl_unit(), Minus, literal(1f64))
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
call_with_decl_unit_and_literal
|
|
|
|
.invert(&mut symbol_table)
|
|
|
|
.unwrap(),
|
|
|
|
*binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
binary(decl_unit(), Minus, literal(1f64)),
|
2020-06-17 17:45:46 +02:00
|
|
|
Minus,
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(2f64)
|
2020-06-17 17:45:46 +02:00
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_group() {
|
|
|
|
let group_x = binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
group(binary(decl_unit(), Plus, literal(3f64))),
|
2020-06-17 17:45:46 +02:00
|
|
|
Star,
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(2f64),
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
let group_unary_minus = binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(2f64),
|
2020-06-17 17:45:46 +02:00
|
|
|
Minus,
|
2020-12-13 15:52:22 +01:00
|
|
|
group(binary(decl_unit(), Plus, literal(3f64))),
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
let x_group_add = binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(2f64),
|
2020-06-17 17:45:46 +02:00
|
|
|
Star,
|
2020-12-13 15:52:22 +01:00
|
|
|
group(binary(decl_unit(), Plus, literal(3f64))),
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
let x_group_sub = binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(2f64),
|
2020-06-17 17:45:46 +02:00
|
|
|
Star,
|
2020-12-13 15:52:22 +01:00
|
|
|
group(binary(decl_unit(), Minus, literal(3f64))),
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
let x_group_mul = binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(2f64),
|
2020-06-17 17:45:46 +02:00
|
|
|
Star,
|
2020-12-13 15:52:22 +01:00
|
|
|
group(binary(decl_unit(), Star, literal(3f64))),
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
let x_group_div = binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(2f64),
|
2020-06-17 17:45:46 +02:00
|
|
|
Star,
|
2020-12-13 15:52:22 +01:00
|
|
|
group(binary(decl_unit(), Slash, literal(3f64))),
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
let mut symbol_table = SymbolTable::new();
|
|
|
|
assert_eq!(
|
|
|
|
group_x.invert(&mut symbol_table).unwrap(),
|
|
|
|
*binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
binary(
|
|
|
|
decl_unit(),
|
|
|
|
Minus,
|
|
|
|
binary(literal(2f64), Star, literal(3f64))
|
|
|
|
),
|
2020-06-17 17:45:46 +02:00
|
|
|
Slash,
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(2f64)
|
2020-06-17 17:45:46 +02:00
|
|
|
)
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
group_unary_minus.invert(&mut symbol_table).unwrap(),
|
|
|
|
*binary(
|
|
|
|
binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
binary(decl_unit(), Minus, literal(2f64)),
|
2020-06-17 17:45:46 +02:00
|
|
|
Minus,
|
2020-12-13 15:52:22 +01:00
|
|
|
binary(literal(-1f64), Star, literal(3f64))
|
2020-06-17 17:45:46 +02:00
|
|
|
),
|
|
|
|
Slash,
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(-1f64)
|
2020-06-17 17:45:46 +02:00
|
|
|
)
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
x_group_add.invert(&mut symbol_table).unwrap(),
|
|
|
|
*binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
binary(
|
|
|
|
decl_unit(),
|
|
|
|
Minus,
|
|
|
|
binary(literal(2f64), Star, literal(3f64))
|
|
|
|
),
|
2020-06-17 17:45:46 +02:00
|
|
|
Slash,
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(2f64)
|
2020-06-17 17:45:46 +02:00
|
|
|
)
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
x_group_sub.invert(&mut symbol_table).unwrap(),
|
|
|
|
*binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
binary(
|
|
|
|
decl_unit(),
|
|
|
|
Plus,
|
|
|
|
binary(literal(2f64), Star, literal(3f64))
|
|
|
|
),
|
2020-06-17 17:45:46 +02:00
|
|
|
Slash,
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(2f64)
|
2020-06-17 17:45:46 +02:00
|
|
|
)
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
x_group_mul.invert(&mut symbol_table).unwrap(),
|
|
|
|
*binary(
|
2020-12-13 15:52:22 +01:00
|
|
|
binary(decl_unit(), Slash, literal(3f64)),
|
2020-06-17 17:45:46 +02:00
|
|
|
Slash,
|
2020-12-13 15:52:22 +01:00
|
|
|
literal(2f64)
|
2020-06-17 17:45:46 +02:00
|
|
|
)
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
x_group_div.invert(&mut symbol_table).unwrap(),
|
2020-12-13 15:52:22 +01:00
|
|
|
*binary(
|
|
|
|
binary(decl_unit(), Star, literal(3f64)),
|
|
|
|
Slash,
|
|
|
|
literal(2f64)
|
|
|
|
)
|
2020-06-17 17:45:46 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_multiple_decl_units() {
|
2020-06-18 02:02:48 +02:00
|
|
|
/*let add_two = binary(decl_unit(), Plus, decl_unit());
|
2020-06-17 17:45:46 +02:00
|
|
|
|
|
|
|
let mut symbol_table = SymbolTable::new();
|
|
|
|
assert_eq!(
|
|
|
|
add_two.invert(&mut symbol_table).unwrap(),
|
|
|
|
*binary(decl_unit(), Slash, literal("2"))
|
2020-06-18 02:02:48 +02:00
|
|
|
);*/
|
2020-06-17 17:45:46 +02:00
|
|
|
}
|
|
|
|
}
|