2015-01-30 02:58:25 +00:00
|
|
|
// This is a part of rust-chrono.
|
|
|
|
// Copyright (c) 2015, Kang Seonghoon.
|
|
|
|
// See README.md and LICENSE.txt for details.
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* A collection of parsed date and time items.
|
|
|
|
* They can be constructed incrementally while being checked for consistency.
|
|
|
|
*/
|
|
|
|
|
|
|
|
use std::num::{Int, ToPrimitive};
|
|
|
|
|
|
|
|
use {Datelike, Timelike};
|
|
|
|
use Weekday;
|
|
|
|
use div::div_rem;
|
|
|
|
use duration::Duration;
|
2015-02-18 16:48:29 +00:00
|
|
|
use offset::{TimeZone, Offset, LocalResult};
|
|
|
|
use offset::fixed::FixedOffset;
|
2015-01-30 02:58:25 +00:00
|
|
|
use naive::date::NaiveDate;
|
|
|
|
use naive::time::NaiveTime;
|
|
|
|
use naive::datetime::NaiveDateTime;
|
2015-01-30 17:14:29 +00:00
|
|
|
use datetime::DateTime;
|
2015-02-04 15:54:25 +00:00
|
|
|
use super::{ParseResult, OUT_OF_RANGE, IMPOSSIBLE, NOT_ENOUGH};
|
|
|
|
|
|
|
|
/// Parsed parts of date and time. There are two classes of methods:
|
|
|
|
///
|
|
|
|
/// - `set_*` methods try to set given field(s) while checking for the consistency.
|
|
|
|
/// It may or may not check for the range constraint immediately (for efficiency reasons).
|
|
|
|
/// - `to_*` methods try to make a concrete date and time value out of set fields.
|
|
|
|
/// It fully checks any remaining out-of-range conditions and inconsistent/impossible fields,
|
2015-01-30 02:58:25 +00:00
|
|
|
#[allow(missing_copy_implementations)]
|
2015-02-03 17:58:40 +00:00
|
|
|
#[derive(Clone, PartialEq, Debug)]
|
2015-01-30 02:58:25 +00:00
|
|
|
pub struct Parsed {
|
2015-02-15 17:16:47 +00:00
|
|
|
/// Year. This can be negative unlike `year_{div,mod}_100` fields.
|
|
|
|
pub year: Option<i32>,
|
|
|
|
/// Year divided by 100. Implies that the year is >= 1 BCE when set.
|
2015-01-30 02:58:25 +00:00
|
|
|
///
|
|
|
|
/// Due to the common usage, if this field is missing but `year_mod_100` is present,
|
|
|
|
/// it is inferred to 19 when `year_mod_100 >= 70` and 20 otherwise.
|
2015-02-15 17:16:47 +00:00
|
|
|
pub year_div_100: Option<i32>,
|
|
|
|
/// Year modulo 100. Implies that the year is >= 1 BCE when set.
|
|
|
|
pub year_mod_100: Option<i32>,
|
|
|
|
/// Year in the ISO week date. This can be negative unlike `isoyear_{div,mod}_100` fields.
|
|
|
|
pub isoyear: Option<i32>,
|
|
|
|
/// Year in the ISO week date, divided by 100. Implies that the year is >= 1 BCE when set.
|
2015-01-30 02:58:25 +00:00
|
|
|
///
|
|
|
|
/// Due to the common usage, if this field is missing but `isoyear_mod_100` is present,
|
|
|
|
/// it is inferred to 19 when `isoyear_mod_100 >= 70` and 20 otherwise.
|
2015-02-15 17:16:47 +00:00
|
|
|
pub isoyear_div_100: Option<i32>,
|
|
|
|
/// Year in the ISO week date, modulo 100. Implies that the year is >= 1 BCE when set.
|
|
|
|
pub isoyear_mod_100: Option<i32>,
|
2015-01-30 02:58:25 +00:00
|
|
|
/// Month (1--12).
|
|
|
|
pub month: Option<u32>,
|
|
|
|
/// Week number, where the week 1 starts at the first Sunday of January.
|
|
|
|
/// (0--53, 1--53 or 1--52 depending on the year).
|
|
|
|
pub week_from_sun: Option<u32>,
|
|
|
|
/// Week number, where the week 1 starts at the first Monday of January.
|
|
|
|
/// (0--53, 1--53 or 1--52 depending on the year).
|
|
|
|
pub week_from_mon: Option<u32>,
|
|
|
|
/// ISO week number (1--52 or 1--53 depending on the year).
|
|
|
|
pub isoweek: Option<u32>,
|
|
|
|
/// Day of the week.
|
|
|
|
pub weekday: Option<Weekday>,
|
|
|
|
/// Day of the year (1--365 or 1--366 depending on the year).
|
|
|
|
pub ordinal: Option<u32>,
|
|
|
|
/// Day of the month (1--28, 1--29, 1--30 or 1--31 depending on the month).
|
|
|
|
pub day: Option<u32>,
|
|
|
|
/// Hour number divided by 12 (0--1). 0 indicates AM and 1 indicates PM.
|
|
|
|
pub hour_div_12: Option<u32>,
|
|
|
|
/// Hour number modulo 12 (0--11).
|
|
|
|
pub hour_mod_12: Option<u32>,
|
|
|
|
/// Minute number (0--59).
|
|
|
|
pub minute: Option<u32>,
|
|
|
|
/// Second number (0--60, accounting for leap seconds).
|
|
|
|
pub second: Option<u32>,
|
|
|
|
/// The number of nanoseconds since the whole second (0--999,999,999).
|
|
|
|
pub nanosecond: Option<u32>,
|
|
|
|
/// The number of non-leap seconds since January 1, 1970 0:00:00 UTC.
|
|
|
|
///
|
|
|
|
/// This can be off by one if `second` is 60 (a leap second).
|
|
|
|
pub timestamp: Option<i64>,
|
|
|
|
/// Offset from the local time to UTC, in seconds.
|
|
|
|
pub offset: Option<i32>,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Checks if `old` is either empty or has the same value to `new` (i.e. "consistent"),
|
|
|
|
/// and if it is empty, set `old` to `new` as well.
|
2015-02-04 15:54:25 +00:00
|
|
|
fn set_if_consistent<T: PartialEq>(old: &mut Option<T>, new: T) -> ParseResult<()> {
|
2015-01-30 02:58:25 +00:00
|
|
|
if let Some(ref old) = *old {
|
2015-02-04 15:54:25 +00:00
|
|
|
if *old == new {Ok(())} else {Err(IMPOSSIBLE)}
|
2015-01-30 02:58:25 +00:00
|
|
|
} else {
|
|
|
|
*old = Some(new);
|
2015-02-04 15:54:25 +00:00
|
|
|
Ok(())
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Parsed {
|
|
|
|
/// Returns the initial value of parsed parts.
|
|
|
|
pub fn new() -> Parsed {
|
2015-02-15 17:16:47 +00:00
|
|
|
Parsed { year: None, year_div_100: None, year_mod_100: None, isoyear: None,
|
|
|
|
isoyear_div_100: None, isoyear_mod_100: None, month: None,
|
|
|
|
week_from_sun: None, week_from_mon: None, isoweek: None, weekday: None,
|
|
|
|
ordinal: None, day: None, hour_div_12: None, hour_mod_12: None, minute: None,
|
|
|
|
second: None, nanosecond: None, timestamp: None, offset: None }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Tries to set the `year` field from given value.
|
|
|
|
pub fn set_year(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.year, try!(value.to_i32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `year_div_100` field from given value.
|
|
|
|
pub fn set_year_div_100(&mut self, value: i64) -> ParseResult<()> {
|
2015-02-15 17:16:47 +00:00
|
|
|
if value < 0 { return Err(OUT_OF_RANGE); }
|
|
|
|
set_if_consistent(&mut self.year_div_100, try!(value.to_i32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `year_mod_100` field from given value.
|
|
|
|
pub fn set_year_mod_100(&mut self, value: i64) -> ParseResult<()> {
|
2015-02-15 17:16:47 +00:00
|
|
|
if value < 0 { return Err(OUT_OF_RANGE); }
|
|
|
|
set_if_consistent(&mut self.year_mod_100, try!(value.to_i32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-15 17:16:47 +00:00
|
|
|
/// Tries to set the `isoyear` field from given value.
|
|
|
|
pub fn set_isoyear(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.isoyear, try!(value.to_i32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `isoyear_div_100` field from given value.
|
|
|
|
pub fn set_isoyear_div_100(&mut self, value: i64) -> ParseResult<()> {
|
2015-02-15 17:16:47 +00:00
|
|
|
if value < 0 { return Err(OUT_OF_RANGE); }
|
|
|
|
set_if_consistent(&mut self.isoyear_div_100, try!(value.to_i32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `isoyear_mod_100` field from given value.
|
|
|
|
pub fn set_isoyear_mod_100(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
if value < 0 { return Err(OUT_OF_RANGE); }
|
2015-02-15 17:16:47 +00:00
|
|
|
set_if_consistent(&mut self.isoyear_mod_100, try!(value.to_i32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `month` field from given value.
|
|
|
|
pub fn set_month(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.month, try!(value.to_u32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `week_from_sun` field from given value.
|
|
|
|
pub fn set_week_from_sun(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.week_from_sun, try!(value.to_u32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `week_from_mon` field from given value.
|
|
|
|
pub fn set_week_from_mon(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.week_from_mon, try!(value.to_u32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `isoweek` field from given value.
|
|
|
|
pub fn set_isoweek(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.isoweek, try!(value.to_u32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `weekday` field from given value.
|
|
|
|
pub fn set_weekday(&mut self, value: Weekday) -> ParseResult<()> {
|
2015-01-30 02:58:25 +00:00
|
|
|
set_if_consistent(&mut self.weekday, value)
|
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `ordinal` field from given value.
|
|
|
|
pub fn set_ordinal(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.ordinal, try!(value.to_u32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `day` field from given value.
|
|
|
|
pub fn set_day(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.day, try!(value.to_u32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Tries to set the `hour_div_12` field from given value. (`false` for AM, `true` for PM)
|
2015-02-04 15:54:25 +00:00
|
|
|
pub fn set_ampm(&mut self, value: bool) -> ParseResult<()> {
|
2015-01-30 02:58:25 +00:00
|
|
|
set_if_consistent(&mut self.hour_div_12, if value {1} else {0})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Tries to set the `hour_mod_12` field from given hour number in 12-hour clocks.
|
2015-02-04 15:54:25 +00:00
|
|
|
pub fn set_hour12(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
if value < 1 || value > 12 { return Err(OUT_OF_RANGE); }
|
|
|
|
set_if_consistent(&mut self.hour_mod_12, value as u32 % 12)
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Tries to set both `hour_div_12` and `hour_mod_12` fields from given value.
|
2015-02-04 15:54:25 +00:00
|
|
|
pub fn set_hour(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
let v = try!(value.to_u32().ok_or(OUT_OF_RANGE));
|
|
|
|
try!(set_if_consistent(&mut self.hour_div_12, v / 12));
|
|
|
|
try!(set_if_consistent(&mut self.hour_mod_12, v % 12));
|
|
|
|
Ok(())
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `minute` field from given value.
|
|
|
|
pub fn set_minute(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.minute, try!(value.to_u32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `second` field from given value.
|
|
|
|
pub fn set_second(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.second, try!(value.to_u32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `nanosecond` field from given value.
|
|
|
|
pub fn set_nanosecond(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.nanosecond, try!(value.to_u32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `timestamp` field from given value.
|
|
|
|
pub fn set_timestamp(&mut self, value: i64) -> ParseResult<()> {
|
2015-01-30 02:58:25 +00:00
|
|
|
set_if_consistent(&mut self.timestamp, value)
|
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
/// Tries to set the `offset` field from given value.
|
|
|
|
pub fn set_offset(&mut self, value: i64) -> ParseResult<()> {
|
|
|
|
set_if_consistent(&mut self.offset, try!(value.to_i32().ok_or(OUT_OF_RANGE)))
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a parsed naive date out of given fields.
|
|
|
|
///
|
|
|
|
/// This method is able to determine the date from given subset of fields:
|
|
|
|
///
|
|
|
|
/// - Year, month, day.
|
|
|
|
/// - Year, day of the year (ordinal).
|
|
|
|
/// - Year, week number counted from Sunday or Monday, day of the week.
|
|
|
|
/// - ISO week date.
|
|
|
|
///
|
|
|
|
/// Gregorian year and ISO week date year can have their century number (`*_div_100`) omitted,
|
|
|
|
/// the two-digit year is used to guess the century number then.
|
2015-02-04 15:54:25 +00:00
|
|
|
pub fn to_naive_date(&self) -> ParseResult<NaiveDate> {
|
2015-02-15 17:16:47 +00:00
|
|
|
fn resolve_year(y: Option<i32>, q: Option<i32>,
|
|
|
|
r: Option<i32>) -> ParseResult<Option<i32>> {
|
|
|
|
match (y, q, r) {
|
|
|
|
// if there is no further information, simply return the given full year.
|
|
|
|
// this is a common case, so let's avoid division here.
|
|
|
|
(y, None, None) => Ok(y),
|
|
|
|
|
|
|
|
// if there is a full year *and* also quotient and/or modulo,
|
|
|
|
// check if present quotient and/or modulo is consistent to the full year.
|
|
|
|
// since the presence of those fields means a positive full year,
|
|
|
|
// we should filter a negative full year first.
|
|
|
|
(Some(y), q, r @ Some(0...99)) | (Some(y), q, r @ None) => {
|
|
|
|
if y < 0 { return Err(OUT_OF_RANGE); }
|
|
|
|
let (q_, r_) = div_rem(y, 100);
|
|
|
|
if q.unwrap_or(q_) == q_ && r.unwrap_or(r_) == r_ {
|
|
|
|
Ok(Some(y))
|
|
|
|
} else {
|
|
|
|
Err(IMPOSSIBLE)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
// the full year is missing but we have quotient and modulo.
|
|
|
|
// reconstruct the full year. make sure that the result is always positive.
|
|
|
|
(None, Some(q), Some(r @ 0...99)) => {
|
|
|
|
if q < 0 { return Err(OUT_OF_RANGE); }
|
|
|
|
let y = q.checked_mul(100).and_then(|v| v.checked_add(r));
|
|
|
|
Ok(Some(try!(y.ok_or(OUT_OF_RANGE))))
|
|
|
|
},
|
|
|
|
|
|
|
|
// we only have modulo. try to interpret a modulo as a conventional two-digit year.
|
|
|
|
// note: we are affected by Rust issue #18060. avoid multiple range patterns.
|
|
|
|
(None, None, Some(r @ 0...99)) => Ok(Some(r + if r < 70 {2000} else {1900})),
|
|
|
|
|
|
|
|
// otherwise it is an out-of-bound or insufficient condition.
|
|
|
|
(None, Some(_), None) => Err(NOT_ENOUGH),
|
|
|
|
(_, _, Some(_)) => Err(OUT_OF_RANGE),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let given_year =
|
|
|
|
try!(resolve_year(self.year, self.year_div_100, self.year_mod_100));
|
|
|
|
let given_isoyear =
|
|
|
|
try!(resolve_year(self.isoyear, self.isoyear_div_100, self.isoyear_mod_100));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
// verify the normal year-month-day date.
|
2015-02-15 17:16:47 +00:00
|
|
|
let verify_ymd = |date: NaiveDate| {
|
2015-01-30 02:58:25 +00:00
|
|
|
let year = date.year();
|
2015-02-15 17:16:47 +00:00
|
|
|
let (year_div_100, year_mod_100) = if year >= 0 {
|
|
|
|
let (q, r) = div_rem(year, 100);
|
|
|
|
(Some(q), Some(r))
|
|
|
|
} else {
|
|
|
|
(None, None) // they should be empty to be consistent
|
|
|
|
};
|
2015-01-30 02:58:25 +00:00
|
|
|
let month = date.month();
|
|
|
|
let day = date.day();
|
2015-02-15 17:16:47 +00:00
|
|
|
(self.year.unwrap_or(year) == year &&
|
|
|
|
self.year_div_100.or(year_div_100) == year_div_100 &&
|
|
|
|
self.year_mod_100.or(year_mod_100) == year_mod_100 &&
|
2015-01-30 02:58:25 +00:00
|
|
|
self.month.unwrap_or(month) == month &&
|
|
|
|
self.day.unwrap_or(day) == day)
|
|
|
|
};
|
|
|
|
|
|
|
|
// verify the ISO week date.
|
2015-02-15 17:16:47 +00:00
|
|
|
let verify_isoweekdate = |date: NaiveDate| {
|
2015-01-30 02:58:25 +00:00
|
|
|
let (isoyear, isoweek, weekday) = date.isoweekdate();
|
2015-02-15 17:16:47 +00:00
|
|
|
let (isoyear_div_100, isoyear_mod_100) = if isoyear >= 0 {
|
|
|
|
let (q, r) = div_rem(isoyear, 100);
|
|
|
|
(Some(q), Some(r))
|
|
|
|
} else {
|
|
|
|
(None, None) // they should be empty to be consistent
|
|
|
|
};
|
|
|
|
(self.isoyear.unwrap_or(isoyear) == isoyear &&
|
|
|
|
self.isoyear_div_100.or(isoyear_div_100) == isoyear_div_100 &&
|
|
|
|
self.isoyear_mod_100.or(isoyear_mod_100) == isoyear_mod_100 &&
|
2015-01-30 02:58:25 +00:00
|
|
|
self.isoweek.unwrap_or(isoweek) == isoweek &&
|
|
|
|
self.weekday.unwrap_or(weekday) == weekday)
|
|
|
|
};
|
|
|
|
|
|
|
|
// verify the ordinal and other (non-ISO) week dates.
|
2015-02-15 17:16:47 +00:00
|
|
|
let verify_ordinal = |date: NaiveDate| {
|
2015-01-30 02:58:25 +00:00
|
|
|
let ordinal = date.ordinal();
|
|
|
|
let weekday = date.weekday();
|
2015-03-05 15:23:51 +00:00
|
|
|
let week_from_sun = (ordinal as i32 - weekday.num_days_from_sunday() as i32 + 7) / 7;
|
|
|
|
let week_from_mon = (ordinal as i32 - weekday.num_days_from_monday() as i32 + 7) / 7;
|
2015-01-30 02:58:25 +00:00
|
|
|
(self.ordinal.unwrap_or(ordinal) == ordinal &&
|
2015-03-05 15:23:51 +00:00
|
|
|
self.week_from_sun.map_or(week_from_sun, |v| v as i32) == week_from_sun &&
|
|
|
|
self.week_from_mon.map_or(week_from_mon, |v| v as i32) == week_from_mon)
|
2015-01-30 02:58:25 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// test several possibilities.
|
|
|
|
// tries to construct a full `NaiveDate` as much as possible, then verifies that
|
|
|
|
// it is consistent with other given fields.
|
|
|
|
let (verified, parsed_date) = match (given_year, given_isoyear, self) {
|
|
|
|
(Some(year), _, &Parsed { month: Some(month), day: Some(day), .. }) => {
|
|
|
|
// year, month, day
|
2015-02-04 15:54:25 +00:00
|
|
|
let date = try!(NaiveDate::from_ymd_opt(year, month, day).ok_or(OUT_OF_RANGE));
|
2015-01-30 02:58:25 +00:00
|
|
|
(verify_isoweekdate(date) && verify_ordinal(date), date)
|
|
|
|
},
|
|
|
|
|
|
|
|
(Some(year), _, &Parsed { ordinal: Some(ordinal), .. }) => {
|
|
|
|
// year, day of the year
|
2015-02-04 15:54:25 +00:00
|
|
|
let date = try!(NaiveDate::from_yo_opt(year, ordinal).ok_or(OUT_OF_RANGE));
|
2015-01-30 02:58:25 +00:00
|
|
|
(verify_ymd(date) && verify_isoweekdate(date) && verify_ordinal(date), date)
|
|
|
|
},
|
|
|
|
|
|
|
|
(Some(year), _, &Parsed { week_from_sun: Some(week_from_sun),
|
|
|
|
weekday: Some(weekday), .. }) => {
|
|
|
|
// year, week (starting at 1st Sunday), day of the week
|
2015-02-04 15:54:25 +00:00
|
|
|
let newyear = try!(NaiveDate::from_yo_opt(year, 1).ok_or(OUT_OF_RANGE));
|
2015-01-30 17:14:29 +00:00
|
|
|
let firstweek = match newyear.weekday() {
|
|
|
|
Weekday::Sun => 0,
|
|
|
|
Weekday::Mon => 6,
|
|
|
|
Weekday::Tue => 5,
|
|
|
|
Weekday::Wed => 4,
|
|
|
|
Weekday::Thu => 3,
|
|
|
|
Weekday::Fri => 2,
|
|
|
|
Weekday::Sat => 1,
|
|
|
|
};
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
// `firstweek+1`-th day of January is the beginning of the week 1.
|
2015-02-04 15:54:25 +00:00
|
|
|
if week_from_sun > 53 { return Err(OUT_OF_RANGE); } // can it overflow?
|
2015-03-05 15:23:51 +00:00
|
|
|
let ndays = firstweek + (week_from_sun as i32 - 1) * 7 +
|
|
|
|
weekday.num_days_from_sunday() as i32;
|
2015-02-04 15:54:25 +00:00
|
|
|
let date = try!(newyear.checked_add(Duration::days(ndays as i64))
|
|
|
|
.ok_or(OUT_OF_RANGE));
|
|
|
|
if date.year() != year { return Err(OUT_OF_RANGE); } // early exit for correct error
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
(verify_ymd(date) && verify_isoweekdate(date) && verify_ordinal(date), date)
|
|
|
|
},
|
|
|
|
|
|
|
|
(Some(year), _, &Parsed { week_from_mon: Some(week_from_mon),
|
|
|
|
weekday: Some(weekday), .. }) => {
|
|
|
|
// year, week (starting at 1st Monday), day of the week
|
2015-02-04 15:54:25 +00:00
|
|
|
let newyear = try!(NaiveDate::from_yo_opt(year, 1).ok_or(OUT_OF_RANGE));
|
2015-01-30 17:14:29 +00:00
|
|
|
let firstweek = match newyear.weekday() {
|
|
|
|
Weekday::Sun => 1,
|
|
|
|
Weekday::Mon => 0,
|
|
|
|
Weekday::Tue => 6,
|
|
|
|
Weekday::Wed => 5,
|
|
|
|
Weekday::Thu => 4,
|
|
|
|
Weekday::Fri => 3,
|
|
|
|
Weekday::Sat => 2,
|
|
|
|
};
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
// `firstweek+1`-th day of January is the beginning of the week 1.
|
2015-02-04 15:54:25 +00:00
|
|
|
if week_from_mon > 53 { return Err(OUT_OF_RANGE); } // can it overflow?
|
2015-03-05 15:23:51 +00:00
|
|
|
let ndays = firstweek + (week_from_mon as i32 - 1) * 7 +
|
|
|
|
weekday.num_days_from_monday() as i32;
|
2015-02-04 15:54:25 +00:00
|
|
|
let date = try!(newyear.checked_add(Duration::days(ndays as i64))
|
|
|
|
.ok_or(OUT_OF_RANGE));
|
|
|
|
if date.year() != year { return Err(OUT_OF_RANGE); } // early exit for correct error
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
(verify_ymd(date) && verify_isoweekdate(date) && verify_ordinal(date), date)
|
|
|
|
},
|
|
|
|
|
|
|
|
(_, Some(isoyear), &Parsed { isoweek: Some(isoweek), weekday: Some(weekday), .. }) => {
|
|
|
|
// ISO year, week, day of the week
|
2015-02-04 15:54:25 +00:00
|
|
|
let date = NaiveDate::from_isoywd_opt(isoyear, isoweek, weekday);
|
|
|
|
let date = try!(date.ok_or(OUT_OF_RANGE));
|
2015-01-30 02:58:25 +00:00
|
|
|
(verify_ymd(date) && verify_ordinal(date), date)
|
|
|
|
},
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
(_, _, _) => return Err(NOT_ENOUGH)
|
2015-01-30 02:58:25 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
if verified {
|
2015-02-04 15:54:25 +00:00
|
|
|
Ok(parsed_date)
|
2015-01-30 02:58:25 +00:00
|
|
|
} else {
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(IMPOSSIBLE)
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a parsed naive time out of given fields.
|
|
|
|
///
|
|
|
|
/// This method is able to determine the time from given subset of fields:
|
|
|
|
///
|
|
|
|
/// - Hour, minute. (second and nanosecond assumed to be 0)
|
|
|
|
/// - Hour, minute, second. (nanosecond assumed to be 0)
|
|
|
|
/// - Hour, minute, second, nanosecond.
|
|
|
|
///
|
|
|
|
/// It is able to handle leap seconds when given second is 60.
|
2015-02-04 15:54:25 +00:00
|
|
|
pub fn to_naive_time(&self) -> ParseResult<NaiveTime> {
|
|
|
|
let hour_div_12 = match self.hour_div_12 {
|
|
|
|
Some(v @ 0...1) => v,
|
|
|
|
Some(_) => return Err(OUT_OF_RANGE),
|
|
|
|
None => return Err(NOT_ENOUGH),
|
|
|
|
};
|
|
|
|
let hour_mod_12 = match self.hour_mod_12 {
|
|
|
|
Some(v @ 0...11) => v,
|
|
|
|
Some(_) => return Err(OUT_OF_RANGE),
|
|
|
|
None => return Err(NOT_ENOUGH),
|
|
|
|
};
|
2015-01-30 02:58:25 +00:00
|
|
|
let hour = hour_div_12 * 12 + hour_mod_12;
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
let minute = match self.minute {
|
|
|
|
Some(v @ 0...59) => v,
|
|
|
|
Some(_) => return Err(OUT_OF_RANGE),
|
|
|
|
None => return Err(NOT_ENOUGH),
|
|
|
|
};
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
// we allow omitting seconds or nanoseconds, but they should be in the range.
|
|
|
|
let (second, mut nano) = match self.second.unwrap_or(0) {
|
|
|
|
v @ 0...59 => (v, 0),
|
2015-01-30 17:14:29 +00:00
|
|
|
60 => (59, 1_000_000_000),
|
2015-02-04 15:54:25 +00:00
|
|
|
_ => return Err(OUT_OF_RANGE),
|
2015-01-30 02:58:25 +00:00
|
|
|
};
|
|
|
|
nano += match self.nanosecond {
|
|
|
|
Some(v @ 0...999_999_999) if self.second.is_some() => v,
|
2015-02-04 15:54:25 +00:00
|
|
|
Some(0...999_999_999) => return Err(NOT_ENOUGH), // second is missing
|
|
|
|
Some(_) => return Err(OUT_OF_RANGE),
|
|
|
|
None => 0,
|
2015-01-30 02:58:25 +00:00
|
|
|
};
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
NaiveTime::from_hms_nano_opt(hour, minute, second, nano).ok_or(OUT_OF_RANGE)
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2015-01-30 17:14:29 +00:00
|
|
|
/// Returns a parsed naive date and time out of given fields,
|
|
|
|
/// except for the `offset` field (assumed to have a given value).
|
2015-02-04 15:54:25 +00:00
|
|
|
/// This is required for parsing a local time or other known-timezone inputs.
|
2015-01-30 02:58:25 +00:00
|
|
|
///
|
|
|
|
/// This method is able to determine the combined date and time
|
|
|
|
/// from date and time fields or a single `timestamp` field.
|
|
|
|
/// Either way those fields have to be consistent to each other.
|
2015-02-04 15:54:25 +00:00
|
|
|
pub fn to_naive_datetime_with_offset(&self, offset: i32) -> ParseResult<NaiveDateTime> {
|
2015-01-30 02:58:25 +00:00
|
|
|
let date = self.to_naive_date();
|
|
|
|
let time = self.to_naive_time();
|
2015-02-04 15:54:25 +00:00
|
|
|
if let (Ok(date), Ok(time)) = (date, time) {
|
2015-01-30 02:58:25 +00:00
|
|
|
let datetime = date.and_time(time);
|
|
|
|
|
|
|
|
// verify the timestamp field if any
|
2015-02-18 17:45:29 +00:00
|
|
|
// the following is safe, `timestamp` is very limited in range
|
|
|
|
let timestamp = datetime.timestamp() - offset as i64;
|
2015-01-30 02:58:25 +00:00
|
|
|
if let Some(given_timestamp) = self.timestamp {
|
|
|
|
// if `datetime` represents a leap second, it might be off by one second.
|
|
|
|
if given_timestamp != timestamp &&
|
2015-01-30 17:14:29 +00:00
|
|
|
!(datetime.nanosecond() >= 1_000_000_000 && given_timestamp == timestamp + 1) {
|
2015-02-04 15:54:25 +00:00
|
|
|
return Err(IMPOSSIBLE);
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
Ok(datetime)
|
2015-01-30 02:58:25 +00:00
|
|
|
} else if let Some(timestamp) = self.timestamp {
|
2015-02-04 15:54:25 +00:00
|
|
|
// if date and time is problematic already, there is no point proceeding.
|
|
|
|
// we at least try to give a correct error though.
|
|
|
|
match (date, time) {
|
|
|
|
(Err(OUT_OF_RANGE), _) | (_, Err(OUT_OF_RANGE)) => return Err(OUT_OF_RANGE),
|
|
|
|
(Err(IMPOSSIBLE), _) | (_, Err(IMPOSSIBLE)) => return Err(IMPOSSIBLE),
|
|
|
|
(_, _) => {} // one of them is insufficient
|
|
|
|
}
|
|
|
|
|
2015-01-30 02:58:25 +00:00
|
|
|
// reconstruct date and time fields from timestamp
|
2015-02-04 15:54:25 +00:00
|
|
|
let ts = try!(timestamp.checked_add(offset as i64).ok_or(OUT_OF_RANGE));
|
2015-02-18 17:45:29 +00:00
|
|
|
let datetime = NaiveDateTime::from_timestamp_opt(ts, 0);
|
2015-02-04 15:54:25 +00:00
|
|
|
let mut datetime = try!(datetime.ok_or(OUT_OF_RANGE));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
2015-01-30 17:14:29 +00:00
|
|
|
// fill year, ordinal, hour, minute and second fields from timestamp.
|
2015-01-30 02:58:25 +00:00
|
|
|
// if existing fields are consistent, this will allow the full date/time reconstruction.
|
|
|
|
let mut parsed = self.clone();
|
2015-01-30 17:40:19 +00:00
|
|
|
if parsed.second == Some(60) {
|
|
|
|
// `datetime.second()` cannot be 60, so this is the only case for a leap second.
|
|
|
|
match datetime.second() {
|
|
|
|
// it's okay, just do not try to overwrite the existing field.
|
|
|
|
59 => {}
|
|
|
|
// `datetime` is known to be off by one second.
|
|
|
|
0 => { datetime = datetime - Duration::seconds(1); }
|
|
|
|
// otherwise it is impossible.
|
2015-02-04 15:54:25 +00:00
|
|
|
_ => return Err(IMPOSSIBLE)
|
2015-01-30 17:40:19 +00:00
|
|
|
}
|
|
|
|
// ...and we have the correct candidates for other fields.
|
|
|
|
} else {
|
2015-02-04 15:54:25 +00:00
|
|
|
try!(parsed.set_second(datetime.second() as i64));
|
2015-01-30 17:40:19 +00:00
|
|
|
}
|
2015-02-04 15:54:25 +00:00
|
|
|
try!(parsed.set_year (datetime.year() as i64));
|
|
|
|
try!(parsed.set_ordinal(datetime.ordinal() as i64)); // more efficient than ymd
|
|
|
|
try!(parsed.set_hour (datetime.hour() as i64));
|
|
|
|
try!(parsed.set_minute (datetime.minute() as i64));
|
|
|
|
try!(parsed.set_nanosecond(0)); // no nanosecond precision in timestamp
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
// validate other fields (e.g. week) and return
|
2015-02-04 15:54:25 +00:00
|
|
|
let date = try!(parsed.to_naive_date());
|
|
|
|
let time = try!(parsed.to_naive_time());
|
|
|
|
Ok(date.and_time(time))
|
2015-01-30 02:58:25 +00:00
|
|
|
} else {
|
2015-02-15 12:01:36 +00:00
|
|
|
// reproduce the previous error(s)
|
|
|
|
try!(date);
|
|
|
|
try!(time);
|
|
|
|
unreachable!()
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a parsed fixed time zone offset out of given fields.
|
2015-02-04 15:54:25 +00:00
|
|
|
pub fn to_fixed_offset(&self) -> ParseResult<FixedOffset> {
|
|
|
|
self.offset.and_then(|offset| FixedOffset::east_opt(offset)).ok_or(OUT_OF_RANGE)
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
2015-01-30 17:14:29 +00:00
|
|
|
|
|
|
|
/// Returns a parsed timezone-aware date and time out of given fields.
|
|
|
|
///
|
|
|
|
/// This method is able to determine the combined date and time
|
|
|
|
/// from date and time fields or a single `timestamp` field, plus a time zone offset.
|
|
|
|
/// Either way those fields have to be consistent to each other.
|
2015-02-04 15:54:25 +00:00
|
|
|
pub fn to_datetime(&self) -> ParseResult<DateTime<FixedOffset>> {
|
|
|
|
let offset = try!(self.offset.ok_or(NOT_ENOUGH));
|
|
|
|
let datetime = try!(self.to_naive_datetime_with_offset(offset));
|
|
|
|
let offset = try!(FixedOffset::east_opt(offset).ok_or(OUT_OF_RANGE));
|
|
|
|
match offset.from_local_datetime(&datetime) {
|
|
|
|
LocalResult::None => Err(IMPOSSIBLE),
|
|
|
|
LocalResult::Single(t) => Ok(t),
|
|
|
|
LocalResult::Ambiguous(..) => Err(NOT_ENOUGH),
|
|
|
|
}
|
2015-02-04 17:16:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a parsed timezone-aware date and time out of given fields,
|
2015-02-18 16:48:29 +00:00
|
|
|
/// with an additional `TimeZone` used to interpret and validate the local date.
|
2015-02-04 17:16:35 +00:00
|
|
|
///
|
|
|
|
/// This method is able to determine the combined date and time
|
|
|
|
/// from date and time fields or a single `timestamp` field, plus a time zone offset.
|
|
|
|
/// Either way those fields have to be consistent to each other.
|
|
|
|
/// If parsed fields include an UTC offset, it also has to be consistent to `offset`.
|
2015-02-18 16:48:29 +00:00
|
|
|
pub fn to_datetime_with_timezone<Tz: TimeZone>(&self, tz: &Tz) -> ParseResult<DateTime<Tz>> {
|
|
|
|
// if we have `timestamp` specified, guess an offset from that.
|
|
|
|
let mut guessed_offset = 0;
|
|
|
|
if let Some(timestamp) = self.timestamp {
|
|
|
|
// make a naive `DateTime` from given timestamp and (if any) nanosecond.
|
|
|
|
// an empty `nanosecond` is always equal to zero, so missing nanosecond is fine.
|
|
|
|
let nanosecond = self.nanosecond.unwrap_or(0);
|
2015-02-18 17:45:29 +00:00
|
|
|
let dt = NaiveDateTime::from_timestamp_opt(timestamp, nanosecond);
|
2015-02-18 16:48:29 +00:00
|
|
|
let dt = try!(dt.ok_or(OUT_OF_RANGE));
|
|
|
|
|
|
|
|
// we cannot handle offsets larger than i32 at all. give up if so.
|
|
|
|
// we can instead make `to_naive_datetime_with_offset` to accept i64, but this makes
|
|
|
|
// the algorithm too complex and tons of edge cases. i32 should be enough for all.
|
|
|
|
let offset = tz.offset_from_utc_datetime(&dt).local_minus_utc().num_seconds();
|
|
|
|
guessed_offset = try!(offset.to_i32().ok_or(OUT_OF_RANGE));
|
|
|
|
}
|
|
|
|
|
|
|
|
// checks if the given `DateTime` has a consistent `Offset` with given `self.offset`.
|
|
|
|
let check_offset = |dt: &DateTime<Tz>| {
|
|
|
|
if let Some(offset) = self.offset {
|
|
|
|
let delta = dt.offset().local_minus_utc().num_seconds();
|
|
|
|
// if `delta` does not fit in `i32`, it cannot equal to `self.offset` anyway.
|
|
|
|
delta.to_i32() == Some(offset)
|
|
|
|
} else {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// `guessed_offset` should be correct when `self.timestamp` is given.
|
|
|
|
// it will be 0 otherwise, but this is fine as the algorithm ignores offset for that case.
|
|
|
|
let datetime = try!(self.to_naive_datetime_with_offset(guessed_offset));
|
|
|
|
match tz.from_local_datetime(&datetime) {
|
2015-02-04 17:16:35 +00:00
|
|
|
LocalResult::None => Err(IMPOSSIBLE),
|
2015-02-18 16:48:29 +00:00
|
|
|
LocalResult::Single(t) => if check_offset(&t) {Ok(t)} else {Err(IMPOSSIBLE)},
|
|
|
|
LocalResult::Ambiguous(min, max) => {
|
|
|
|
// try to disambiguate two possible local dates by offset.
|
|
|
|
match (check_offset(&min), check_offset(&max)) {
|
|
|
|
(false, false) => Err(IMPOSSIBLE),
|
|
|
|
(false, true) => Ok(max),
|
|
|
|
(true, false) => Ok(min),
|
|
|
|
(true, true) => Err(NOT_ENOUGH),
|
|
|
|
}
|
|
|
|
}
|
2015-02-04 17:16:35 +00:00
|
|
|
}
|
2015-01-30 17:14:29 +00:00
|
|
|
}
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::Parsed;
|
2015-02-04 15:54:25 +00:00
|
|
|
use super::super::{OUT_OF_RANGE, IMPOSSIBLE, NOT_ENOUGH};
|
2015-01-30 02:58:25 +00:00
|
|
|
use Datelike;
|
2015-02-04 15:54:25 +00:00
|
|
|
use Weekday::*;
|
2015-01-30 02:58:25 +00:00
|
|
|
use naive::date::{self, NaiveDate};
|
|
|
|
use naive::time::NaiveTime;
|
2015-02-18 16:48:29 +00:00
|
|
|
use offset::TimeZone;
|
|
|
|
use offset::utc::UTC;
|
|
|
|
use offset::fixed::FixedOffset;
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_parsed_set_fields() {
|
|
|
|
// year*, isoyear*
|
|
|
|
let mut p = Parsed::new();
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(p.set_year(1987), Ok(()));
|
|
|
|
assert_eq!(p.set_year(1986), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_year(1988), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_year(1987), Ok(()));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(p.set_year_div_100(20), Ok(())); // independent to `year`
|
|
|
|
assert_eq!(p.set_year_div_100(21), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_year_div_100(19), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_year_mod_100(37), Ok(())); // ditto
|
|
|
|
assert_eq!(p.set_year_mod_100(38), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_year_mod_100(36), Err(IMPOSSIBLE));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
let mut p = Parsed::new();
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(p.set_year(0), Ok(()));
|
|
|
|
assert_eq!(p.set_year_div_100(0), Ok(()));
|
|
|
|
assert_eq!(p.set_year_mod_100(0), Ok(()));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
let mut p = Parsed::new();
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(p.set_year_div_100(-1), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(p.set_year_mod_100(-1), Err(OUT_OF_RANGE));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(p.set_year(-1), Ok(()));
|
|
|
|
assert_eq!(p.set_year(-2), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_year(0), Err(IMPOSSIBLE));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
let mut p = Parsed::new();
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(p.set_year_div_100(0x1_0000_0008), Err(OUT_OF_RANGE));
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(p.set_year_div_100(8), Ok(()));
|
|
|
|
assert_eq!(p.set_year_div_100(0x1_0000_0008), Err(OUT_OF_RANGE));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
// month, week*, isoweek, ordinal, day, minute, second, nanosecond, offset
|
|
|
|
let mut p = Parsed::new();
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(p.set_month(7), Ok(()));
|
|
|
|
assert_eq!(p.set_month(1), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_month(6), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_month(8), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_month(12), Err(IMPOSSIBLE));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
let mut p = Parsed::new();
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(p.set_month(8), Ok(()));
|
|
|
|
assert_eq!(p.set_month(0x1_0000_0008), Err(OUT_OF_RANGE));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
// hour
|
|
|
|
let mut p = Parsed::new();
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(p.set_hour(12), Ok(()));
|
|
|
|
assert_eq!(p.set_hour(11), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_hour(13), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_hour(12), Ok(()));
|
|
|
|
assert_eq!(p.set_ampm(false), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_ampm(true), Ok(()));
|
|
|
|
assert_eq!(p.set_hour12(12), Ok(()));
|
|
|
|
assert_eq!(p.set_hour12(0), Err(OUT_OF_RANGE)); // requires canonical representation
|
|
|
|
assert_eq!(p.set_hour12(1), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_hour12(11), Err(IMPOSSIBLE));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
let mut p = Parsed::new();
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(p.set_ampm(true), Ok(()));
|
|
|
|
assert_eq!(p.set_hour12(7), Ok(()));
|
|
|
|
assert_eq!(p.set_hour(7), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_hour(18), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_hour(19), Ok(()));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
// timestamp
|
|
|
|
let mut p = Parsed::new();
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(p.set_timestamp(1_234_567_890), Ok(()));
|
|
|
|
assert_eq!(p.set_timestamp(1_234_567_889), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(p.set_timestamp(1_234_567_891), Err(IMPOSSIBLE));
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_parsed_to_naive_date() {
|
|
|
|
macro_rules! parse {
|
|
|
|
($($k:ident: $v:expr),*) => (
|
|
|
|
Parsed { $($k: Some($v),)* ..Parsed::new() }.to_naive_date()
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2015-02-15 17:16:47 +00:00
|
|
|
let ymd = |y,m,d| Ok(NaiveDate::from_ymd(y, m, d));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
2015-01-30 17:14:29 +00:00
|
|
|
// ymd: omission of fields
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(), Err(NOT_ENOUGH));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 1984), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(year: 1984, month: 1), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(year: 1984, month: 1, day: 2), ymd(1984, 1, 2));
|
|
|
|
assert_eq!(parse!(year: 1984, day: 2), Err(NOT_ENOUGH));
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(year_div_100: 19), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: 84), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: 84, month: 1), Err(NOT_ENOUGH));
|
2015-01-30 02:58:25 +00:00
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: 84, month: 1, day: 2), ymd(1984, 1, 2));
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: 84, day: 2), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(year_div_100: 19, month: 1, day: 2), Err(NOT_ENOUGH));
|
2015-01-30 17:14:29 +00:00
|
|
|
assert_eq!(parse!(year_mod_100: 70, month: 1, day: 2), ymd(1970, 1, 2));
|
|
|
|
assert_eq!(parse!(year_mod_100: 69, month: 1, day: 2), ymd(2069, 1, 2));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
2015-01-30 17:14:29 +00:00
|
|
|
// ymd: out-of-range conditions
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: 84, month: 2, day: 29),
|
|
|
|
ymd(1984, 2, 29));
|
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: 83, month: 2, day: 29),
|
|
|
|
Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: 83, month: 13, day: 1),
|
|
|
|
Err(OUT_OF_RANGE));
|
2015-01-30 02:58:25 +00:00
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: 83, month: 12, day: 31),
|
|
|
|
ymd(1983, 12, 31));
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: 83, month: 12, day: 32),
|
|
|
|
Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: 83, month: 12, day: 0),
|
|
|
|
Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: 100, month: 1, day: 1),
|
|
|
|
Err(OUT_OF_RANGE));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year_div_100: 19, year_mod_100: -1, month: 1, day: 1),
|
|
|
|
Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year_div_100: 0, year_mod_100: 0, month: 1, day: 1),
|
|
|
|
ymd(0, 1, 1));
|
|
|
|
assert_eq!(parse!(year_div_100: -1, year_mod_100: 42, month: 1, day: 1),
|
|
|
|
Err(OUT_OF_RANGE));
|
2015-01-30 02:58:25 +00:00
|
|
|
let max_year = date::MAX.year();
|
2015-03-27 02:35:34 +00:00
|
|
|
assert_eq!(parse!(year_div_100: max_year / 100,
|
|
|
|
year_mod_100: max_year % 100, month: 1, day: 1),
|
2015-01-30 02:58:25 +00:00
|
|
|
ymd(max_year, 1, 1));
|
2015-03-27 02:35:34 +00:00
|
|
|
assert_eq!(parse!(year_div_100: (max_year + 1) / 100,
|
|
|
|
year_mod_100: (max_year + 1) % 100, month: 1, day: 1),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(OUT_OF_RANGE));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
2015-02-15 17:16:47 +00:00
|
|
|
// ymd: conflicting inputs
|
|
|
|
assert_eq!(parse!(year: 1984, year_div_100: 19, month: 1, day: 1), ymd(1984, 1, 1));
|
|
|
|
assert_eq!(parse!(year: 1984, year_div_100: 20, month: 1, day: 1), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(parse!(year: 1984, year_mod_100: 84, month: 1, day: 1), ymd(1984, 1, 1));
|
|
|
|
assert_eq!(parse!(year: 1984, year_mod_100: 83, month: 1, day: 1), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(parse!(year: 1984, year_div_100: 19, year_mod_100: 84, month: 1, day: 1),
|
|
|
|
ymd(1984, 1, 1));
|
|
|
|
assert_eq!(parse!(year: 1984, year_div_100: 18, year_mod_100: 94, month: 1, day: 1),
|
|
|
|
Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(parse!(year: 1984, year_div_100: 18, year_mod_100: 184, month: 1, day: 1),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(OUT_OF_RANGE));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: -1, year_div_100: 0, year_mod_100: -1, month: 1, day: 1),
|
|
|
|
Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: -1, year_div_100: -1, year_mod_100: 99, month: 1, day: 1),
|
|
|
|
Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: -1, year_div_100: 0, month: 1, day: 1), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: -1, year_mod_100: 99, month: 1, day: 1), Err(OUT_OF_RANGE));
|
|
|
|
|
|
|
|
// weekdates
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 0), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_sun: 0), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(year: 2000, weekday: Sun), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 0, weekday: Fri), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_sun: 0, weekday: Fri), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 0, weekday: Sat), ymd(2000, 1, 1));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_sun: 0, weekday: Sat), ymd(2000, 1, 1));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 0, weekday: Sun), ymd(2000, 1, 2));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_sun: 1, weekday: Sun), ymd(2000, 1, 2));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 1, weekday: Mon), ymd(2000, 1, 3));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_sun: 1, weekday: Mon), ymd(2000, 1, 3));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 1, weekday: Sat), ymd(2000, 1, 8));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_sun: 1, weekday: Sat), ymd(2000, 1, 8));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 1, weekday: Sun), ymd(2000, 1, 9));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_sun: 2, weekday: Sun), ymd(2000, 1, 9));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 2, weekday: Mon), ymd(2000, 1, 10));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_sun: 52, weekday: Sat), ymd(2000, 12, 30));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_sun: 53, weekday: Sun), ymd(2000, 12, 31));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_sun: 53, weekday: Mon), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: 2000, week_from_sun: 0xffffffff, weekday: Mon), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: 2006, week_from_sun: 0, weekday: Sat), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: 2006, week_from_sun: 1, weekday: Sun), ymd(2006, 1, 1));
|
2015-01-30 17:14:29 +00:00
|
|
|
|
|
|
|
// weekdates: conflicting inputs
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 1, week_from_sun: 1, weekday: Sat),
|
2015-02-04 15:54:25 +00:00
|
|
|
ymd(2000, 1, 8));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 1, week_from_sun: 2, weekday: Sun),
|
2015-02-04 15:54:25 +00:00
|
|
|
ymd(2000, 1, 9));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 1, week_from_sun: 1, weekday: Sun),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(IMPOSSIBLE));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2000, week_from_mon: 2, week_from_sun: 2, weekday: Sun),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(IMPOSSIBLE));
|
2015-01-30 17:14:29 +00:00
|
|
|
|
|
|
|
// ISO weekdates
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(isoyear: 2004, isoweek: 53), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(isoyear: 2004, isoweek: 53, weekday: Fri), ymd(2004, 12, 31));
|
|
|
|
assert_eq!(parse!(isoyear: 2004, isoweek: 53, weekday: Sat), ymd(2005, 1, 1));
|
|
|
|
assert_eq!(parse!(isoyear: 2004, isoweek: 0xffffffff, weekday: Sat), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(isoyear: 2005, isoweek: 0, weekday: Thu), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(isoyear: 2005, isoweek: 5, weekday: Thu), ymd(2005, 2, 3));
|
|
|
|
assert_eq!(parse!(isoyear: 2005, weekday: Thu), Err(NOT_ENOUGH));
|
2015-01-30 17:14:29 +00:00
|
|
|
|
|
|
|
// year and ordinal
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(ordinal: 123), Err(NOT_ENOUGH));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2000, ordinal: 0), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: 2000, ordinal: 1), ymd(2000, 1, 1));
|
|
|
|
assert_eq!(parse!(year: 2000, ordinal: 60), ymd(2000, 2, 29));
|
|
|
|
assert_eq!(parse!(year: 2000, ordinal: 61), ymd(2000, 3, 1));
|
|
|
|
assert_eq!(parse!(year: 2000, ordinal: 366), ymd(2000, 12, 31));
|
|
|
|
assert_eq!(parse!(year: 2000, ordinal: 367), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: 2000, ordinal: 0xffffffff), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: 2100, ordinal: 0), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: 2100, ordinal: 1), ymd(2100, 1, 1));
|
|
|
|
assert_eq!(parse!(year: 2100, ordinal: 59), ymd(2100, 2, 28));
|
|
|
|
assert_eq!(parse!(year: 2100, ordinal: 60), ymd(2100, 3, 1));
|
|
|
|
assert_eq!(parse!(year: 2100, ordinal: 365), ymd(2100, 12, 31));
|
|
|
|
assert_eq!(parse!(year: 2100, ordinal: 366), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(year: 2100, ordinal: 0xffffffff), Err(OUT_OF_RANGE));
|
2015-01-30 17:14:29 +00:00
|
|
|
|
|
|
|
// more complex cases
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2014, month: 12, day: 31, ordinal: 365, isoyear: 2015, isoweek: 1,
|
2015-02-04 15:54:25 +00:00
|
|
|
week_from_sun: 52, week_from_mon: 52, weekday: Wed),
|
2015-01-30 17:14:29 +00:00
|
|
|
ymd(2014, 12, 31));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2014, month: 12, ordinal: 365, isoyear: 2015, isoweek: 1,
|
2015-01-30 17:14:29 +00:00
|
|
|
week_from_sun: 52, week_from_mon: 52),
|
|
|
|
ymd(2014, 12, 31));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2014, month: 12, day: 31, ordinal: 365, isoyear: 2014, isoweek: 53,
|
2015-02-04 15:54:25 +00:00
|
|
|
week_from_sun: 52, week_from_mon: 52, weekday: Wed),
|
|
|
|
Err(IMPOSSIBLE)); // no ISO week date 2014-W53-3
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2012, isoyear: 2015, isoweek: 1,
|
|
|
|
week_from_sun: 52, week_from_mon: 52),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(NOT_ENOUGH)); // ambiguous (2014-12-29, 2014-12-30, 2014-12-31)
|
2015-01-30 17:14:29 +00:00
|
|
|
assert_eq!(parse!(year_div_100: 20, isoyear_mod_100: 15, ordinal: 366),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(NOT_ENOUGH)); // technically unique (2014-12-31) but Chrono gives up
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_parsed_to_naive_time() {
|
|
|
|
macro_rules! parse {
|
|
|
|
($($k:ident: $v:expr),*) => (
|
|
|
|
Parsed { $($k: Some($v),)* ..Parsed::new() }.to_naive_time()
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2015-02-15 17:16:47 +00:00
|
|
|
let hms = |h,m,s| Ok(NaiveTime::from_hms(h, m, s));
|
|
|
|
let hmsn = |h,m,s,n| Ok(NaiveTime::from_hms_nano(h, m, s, n));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
|
|
|
// omission of fields
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(hour_div_12: 0), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(hour_div_12: 0, hour_mod_12: 1), Err(NOT_ENOUGH));
|
2015-01-30 02:58:25 +00:00
|
|
|
assert_eq!(parse!(hour_div_12: 0, hour_mod_12: 1, minute: 23), hms(1,23,0));
|
|
|
|
assert_eq!(parse!(hour_div_12: 0, hour_mod_12: 1, minute: 23, second: 45), hms(1,23,45));
|
|
|
|
assert_eq!(parse!(hour_div_12: 0, hour_mod_12: 1, minute: 23, second: 45,
|
|
|
|
nanosecond: 678_901_234),
|
|
|
|
hmsn(1,23,45,678_901_234));
|
|
|
|
assert_eq!(parse!(hour_div_12: 1, hour_mod_12: 11, minute: 45, second: 6), hms(23,45,6));
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(hour_mod_12: 1, minute: 23), Err(NOT_ENOUGH));
|
|
|
|
assert_eq!(parse!(hour_div_12: 0, hour_mod_12: 1, minute: 23, nanosecond: 456_789_012),
|
|
|
|
Err(NOT_ENOUGH));
|
2015-01-30 02:58:25 +00:00
|
|
|
|
2015-01-30 17:14:29 +00:00
|
|
|
// out-of-range conditions
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(hour_div_12: 2, hour_mod_12: 0, minute: 0), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(hour_div_12: 1, hour_mod_12: 12, minute: 0), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(hour_div_12: 0, hour_mod_12: 1, minute: 60), Err(OUT_OF_RANGE));
|
|
|
|
assert_eq!(parse!(hour_div_12: 0, hour_mod_12: 1, minute: 23, second: 61),
|
|
|
|
Err(OUT_OF_RANGE));
|
2015-01-30 17:14:29 +00:00
|
|
|
assert_eq!(parse!(hour_div_12: 0, hour_mod_12: 1, minute: 23, second: 34,
|
|
|
|
nanosecond: 1_000_000_000),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(OUT_OF_RANGE));
|
2015-01-30 17:14:29 +00:00
|
|
|
|
|
|
|
// leap seconds
|
|
|
|
assert_eq!(parse!(hour_div_12: 0, hour_mod_12: 1, minute: 23, second: 60),
|
|
|
|
hmsn(1,23,59,1_000_000_000));
|
|
|
|
assert_eq!(parse!(hour_div_12: 0, hour_mod_12: 1, minute: 23, second: 60,
|
|
|
|
nanosecond: 999_999_999),
|
|
|
|
hmsn(1,23,59,1_999_999_999));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2015-02-04 15:54:25 +00:00
|
|
|
fn test_parsed_to_naive_datetime_with_offset() {
|
2015-01-30 17:14:29 +00:00
|
|
|
macro_rules! parse {
|
2015-02-04 15:54:25 +00:00
|
|
|
(offset = $offset:expr; $($k:ident: $v:expr),*) => (
|
|
|
|
Parsed { $($k: Some($v),)* ..Parsed::new() }.to_naive_datetime_with_offset($offset)
|
|
|
|
);
|
|
|
|
($($k:ident: $v:expr),*) => (parse!(offset = 0; $($k: $v),*))
|
2015-01-30 17:14:29 +00:00
|
|
|
}
|
|
|
|
|
2015-02-15 17:16:47 +00:00
|
|
|
let ymdhms = |y,m,d,h,n,s| Ok(NaiveDate::from_ymd(y, m, d).and_hms(h, n, s));
|
2015-01-30 17:14:29 +00:00
|
|
|
let ymdhmsn =
|
2015-02-15 17:16:47 +00:00
|
|
|
|y,m,d,h,n,s,nano| Ok(NaiveDate::from_ymd(y, m, d).and_hms_nano(h, n, s, nano));
|
2015-01-30 17:14:29 +00:00
|
|
|
|
|
|
|
// omission of fields
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(), Err(NOT_ENOUGH));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2015, month: 1, day: 30,
|
2015-01-30 17:14:29 +00:00
|
|
|
hour_div_12: 1, hour_mod_12: 2, minute: 38),
|
|
|
|
ymdhms(2015,1,30, 14,38,0));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 1997, month: 1, day: 30,
|
2015-01-30 17:14:29 +00:00
|
|
|
hour_div_12: 1, hour_mod_12: 2, minute: 38, second: 5),
|
|
|
|
ymdhms(1997,1,30, 14,38,5));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2012, ordinal: 34, hour_div_12: 0, hour_mod_12: 5,
|
2015-01-30 17:14:29 +00:00
|
|
|
minute: 6, second: 7, nanosecond: 890_123_456),
|
|
|
|
ymdhmsn(2012,2,3, 5,6,7,890_123_456));
|
|
|
|
assert_eq!(parse!(timestamp: 0), ymdhms(1970,1,1, 0,0,0));
|
|
|
|
assert_eq!(parse!(timestamp: 1, nanosecond: 0), ymdhms(1970,1,1, 0,0,1));
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(timestamp: 1, nanosecond: 1), Err(IMPOSSIBLE));
|
2015-01-30 17:14:29 +00:00
|
|
|
assert_eq!(parse!(timestamp: 1_420_000_000), ymdhms(2014,12,31, 4,26,40));
|
|
|
|
assert_eq!(parse!(timestamp: -0x1_0000_0000), ymdhms(1833,11,24, 17,31,44));
|
|
|
|
|
|
|
|
// full fields
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2014, year_div_100: 20, year_mod_100: 14, month: 12, day: 31,
|
|
|
|
ordinal: 365, isoyear: 2015, isoyear_div_100: 20, isoyear_mod_100: 15,
|
|
|
|
isoweek: 1, week_from_sun: 52, week_from_mon: 52, weekday: Wed,
|
2015-01-30 17:14:29 +00:00
|
|
|
hour_div_12: 0, hour_mod_12: 4, minute: 26, second: 40,
|
|
|
|
nanosecond: 12_345_678, timestamp: 1_420_000_000),
|
|
|
|
ymdhmsn(2014,12,31, 4,26,40,12_345_678));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2014, year_div_100: 20, year_mod_100: 14, month: 12, day: 31,
|
|
|
|
ordinal: 365, isoyear: 2015, isoyear_div_100: 20, isoyear_mod_100: 15,
|
|
|
|
isoweek: 1, week_from_sun: 52, week_from_mon: 52, weekday: Wed,
|
2015-01-30 17:14:29 +00:00
|
|
|
hour_div_12: 0, hour_mod_12: 4, minute: 26, second: 40,
|
|
|
|
nanosecond: 12_345_678, timestamp: 1_419_999_999),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(parse!(offset = 32400;
|
2015-02-15 17:16:47 +00:00
|
|
|
year: 2014, year_div_100: 20, year_mod_100: 14, month: 12, day: 31,
|
|
|
|
ordinal: 365, isoyear: 2015, isoyear_div_100: 20, isoyear_mod_100: 15,
|
|
|
|
isoweek: 1, week_from_sun: 52, week_from_mon: 52, weekday: Wed,
|
2015-02-04 15:54:25 +00:00
|
|
|
hour_div_12: 0, hour_mod_12: 4, minute: 26, second: 40,
|
|
|
|
nanosecond: 12_345_678, timestamp: 1_419_967_600),
|
|
|
|
ymdhmsn(2014,12,31, 4,26,40,12_345_678));
|
2015-01-30 17:14:29 +00:00
|
|
|
|
|
|
|
// more timestamps
|
|
|
|
let max_days_from_year_1970 = date::MAX - NaiveDate::from_ymd(1970,1,1);
|
|
|
|
let year_0_from_year_1970 = NaiveDate::from_ymd(0,1,1) - NaiveDate::from_ymd(1970,1,1);
|
2015-02-15 17:16:47 +00:00
|
|
|
let min_days_from_year_1970 = date::MIN - NaiveDate::from_ymd(1970,1,1);
|
|
|
|
assert_eq!(parse!(timestamp: min_days_from_year_1970.num_seconds()),
|
|
|
|
ymdhms(date::MIN.year(),1,1, 0,0,0));
|
2015-01-30 17:14:29 +00:00
|
|
|
assert_eq!(parse!(timestamp: year_0_from_year_1970.num_seconds()),
|
|
|
|
ymdhms(0,1,1, 0,0,0));
|
|
|
|
assert_eq!(parse!(timestamp: max_days_from_year_1970.num_seconds() + 86399),
|
|
|
|
ymdhms(date::MAX.year(),12,31, 23,59,59));
|
2015-01-30 17:40:19 +00:00
|
|
|
|
|
|
|
// leap seconds #1: partial fields
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(second: 59, timestamp: 1_341_100_798), Err(IMPOSSIBLE));
|
2015-01-30 17:40:19 +00:00
|
|
|
assert_eq!(parse!(second: 59, timestamp: 1_341_100_799), ymdhms(2012,6,30, 23,59,59));
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(second: 59, timestamp: 1_341_100_800), Err(IMPOSSIBLE));
|
2015-01-30 17:40:19 +00:00
|
|
|
assert_eq!(parse!(second: 60, timestamp: 1_341_100_799),
|
|
|
|
ymdhmsn(2012,6,30, 23,59,59,1_000_000_000));
|
|
|
|
assert_eq!(parse!(second: 60, timestamp: 1_341_100_800),
|
|
|
|
ymdhmsn(2012,6,30, 23,59,59,1_000_000_000));
|
|
|
|
assert_eq!(parse!(second: 0, timestamp: 1_341_100_800), ymdhms(2012,7,1, 0,0,0));
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(second: 1, timestamp: 1_341_100_800), Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(parse!(second: 60, timestamp: 1_341_100_801), Err(IMPOSSIBLE));
|
2015-01-30 17:40:19 +00:00
|
|
|
|
|
|
|
// leap seconds #2: full fields
|
|
|
|
// we need to have separate tests for them since it uses another control flow.
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2012, ordinal: 182, hour_div_12: 1, hour_mod_12: 11,
|
2015-01-30 17:40:19 +00:00
|
|
|
minute: 59, second: 59, timestamp: 1_341_100_798),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(IMPOSSIBLE));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2012, ordinal: 182, hour_div_12: 1, hour_mod_12: 11,
|
2015-01-30 17:40:19 +00:00
|
|
|
minute: 59, second: 59, timestamp: 1_341_100_799),
|
|
|
|
ymdhms(2012,6,30, 23,59,59));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2012, ordinal: 182, hour_div_12: 1, hour_mod_12: 11,
|
2015-01-30 17:40:19 +00:00
|
|
|
minute: 59, second: 59, timestamp: 1_341_100_800),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(IMPOSSIBLE));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2012, ordinal: 182, hour_div_12: 1, hour_mod_12: 11,
|
2015-01-30 17:40:19 +00:00
|
|
|
minute: 59, second: 60, timestamp: 1_341_100_799),
|
|
|
|
ymdhmsn(2012,6,30, 23,59,59,1_000_000_000));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2012, ordinal: 182, hour_div_12: 1, hour_mod_12: 11,
|
2015-01-30 17:40:19 +00:00
|
|
|
minute: 59, second: 60, timestamp: 1_341_100_800),
|
|
|
|
ymdhmsn(2012,6,30, 23,59,59,1_000_000_000));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2012, ordinal: 183, hour_div_12: 0, hour_mod_12: 0,
|
2015-01-30 17:40:19 +00:00
|
|
|
minute: 0, second: 0, timestamp: 1_341_100_800),
|
|
|
|
ymdhms(2012,7,1, 0,0,0));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2012, ordinal: 183, hour_div_12: 0, hour_mod_12: 0,
|
2015-01-30 17:40:19 +00:00
|
|
|
minute: 0, second: 1, timestamp: 1_341_100_800),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(IMPOSSIBLE));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2012, ordinal: 182, hour_div_12: 1, hour_mod_12: 11,
|
2015-01-30 17:40:19 +00:00
|
|
|
minute: 59, second: 60, timestamp: 1_341_100_801),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(IMPOSSIBLE));
|
2015-02-15 12:01:36 +00:00
|
|
|
|
|
|
|
// error codes
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2015, month: 1, day: 20, weekday: Tue,
|
2015-02-15 12:01:36 +00:00
|
|
|
hour_div_12: 2, hour_mod_12: 1, minute: 35, second: 20),
|
|
|
|
Err(OUT_OF_RANGE)); // `hour_div_12` is out of range
|
2015-01-30 17:14:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_parsed_to_datetime() {
|
|
|
|
macro_rules! parse {
|
|
|
|
($($k:ident: $v:expr),*) => (
|
|
|
|
Parsed { $($k: Some($v),)* ..Parsed::new() }.to_datetime()
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2015-02-15 17:16:47 +00:00
|
|
|
let ymdhmsn = |y,m,d,h,n,s,nano,off| Ok(FixedOffset::east(off).ymd(y, m, d)
|
|
|
|
.and_hms_nano(h, n, s, nano));
|
2015-01-30 17:14:29 +00:00
|
|
|
|
2015-02-04 15:54:25 +00:00
|
|
|
assert_eq!(parse!(offset: 0), Err(NOT_ENOUGH));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2014, ordinal: 365, hour_div_12: 0, hour_mod_12: 4,
|
|
|
|
minute: 26, second: 40, nanosecond: 12_345_678),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(NOT_ENOUGH));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2014, ordinal: 365, hour_div_12: 0, hour_mod_12: 4,
|
|
|
|
minute: 26, second: 40, nanosecond: 12_345_678, offset: 0),
|
2015-01-30 17:14:29 +00:00
|
|
|
ymdhmsn(2014,12,31, 4,26,40,12_345_678, 0));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2014, ordinal: 365, hour_div_12: 1, hour_mod_12: 1,
|
|
|
|
minute: 26, second: 40, nanosecond: 12_345_678, offset: 32400),
|
2015-01-30 17:14:29 +00:00
|
|
|
ymdhmsn(2014,12,31, 13,26,40,12_345_678, 32400));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2014, ordinal: 365, hour_div_12: 0, hour_mod_12: 1,
|
|
|
|
minute: 42, second: 4, nanosecond: 12_345_678, offset: -9876),
|
2015-01-30 17:14:29 +00:00
|
|
|
ymdhmsn(2014,12,31, 1,42,4,12_345_678, -9876));
|
2015-02-15 17:16:47 +00:00
|
|
|
assert_eq!(parse!(year: 2015, ordinal: 1, hour_div_12: 0, hour_mod_12: 4,
|
|
|
|
minute: 26, second: 40, nanosecond: 12_345_678, offset: 86400),
|
2015-02-04 15:54:25 +00:00
|
|
|
Err(OUT_OF_RANGE)); // `FixedOffset` does not support such huge offset
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
2015-02-18 16:48:29 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_parsed_to_datetime_with_timezone() {
|
|
|
|
macro_rules! parse {
|
|
|
|
($tz:expr; $($k:ident: $v:expr),*) => (
|
|
|
|
Parsed { $($k: Some($v),)* ..Parsed::new() }.to_datetime_with_timezone(&$tz)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// single result from ymdhms
|
|
|
|
assert_eq!(parse!(UTC;
|
|
|
|
year: 2014, ordinal: 365, hour_div_12: 0, hour_mod_12: 4,
|
|
|
|
minute: 26, second: 40, nanosecond: 12_345_678, offset: 0),
|
|
|
|
Ok(UTC.ymd(2014, 12, 31).and_hms_nano(4, 26, 40, 12_345_678)));
|
|
|
|
assert_eq!(parse!(UTC;
|
|
|
|
year: 2014, ordinal: 365, hour_div_12: 1, hour_mod_12: 1,
|
|
|
|
minute: 26, second: 40, nanosecond: 12_345_678, offset: 32400),
|
|
|
|
Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(parse!(FixedOffset::east(32400);
|
|
|
|
year: 2014, ordinal: 365, hour_div_12: 0, hour_mod_12: 4,
|
|
|
|
minute: 26, second: 40, nanosecond: 12_345_678, offset: 0),
|
|
|
|
Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(parse!(FixedOffset::east(32400);
|
|
|
|
year: 2014, ordinal: 365, hour_div_12: 1, hour_mod_12: 1,
|
|
|
|
minute: 26, second: 40, nanosecond: 12_345_678, offset: 32400),
|
|
|
|
Ok(FixedOffset::east(32400).ymd(2014, 12, 31)
|
|
|
|
.and_hms_nano(13, 26, 40, 12_345_678)));
|
|
|
|
|
|
|
|
// single result from timestamp
|
|
|
|
assert_eq!(parse!(UTC; timestamp: 1_420_000_000, offset: 0),
|
|
|
|
Ok(UTC.ymd(2014, 12, 31).and_hms(4, 26, 40)));
|
|
|
|
assert_eq!(parse!(UTC; timestamp: 1_420_000_000, offset: 32400),
|
|
|
|
Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(parse!(FixedOffset::east(32400); timestamp: 1_420_000_000, offset: 0),
|
|
|
|
Err(IMPOSSIBLE));
|
|
|
|
assert_eq!(parse!(FixedOffset::east(32400); timestamp: 1_420_000_000, offset: 32400),
|
|
|
|
Ok(FixedOffset::east(32400).ymd(2014, 12, 31).and_hms(13, 26, 40)));
|
|
|
|
|
|
|
|
// TODO test with a variable time zone (for None and Ambiguous cases)
|
|
|
|
}
|
2015-01-30 02:58:25 +00:00
|
|
|
}
|
|
|
|
|