use chrono::{DateTime, Utc, TimeZone, NaiveDateTime, LocalResult};
use chrono_tz::Tz;
use serde::Serialize;
use serde_json::{Value, json};
use std::hash::Hash;
use anyhow::{anyhow, Result};
#[derive(Eq, PartialEq, Clone, Hash, PartialOrd, Ord)]
pub struct Timestamp {
ts: i64,
}
impl Serialize for Timestamp {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer {
serializer.serialize_i64(self.ts)
}
}
impl<Tz> From<DateTime<Tz>> for Timestamp where Tz: TimeZone {
fn from(d: DateTime<Tz>) -> Self {
let ts = d.with_timezone(&Utc);
Self {
ts: ts.timestamp_millis(),
}
}
}
impl TryFrom<(i64, &Tz)> for Timestamp {
type Error = anyhow::Error;
fn try_from((unix_ts, src_tz): (i64, &Tz)) -> Result<Self, Self::Error> {
let ts = match src_tz.from_local_datetime(&NaiveDateTime::from_timestamp_opt(unix_ts, 0).unwrap()) {
LocalResult::None => {
return Err(anyhow!("INVALID DATETIME"));
}
LocalResult::Single(t) => t,
LocalResult::Ambiguous(t1, _t2) => t1,
};
Ok(
Self {
ts: ts.timestamp_millis(),
}
)
}
}
impl From<Timestamp> for i64 {
fn from(ts: Timestamp) -> Self {
ts.ts
}
}
impl From<&Timestamp> for Value {
fn from(ts: &Timestamp) -> Self {
json!(ts.ts)
}
}
impl Timestamp {
pub fn timestamp_millis(&self) -> i64 {
self.ts
}
}