use core::fmt;
use crate::Duration;
use crate::Timestamp;
use crate::TimestampError;
#[derive(Debug, Default, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
pub(crate) struct DateTime {
pub(crate) year: i64,
pub(crate) month: u8,
pub(crate) day: u8,
pub(crate) hour: u8,
pub(crate) minute: u8,
pub(crate) second: u8,
pub(crate) nanos: u32,
}
impl DateTime {
pub(crate) const MIN: DateTime = DateTime {
year: -292_277_022_657,
month: 1,
day: 27,
hour: 8,
minute: 29,
second: 52,
nanos: 0,
};
pub(crate) const MAX: DateTime = DateTime {
year: 292_277_026_596,
month: 12,
day: 4,
hour: 15,
minute: 30,
second: 7,
nanos: 999_999_999,
};
pub(crate) fn is_valid(&self) -> bool {
self >= &DateTime::MIN
&& self <= &DateTime::MAX
&& self.month > 0
&& self.month <= 12
&& self.day > 0
&& self.day <= days_in_month(self.year, self.month)
&& self.hour < 24
&& self.minute < 60
&& self.second < 60
&& self.nanos < 1_000_000_000
}
}
impl fmt::Display for DateTime {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
if self.year > 9999 {
write!(f, "+{}", self.year)?;
} else if self.year < 0 {
write!(f, "{:05}", self.year)?;
} else {
write!(f, "{:04}", self.year)?;
};
write!(
f,
"-{:02}-{:02}T{:02}:{:02}:{:02}",
self.month, self.day, self.hour, self.minute, self.second,
)?;
let nanos = self.nanos;
if nanos == 0 {
write!(f, "Z")
} else if nanos % 1_000_000 == 0 {
write!(f, ".{:03}Z", nanos / 1_000_000)
} else if nanos % 1_000 == 0 {
write!(f, ".{:06}Z", nanos / 1_000)
} else {
write!(f, ".{:09}Z", nanos)
}
}
}
impl From<Timestamp> for DateTime {
fn from(mut timestamp: Timestamp) -> DateTime {
timestamp.normalize();
let t = timestamp.seconds;
let nanos = timestamp.nanos;
const LEAPOCH: i64 = 946_684_800 + 86400 * (31 + 29);
const DAYS_PER_400Y: i32 = 365 * 400 + 97;
const DAYS_PER_100Y: i32 = 365 * 100 + 24;
const DAYS_PER_4Y: i32 = 365 * 4 + 1;
const DAYS_IN_MONTH: [u8; 12] = [31, 30, 31, 30, 31, 31, 30, 31, 30, 31, 31, 29];
let mut days: i64 = (t / 86_400) - (LEAPOCH / 86_400);
let mut remsecs: i32 = (t % 86_400) as i32;
if remsecs < 0i32 {
remsecs += 86_400;
days -= 1
}
let mut qc_cycles: i32 = (days / i64::from(DAYS_PER_400Y)) as i32;
let mut remdays: i32 = (days % i64::from(DAYS_PER_400Y)) as i32;
if remdays < 0 {
remdays += DAYS_PER_400Y;
qc_cycles -= 1;
}
let mut c_cycles: i32 = remdays / DAYS_PER_100Y;
if c_cycles == 4 {
c_cycles -= 1;
}
remdays -= c_cycles * DAYS_PER_100Y;
let mut q_cycles: i32 = remdays / DAYS_PER_4Y;
if q_cycles == 25 {
q_cycles -= 1;
}
remdays -= q_cycles * DAYS_PER_4Y;
let mut remyears: i32 = remdays / 365;
if remyears == 4 {
remyears -= 1;
}
remdays -= remyears * 365;
let mut years: i64 = i64::from(remyears)
+ 4 * i64::from(q_cycles)
+ 100 * i64::from(c_cycles)
+ 400 * i64::from(qc_cycles);
let mut months: i32 = 0;
while i32::from(DAYS_IN_MONTH[months as usize]) <= remdays {
remdays -= i32::from(DAYS_IN_MONTH[months as usize]);
months += 1
}
if months >= 10 {
months -= 12;
years += 1;
}
let date_time = DateTime {
year: years + 2000,
month: (months + 3) as u8,
day: (remdays + 1) as u8,
hour: (remsecs / 3600) as u8,
minute: (remsecs / 60 % 60) as u8,
second: (remsecs % 60) as u8,
nanos: nanos as u32,
};
debug_assert!(date_time.is_valid());
date_time
}
}
fn days_in_month(year: i64, month: u8) -> u8 {
const DAYS_IN_MONTH: [u8; 12] = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31];
let (_, is_leap) = year_to_seconds(year);
DAYS_IN_MONTH[usize::from(month - 1)] + u8::from(is_leap && month == 2)
}
macro_rules! ensure {
($expr:expr) => {{
if !$expr {
return None;
}
}};
}
fn parse_date(s: &str) -> Option<(i64, u8, u8, &str)> {
debug_assert!(s.is_ascii());
ensure!(s.len() >= 10);
let (year, s) = match s.as_bytes()[0] {
b'+' => {
let (digits, s) = parse_digits(&s[1..]);
ensure!(digits.len() >= 5);
let date: i64 = digits.parse().ok()?;
(date, s)
}
b'-' => {
let (digits, s) = parse_digits(&s[1..]);
ensure!(digits.len() >= 4);
let date: i64 = digits.parse().ok()?;
(-date, s)
}
_ => {
let (n1, s) = parse_two_digit_numeric(s)?;
let (n2, s) = parse_two_digit_numeric(s)?;
(i64::from(n1) * 100 + i64::from(n2), s)
}
};
let s = parse_char(s, b'-')?;
let (month, s) = parse_two_digit_numeric(s)?;
let s = parse_char(s, b'-')?;
let (day, s) = parse_two_digit_numeric(s)?;
Some((year, month, day, s))
}
fn parse_time(s: &str) -> Option<(u8, u8, u8, u32, &str)> {
debug_assert!(s.is_ascii());
let (hour, s) = parse_two_digit_numeric(s)?;
let s = parse_char(s, b':')?;
let (minute, s) = parse_two_digit_numeric(s)?;
let s = parse_char(s, b':')?;
let (second, s) = parse_two_digit_numeric(s)?;
let (nanos, s) = parse_nanos(s)?;
Some((hour, minute, second, nanos, s))
}
fn parse_nanos(s: &str) -> Option<(u32, &str)> {
debug_assert!(s.is_ascii());
let (nanos, s) = if let Some(s) = parse_char(s, b'.') {
let (mut digits, s) = parse_digits(s);
if digits.len() > 9 {
digits = digits.split_at(9).0;
}
let nanos = 10u32.pow(9 - digits.len() as u32) * digits.parse::<u32>().ok()?;
(nanos, s)
} else {
(0, s)
};
Some((nanos, s))
}
fn parse_offset(s: &str) -> Option<(i8, i8, &str)> {
debug_assert!(s.is_ascii());
if s.is_empty() {
return Some((0, 0, s));
}
let s = parse_char(s, b' ').unwrap_or(s);
if let Some(s) = parse_char_ignore_case(s, b'Z') {
Some((0, 0, s))
} else {
let (is_positive, s) = if let Some(s) = parse_char(s, b'+') {
(true, s)
} else if let Some(s) = parse_char(s, b'-') {
(false, s)
} else {
return None;
};
let (hour, s) = parse_two_digit_numeric(s)?;
let (minute, s) = if s.is_empty() {
(0, s)
} else {
let s = parse_char(s, b':').unwrap_or(s);
let (minute, s) = parse_two_digit_numeric(s)?;
(minute, s)
};
ensure!(hour < 24 && minute < 60);
let hour = hour as i8;
let minute = minute as i8;
if is_positive {
Some((hour, minute, s))
} else {
Some((-hour, -minute, s))
}
}
}
fn parse_two_digit_numeric(s: &str) -> Option<(u8, &str)> {
debug_assert!(s.is_ascii());
if s.len() < 2 {
return None;
}
if s.starts_with('+') {
return None;
}
let (digits, s) = s.split_at(2);
Some((digits.parse().ok()?, s))
}
fn parse_digits(s: &str) -> (&str, &str) {
debug_assert!(s.is_ascii());
let idx = s
.as_bytes()
.iter()
.position(|c| !c.is_ascii_digit())
.unwrap_or(s.len());
s.split_at(idx)
}
fn parse_char(s: &str, c: u8) -> Option<&str> {
debug_assert!(s.is_ascii());
ensure!(*s.as_bytes().first()? == c);
Some(&s[1..])
}
fn parse_char_ignore_case(s: &str, c: u8) -> Option<&str> {
debug_assert!(s.is_ascii());
ensure!(s.as_bytes().first()?.eq_ignore_ascii_case(&c));
Some(&s[1..])
}
fn date_time_to_seconds(tm: &DateTime) -> i64 {
let (start_of_year, is_leap) = year_to_seconds(tm.year);
let seconds_within_year = month_to_seconds(tm.month, is_leap)
+ 86400 * u32::from(tm.day - 1)
+ 3600 * u32::from(tm.hour)
+ 60 * u32::from(tm.minute)
+ u32::from(tm.second);
(start_of_year + i128::from(seconds_within_year)) as i64
}
fn month_to_seconds(month: u8, is_leap: bool) -> u32 {
const SECS_THROUGH_MONTH: [u32; 12] = [
0,
31 * 86400,
59 * 86400,
90 * 86400,
120 * 86400,
151 * 86400,
181 * 86400,
212 * 86400,
243 * 86400,
273 * 86400,
304 * 86400,
334 * 86400,
];
let t = SECS_THROUGH_MONTH[usize::from(month - 1)];
if is_leap && month > 2 {
t + 86400
} else {
t
}
}
pub(crate) fn year_to_seconds(year: i64) -> (i128, bool) {
let is_leap;
let year = year - 1900;
if (1..=138).contains(&year) {
let mut leaps: i64 = (year - 68) >> 2;
if (year - 68).trailing_zeros() >= 2 {
leaps -= 1;
is_leap = true;
} else {
is_leap = false;
}
return (
i128::from(31_536_000 * (year - 70) + 86400 * leaps),
is_leap,
);
}
let centuries: i64;
let mut leaps: i64;
let mut cycles: i64 = (year - 100) / 400;
let mut rem: i64 = (year - 100) % 400;
if rem < 0 {
cycles -= 1;
rem += 400
}
if rem == 0 {
is_leap = true;
centuries = 0;
leaps = 0;
} else {
if rem >= 200 {
if rem >= 300 {
centuries = 3;
rem -= 300;
} else {
centuries = 2;
rem -= 200;
}
} else if rem >= 100 {
centuries = 1;
rem -= 100;
} else {
centuries = 0;
}
if rem == 0 {
is_leap = false;
leaps = 0;
} else {
leaps = rem / 4;
rem %= 4;
is_leap = rem == 0;
}
}
leaps += 97 * cycles + 24 * centuries - i64::from(is_leap);
(
i128::from((year - 100) * 31_536_000) + i128::from(leaps * 86400 + 946_684_800 + 86400),
is_leap,
)
}
pub(crate) fn parse_timestamp(s: &str) -> Option<Timestamp> {
ensure!(s.is_ascii());
let (year, month, day, s) = parse_date(s)?;
if s.is_empty() {
let date_time = DateTime {
year,
month,
day,
..DateTime::default()
};
return Timestamp::try_from(date_time).ok();
}
let s = parse_char_ignore_case(s, b'T').or_else(|| parse_char(s, b' '))?;
let (hour, minute, mut second, nanos, s) = parse_time(s)?;
let (offset_hour, offset_minute, s) = parse_offset(s)?;
ensure!(s.is_empty());
if second == 60 {
second = 59;
}
let date_time = DateTime {
year,
month,
day,
hour,
minute,
second,
nanos,
};
let Timestamp { seconds, nanos } = Timestamp::try_from(date_time).ok()?;
let seconds =
seconds.checked_sub(i64::from(offset_hour) * 3600 + i64::from(offset_minute) * 60)?;
Some(Timestamp { seconds, nanos })
}
pub(crate) fn parse_duration(s: &str) -> Option<Duration> {
ensure!(s.is_ascii());
let (is_negative, s) = match parse_char(s, b'-') {
Some(s) => (true, s),
None => (false, s),
};
let (digits, s) = parse_digits(s);
let seconds = digits.parse::<i64>().ok()?;
let (nanos, s) = parse_nanos(s)?;
let s = parse_char(s, b's')?;
ensure!(s.is_empty());
ensure!(nanos < crate::NANOS_PER_SECOND as u32);
let (seconds, nanos) = if is_negative {
(-seconds, -(nanos as i32))
} else {
(seconds, nanos as i32)
};
Some(Duration { seconds, nanos })
}
impl TryFrom<DateTime> for Timestamp {
type Error = TimestampError;
fn try_from(date_time: DateTime) -> Result<Timestamp, TimestampError> {
if !date_time.is_valid() {
return Err(TimestampError::InvalidDateTime);
}
let seconds = date_time_to_seconds(&date_time);
let nanos = date_time.nanos;
Ok(Timestamp {
seconds,
nanos: nanos as i32,
})
}
}
#[cfg(test)]
mod tests {
use super::*;
use proptest::prelude::*;
use prost::alloc::format;
#[test]
fn test_min_max() {
assert_eq!(
DateTime::MIN,
DateTime::from(Timestamp {
seconds: i64::MIN,
nanos: 0
}),
);
assert_eq!(
DateTime::MAX,
DateTime::from(Timestamp {
seconds: i64::MAX,
nanos: 999_999_999
}),
);
}
#[cfg(feature = "std")]
#[test]
fn test_datetime_from_timestamp() {
let case = |expected: &str, secs: i64, nanos: i32| {
let timestamp = Timestamp {
seconds: secs,
nanos,
};
assert_eq!(
expected,
format!("{}", DateTime::from(timestamp)),
"timestamp: {:?}",
timestamp
);
};
case("1970-01-01T00:00:00Z", 0, 0);
case("1970-01-01T00:00:00.000000001Z", 0, 1);
case("1970-01-01T00:00:00.123450Z", 0, 123_450_000);
case("1970-01-01T00:00:00.050Z", 0, 50_000_000);
case("1970-01-01T00:00:01.000000001Z", 1, 1);
case("1970-01-01T00:01:01.000000001Z", 60 + 1, 1);
case("1970-01-01T01:01:01.000000001Z", 60 * 60 + 60 + 1, 1);
case(
"1970-01-02T01:01:01.000000001Z",
24 * 60 * 60 + 60 * 60 + 60 + 1,
1,
);
case("1969-12-31T23:59:59Z", -1, 0);
case("1969-12-31T23:59:59.000001Z", -1, 1_000);
case("1969-12-31T23:59:59.500Z", -1, 500_000_000);
case("1969-12-31T23:58:59.000001Z", -60 - 1, 1_000);
case("1969-12-31T22:58:59.000001Z", -60 * 60 - 60 - 1, 1_000);
case(
"1969-12-30T22:58:59.000000001Z",
-24 * 60 * 60 - 60 * 60 - 60 - 1,
1,
);
case("2038-01-19T03:14:07Z", i32::MAX as i64, 0);
case("2038-01-19T03:14:08Z", i32::MAX as i64 + 1, 0);
case("1901-12-13T20:45:52Z", i32::MIN as i64, 0);
case("1901-12-13T20:45:51Z", i32::MIN as i64 - 1, 0);
#[cfg(not(target_os = "windows"))]
{
case("+292277026596-12-04T15:30:07Z", i64::MAX, 0);
case("+292277026596-12-04T15:30:06Z", i64::MAX - 1, 0);
case("-292277022657-01-27T08:29:53Z", i64::MIN + 1, 0);
}
case("1900-01-01T00:00:00Z", -2_208_988_800, 0);
case("1899-12-31T23:59:59Z", -2_208_988_801, 0);
case("0000-01-01T00:00:00Z", -62_167_219_200, 0);
case("-0001-12-31T23:59:59Z", -62_167_219_201, 0);
case("1234-05-06T07:08:09Z", -23_215_049_511, 0);
case("-1234-05-06T07:08:09Z", -101_097_651_111, 0);
case("2345-06-07T08:09:01Z", 11_847_456_541, 0);
case("-2345-06-07T08:09:01Z", -136_154_620_259, 0);
}
#[test]
fn test_parse_timestamp() {
assert_eq!(
"1985-04-12T23:20:50.52Z".parse::<Timestamp>(),
Timestamp::date_time_nanos(1985, 4, 12, 23, 20, 50, 520_000_000),
);
assert_eq!(
"1996-12-19T16:39:57-08:00".parse::<Timestamp>(),
Timestamp::date_time(1996, 12, 20, 0, 39, 57),
);
assert_eq!(
"1996-12-19T16:39:57-08:00".parse::<Timestamp>(),
Timestamp::date_time(1996, 12, 20, 0, 39, 57),
);
assert_eq!(
"1990-12-31T23:59:60Z".parse::<Timestamp>(),
Timestamp::date_time(1990, 12, 31, 23, 59, 59),
);
assert_eq!(
"1990-12-31T15:59:60-08:00".parse::<Timestamp>(),
Timestamp::date_time(1990, 12, 31, 23, 59, 59),
);
assert_eq!(
"1937-01-01T12:00:27.87+00:20".parse::<Timestamp>(),
Timestamp::date_time_nanos(1937, 1, 1, 11, 40, 27, 870_000_000),
);
assert_eq!(
"1937-01-01".parse::<Timestamp>(),
Timestamp::date(1937, 1, 1),
);
assert_eq!(
"-0008-01-01".parse::<Timestamp>(),
Timestamp::date(-8, 1, 1),
);
assert_eq!(
"+19370-01-01".parse::<Timestamp>(),
Timestamp::date(19370, 1, 1),
);
assert_eq!(
"2020-02-03T01:02:03.123456789Z".parse::<Timestamp>(),
Timestamp::date_time_nanos(2020, 2, 3, 1, 2, 3, 123_456_789),
);
assert_eq!(
"2020-02-29T01:02:03.00Z".parse::<Timestamp>(),
Timestamp::try_from(DateTime {
year: 2020,
month: 2,
day: 29,
hour: 1,
minute: 2,
second: 3,
nanos: 0,
})
);
assert_eq!(
"1985-04-12 23:20:50.52Z".parse::<Timestamp>(),
Timestamp::date_time_nanos(1985, 4, 12, 23, 20, 50, 520_000_000),
);
assert_eq!(
"1985-04-12T23:20:50.52".parse::<Timestamp>(),
Timestamp::date_time_nanos(1985, 4, 12, 23, 20, 50, 520_000_000),
);
assert_eq!(
"1996-12-19T16:39:57-08".parse::<Timestamp>(),
Timestamp::date_time(1996, 12, 20, 0, 39, 57),
);
assert_eq!(
"2015-09-12 00:47:19.591 Z".parse::<Timestamp>(),
Timestamp::date_time_nanos(2015, 9, 12, 0, 47, 19, 591_000_000),
);
assert_eq!(
"2020-06-15 00:01:02.123 +0800".parse::<Timestamp>(),
Timestamp::date_time_nanos(2020, 6, 14, 16, 1, 2, 123_000_000),
);
assert_eq!(
"-11111111-z".parse::<Timestamp>(),
Err(crate::TimestampError::ParseFailure),
);
assert_eq!(
"1900-01-10".parse::<Timestamp>(),
Ok(Timestamp {
seconds: -2208211200,
nanos: 0
}),
);
assert_eq!(
"19+1-+2-+3T+4:+5:+6Z".parse::<Timestamp>(),
Err(crate::TimestampError::ParseFailure),
);
assert_eq!(
"1343-08-16 18:33:44.1666666666666666666666666666666666666666666666666666666666666666666666666666666666666666666666666666666666666666666666666660404z".parse::<Timestamp>(),
Timestamp::date_time_nanos(1343, 8, 16, 18, 33, 44, 166_666_666),
);
}
#[test]
fn test_parse_duration() {
let case = |s: &str, seconds: i64, nanos: i32| {
assert_eq!(
s.parse::<Duration>().unwrap(),
Duration { seconds, nanos },
"duration: {}",
s
);
};
case("0s", 0, 0);
case("0.0s", 0, 0);
case("0.000s", 0, 0);
case("-0s", 0, 0);
case("-0.0s", 0, 0);
case("-0.000s", 0, 0);
case("-0s", 0, 0);
case("-0.0s", 0, 0);
case("-0.000s", 0, 0);
case("0.05s", 0, 50_000_000);
case("0.050s", 0, 50_000_000);
case("-0.05s", 0, -50_000_000);
case("-0.050s", 0, -50_000_000);
case("1s", 1, 0);
case("1.0s", 1, 0);
case("1.000s", 1, 0);
case("-1s", -1, 0);
case("-1.0s", -1, 0);
case("-1.000s", -1, 0);
case("15s", 15, 0);
case("15.1s", 15, 100_000_000);
case("15.100s", 15, 100_000_000);
case("-15s", -15, 0);
case("-15.1s", -15, -100_000_000);
case("-15.100s", -15, -100_000_000);
case("100.000000009s", 100, 9);
case("-100.000000009s", -100, -9);
}
#[test]
fn test_parse_non_ascii() {
assert!("2021️⃣-06-15 00:01:02.123 +0800"
.parse::<Timestamp>()
.is_err());
assert!("1️⃣s".parse::<Duration>().is_err());
}
#[test]
fn check_invalid_datetimes() {
assert_eq!(
Timestamp::try_from(DateTime {
year: i64::from_le_bytes([178, 2, 0, 0, 0, 0, 0, 128]),
month: 2,
day: 2,
hour: 8,
minute: 58,
second: 8,
nanos: u32::from_le_bytes([0, 0, 0, 50]),
}),
Err(TimestampError::InvalidDateTime)
);
assert_eq!(
Timestamp::try_from(DateTime {
year: i64::from_le_bytes([132, 7, 0, 0, 0, 0, 0, 128]),
month: 2,
day: 2,
hour: 8,
minute: 58,
second: 8,
nanos: u32::from_le_bytes([0, 0, 0, 50]),
}),
Err(TimestampError::InvalidDateTime)
);
assert_eq!(
Timestamp::try_from(DateTime {
year: i64::from_le_bytes([80, 96, 32, 240, 99, 0, 32, 180]),
month: 1,
day: 18,
hour: 19,
minute: 26,
second: 8,
nanos: u32::from_le_bytes([0, 0, 0, 50]),
}),
Err(TimestampError::InvalidDateTime)
);
assert_eq!(
Timestamp::try_from(DateTime {
year: DateTime::MIN.year - 1,
month: 0,
day: 0,
hour: 0,
minute: 0,
second: 0,
nanos: 0,
}),
Err(TimestampError::InvalidDateTime)
);
assert_eq!(
Timestamp::try_from(DateTime {
year: i64::MIN,
month: 0,
day: 0,
hour: 0,
minute: 0,
second: 0,
nanos: 0,
}),
Err(TimestampError::InvalidDateTime)
);
assert_eq!(
Timestamp::try_from(DateTime {
year: DateTime::MAX.year + 1,
month: u8::MAX,
day: u8::MAX,
hour: u8::MAX,
minute: u8::MAX,
second: u8::MAX,
nanos: u32::MAX,
}),
Err(TimestampError::InvalidDateTime)
);
assert_eq!(
Timestamp::try_from(DateTime {
year: i64::MAX,
month: u8::MAX,
day: u8::MAX,
hour: u8::MAX,
minute: u8::MAX,
second: u8::MAX,
nanos: u32::MAX,
}),
Err(TimestampError::InvalidDateTime)
);
}
proptest! {
#[cfg(feature = "std")]
#[test]
fn check_timestamp_parse_to_string_roundtrip(
system_time in std::time::SystemTime::arbitrary(),
) {
let ts = Timestamp::from(system_time);
assert_eq!(
ts,
ts.to_string().parse::<Timestamp>().unwrap(),
)
}
#[cfg(feature = "std")]
#[test]
fn check_duration_parse_to_string_roundtrip(
duration in core::time::Duration::arbitrary(),
) {
let duration = match Duration::try_from(duration) {
Ok(duration) => duration,
Err(_) => return Err(TestCaseError::reject("duration out of range")),
};
prop_assert_eq!(
&duration,
&duration.to_string().parse::<Duration>().unwrap(),
"{}", duration.to_string()
);
}
#[test]
fn check_timestamp_roundtrip_with_date_time(
seconds in i64::arbitrary(),
nanos in i32::arbitrary(),
) {
let timestamp = Timestamp { seconds, nanos };
let date_time = DateTime::from(timestamp);
let roundtrip = Timestamp::try_from(date_time).unwrap();
prop_assert_eq!(timestamp.normalized(), roundtrip);
}
#[test]
fn check_date_time_roundtrip_with_timestamp(
year in i64::arbitrary(),
month in u8::arbitrary(),
day in u8::arbitrary(),
hour in u8::arbitrary(),
minute in u8::arbitrary(),
second in u8::arbitrary(),
nanos in u32::arbitrary(),
) {
let date_time = DateTime {
year,
month,
day,
hour,
minute,
second,
nanos
};
if date_time.is_valid() {
let timestamp = Timestamp::try_from(date_time).unwrap();
let roundtrip = DateTime::from(timestamp);
prop_assert_eq!(date_time, roundtrip);
} else {
prop_assert_eq!(Timestamp::try_from(date_time), Err(TimestampError::InvalidDateTime));
}
}
}
}