use crate::sql::common::{take_digits, take_digits_range, take_u32_len}; use crate::sql::duration::Duration; use crate::sql::error::IResult; use crate::sql::escape::escape_str; use crate::sql::serde::is_internal_serialization; use chrono::{DateTime, FixedOffset, Offset, SecondsFormat, TimeZone, Utc}; use chrono::{NaiveDate, NaiveDateTime, NaiveTime}; use nom::branch::alt; use nom::character::complete::char; use nom::combinator::map; use nom::error::ErrorKind; use nom::sequence::delimited; use nom::{error_position, Err}; use serde::{Deserialize, Serialize}; use std::fmt::{self, Display, Formatter}; use std::ops; use std::ops::Deref; use std::str; pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Datetime"; #[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Deserialize, Hash)] pub struct Datetime(pub DateTime); impl Default for Datetime { fn default() -> Self { Self(Utc::now()) } } impl From> for Datetime { fn from(v: DateTime) -> Self { Self(v) } } impl From<&str> for Datetime { fn from(s: &str) -> Self { match datetime_all_raw(s) { Ok((_, v)) => v, Err(_) => Self::default(), } } } impl Deref for Datetime { type Target = DateTime; fn deref(&self) -> &Self::Target { &self.0 } } impl From for DateTime { fn from(x: Datetime) -> Self { x.0 } } impl Datetime { /// Convert the Datetime to a raw String pub fn to_raw(&self) -> String { self.0.to_rfc3339_opts(SecondsFormat::AutoSi, true) } } impl Display for Datetime { fn fmt(&self, f: &mut Formatter) -> fmt::Result { Display::fmt(&escape_str(&self.0.to_rfc3339_opts(SecondsFormat::AutoSi, true)), f) } } impl Serialize for Datetime { fn serialize(&self, serializer: S) -> Result where S: serde::Serializer, { if is_internal_serialization() { serializer.serialize_newtype_struct(TOKEN, &self.0) } else { serializer.serialize_some(&self.to_string()) } } } impl ops::Sub for Datetime { type Output = Duration; fn sub(self, other: Self) -> Duration { match (self.0 - other.0).to_std() { Ok(d) => Duration::from(d), Err(_) => Duration::default(), } } } pub fn datetime(i: &str) -> IResult<&str, Datetime> { alt((datetime_single, datetime_double))(i) } fn datetime_single(i: &str) -> IResult<&str, Datetime> { delimited(char('\''), datetime_raw, char('\''))(i) } fn datetime_double(i: &str) -> IResult<&str, Datetime> { delimited(char('\"'), datetime_raw, char('\"'))(i) } fn datetime_all_raw(i: &str) -> IResult<&str, Datetime> { alt((nano, time, date))(i) } fn datetime_raw(i: &str) -> IResult<&str, Datetime> { alt((nano, time))(i) } fn date(i: &str) -> IResult<&str, Datetime> { let (i, year) = year(i)?; let (i, _) = char('-')(i)?; let (i, mon) = month(i)?; let (i, _) = char('-')(i)?; let (i, day) = day(i)?; convert(i, (year, mon, day), (0, 0, 0, 0), Utc.fix()) } fn time(i: &str) -> IResult<&str, Datetime> { let (i, year) = year(i)?; let (i, _) = char('-')(i)?; let (i, mon) = month(i)?; let (i, _) = char('-')(i)?; let (i, day) = day(i)?; let (i, _) = char('T')(i)?; let (i, hour) = hour(i)?; let (i, _) = char(':')(i)?; let (i, min) = minute(i)?; let (i, _) = char(':')(i)?; let (i, sec) = second(i)?; let (i, zone) = zone(i)?; convert(i, (year, mon, day), (hour, min, sec, 0), zone) } fn nano(i: &str) -> IResult<&str, Datetime> { let (i, year) = year(i)?; let (i, _) = char('-')(i)?; let (i, mon) = month(i)?; let (i, _) = char('-')(i)?; let (i, day) = day(i)?; let (i, _) = char('T')(i)?; let (i, hour) = hour(i)?; let (i, _) = char(':')(i)?; let (i, min) = minute(i)?; let (i, _) = char(':')(i)?; let (i, sec) = second(i)?; let (i, nano) = nanosecond(i)?; let (i, zone) = zone(i)?; convert(i, (year, mon, day), (hour, min, sec, nano), zone) } fn convert( i: &str, (year, mon, day): (i32, u32, u32), (hour, min, sec, nano): (u32, u32, u32, u32), zone: FixedOffset, ) -> IResult<&str, Datetime> { // Attempt to create date let d = NaiveDate::from_ymd_opt(year, mon, day) .ok_or_else(|| Err::Error(error_position!(i, ErrorKind::Verify)))?; // Attempt to create time let t = NaiveTime::from_hms_nano_opt(hour, min, sec, nano) .ok_or_else(|| Err::Error(error_position!(i, ErrorKind::Verify)))?; // let v = NaiveDateTime::new(d, t); // Attempt to create time let d = zone .from_local_datetime(&v) .earliest() .ok_or_else(|| Err::Error(error_position!(i, ErrorKind::Verify)))? .with_timezone(&Utc); // This is a valid datetime Ok((i, Datetime(d))) } fn year(i: &str) -> IResult<&str, i32> { let (i, s) = sign(i).unwrap_or((i, 1)); let (i, y) = take_digits(i, 4)?; let v = s * y as i32; Ok((i, v)) } fn month(i: &str) -> IResult<&str, u32> { take_digits_range(i, 2, 1..=12) } fn day(i: &str) -> IResult<&str, u32> { take_digits_range(i, 2, 1..=31) } fn hour(i: &str) -> IResult<&str, u32> { take_digits_range(i, 2, 0..=24) } fn minute(i: &str) -> IResult<&str, u32> { take_digits_range(i, 2, 0..=59) } fn second(i: &str) -> IResult<&str, u32> { take_digits_range(i, 2, 0..=60) } fn nanosecond(i: &str) -> IResult<&str, u32> { let (i, _) = char('.')(i)?; let (i, (v, l)) = take_u32_len(i)?; let v = match l { l if l <= 2 => v * 10000000, l if l <= 3 => v * 1000000, l if l <= 4 => v * 100000, l if l <= 5 => v * 10000, l if l <= 6 => v * 1000, l if l <= 7 => v * 100, l if l <= 8 => v * 10, _ => v, }; Ok((i, v)) } fn zone(i: &str) -> IResult<&str, FixedOffset> { alt((zone_utc, zone_all))(i) } fn zone_utc(i: &str) -> IResult<&str, FixedOffset> { let (i, _) = char('Z')(i)?; Ok((i, Utc.fix())) } fn zone_all(i: &str) -> IResult<&str, FixedOffset> { let (i, s) = sign(i)?; let (i, h) = hour(i)?; let (i, _) = char(':')(i)?; let (i, m) = minute(i)?; if h == 0 && m == 0 { Ok((i, Utc.fix())) } else if s < 0 { match FixedOffset::west_opt((h * 3600 + m * 60) as i32) { Some(v) => Ok((i, v)), None => Err(Err::Error(error_position!(i, ErrorKind::Verify))), } } else if s > 0 { match FixedOffset::east_opt((h * 3600 + m * 60) as i32) { Some(v) => Ok((i, v)), None => Err(Err::Error(error_position!(i, ErrorKind::Verify))), } } else { Ok((i, Utc.fix())) } } fn sign(i: &str) -> IResult<&str, i32> { map(alt((char('-'), char('+'))), |s: char| match s { '-' => -1, _ => 1, })(i) } #[cfg(test)] mod tests { use super::*; #[test] fn date_time() { let sql = "2012-04-23T18:25:43Z"; let res = datetime_raw(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!("'2012-04-23T18:25:43Z'", format!("{}", out)); } #[test] fn date_time_nanos() { let sql = "2012-04-23T18:25:43.5631Z"; let res = datetime_raw(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!("'2012-04-23T18:25:43.563100Z'", format!("{}", out)); } #[test] fn date_time_timezone_utc() { let sql = "2012-04-23T18:25:43.0000511Z"; let res = datetime_raw(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!("'2012-04-23T18:25:43.000051100Z'", format!("{}", out)); } #[test] fn date_time_timezone_pacific() { let sql = "2012-04-23T18:25:43.511-08:00"; let res = datetime_raw(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!("'2012-04-24T02:25:43.511Z'", format!("{}", out)); } #[test] fn date_time_timezone_pacific_partial() { let sql = "2012-04-23T18:25:43.511-08:30"; let res = datetime_raw(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!("'2012-04-24T02:55:43.511Z'", format!("{}", out)); } #[test] fn date_time_timezone_utc_nanoseconds() { let sql = "2012-04-23T18:25:43.5110000Z"; let res = datetime_raw(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!("'2012-04-23T18:25:43.511Z'", format!("{}", out)); } #[test] fn date_time_timezone_utc_sub_nanoseconds() { let sql = "2012-04-23T18:25:43.0000511Z"; let res = datetime_raw(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!("'2012-04-23T18:25:43.000051100Z'", format!("{}", out)); } #[test] fn date_time_illegal_date() { // Hey! There's not a 31st of November! let sql = "2022-11-31T12:00:00.000Z"; let res = datetime_raw(sql); assert!(res.is_err()); } }