extern crate chrono;
use self::chrono::{DateTime, Duration, Local, NaiveDate, NaiveDateTime, NaiveTime, TimeZone, Utc};
use super::{PgDate, PgTime, PgTimestamp};
use crate::deserialize::{self, FromSql};
use crate::pg::{Pg, PgValue};
use crate::serialize::{self, Output, ToSql};
use crate::sql_types::{Date, Time, Timestamp, Timestamptz};
fn pg_epoch() -> NaiveDateTime {
NaiveDate::from_ymd_opt(2000, 1, 1)
.expect("This is in supported range of chrono dates")
.and_hms_opt(0, 0, 0)
.expect("This is a valid input")
}
#[cfg(all(feature = "chrono", feature = "postgres_backend"))]
impl FromSql<Timestamp, Pg> for NaiveDateTime {
fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
let PgTimestamp(offset) = FromSql::<Timestamp, Pg>::from_sql(bytes)?;
match pg_epoch().checked_add_signed(Duration::microseconds(offset)) {
Some(v) => Ok(v),
None => {
let message = "Tried to deserialize a timestamp that is too large for Chrono";
Err(message.into())
}
}
}
}
#[cfg(all(feature = "chrono", feature = "postgres_backend"))]
impl ToSql<Timestamp, Pg> for NaiveDateTime {
fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
let time = match (self.signed_duration_since(pg_epoch())).num_microseconds() {
Some(time) => time,
None => {
let error_message =
format!("{self:?} as microseconds is too large to fit in an i64");
return Err(error_message.into());
}
};
ToSql::<Timestamp, Pg>::to_sql(&PgTimestamp(time), &mut out.reborrow())
}
}
#[cfg(all(feature = "chrono", feature = "postgres_backend"))]
impl FromSql<Timestamptz, Pg> for NaiveDateTime {
fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
FromSql::<Timestamp, Pg>::from_sql(bytes)
}
}
#[cfg(all(feature = "chrono", feature = "postgres_backend"))]
impl ToSql<Timestamptz, Pg> for NaiveDateTime {
fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
ToSql::<Timestamp, Pg>::to_sql(self, out)
}
}
#[cfg(all(feature = "chrono", feature = "postgres_backend"))]
impl FromSql<Timestamptz, Pg> for DateTime<Utc> {
fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
let naive_date_time = <NaiveDateTime as FromSql<Timestamptz, Pg>>::from_sql(bytes)?;
Ok(Utc.from_utc_datetime(&naive_date_time))
}
}
#[cfg(all(feature = "chrono", feature = "postgres_backend"))]
impl FromSql<Timestamptz, Pg> for DateTime<Local> {
fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
let naive_date_time = <NaiveDateTime as FromSql<Timestamptz, Pg>>::from_sql(bytes)?;
Ok(Local::from_utc_datetime(&Local, &naive_date_time))
}
}
#[cfg(all(feature = "chrono", feature = "postgres_backend"))]
impl<TZ: TimeZone> ToSql<Timestamptz, Pg> for DateTime<TZ> {
fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
ToSql::<Timestamptz, Pg>::to_sql(&self.naive_utc(), &mut out.reborrow())
}
}
fn midnight() -> NaiveTime {
NaiveTime::from_hms_opt(0, 0, 0).expect("This is a valid hms spec")
}
#[cfg(all(feature = "chrono", feature = "postgres_backend"))]
impl ToSql<Time, Pg> for NaiveTime {
fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
let duration = self.signed_duration_since(midnight());
match duration.num_microseconds() {
Some(offset) => ToSql::<Time, Pg>::to_sql(&PgTime(offset), &mut out.reborrow()),
None => unreachable!(),
}
}
}
#[cfg(all(feature = "chrono", feature = "postgres_backend"))]
impl FromSql<Time, Pg> for NaiveTime {
fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
let PgTime(offset) = FromSql::<Time, Pg>::from_sql(bytes)?;
let duration = Duration::microseconds(offset);
Ok(midnight() + duration)
}
}
fn pg_epoch_date() -> NaiveDate {
NaiveDate::from_ymd_opt(2000, 1, 1).expect("This is in supported range of chrono dates")
}
#[cfg(all(feature = "chrono", feature = "postgres_backend"))]
impl ToSql<Date, Pg> for NaiveDate {
fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
let days_since_epoch = self.signed_duration_since(pg_epoch_date()).num_days();
ToSql::<Date, Pg>::to_sql(&PgDate(days_since_epoch as i32), &mut out.reborrow())
}
}
#[cfg(all(feature = "chrono", feature = "postgres_backend"))]
impl FromSql<Date, Pg> for NaiveDate {
fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
let PgDate(offset) = FromSql::<Date, Pg>::from_sql(bytes)?;
match pg_epoch_date().checked_add_signed(Duration::days(i64::from(offset))) {
Some(date) => Ok(date),
None => {
let error_message = format!(
"Chrono can only represent dates up to {:?}",
chrono::NaiveDate::MAX
);
Err(error_message.into())
}
}
}
}
#[cfg(test)]
mod tests {
extern crate chrono;
extern crate dotenvy;
use self::chrono::{Duration, FixedOffset, NaiveDate, NaiveTime, TimeZone, Utc};
use crate::dsl::{now, sql};
use crate::prelude::*;
use crate::select;
use crate::sql_types::{Date, Time, Timestamp, Timestamptz};
use crate::test_helpers::connection;
#[test]
fn unix_epoch_encodes_correctly() {
let connection = &mut connection();
let time = NaiveDate::from_ymd_opt(1970, 1, 1)
.unwrap()
.and_hms_opt(0, 0, 0)
.unwrap();
let query = select(sql::<Timestamp>("'1970-01-01'").eq(time));
assert!(query.get_result::<bool>(connection).unwrap());
}
#[test]
fn unix_epoch_encodes_correctly_with_utc_timezone() {
let connection = &mut connection();
let time = Utc.with_ymd_and_hms(1970, 1, 1, 0, 0, 0).single().unwrap();
let query = select(sql::<Timestamptz>("'1970-01-01Z'::timestamptz").eq(time));
assert!(query.get_result::<bool>(connection).unwrap());
}
#[test]
fn unix_epoch_encodes_correctly_with_timezone() {
let connection = &mut connection();
let time = FixedOffset::west_opt(3600)
.unwrap()
.with_ymd_and_hms(1970, 1, 1, 0, 0, 0)
.single()
.unwrap();
let query = select(sql::<Timestamptz>("'1970-01-01 01:00:00Z'::timestamptz").eq(time));
assert!(query.get_result::<bool>(connection).unwrap());
}
#[test]
fn unix_epoch_decodes_correctly() {
let connection = &mut connection();
let time = NaiveDate::from_ymd_opt(1970, 1, 1)
.unwrap()
.and_hms_opt(0, 0, 0)
.unwrap();
let epoch_from_sql =
select(sql::<Timestamp>("'1970-01-01'::timestamp")).get_result(connection);
assert_eq!(Ok(time), epoch_from_sql);
}
#[test]
fn unix_epoch_decodes_correctly_with_timezone() {
let connection = &mut connection();
let time = Utc.with_ymd_and_hms(1970, 1, 1, 0, 0, 0).single().unwrap();
let epoch_from_sql =
select(sql::<Timestamptz>("'1970-01-01Z'::timestamptz")).get_result(connection);
assert_eq!(Ok(time), epoch_from_sql);
}
#[test]
fn times_relative_to_now_encode_correctly() {
let connection = &mut connection();
let time = Utc::now().naive_utc() + Duration::seconds(60);
let query = select(now.at_time_zone("utc").lt(time));
assert!(query.get_result::<bool>(connection).unwrap());
let time = Utc::now().naive_utc() - Duration::seconds(60);
let query = select(now.at_time_zone("utc").gt(time));
assert!(query.get_result::<bool>(connection).unwrap());
}
#[test]
fn times_with_timezones_round_trip_after_conversion() {
let connection = &mut connection();
let time = FixedOffset::east_opt(3600)
.unwrap()
.with_ymd_and_hms(2016, 1, 2, 1, 0, 0)
.unwrap();
let expected = NaiveDate::from_ymd_opt(2016, 1, 1)
.unwrap()
.and_hms_opt(20, 0, 0)
.unwrap();
let query = select(time.into_sql::<Timestamptz>().at_time_zone("EDT"));
assert_eq!(Ok(expected), query.get_result(connection));
}
#[test]
fn times_of_day_encode_correctly() {
let connection = &mut connection();
let midnight = NaiveTime::from_hms_opt(0, 0, 0).unwrap();
let query = select(sql::<Time>("'00:00:00'::time").eq(midnight));
assert!(query.get_result::<bool>(connection).unwrap());
let noon = NaiveTime::from_hms_opt(12, 0, 0).unwrap();
let query = select(sql::<Time>("'12:00:00'::time").eq(noon));
assert!(query.get_result::<bool>(connection).unwrap());
let roughly_half_past_eleven = NaiveTime::from_hms_micro_opt(23, 37, 4, 2200).unwrap();
let query = select(sql::<Time>("'23:37:04.002200'::time").eq(roughly_half_past_eleven));
assert!(query.get_result::<bool>(connection).unwrap());
}
#[test]
fn times_of_day_decode_correctly() {
let connection = &mut connection();
let midnight = NaiveTime::from_hms_opt(0, 0, 0).unwrap();
let query = select(sql::<Time>("'00:00:00'::time"));
assert_eq!(Ok(midnight), query.get_result::<NaiveTime>(connection));
let noon = NaiveTime::from_hms_opt(12, 0, 0).unwrap();
let query = select(sql::<Time>("'12:00:00'::time"));
assert_eq!(Ok(noon), query.get_result::<NaiveTime>(connection));
let roughly_half_past_eleven = NaiveTime::from_hms_micro_opt(23, 37, 4, 2200).unwrap();
let query = select(sql::<Time>("'23:37:04.002200'::time"));
assert_eq!(
Ok(roughly_half_past_eleven),
query.get_result::<NaiveTime>(connection)
);
}
#[test]
fn dates_encode_correctly() {
let connection = &mut connection();
let january_first_2000 = NaiveDate::from_ymd_opt(2000, 1, 1).unwrap();
let query = select(sql::<Date>("'2000-1-1'").eq(january_first_2000));
assert!(query.get_result::<bool>(connection).unwrap());
let distant_past = NaiveDate::from_ymd_opt(-398, 4, 11).unwrap(); let query = select(sql::<Date>("'399-4-11 BC'").eq(distant_past));
assert!(query.get_result::<bool>(connection).unwrap());
let julian_epoch = NaiveDate::from_ymd_opt(-4713, 11, 24).unwrap();
let query = select(sql::<Date>("'J0'::date").eq(julian_epoch));
assert!(query.get_result::<bool>(connection).unwrap());
let max_date = NaiveDate::MAX;
let query = select(sql::<Date>("'262143-12-31'::date").eq(max_date));
assert!(query.get_result::<bool>(connection).unwrap());
let january_first_2018 = NaiveDate::from_ymd_opt(2018, 1, 1).unwrap();
let query = select(sql::<Date>("'2018-1-1'::date").eq(january_first_2018));
assert!(query.get_result::<bool>(connection).unwrap());
let distant_future = NaiveDate::from_ymd_opt(72_400, 1, 8).unwrap();
let query = select(sql::<Date>("'72400-1-8'::date").eq(distant_future));
assert!(query.get_result::<bool>(connection).unwrap());
}
#[test]
fn dates_decode_correctly() {
let connection = &mut connection();
let january_first_2000 = NaiveDate::from_ymd_opt(2000, 1, 1).unwrap();
let query = select(sql::<Date>("'2000-1-1'::date"));
assert_eq!(
Ok(january_first_2000),
query.get_result::<NaiveDate>(connection)
);
let distant_past = NaiveDate::from_ymd_opt(-398, 4, 11).unwrap();
let query = select(sql::<Date>("'399-4-11 BC'::date"));
assert_eq!(Ok(distant_past), query.get_result::<NaiveDate>(connection));
let julian_epoch = NaiveDate::from_ymd_opt(-4713, 11, 24).unwrap();
let query = select(sql::<Date>("'J0'::date"));
assert_eq!(Ok(julian_epoch), query.get_result::<NaiveDate>(connection));
let max_date = NaiveDate::MAX;
let query = select(sql::<Date>("'262143-12-31'::date"));
assert_eq!(Ok(max_date), query.get_result::<NaiveDate>(connection));
let january_first_2018 = NaiveDate::from_ymd_opt(2018, 1, 1).unwrap();
let query = select(sql::<Date>("'2018-1-1'::date"));
assert_eq!(
Ok(january_first_2018),
query.get_result::<NaiveDate>(connection)
);
let distant_future = NaiveDate::from_ymd_opt(72_400, 1, 8).unwrap();
let query = select(sql::<Date>("'72400-1-8'::date"));
assert_eq!(
Ok(distant_future),
query.get_result::<NaiveDate>(connection)
);
}
}