use crate::ast::{Expr, Ref};
use crate::builtins;
use crate::builtins::utils::{ensure_args_count, ensure_numeric, ensure_string};
use crate::lexer::Span;
use crate::value::Value;
use std::collections::HashMap;
use anyhow::{anyhow, bail, Result};
use chrono::{
DateTime, Datelike, Days, FixedOffset, Local, Months, NaiveDateTime, SecondsFormat, TimeZone,
Timelike, Utc, Weekday,
};
use chrono_tz::Tz;
pub(in crate::builtins) mod compat;
mod diff;
pub fn register(m: &mut HashMap<&'static str, builtins::BuiltinFcn>) {
m.insert("time.add_date", (add_date, 4));
m.insert("time.clock", (clock, 1));
m.insert("time.date", (date, 1));
m.insert("time.diff", (diff, 2));
m.insert("time.format", (format, 1));
m.insert("time.now_ns", (now_ns, 0));
m.insert("time.parse_duration_ns", (parse_duration_ns, 1));
m.insert("time.parse_ns", (parse_ns, 2));
m.insert("time.parse_rfc3339_ns", (parse_rfc3339_ns, 1));
m.insert("time.weekday", (weekday, 1));
}
fn add_date(span: &Span, params: &[Ref<Expr>], args: &[Value], strict: bool) -> Result<Value> {
let name = "time.add_date";
ensure_args_count(span, name, params, args, 4)?;
let (datetime, _) = parse_epoch(name, ¶ms[0], &args[0])?;
let years = ensure_i32(name, ¶ms[1], &args[1])?;
let months = ensure_i32(name, ¶ms[2], &args[2])?;
let days = ensure_i32(name, ¶ms[3], &args[3])?;
let Some(new_year) = datetime.year().checked_add(years) else {
return Ok(Value::Undefined);
};
datetime
.with_year(new_year)
.and_then(|d| {
let rhs = Months::new(months.unsigned_abs());
if months >= 0 {
d.checked_add_months(rhs)
} else {
d.checked_sub_months(rhs)
}
})
.and_then(|d| {
let rhs = Days::new(days.unsigned_abs() as u64);
if days >= 0 {
d.checked_add_days(rhs)
} else {
d.checked_sub_days(rhs)
}
})
.map_or(Ok(Value::Undefined), |d| {
safe_timestamp_nanos(span, strict, d.timestamp_nanos_opt())
})
}
fn clock(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "time.clock";
ensure_args_count(span, name, params, args, 1)?;
let (datetime, _) = parse_epoch(name, ¶ms[0], &args[0])?;
Ok(Vec::from([
(datetime.hour() as u64).into(),
(datetime.minute() as u64).into(),
(datetime.second() as u64).into(),
])
.into())
}
fn date(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "time.date";
ensure_args_count(span, name, params, args, 1)?;
let (datetime, _) = parse_epoch(name, ¶ms[0], &args[0])?;
Ok(Vec::from([
(datetime.year() as u64).into(),
(datetime.month() as u64).into(),
(datetime.day() as u64).into(),
])
.into())
}
fn diff(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "time.diff";
ensure_args_count(span, name, params, args, 2)?;
let (datetime1, _) = parse_epoch(name, ¶ms[0], &args[0])?;
let (datetime2, _) = parse_epoch(name, ¶ms[1], &args[1])?;
let (year, month, day, hour, min, sec) = diff::diff_between_datetimes(datetime1, datetime2)?;
Ok(Vec::from([
(year as i64).into(),
(month as i64).into(),
(day as i64).into(),
(hour as i64).into(),
(min as i64).into(),
(sec as i64).into(),
])
.into())
}
fn format(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "time.format";
ensure_args_count(span, name, params, args, 1)?;
let (datetime, format) = parse_epoch(name, ¶ms[0], &args[0])?;
let result = match format {
Some(format) => datetime.format(&format).to_string(),
None => datetime.to_rfc3339_opts(SecondsFormat::AutoSi, true),
};
Ok(Value::String(result.into()))
}
fn now_ns(span: &Span, params: &[Ref<Expr>], args: &[Value], strict: bool) -> Result<Value> {
let name = "time.now_ns";
ensure_args_count(span, name, params, args, 0)?;
safe_timestamp_nanos(span, strict, Utc::now().timestamp_nanos_opt())
}
fn parse_duration_ns(
span: &Span,
params: &[Ref<Expr>],
args: &[Value],
strict: bool,
) -> Result<Value> {
let name = "time.parse_duration_ns";
ensure_args_count(span, name, params, args, 1)?;
let value = ensure_string(name, ¶ms[0], &args[0])?;
let dur = compat::parse_duration(value.as_ref())?;
safe_timestamp_nanos(span, strict, dur.num_nanoseconds())
}
fn parse_ns(span: &Span, params: &[Ref<Expr>], args: &[Value], strict: bool) -> Result<Value> {
let name = "time.parse_ns";
ensure_args_count(span, name, params, args, 2)?;
let layout = ensure_string(name, ¶ms[0], &args[0])?;
let value = ensure_string(name, ¶ms[1], &args[1])?;
let datetime = NaiveDateTime::parse_from_str(&value, &layout)?;
safe_timestamp_nanos(span, strict, datetime.timestamp_nanos_opt())
}
fn parse_rfc3339_ns(
span: &Span,
params: &[Ref<Expr>],
args: &[Value],
strict: bool,
) -> Result<Value> {
let name = "time.parse_rfc3339_ns";
ensure_args_count(span, name, params, args, 1)?;
let value = ensure_string(name, ¶ms[0], &args[0])?;
let datetime = DateTime::parse_from_rfc3339(&value)?;
safe_timestamp_nanos(span, strict, datetime.timestamp_nanos_opt())
}
fn weekday(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "time.weekday";
ensure_args_count(span, name, params, args, 1)?;
let (datetime, _) = parse_epoch(name, ¶ms[0], &args[0])?;
let weekday = match datetime.weekday() {
Weekday::Mon => "Monday",
Weekday::Tue => "Tuesday",
Weekday::Wed => "Wednesday",
Weekday::Thu => "Thursday",
Weekday::Fri => "Friday",
Weekday::Sat => "Saturday",
Weekday::Sun => "Sunday",
};
Ok(Value::String(weekday.into()))
}
fn ensure_i32(name: &str, arg: &Expr, v: &Value) -> Result<i32> {
ensure_numeric(name, arg, v)?
.as_i64()
.and_then(|n| n.try_into().ok())
.ok_or_else(|| arg.span().error("could not convert to int32"))
}
fn safe_timestamp_nanos(span: &Span, strict: bool, nanos: Option<i64>) -> Result<Value> {
match nanos {
Some(ns) => Ok(Value::Number(ns.into())),
None if strict => {
bail!(span.error("time outside of valid range"))
}
None => Ok(Value::Undefined),
}
}
fn parse_epoch(
fcn: &str,
arg: &Expr,
val: &Value,
) -> Result<(DateTime<FixedOffset>, Option<String>)> {
match val {
Value::Number(num) => {
let ns = num.as_i64().ok_or_else(|| {
arg.span()
.error("could not convert numeric value of `ns` to int64")
})?;
return Ok((Utc.timestamp_nanos(ns).fixed_offset(), None));
}
Value::Array(arr) => match arr.as_slice() {
[Value::Number(num)] => {
let ns = num.as_i64().ok_or_else(|| {
arg.span()
.error("could not convert numeric value of `ns` to int64")
})?;
return Ok((Utc.timestamp_nanos(ns).fixed_offset(), None));
}
[Value::Number(num), Value::String(tz), rest @ ..] => {
let ns = num.as_i64().ok_or_else(|| {
arg.span()
.error("could not convert numeric value of `ns` to int64")
})?;
let datetime = match tz.as_ref() {
"UTC" | "" => Utc.timestamp_nanos(ns).fixed_offset(),
"Local" => Local.timestamp_nanos(ns).fixed_offset(),
_ => {
let tz: Tz = tz.parse().map_err(|err: String| anyhow!(err))?;
tz.timestamp_nanos(ns).fixed_offset()
}
};
let format = match rest.first() {
Some(Value::String(format)) => Some(format.to_string()),
Some(other) => {
bail!(arg.span().error(&format!(
"`{fcn}` expects 3rd element of `ns` to be a `string`. Got `{other}` instead"
)))
}
None => None,
};
return Ok((datetime, format));
}
_ => {}
},
_ => {}
}
bail!(arg.span().error(&format!(
"`{fcn}` expects `ns` to be a `number` or `array[number, string]`. Got `{val}` instead"
)))
}