use reifydb_core::value::column::columns::Columns;
use reifydb_runtime::context::clock::Clock;
use reifydb_transaction::transaction::Transaction;
use reifydb_type::{
fragment::Fragment,
params::Params,
value::{Value, datetime::DateTime, r#type::Type},
};
use super::set::extract_millis;
use crate::procedure::{Procedure, context::ProcedureContext, error::ProcedureError};
pub struct ClockAdvanceProcedure;
impl Default for ClockAdvanceProcedure {
fn default() -> Self {
Self::new()
}
}
impl ClockAdvanceProcedure {
pub fn new() -> Self {
Self
}
}
impl Procedure for ClockAdvanceProcedure {
fn call(&self, ctx: &ProcedureContext, _tx: &mut Transaction<'_>) -> Result<Columns, ProcedureError> {
let arg = match ctx.params {
Params::Positional(args) if args.len() == 1 => &args[0],
Params::Positional(args) => {
return Err(ProcedureError::ArityMismatch {
procedure: Fragment::internal("clock::advance"),
expected: 1,
actual: args.len(),
});
}
_ => {
return Err(ProcedureError::ArityMismatch {
procedure: Fragment::internal("clock::advance"),
expected: 1,
actual: 0,
});
}
};
match &ctx.runtime_context.clock {
Clock::Mock(mock) => {
match arg {
Value::Duration(dur) => {
if dur.get_months() == 0 && dur.get_days() == 0 {
let nanos = dur.get_nanos();
if nanos >= 0 {
mock.advance_nanos(nanos as u64);
} else {
let current = mock.now_nanos();
let abs_nanos = nanos.unsigned_abs();
if abs_nanos > current {
return Err(ProcedureError::ExecutionFailed {
procedure: Fragment::internal("clock::advance"),
reason: "clock cannot be set before Unix epoch".to_string(),
});
}
mock.set_nanos(current - abs_nanos);
}
} else {
let current_nanos = mock.now_nanos();
let current_dt = DateTime::from_nanos(current_nanos);
let new_dt = current_dt.add_duration(dur)?;
mock.set_nanos(new_dt.to_nanos());
}
}
other => {
let millis = extract_millis(other).ok_or_else(|| {
ProcedureError::InvalidArgumentType {
procedure: Fragment::internal("clock::advance"),
argument_index: 0,
expected: EXPECTED_ADVANCE_TYPES.to_vec(),
actual: other.get_type(),
}
})?;
mock.advance_millis(millis);
}
}
let current_nanos = mock.now_nanos();
let dt = DateTime::from_nanos(current_nanos);
Ok(Columns::single_row([("clock", Value::DateTime(dt))]))
}
Clock::Real => Err(ProcedureError::ExecutionFailed {
procedure: Fragment::internal("clock::advance"),
reason: "clock::advance can only be used with a mock clock".to_string(),
}),
}
}
}
const EXPECTED_ADVANCE_TYPES: &[Type] = &[
Type::Duration,
Type::Int1,
Type::Int2,
Type::Int4,
Type::Int8,
Type::Int16,
Type::Uint1,
Type::Uint2,
Type::Uint4,
Type::Uint8,
Type::Uint16,
];