use crate::eval::functions::check_arity;
use crate::types::{ErrorKind, Value};
pub fn percentile_inc_fn(args: &[Value]) -> Value {
if let Some(err) = check_arity(args, 2, 2) {
return err;
}
let k = match &args[1] {
Value::Number(n) => *n,
_ => return Value::Error(ErrorKind::Num),
};
if !(0.0..=1.0).contains(&k) {
return Value::Error(ErrorKind::Num);
}
let mut nums = collect_numbers(&args[0]);
if nums.is_empty() {
return Value::Error(ErrorKind::Num);
}
nums.sort_by(|a, b| a.partial_cmp(b).unwrap());
Value::Number(percentile_inc_calc(&nums, k))
}
pub fn percentile_inc_calc(sorted: &[f64], k: f64) -> f64 {
let n = sorted.len();
if n == 1 {
return sorted[0];
}
let idx = k * (n - 1) as f64;
let lo = idx.floor() as usize;
let hi = idx.ceil() as usize;
if lo == hi {
sorted[lo]
} else {
let frac = idx - lo as f64;
sorted[lo] * (1.0 - frac) + sorted[hi] * frac
}
}
pub fn collect_numbers(v: &Value) -> Vec<f64> {
match v {
Value::Array(arr) => arr.iter().filter_map(|x| {
if let Value::Number(n) = x { Some(*n) } else { None }
}).collect(),
Value::Number(n) => vec![*n],
_ => vec![],
}
}
#[cfg(test)]
mod tests;