use crate::eval::coercion::to_number;
use crate::eval::functions::check_arity;
use crate::types::{ErrorKind, Value};
fn combinations(n: u64, k: u64) -> f64 {
if k == 0 {
return 1.0;
}
if k > n {
return 0.0;
}
let k = k.min(n - k);
let mut result = 1.0f64;
for i in 0..k {
result *= (n - i) as f64;
result /= (i + 1) as f64;
}
result
}
pub fn combina_fn(args: &[Value]) -> Value {
if let Some(err) = check_arity(args, 2, 2) {
return err;
}
let n = match to_number(args[0].clone()) {
Err(e) => return e,
Ok(v) => v,
};
let k = match to_number(args[1].clone()) {
Err(e) => return e,
Ok(v) => v,
};
if n < 0.0 || k < 0.0 {
return Value::Error(ErrorKind::Num);
}
let n_int = n.trunc() as u64;
let k_int = k.trunc() as u64;
if n_int == 0 && k_int > 0 {
return Value::Number(0.0);
}
if k_int == 0 {
return Value::Number(1.0);
}
Value::Number(combinations(n_int + k_int - 1, k_int))
}
#[cfg(test)]
mod tests;