#![cfg(not(feature = "no_std"))] use super::arithmetic::make_err as make_arithmetic_err; use crate::plugin::*; use crate::{def_package, Dynamic, EvalAltResult, RhaiResult, RhaiResultOf, INT}; #[cfg(not(feature = "no_float"))] use crate::FLOAT; #[cfg(not(target_family = "wasm"))] use std::time::{Duration, Instant}; #[cfg(target_family = "wasm")] use instant::{Duration, Instant}; def_package! { /// Package of basic timing utilities. pub BasicTimePackage(lib) { lib.standard = true; // Register date/time functions combine_with_exported_module!(lib, "time", time_functions); } } #[export_module] mod time_functions { /// Create a timestamp containing the current system time. /// /// # Example /// /// ```rhai /// let now = timestamp(); /// /// sleep(10.0); // sleep for 10 seconds /// /// print(now.elapsed); // prints 10.??? /// ``` pub fn timestamp() -> Instant { Instant::now() } /// Return the number of seconds between the current system time and the timestamp. /// /// # Example /// /// ```rhai /// let now = timestamp(); /// /// sleep(10.0); // sleep for 10 seconds /// /// print(now.elapsed); // prints 10.??? /// ``` #[rhai_fn(name = "elapsed", get = "elapsed", return_raw)] pub fn elapsed(timestamp: Instant) -> RhaiResult { #[cfg(not(feature = "no_float"))] if timestamp > Instant::now() { Err(make_arithmetic_err("Time-stamp is later than now")) } else { Ok((timestamp.elapsed().as_secs_f64() as FLOAT).into()) } #[cfg(feature = "no_float")] { let seconds = timestamp.elapsed().as_secs(); if cfg!(not(feature = "unchecked")) && seconds > (INT::MAX as u64) { Err(make_arithmetic_err(format!( "Integer overflow for timestamp.elapsed: {seconds}" ))) } else if timestamp > Instant::now() { Err(make_arithmetic_err("Time-stamp is later than now")) } else { Ok((seconds as INT).into()) } } } /// Return the number of seconds between two timestamps. #[rhai_fn(return_raw, name = "-")] pub fn time_diff(timestamp1: Instant, timestamp2: Instant) -> RhaiResult { #[cfg(not(feature = "no_float"))] return Ok(if timestamp2 > timestamp1 { -(timestamp2 - timestamp1).as_secs_f64() as FLOAT } else { (timestamp1 - timestamp2).as_secs_f64() as FLOAT } .into()); #[cfg(feature = "no_float")] if timestamp2 > timestamp1 { let seconds = (timestamp2 - timestamp1).as_secs(); if cfg!(not(feature = "unchecked")) && seconds > (INT::MAX as u64) { Err(make_arithmetic_err(format!( "Integer overflow for timestamp duration: -{seconds}" ))) } else { Ok((-(seconds as INT)).into()) } } else { let seconds = (timestamp1 - timestamp2).as_secs(); if cfg!(not(feature = "unchecked")) && seconds > (INT::MAX as u64) { Err(make_arithmetic_err(format!( "Integer overflow for timestamp duration: {seconds}" ))) } else { Ok((seconds as INT).into()) } } } #[cfg(not(feature = "no_float"))] pub mod float_functions { fn add_impl(timestamp: Instant, seconds: FLOAT) -> RhaiResultOf { if seconds < 0.0 { subtract_impl(timestamp, -seconds) } else if cfg!(not(feature = "unchecked")) { if seconds > (INT::MAX as FLOAT) { Err(make_arithmetic_err(format!( "Integer overflow for timestamp add: {seconds}" ))) } else { timestamp .checked_add(Duration::from_millis((seconds * 1000.0) as u64)) .ok_or_else(|| { make_arithmetic_err(format!( "Timestamp overflow when adding {seconds} second(s)" )) }) } } else { Ok(timestamp + Duration::from_millis((seconds * 1000.0) as u64)) } } fn subtract_impl(timestamp: Instant, seconds: FLOAT) -> RhaiResultOf { if seconds < 0.0 { add_impl(timestamp, -seconds) } else if cfg!(not(feature = "unchecked")) { if seconds > (INT::MAX as FLOAT) { Err(make_arithmetic_err(format!( "Integer overflow for timestamp add: {seconds}" ))) } else { timestamp .checked_sub(Duration::from_millis((seconds * 1000.0) as u64)) .ok_or_else(|| { make_arithmetic_err(format!( "Timestamp overflow when adding {seconds} second(s)" )) }) } } else { Ok(timestamp - Duration::from_millis((seconds * 1000.0) as u64)) } } /// Add the specified number of `seconds` to the timestamp and return it as a new timestamp. #[rhai_fn(return_raw, name = "+")] pub fn add(timestamp: Instant, seconds: FLOAT) -> RhaiResultOf { add_impl(timestamp, seconds) } /// Add the specified number of `seconds` to the timestamp. #[rhai_fn(return_raw, name = "+=")] pub fn add_assign(timestamp: &mut Instant, seconds: FLOAT) -> RhaiResultOf<()> { *timestamp = add_impl(*timestamp, seconds)?; Ok(()) } /// Subtract the specified number of `seconds` from the timestamp and return it as a new timestamp. #[rhai_fn(return_raw, name = "-")] pub fn subtract(timestamp: Instant, seconds: FLOAT) -> RhaiResultOf { subtract_impl(timestamp, seconds) } /// Subtract the specified number of `seconds` from the timestamp. #[rhai_fn(return_raw, name = "-=")] pub fn subtract_assign(timestamp: &mut Instant, seconds: FLOAT) -> RhaiResultOf<()> { *timestamp = subtract_impl(*timestamp, seconds)?; Ok(()) } } fn add_impl(timestamp: Instant, seconds: INT) -> RhaiResultOf { if seconds < 0 { subtract_impl(timestamp, -seconds) } else if cfg!(not(feature = "unchecked")) { timestamp .checked_add(Duration::from_secs(seconds as u64)) .ok_or_else(|| { make_arithmetic_err(format!( "Timestamp overflow when adding {seconds} second(s)" )) }) } else { Ok(timestamp + Duration::from_secs(seconds as u64)) } } fn subtract_impl(timestamp: Instant, seconds: INT) -> RhaiResultOf { if seconds < 0 { add_impl(timestamp, -seconds) } else if cfg!(not(feature = "unchecked")) { timestamp .checked_sub(Duration::from_secs(seconds as u64)) .ok_or_else(|| { make_arithmetic_err(format!( "Timestamp overflow when adding {seconds} second(s)" )) }) } else { Ok(timestamp - Duration::from_secs(seconds as u64)) } } /// Add the specified number of `seconds` to the timestamp and return it as a new timestamp. #[rhai_fn(return_raw, name = "+")] pub fn add(timestamp: Instant, seconds: INT) -> RhaiResultOf { add_impl(timestamp, seconds) } /// Add the specified number of `seconds` to the timestamp. #[rhai_fn(return_raw, name = "+=")] pub fn add_assign(timestamp: &mut Instant, seconds: INT) -> RhaiResultOf<()> { *timestamp = add_impl(*timestamp, seconds)?; Ok(()) } /// Subtract the specified number of `seconds` from the timestamp and return it as a new timestamp. #[rhai_fn(return_raw, name = "-")] pub fn subtract(timestamp: Instant, seconds: INT) -> RhaiResultOf { subtract_impl(timestamp, seconds) } /// Subtract the specified number of `seconds` from the timestamp. #[rhai_fn(return_raw, name = "-=")] pub fn subtract_assign(timestamp: &mut Instant, seconds: INT) -> RhaiResultOf<()> { *timestamp = subtract_impl(*timestamp, seconds)?; Ok(()) } /// Return `true` if two timestamps are equal. #[rhai_fn(name = "==")] pub fn eq(timestamp1: Instant, timestamp2: Instant) -> bool { timestamp1 == timestamp2 } /// Return `true` if two timestamps are not equal. #[rhai_fn(name = "!=")] pub fn ne(timestamp1: Instant, timestamp2: Instant) -> bool { timestamp1 != timestamp2 } /// Return `true` if the first timestamp is earlier than the second. #[rhai_fn(name = "<")] pub fn lt(timestamp1: Instant, timestamp2: Instant) -> bool { timestamp1 < timestamp2 } /// Return `true` if the first timestamp is earlier than or equals to the second. #[rhai_fn(name = "<=")] pub fn lte(timestamp1: Instant, timestamp2: Instant) -> bool { timestamp1 <= timestamp2 } /// Return `true` if the first timestamp is later than the second. #[rhai_fn(name = ">")] pub fn gt(timestamp1: Instant, timestamp2: Instant) -> bool { timestamp1 > timestamp2 } /// Return `true` if the first timestamp is later than or equals to the second. #[rhai_fn(name = ">=")] pub fn gte(timestamp1: Instant, timestamp2: Instant) -> bool { timestamp1 >= timestamp2 } }