use std::sync::Arc;
use drasi_query_ast::ast;
use crate::evaluation::context::QueryVariables;
use crate::evaluation::functions::numeric::Abs;
use crate::evaluation::functions::ScalarFunction;
use crate::evaluation::variable_value::VariableValue;
use crate::evaluation::{
ExpressionEvaluationContext, FunctionError, FunctionEvaluationError, InstantQueryClock,
};
#[tokio::test]
async fn abs_too_few_args() {
let abs = Abs {};
let binding = QueryVariables::new();
let context =
ExpressionEvaluationContext::new(&binding, Arc::new(InstantQueryClock::new(0, 0)));
let args = vec![];
let func_expr = ast::FunctionExpression {
name: Arc::from("abs"),
args: vec![ast::Expression::UnaryExpression(
ast::UnaryExpression::Literal(ast::Literal::Integer(0)),
)],
position_in_query: 10,
};
let result = abs.call(&context, &func_expr, args.clone()).await;
assert!(matches!(
result.unwrap_err(),
FunctionError {
function_name: _,
error: FunctionEvaluationError::InvalidArgumentCount
}
));
}
#[tokio::test]
async fn abs_too_many_args() {
let abs = Abs {};
let binding = QueryVariables::new();
let context =
ExpressionEvaluationContext::new(&binding, Arc::new(InstantQueryClock::new(0, 0)));
let args = vec![
VariableValue::Integer(0.into()),
VariableValue::Integer(0.into()),
];
let func_expr = ast::FunctionExpression {
name: Arc::from("abs"),
args: vec![ast::Expression::UnaryExpression(
ast::UnaryExpression::Literal(ast::Literal::Integer(0)),
)],
position_in_query: 10,
};
let result = abs.call(&context, &func_expr, args.clone()).await;
assert!(matches!(
result.unwrap_err(),
FunctionError {
function_name: _,
error: FunctionEvaluationError::InvalidArgumentCount
}
));
}
#[tokio::test]
async fn abs_null() {
let abs = Abs {};
let binding = QueryVariables::new();
let context =
ExpressionEvaluationContext::new(&binding, Arc::new(InstantQueryClock::new(0, 0)));
let args = vec![VariableValue::Null];
let func_expr = ast::FunctionExpression {
name: Arc::from("abs"),
args: vec![ast::Expression::UnaryExpression(
ast::UnaryExpression::Literal(ast::Literal::Integer(0)),
)],
position_in_query: 10,
};
let result = abs.call(&context, &func_expr, args.clone()).await.unwrap();
assert_eq!(result, VariableValue::Null);
}
#[tokio::test]
async fn abs_zero_int() {
let abs = Abs {};
let binding = QueryVariables::new();
let context =
ExpressionEvaluationContext::new(&binding, Arc::new(InstantQueryClock::new(0, 0)));
let args = vec![VariableValue::Integer(0.into())];
let func_expr = ast::FunctionExpression {
name: Arc::from("abs"),
args: vec![ast::Expression::UnaryExpression(
ast::UnaryExpression::Literal(ast::Literal::Integer(0)),
)],
position_in_query: 10,
};
let result = abs.call(&context, &func_expr, args.clone()).await.unwrap();
assert_eq!(result, 0);
}
#[tokio::test]
async fn abs_negative_int() {
let abs = Abs {};
let binding = QueryVariables::new();
let context =
ExpressionEvaluationContext::new(&binding, Arc::new(InstantQueryClock::new(0, 0)));
let args = vec![VariableValue::Integer((-1).into())];
let func_expr = ast::FunctionExpression {
name: Arc::from("abs"),
args: vec![ast::Expression::UnaryExpression(
ast::UnaryExpression::Literal(ast::Literal::Integer(0)),
)],
position_in_query: 10,
};
let result = abs.call(&context, &func_expr, args.clone()).await.unwrap();
assert_eq!(result, 1);
}
#[tokio::test]
async fn abs_positive_int() {
let abs = Abs {};
let binding = QueryVariables::new();
let context =
ExpressionEvaluationContext::new(&binding, Arc::new(InstantQueryClock::new(0, 0)));
let args = vec![VariableValue::Integer(1.into())];
let func_expr = ast::FunctionExpression {
name: Arc::from("abs"),
args: vec![ast::Expression::UnaryExpression(
ast::UnaryExpression::Literal(ast::Literal::Integer(0)),
)],
position_in_query: 10,
};
let result = abs.call(&context, &func_expr, args.clone()).await.unwrap();
assert_eq!(result, 1);
}
#[tokio::test]
async fn abs_zero_float() {
let abs = Abs {};
let binding = QueryVariables::new();
let context =
ExpressionEvaluationContext::new(&binding, Arc::new(InstantQueryClock::new(0, 0)));
let args = vec![VariableValue::Float((0_f64).into())];
let func_expr = ast::FunctionExpression {
name: Arc::from("abs"),
args: vec![ast::Expression::UnaryExpression(
ast::UnaryExpression::Literal(ast::Literal::Integer(0)),
)],
position_in_query: 10,
};
let result = abs.call(&context, &func_expr, args.clone()).await.unwrap();
assert_eq!(result, 0_f64);
}
#[tokio::test]
async fn abs_negative_float() {
let abs = Abs {};
let binding = QueryVariables::new();
let context =
ExpressionEvaluationContext::new(&binding, Arc::new(InstantQueryClock::new(0, 0)));
let args = vec![VariableValue::Float((-1.001).into())];
let func_expr = ast::FunctionExpression {
name: Arc::from("abs"),
args: vec![ast::Expression::UnaryExpression(
ast::UnaryExpression::Literal(ast::Literal::Integer(0)),
)],
position_in_query: 10,
};
let result = abs.call(&context, &func_expr, args.clone()).await.unwrap();
assert_eq!(result, 1.001_f64);
}
#[tokio::test]
async fn abs_positive_float() {
let abs = Abs {};
let binding = QueryVariables::new();
let context =
ExpressionEvaluationContext::new(&binding, Arc::new(InstantQueryClock::new(0, 0)));
let args = vec![VariableValue::Float((1.001).into())];
let func_expr = ast::FunctionExpression {
name: Arc::from("abs"),
args: vec![ast::Expression::UnaryExpression(
ast::UnaryExpression::Literal(ast::Literal::Integer(0)),
)],
position_in_query: 10,
};
let result = abs.call(&context, &func_expr, args.clone()).await.unwrap();
assert_eq!(result, 1.001_f64);
}