use std::ops::Add;
use crate::deserialize::{self, FromSql, FromSqlRow};
use crate::expression::AsExpression;
use crate::pg::{Pg, PgValue};
use crate::serialize::{self, IsNull, Output, ToSql};
use crate::sql_types::{self, Date, Interval, Time, Timestamp, Timestamptz};
#[cfg(feature = "chrono")]
mod chrono;
#[cfg(feature = "quickcheck")]
mod quickcheck_impls;
mod std_time;
#[cfg(feature = "time")]
mod time;
#[cfg(feature = "postgres_backend")]
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, AsExpression, FromSqlRow)]
#[diesel(sql_type = Timestamp)]
#[diesel(sql_type = Timestamptz)]
pub struct PgTimestamp(pub i64);
#[cfg(feature = "postgres_backend")]
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, AsExpression, FromSqlRow)]
#[diesel(sql_type = Date)]
pub struct PgDate(pub i32);
#[cfg(feature = "postgres_backend")]
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, AsExpression, FromSqlRow)]
#[diesel(sql_type = Time)]
pub struct PgTime(pub i64);
#[cfg(feature = "postgres_backend")]
#[derive(Debug, Clone, Copy, PartialEq, Eq, AsExpression, FromSqlRow)]
#[diesel(sql_type = Interval)]
pub struct PgInterval {
pub microseconds: i64,
pub days: i32,
pub months: i32,
}
impl PgInterval {
pub fn new(microseconds: i64, days: i32, months: i32) -> Self {
PgInterval {
microseconds,
days,
months,
}
}
pub fn from_microseconds(microseconds: i64) -> Self {
Self::new(microseconds, 0, 0)
}
pub fn from_days(days: i32) -> Self {
Self::new(0, days, 0)
}
pub fn from_months(months: i32) -> Self {
Self::new(0, 0, months)
}
}
#[cfg(feature = "postgres_backend")]
impl ToSql<sql_types::Timestamp, Pg> for PgTimestamp {
fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
ToSql::<sql_types::BigInt, Pg>::to_sql(&self.0, out)
}
}
#[cfg(feature = "postgres_backend")]
impl FromSql<sql_types::Timestamp, Pg> for PgTimestamp {
fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
FromSql::<sql_types::BigInt, Pg>::from_sql(bytes).map(PgTimestamp)
}
}
#[cfg(feature = "postgres_backend")]
impl ToSql<sql_types::Timestamptz, Pg> for PgTimestamp {
fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
ToSql::<sql_types::Timestamp, Pg>::to_sql(self, out)
}
}
#[cfg(feature = "postgres_backend")]
impl FromSql<sql_types::Timestamptz, Pg> for PgTimestamp {
fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
FromSql::<sql_types::Timestamp, Pg>::from_sql(bytes)
}
}
#[cfg(feature = "postgres_backend")]
impl ToSql<sql_types::Date, Pg> for PgDate {
fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
ToSql::<sql_types::Integer, Pg>::to_sql(&self.0, out)
}
}
#[cfg(feature = "postgres_backend")]
impl FromSql<sql_types::Date, Pg> for PgDate {
fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
FromSql::<sql_types::Integer, Pg>::from_sql(bytes).map(PgDate)
}
}
#[cfg(feature = "postgres_backend")]
impl ToSql<sql_types::Time, Pg> for PgTime {
fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
ToSql::<sql_types::BigInt, Pg>::to_sql(&self.0, out)
}
}
#[cfg(feature = "postgres_backend")]
impl FromSql<sql_types::Time, Pg> for PgTime {
fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
FromSql::<sql_types::BigInt, Pg>::from_sql(bytes).map(PgTime)
}
}
#[cfg(feature = "postgres_backend")]
impl ToSql<sql_types::Interval, Pg> for PgInterval {
fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
ToSql::<sql_types::BigInt, Pg>::to_sql(&self.microseconds, out)?;
ToSql::<sql_types::Integer, Pg>::to_sql(&self.days, out)?;
ToSql::<sql_types::Integer, Pg>::to_sql(&self.months, out)?;
Ok(IsNull::No)
}
}
#[cfg(feature = "postgres_backend")]
impl FromSql<sql_types::Interval, Pg> for PgInterval {
fn from_sql(value: PgValue<'_>) -> deserialize::Result<Self> {
Ok(PgInterval {
microseconds: FromSql::<sql_types::BigInt, Pg>::from_sql(value.subslice(0..8))?,
days: FromSql::<sql_types::Integer, Pg>::from_sql(value.subslice(8..12))?,
months: FromSql::<sql_types::Integer, Pg>::from_sql(value.subslice(12..16))?,
})
}
}
impl Add<PgInterval> for PgInterval {
type Output = PgInterval;
fn add(self, other: PgInterval) -> Self::Output {
PgInterval {
microseconds: self.microseconds + other.microseconds,
days: self.days + other.days,
months: self.months + other.months,
}
}
}