use crate::core::Value;
use crate::functions::{
AggregateFunction, FunctionDataType, FunctionInfo, FunctionSignature, FunctionType,
};
#[derive(Default)]
pub struct MaxFunction {
max_value: Option<Value>,
}
impl AggregateFunction for MaxFunction {
fn name(&self) -> &str {
"MAX"
}
fn info(&self) -> FunctionInfo {
FunctionInfo::new(
"MAX",
FunctionType::Aggregate,
"Returns the maximum value of all non-NULL values in the specified column",
FunctionSignature::new(
FunctionDataType::Any, vec![FunctionDataType::Any],
1,
1,
),
)
}
fn accumulate(&mut self, value: &Value, _distinct: bool) {
if value.is_null() {
return;
}
if self.max_value.is_none() {
self.max_value = Some(value.clone());
return;
}
if is_greater_than(value, self.max_value.as_ref().unwrap()) {
self.max_value = Some(value.clone());
}
}
fn result(&self) -> Value {
self.max_value.clone().unwrap_or_else(Value::null_unknown)
}
fn reset(&mut self) {
self.max_value = None;
}
fn clone_box(&self) -> Box<dyn AggregateFunction> {
Box::new(MaxFunction::default())
}
}
fn is_greater_than(a: &Value, b: &Value) -> bool {
match (a, b) {
(Value::Null(_), _) | (_, Value::Null(_)) => false,
(Value::Integer(a), Value::Integer(b)) => a > b,
(Value::Float(a), Value::Float(b)) => a > b,
(Value::Integer(a), Value::Float(b)) => (*a as f64) > *b,
(Value::Float(a), Value::Integer(b)) => *a > (*b as f64),
(Value::Text(a), Value::Text(b)) => a > b,
(Value::Boolean(a), Value::Boolean(b)) => *a && !b, (Value::Timestamp(a), Value::Timestamp(b)) => a > b,
_ => false, }
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_max_integers() {
let mut max = MaxFunction::default();
max.accumulate(&Value::Integer(5), false);
max.accumulate(&Value::Integer(2), false);
max.accumulate(&Value::Integer(8), false);
assert_eq!(max.result(), Value::Integer(8));
}
#[test]
fn test_max_floats() {
let mut max = MaxFunction::default();
max.accumulate(&Value::Float(5.5), false);
max.accumulate(&Value::Float(2.2), false);
max.accumulate(&Value::Float(8.8), false);
assert_eq!(max.result(), Value::Float(8.8));
}
#[test]
fn test_max_strings() {
let mut max = MaxFunction::default();
max.accumulate(&Value::text("banana"), false);
max.accumulate(&Value::text("apple"), false);
max.accumulate(&Value::text("cherry"), false);
assert_eq!(max.result(), Value::text("cherry"));
}
#[test]
fn test_max_ignores_null() {
let mut max = MaxFunction::default();
max.accumulate(&Value::Integer(5), false);
max.accumulate(&Value::null_unknown(), false);
max.accumulate(&Value::Integer(2), false);
assert_eq!(max.result(), Value::Integer(5));
}
#[test]
fn test_max_empty() {
let max = MaxFunction::default();
assert!(max.result().is_null());
}
#[test]
fn test_max_reset() {
let mut max = MaxFunction::default();
max.accumulate(&Value::Integer(5), false);
max.accumulate(&Value::Integer(8), false);
max.reset();
assert!(max.result().is_null());
}
#[test]
fn test_max_single_value() {
let mut max = MaxFunction::default();
max.accumulate(&Value::Integer(42), false);
assert_eq!(max.result(), Value::Integer(42));
}
#[test]
fn test_max_negative() {
let mut max = MaxFunction::default();
max.accumulate(&Value::Integer(-5), false);
max.accumulate(&Value::Integer(-10), false);
max.accumulate(&Value::Integer(-3), false);
assert_eq!(max.result(), Value::Integer(-3));
}
#[test]
fn test_max_booleans() {
let mut max = MaxFunction::default();
max.accumulate(&Value::Boolean(false), false);
max.accumulate(&Value::Boolean(true), false);
max.accumulate(&Value::Boolean(false), false);
assert_eq!(max.result(), Value::Boolean(true));
}
}