268 lines
5.8 KiB
Rust
268 lines
5.8 KiB
Rust
use crate::sql::common::{take_digits, take_digits_range, take_u32};
|
|
use crate::sql::error::IResult;
|
|
use crate::sql::serde::is_internal_serialization;
|
|
use chrono::{DateTime, FixedOffset, TimeZone, Utc};
|
|
use nom::branch::alt;
|
|
use nom::character::complete::char;
|
|
use nom::combinator::map;
|
|
use nom::sequence::delimited;
|
|
use serde::{Deserialize, Serialize};
|
|
use std::fmt;
|
|
use std::ops::Deref;
|
|
use std::str;
|
|
|
|
const SINGLE: char = '\'';
|
|
const DOUBLE: char = '"';
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Deserialize)]
|
|
pub struct Datetime(pub DateTime<Utc>);
|
|
|
|
impl Default for Datetime {
|
|
fn default() -> Self {
|
|
Datetime(Utc::now())
|
|
}
|
|
}
|
|
|
|
impl From<i64> for Datetime {
|
|
fn from(v: i64) -> Self {
|
|
Datetime(Utc.timestamp(v, 0))
|
|
}
|
|
}
|
|
|
|
impl From<DateTime<Utc>> for Datetime {
|
|
fn from(v: DateTime<Utc>) -> Self {
|
|
Datetime(v)
|
|
}
|
|
}
|
|
|
|
impl From<&str> for Datetime {
|
|
fn from(s: &str) -> Self {
|
|
match datetime_raw(s) {
|
|
Ok((_, v)) => v,
|
|
Err(_) => Datetime::default(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Deref for Datetime {
|
|
type Target = DateTime<Utc>;
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.0
|
|
}
|
|
}
|
|
|
|
impl fmt::Display for Datetime {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(f, "\"{:?}\"", self.0)
|
|
}
|
|
}
|
|
|
|
impl Serialize for Datetime {
|
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
|
where
|
|
S: serde::Serializer,
|
|
{
|
|
if is_internal_serialization() {
|
|
serializer.serialize_newtype_struct("Datetime", &self.0)
|
|
} else {
|
|
serializer.serialize_some(&self.0)
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn datetime(i: &str) -> IResult<&str, Datetime> {
|
|
alt((
|
|
delimited(char(DOUBLE), datetime_raw, char(DOUBLE)),
|
|
delimited(char(SINGLE), datetime_raw, char(SINGLE)),
|
|
))(i)
|
|
}
|
|
|
|
fn datetime_raw(i: &str) -> IResult<&str, Datetime> {
|
|
alt((nano, time, date))(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)?;
|
|
|
|
let d = Utc.ymd(year, mon, day).and_hms(0, 0, 0);
|
|
Ok((i, Datetime(d)))
|
|
}
|
|
|
|
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)?;
|
|
|
|
let v = match zone {
|
|
Some(z) => {
|
|
let d = z.ymd(year, mon, day).and_hms(hour, min, sec);
|
|
let d = d.with_timezone(&Utc);
|
|
Datetime(d)
|
|
}
|
|
None => {
|
|
let d = Utc.ymd(year, mon, day).and_hms(hour, min, sec);
|
|
Datetime(d)
|
|
}
|
|
};
|
|
|
|
Ok((i, v))
|
|
}
|
|
|
|
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)?;
|
|
|
|
let v = match zone {
|
|
Some(z) => {
|
|
let d = z.ymd(year, mon, day).and_hms_nano(hour, min, sec, nano);
|
|
let d = d.with_timezone(&Utc);
|
|
Datetime(d)
|
|
}
|
|
None => {
|
|
let d = Utc.ymd(year, mon, day).and_hms_nano(hour, min, sec, nano);
|
|
Datetime(d)
|
|
}
|
|
};
|
|
|
|
Ok((i, v))
|
|
}
|
|
|
|
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..=59)
|
|
}
|
|
|
|
fn nanosecond(i: &str) -> IResult<&str, u32> {
|
|
let (i, _) = char('.')(i)?;
|
|
let (i, v) = take_u32(i)?;
|
|
Ok((i, v))
|
|
}
|
|
|
|
fn zone(i: &str) -> IResult<&str, Option<FixedOffset>> {
|
|
alt((zone_utc, zone_all))(i)
|
|
}
|
|
|
|
fn zone_utc(i: &str) -> IResult<&str, Option<FixedOffset>> {
|
|
let (i, _) = char('Z')(i)?;
|
|
Ok((i, None))
|
|
}
|
|
|
|
fn zone_all(i: &str) -> IResult<&str, Option<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, None))
|
|
} else if s < 0 {
|
|
Ok((i, { Some(FixedOffset::west((h * 3600 + m) as i32)) }))
|
|
} else if s > 0 {
|
|
Ok((i, { Some(FixedOffset::east((h * 3600 + m) as i32)) }))
|
|
} else {
|
|
Ok((i, None))
|
|
}
|
|
}
|
|
|
|
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() {
|
|
let sql = "2012-04-23";
|
|
let res = datetime_raw(sql);
|
|
assert!(res.is_ok());
|
|
let out = res.unwrap().1;
|
|
assert_eq!("\"2012-04-23T00:00:00Z\"", format!("{}", out));
|
|
}
|
|
|
|
#[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.000005631Z\"", format!("{}", out));
|
|
}
|
|
|
|
#[test]
|
|
fn date_time_timezone_utc() {
|
|
let sql = "2012-04-23T18:25:43.511Z";
|
|
let res = datetime_raw(sql);
|
|
assert!(res.is_ok());
|
|
let out = res.unwrap().1;
|
|
assert_eq!("\"2012-04-23T18:25:43.000000511Z\"", 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.000000511Z\"", format!("{}", out));
|
|
}
|
|
}
|