use chrono::{Datelike, Duration, NaiveDate};
use crate::eval::coercion::to_number;
use crate::eval::functions::check_arity;
use crate::eval::functions::date::serial::date_to_serial;
use crate::types::{ErrorKind, Value};
pub fn date_fn(args: &[Value]) -> Value {
if let Some(err) = check_arity(args, 3, 3) {
return err;
}
let year = match to_number(args[0].clone()) { Ok(n) => n, Err(e) => return e };
let month = match to_number(args[1].clone()) { Ok(n) => n, Err(e) => return e };
let day = match to_number(args[2].clone()) { Ok(n) => n, Err(e) => return e };
let mut year_i = year.trunc() as i64;
let month_i = month.trunc() as i64;
let day_i = day.trunc() as i64;
let month_adj = month_i - 1; let year_delta = month_adj.div_euclid(12);
let month_norm = (month_adj.rem_euclid(12) + 1) as u32; year_i += year_delta;
if !(0..=9999).contains(&year_i) {
return Value::Error(ErrorKind::Num);
}
let base = match NaiveDate::from_ymd_opt(year_i as i32, month_norm, 1) {
Some(d) => d,
None => return Value::Error(ErrorKind::Num),
};
let date = match base.checked_add_signed(Duration::days(day_i - 1)) {
Some(d) => d,
None => return Value::Error(ErrorKind::Num),
};
let serial = date_to_serial(date);
if serial < 0.0 || date.year() > 9999 {
return Value::Error(ErrorKind::Num);
}
Value::Date(serial)
}
#[cfg(test)]
mod tests;