use std::str::FromStr;
use chrono::{DateTime, Duration, NaiveDate, NaiveDateTime, TimeZone as _, Utc};
use crate::{Property, ValueType};
const NAIVE_DATE_TIME_FORMAT: &str = "%Y%m%dT%H%M%S";
const UTC_DATE_TIME_FORMAT: &str = "%Y%m%dT%H%M%SZ";
const NAIVE_DATE_FORMAT: &str = "%Y%m%d";
pub(crate) fn parse_utc_date_time(s: &str) -> Option<DateTime<Utc>> {
Utc.datetime_from_str(s, UTC_DATE_TIME_FORMAT).ok()
}
pub(crate) fn parse_naive_date_time(s: &str) -> Option<NaiveDateTime> {
NaiveDateTime::parse_from_str(s, NAIVE_DATE_TIME_FORMAT).ok()
}
pub(crate) fn format_utc_date_time(utc_dt: DateTime<Utc>) -> String {
utc_dt.format(UTC_DATE_TIME_FORMAT).to_string()
}
pub(crate) fn parse_duration(s: &str) -> Option<Duration> {
iso8601::duration(s)
.ok()
.and_then(|iso| Duration::from_std(iso.into()).ok())
}
pub(crate) fn naive_date_to_property(date: NaiveDate, key: &str) -> Property {
Property::new(key, date.format(NAIVE_DATE_FORMAT).to_string())
.append_parameter(ValueType::Date)
.done()
}
#[derive(Clone, Debug, Eq, PartialEq)]
pub enum CalendarDateTime {
Floating(NaiveDateTime),
Utc(DateTime<Utc>),
WithTimezone {
date_time: NaiveDateTime,
tzid: String,
},
}
impl CalendarDateTime {
#[cfg(test)]
pub(crate) fn now() -> Self {
NaiveDate::from_ymd_opt(2015, 10, 26)
.unwrap()
.and_hms_opt(1, 22, 00)
.unwrap()
.into()
}
pub(crate) fn from_property(property: &Property) -> Option<Self> {
let value = property.value();
if let Some(tzid) = property.params().get("TZID") {
Some(Self::WithTimezone {
date_time: NaiveDateTime::parse_from_str(value, NAIVE_DATE_TIME_FORMAT).ok()?,
tzid: tzid.value().to_owned(),
})
} else if let Ok(naive_date_time) =
NaiveDateTime::parse_from_str(value, NAIVE_DATE_TIME_FORMAT)
{
Some(naive_date_time.into())
} else {
Self::from_str(value).ok()
}
}
pub(crate) fn to_property(&self, key: &str) -> Property {
match self {
CalendarDateTime::Floating(naive_dt) => {
Property::new(key, naive_dt.format(NAIVE_DATE_TIME_FORMAT).to_string())
}
CalendarDateTime::Utc(utc_dt) => Property::new(key, format_utc_date_time(*utc_dt)),
CalendarDateTime::WithTimezone { date_time, tzid } => {
Property::new(key, date_time.format(NAIVE_DATE_TIME_FORMAT).to_string())
.add_parameter("TZID", tzid)
.done()
}
}
}
pub(crate) fn from_utc_string(s: &str) -> Option<Self> {
parse_utc_date_time(s).map(CalendarDateTime::Utc)
}
pub(crate) fn from_naive_string(s: &str) -> Option<Self> {
parse_naive_date_time(s).map(CalendarDateTime::Floating)
}
#[cfg(feature = "chrono-tz")]
pub fn try_into_utc(&self) -> Option<DateTime<Utc>> {
match self {
CalendarDateTime::Floating(_) => None, CalendarDateTime::Utc(inner) => Some(*inner),
CalendarDateTime::WithTimezone { date_time, tzid } => tzid
.parse::<chrono_tz::Tz>()
.ok()
.and_then(|tz| tz.from_local_datetime(date_time).single())
.map(|tz| tz.with_timezone(&Utc)),
}
}
#[cfg(feature = "chrono-tz")]
#[allow(dead_code)]
pub(crate) fn with_timezone(dt: NaiveDateTime, tz_id: chrono_tz::Tz) -> Self {
Self::WithTimezone {
date_time: dt,
tzid: tz_id.name().to_owned(),
}
}
#[cfg(feature = "chrono-tz")]
pub fn from_ymd_hm_tzid(
year: i32,
month: u32,
day: u32,
hour: u32,
min: u32,
tz_id: chrono_tz::Tz,
) -> Option<Self> {
NaiveDate::from_ymd_opt(year, month, day)
.and_then(|date| date.and_hms_opt(hour, min, 0))
.zip(Some(tz_id))
.map(|(dt, tz)| Self::with_timezone(dt, tz))
}
#[cfg(feature = "chrono-tz")]
pub fn from_date_time<TZ: chrono::TimeZone<Offset = O>, O: chrono_tz::OffsetName>(
dt: DateTime<TZ>,
) -> Self {
Self::WithTimezone {
date_time: dt.naive_local(),
tzid: dt.offset().tz_id().to_owned(),
}
}
}
#[cfg(feature = "chrono-tz")]
pub fn ymd_hm_tzid(
year: i32,
month: u32,
day: u32,
hour: u32,
min: u32,
tz_id: chrono_tz::Tz,
) -> Option<CalendarDateTime> {
CalendarDateTime::from_ymd_hm_tzid(year, month, day, hour, min, tz_id)
}
impl From<DateTime<Utc>> for CalendarDateTime {
fn from(dt: DateTime<Utc>) -> Self {
Self::Utc(dt)
}
}
impl From<NaiveDateTime> for CalendarDateTime {
fn from(dt: NaiveDateTime) -> Self {
Self::Floating(dt)
}
}
#[cfg(feature = "chrono-tz")]
impl From<(NaiveDateTime, chrono_tz::Tz)> for CalendarDateTime {
fn from((date_time, tzid): (NaiveDateTime, chrono_tz::Tz)) -> Self {
Self::WithTimezone {
date_time,
tzid: tzid.name().into(),
}
}
}
#[cfg(feature = "chrono-tz")]
impl TryFrom<(NaiveDateTime, &str)> for CalendarDateTime {
type Error = String;
fn try_from((dt, maybe_tz): (NaiveDateTime, &str)) -> Result<Self, Self::Error> {
let tzid: chrono_tz::Tz = maybe_tz
.parse()
.map_err(|e: chrono_tz::ParseError| e.to_string())?;
Ok(CalendarDateTime::from((dt, tzid)))
}
}
impl FromStr for CalendarDateTime {
type Err = ();
fn from_str(s: &str) -> Result<Self, Self::Err> {
CalendarDateTime::from_utc_string(s)
.or_else(|| CalendarDateTime::from_naive_string(s))
.ok_or(())
}
}
#[derive(Clone, Debug, Eq, PartialEq)]
pub enum DatePerhapsTime {
DateTime(CalendarDateTime),
Date(NaiveDate),
}
impl DatePerhapsTime {
pub(crate) fn from_property(property: &Property) -> Option<Self> {
if property.value_type() == Some(ValueType::Date) {
Some(
NaiveDate::parse_from_str(property.value(), NAIVE_DATE_FORMAT)
.ok()?
.into(),
)
} else {
Some(CalendarDateTime::from_property(property)?.into())
}
}
pub(crate) fn to_property(&self, key: &str) -> Property {
match self {
Self::DateTime(date_time) => date_time.to_property(key),
Self::Date(date) => naive_date_to_property(*date, key),
}
}
}
#[cfg(feature = "chrono-tz")]
#[allow(dead_code)]
pub fn with_timezone<T: chrono::TimeZone + chrono_tz::OffsetName>(
dt: DateTime<T>,
) -> DatePerhapsTime {
CalendarDateTime::WithTimezone {
date_time: dt.naive_local(),
tzid: dt.timezone().tz_id().to_owned(),
}
.into()
}
impl From<CalendarDateTime> for DatePerhapsTime {
fn from(dt: CalendarDateTime) -> Self {
Self::DateTime(dt)
}
}
impl From<DateTime<Utc>> for DatePerhapsTime {
fn from(dt: DateTime<Utc>) -> Self {
Self::DateTime(CalendarDateTime::Utc(dt))
}
}
#[allow(deprecated)]
impl From<chrono::Date<Utc>> for DatePerhapsTime {
fn from(dt: chrono::Date<Utc>) -> Self {
Self::Date(dt.naive_utc())
}
}
impl From<NaiveDateTime> for DatePerhapsTime {
fn from(dt: NaiveDateTime) -> Self {
Self::DateTime(dt.into())
}
}
#[cfg(feature = "chrono-tz")]
impl TryFrom<(NaiveDateTime, &str)> for DatePerhapsTime {
type Error = String;
fn try_from(value: (NaiveDateTime, &str)) -> Result<Self, Self::Error> {
Ok(Self::DateTime(value.try_into()?))
}
}
#[cfg(feature = "chrono-tz")]
impl From<(NaiveDateTime, chrono_tz::Tz)> for DatePerhapsTime {
fn from(both: (NaiveDateTime, chrono_tz::Tz)) -> Self {
Self::DateTime(both.into())
}
}
impl From<NaiveDate> for DatePerhapsTime {
fn from(date: NaiveDate) -> Self {
Self::Date(date)
}
}
#[cfg(feature = "parser")]
impl TryFrom<&crate::parser::Property<'_>> for DatePerhapsTime {
type Error = &'static str;
fn try_from(value: &crate::parser::Property) -> Result<Self, Self::Error> {
let val = value.val.as_ref();
if let Ok(utc_dt) = Utc.datetime_from_str(val, "%Y%m%dT%H%M%SZ") {
return Ok(Self::DateTime(CalendarDateTime::Utc(utc_dt)));
};
if let Ok(naive_date) = NaiveDate::parse_from_str(val, "%Y%m%d") {
return Ok(Self::Date(naive_date));
};
if let Ok(naive_dt) = NaiveDateTime::parse_from_str(val, "%Y%m%dT%H%M%S") {
if let Some(tz_param) = value.params.iter().find(|p| p.key == "TZID") {
if let Some(tzid) = &tz_param.val {
return Ok(Self::DateTime(CalendarDateTime::WithTimezone {
date_time: naive_dt,
tzid: tzid.as_ref().to_string(),
}));
} else {
return Err("Found empty TZID param.");
}
} else {
return Ok(Self::DateTime(CalendarDateTime::Floating(naive_dt)));
};
};
Err("Value does not look like a known DATE-TIME")
}
}
#[cfg(all(test, feature = "parser"))]
mod try_from_tests {
use super::*;
#[test]
fn try_from_utc_dt() {
let prop = crate::parser::Property {
name: "TRIGGER".into(),
val: "20220716T141500Z".into(),
params: vec![crate::parser::Parameter {
key: "VALUE".into(),
val: Some("DATE-TIME".into()),
}],
};
let result = DatePerhapsTime::try_from(&prop);
let expected = Utc.ymd(2022, 7, 16).and_hms(14, 15, 0);
assert_eq!(
result,
Ok(DatePerhapsTime::DateTime(CalendarDateTime::Utc(expected)))
);
}
#[test]
fn try_from_naive_date() {
let prop = crate::parser::Property {
name: "TRIGGER".into(),
val: "19970714".into(),
params: vec![crate::parser::Parameter {
key: "VALUE".into(),
val: Some("DATE-TIME".into()),
}],
};
let result = DatePerhapsTime::try_from(&prop);
let expected = NaiveDate::from_ymd(1997, 7, 14);
assert_eq!(result, Ok(DatePerhapsTime::Date(expected)));
}
#[test]
fn try_from_dt_with_tz() {
let prop = crate::parser::Property {
name: "TRIGGER".into(),
val: "20220716T141500".into(),
params: vec![
crate::parser::Parameter {
key: "VALUE".into(),
val: Some("DATE-TIME".into()),
},
crate::parser::Parameter {
key: "TZID".into(),
val: Some("MY-TZ-ID".into()),
},
],
};
let result = DatePerhapsTime::try_from(&prop);
let expected = NaiveDate::from_ymd(2022, 7, 16).and_hms(14, 15, 0);
assert_eq!(
result,
Ok(DatePerhapsTime::DateTime(CalendarDateTime::WithTimezone {
date_time: expected,
tzid: "MY-TZ-ID".into(),
}))
);
}
#[test]
fn try_from_dt_with_empty_tz() {
let prop = crate::parser::Property {
name: "TRIGGER".into(),
val: "20220716T141500".into(),
params: vec![
crate::parser::Parameter {
key: "VALUE".into(),
val: Some("DATE-TIME".into()),
},
crate::parser::Parameter {
key: "TZID".into(),
val: None,
},
],
};
let result = DatePerhapsTime::try_from(&prop);
assert_eq!(result, Err("Found empty TZID param."));
}
#[test]
fn try_from_floating_dt() {
let prop = crate::parser::Property {
name: "TRIGGER".into(),
val: "20220716T141500".into(),
params: vec![crate::parser::Parameter {
key: "VALUE".into(),
val: Some("DATE-TIME".into()),
}],
};
let result = DatePerhapsTime::try_from(&prop);
let expected = NaiveDate::from_ymd(2022, 7, 16).and_hms(14, 15, 0);
assert_eq!(
result,
Ok(DatePerhapsTime::DateTime(CalendarDateTime::Floating(
expected
)))
);
}
#[test]
fn try_from_non_dt_prop() {
let prop = crate::parser::Property {
name: "TZNAME".into(),
val: "CET".into(),
params: vec![],
};
let result = DatePerhapsTime::try_from(&prop);
assert_eq!(result, Err("Value does not look like a known DATE-TIME"));
}
}