364 lines
9 KiB
Rust
364 lines
9 KiB
Rust
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::quote_str;
|
|
use crate::sql::strand::Strand;
|
|
use chrono::{
|
|
DateTime, FixedOffset, NaiveDate, NaiveDateTime, NaiveTime, Offset, SecondsFormat, TimeZone,
|
|
Utc,
|
|
};
|
|
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 revision::revisioned;
|
|
use serde::{Deserialize, Serialize};
|
|
use std::fmt::{self, Display, Formatter};
|
|
use std::ops;
|
|
use std::ops::Deref;
|
|
use std::str;
|
|
use std::str::FromStr;
|
|
|
|
use super::error::expected;
|
|
|
|
pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Datetime";
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize, Hash)]
|
|
#[serde(rename = "$surrealdb::private::sql::Datetime")]
|
|
#[revisioned(revision = 1)]
|
|
pub struct Datetime(pub DateTime<Utc>);
|
|
|
|
impl Default for Datetime {
|
|
fn default() -> Self {
|
|
Self(Utc::now())
|
|
}
|
|
}
|
|
|
|
impl From<DateTime<Utc>> for Datetime {
|
|
fn from(v: DateTime<Utc>) -> Self {
|
|
Self(v)
|
|
}
|
|
}
|
|
|
|
impl From<Datetime> for DateTime<Utc> {
|
|
fn from(x: Datetime) -> Self {
|
|
x.0
|
|
}
|
|
}
|
|
|
|
impl FromStr for Datetime {
|
|
type Err = ();
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
Self::try_from(s)
|
|
}
|
|
}
|
|
|
|
impl TryFrom<String> for Datetime {
|
|
type Error = ();
|
|
fn try_from(v: String) -> Result<Self, Self::Error> {
|
|
Self::try_from(v.as_str())
|
|
}
|
|
}
|
|
|
|
impl TryFrom<Strand> for Datetime {
|
|
type Error = ();
|
|
fn try_from(v: Strand) -> Result<Self, Self::Error> {
|
|
Self::try_from(v.as_str())
|
|
}
|
|
}
|
|
|
|
impl TryFrom<&str> for Datetime {
|
|
type Error = ();
|
|
fn try_from(v: &str) -> Result<Self, Self::Error> {
|
|
match datetime_all_raw(v) {
|
|
Ok((_, v)) => Ok(v),
|
|
_ => Err(()),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Deref for Datetime {
|
|
type Target = DateTime<Utc>;
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.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("e_str(&self.to_raw()), f)
|
|
}
|
|
}
|
|
|
|
impl ops::Sub<Self> 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> {
|
|
expected("a 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 chrono::Date;
|
|
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn date_zone() {
|
|
let sql = "2020-01-01T00:00:00Z";
|
|
let res = datetime_all_raw(sql);
|
|
let out = res.unwrap().1;
|
|
assert_eq!("'2020-01-01T00:00:00Z'", format!("{}", out));
|
|
assert_eq!(out, Datetime::try_from("2020-01-01T00:00:00Z").unwrap());
|
|
}
|
|
|
|
#[test]
|
|
fn date_time() {
|
|
let sql = "2012-04-23T18:25:43Z";
|
|
let res = datetime_raw(sql);
|
|
let out = res.unwrap().1;
|
|
assert_eq!("'2012-04-23T18:25:43Z'", format!("{}", out));
|
|
assert_eq!(out, Datetime::try_from("2012-04-23T18:25:43Z").unwrap());
|
|
}
|
|
|
|
#[test]
|
|
fn date_time_nanos() {
|
|
let sql = "2012-04-23T18:25:43.5631Z";
|
|
let res = datetime_raw(sql);
|
|
let out = res.unwrap().1;
|
|
assert_eq!("'2012-04-23T18:25:43.563100Z'", format!("{}", out));
|
|
assert_eq!(out, Datetime::try_from("2012-04-23T18:25:43.563100Z").unwrap());
|
|
}
|
|
|
|
#[test]
|
|
fn date_time_timezone_utc() {
|
|
let sql = "2012-04-23T18:25:43.0000511Z";
|
|
let res = datetime_raw(sql);
|
|
let out = res.unwrap().1;
|
|
assert_eq!("'2012-04-23T18:25:43.000051100Z'", format!("{}", out));
|
|
assert_eq!(out, Datetime::try_from("2012-04-23T18:25:43.000051100Z").unwrap());
|
|
}
|
|
|
|
#[test]
|
|
fn date_time_timezone_pacific() {
|
|
let sql = "2012-04-23T18:25:43.511-08:00";
|
|
let res = datetime_raw(sql);
|
|
let out = res.unwrap().1;
|
|
assert_eq!("'2012-04-24T02:25:43.511Z'", format!("{}", out));
|
|
assert_eq!(out, Datetime::try_from("2012-04-24T02:25:43.511Z").unwrap());
|
|
}
|
|
|
|
#[test]
|
|
fn date_time_timezone_pacific_partial() {
|
|
let sql = "2012-04-23T18:25:43.511-08:30";
|
|
let res = datetime_raw(sql);
|
|
let out = res.unwrap().1;
|
|
assert_eq!("'2012-04-24T02:55:43.511Z'", format!("{}", out));
|
|
assert_eq!(out, Datetime::try_from("2012-04-24T02:55:43.511Z").unwrap());
|
|
}
|
|
|
|
#[test]
|
|
fn date_time_timezone_utc_nanoseconds() {
|
|
let sql = "2012-04-23T18:25:43.5110000Z";
|
|
let res = datetime_raw(sql);
|
|
let out = res.unwrap().1;
|
|
assert_eq!("'2012-04-23T18:25:43.511Z'", format!("{}", out));
|
|
assert_eq!(out, Datetime::try_from("2012-04-23T18:25:43.511Z").unwrap());
|
|
}
|
|
|
|
#[test]
|
|
fn date_time_timezone_utc_sub_nanoseconds() {
|
|
let sql = "2012-04-23T18:25:43.0000511Z";
|
|
let res = datetime_raw(sql);
|
|
let out = res.unwrap().1;
|
|
assert_eq!("'2012-04-23T18:25:43.000051100Z'", format!("{}", out));
|
|
assert_eq!(out, Datetime::try_from("2012-04-23T18:25:43.000051100Z").unwrap());
|
|
}
|
|
|
|
#[test]
|
|
fn date_time_illegal_date() {
|
|
// Hey! There's not a 31st of November!
|
|
let sql = "2022-11-31T12:00:00.000Z";
|
|
datetime_raw(sql).unwrap_err();
|
|
}
|
|
}
|