use crate::{
error::{fmt::friendly::Error as E, ErrorContext},
fmt::{
friendly::parser_label,
util::{parse_temporal_fraction, DurationUnits},
Parsed,
},
util::{b::Sign, parse},
Error, SignedDuration, Span, Unit,
};
#[derive(Clone, Debug, Default)]
pub struct SpanParser {
_private: (),
}
impl SpanParser {
#[inline]
pub const fn new() -> SpanParser {
SpanParser { _private: () }
}
#[inline]
pub fn parse_span<I: AsRef<[u8]>>(&self, input: I) -> Result<Span, Error> {
#[inline(never)]
fn imp(span_parser: &SpanParser, input: &[u8]) -> Result<Span, Error> {
let mut builder = DurationUnits::default();
let parsed = span_parser.parse(input, &mut builder)?;
let parsed = parsed.and_then(|_| builder.to_span())?;
parsed.into_full()
}
let input = input.as_ref();
imp(self, input).context(E::Failed)
}
#[inline]
pub fn parse_duration<I: AsRef<[u8]>>(
&self,
input: I,
) -> Result<SignedDuration, Error> {
#[inline(never)]
fn imp(
span_parser: &SpanParser,
input: &[u8],
) -> Result<SignedDuration, Error> {
let mut builder = DurationUnits::default();
let parsed = span_parser.parse(input, &mut builder)?;
let parsed = parsed.and_then(|_| builder.to_signed_duration())?;
parsed.into_full()
}
let input = input.as_ref();
imp(self, input).context(E::Failed)
}
#[inline]
pub fn parse_unsigned_duration<I: AsRef<[u8]>>(
&self,
input: I,
) -> Result<core::time::Duration, Error> {
#[inline(never)]
fn imp(
span_parser: &SpanParser,
input: &[u8],
) -> Result<core::time::Duration, Error> {
let mut builder = DurationUnits::default();
let parsed = span_parser.parse(input, &mut builder)?;
let parsed =
parsed.and_then(|_| builder.to_unsigned_duration())?;
let d = parsed.value;
parsed.into_full_with(format_args!("{d:?}"))
}
let input = input.as_ref();
imp(self, input).context(E::Failed)
}
#[cfg_attr(feature = "perf-inline", inline(always))]
fn parse<'i>(
&self,
input: &'i [u8],
builder: &mut DurationUnits,
) -> Result<Parsed<'i, ()>, Error> {
if input.is_empty() {
return Err(Error::from(E::Empty));
}
let (sign, input) =
if !input.first().map_or(false, |&b| matches!(b, b'+' | b'-')) {
(None, input)
} else {
let Parsed { value: sign, input } =
self.parse_prefix_sign(input);
(sign, input)
};
let Parsed { value, input } = self.parse_unit_value(input)?;
let Some(first_unit_value) = value else {
return Err(Error::from(E::ExpectedIntegerAfterSign));
};
let Parsed { input, .. } =
self.parse_duration_units(input, first_unit_value, builder)?;
let (sign, input) = if !input.first().map_or(false, is_whitespace) {
(sign.unwrap_or(Sign::Positive), input)
} else {
let parsed = self.parse_suffix_sign(sign, input)?;
(parsed.value, parsed.input)
};
builder.set_sign(sign);
Ok(Parsed { value: (), input })
}
#[cfg_attr(feature = "perf-inline", inline(always))]
fn parse_duration_units<'i>(
&self,
mut input: &'i [u8],
first_unit_value: u64,
builder: &mut DurationUnits,
) -> Result<Parsed<'i, ()>, Error> {
let mut parsed_any_after_comma = true;
let mut value = first_unit_value;
loop {
let parsed = self.parse_hms_maybe(input, value)?;
input = parsed.input;
if let Some(hms) = parsed.value {
builder.set_hms(
hms.hour,
hms.minute,
hms.second,
hms.fraction,
)?;
break;
}
let fraction =
if input.first().map_or(false, |&b| b == b'.' || b == b',') {
let parsed = parse_temporal_fraction(input)?;
input = parsed.input;
parsed.value
} else {
None
};
input = self.parse_optional_whitespace(input).input;
let parsed = self.parse_unit_designator(input)?;
input = parsed.input;
let unit = parsed.value;
if input.first().map_or(false, |&b| b == b',') {
input = self.parse_optional_comma(input)?.input;
parsed_any_after_comma = false;
}
builder.set_unit_value(unit, value)?;
if let Some(fraction) = fraction {
builder.set_fraction(fraction)?;
break;
}
let after_whitespace = self.parse_optional_whitespace(input).input;
let parsed = self.parse_unit_value(after_whitespace)?;
value = match parsed.value {
None => break,
Some(value) => value,
};
input = parsed.input;
parsed_any_after_comma = true;
}
if !parsed_any_after_comma {
return Err(Error::from(E::ExpectedOneMoreUnitAfterComma));
}
Ok(Parsed { value: (), input })
}
#[cfg_attr(feature = "perf-inline", inline(always))]
fn parse_hms_maybe<'i>(
&self,
input: &'i [u8],
hour: u64,
) -> Result<Parsed<'i, Option<HMS>>, Error> {
let Some((&first, tail)) = input.split_first() else {
return Ok(Parsed { input, value: None });
};
if first != b':' {
return Ok(Parsed { input, value: None });
}
let Parsed { input, value } = self.parse_hms(tail, hour)?;
Ok(Parsed { input, value: Some(value) })
}
#[inline(never)]
fn parse_hms<'i>(
&self,
input: &'i [u8],
hour: u64,
) -> Result<Parsed<'i, HMS>, Error> {
let Parsed { input, value } = self.parse_unit_value(input)?;
let minute = value.ok_or(E::ExpectedMinuteAfterHour)?;
let (&first, input) =
input.split_first().ok_or(E::ExpectedColonAfterMinute)?;
if first != b':' {
return Err(Error::from(E::ExpectedColonAfterMinute));
}
let Parsed { input, value } = self.parse_unit_value(input)?;
let second = value.ok_or(E::ExpectedSecondAfterMinute)?;
let (fraction, input) =
if input.first().map_or(false, |&b| b == b'.' || b == b',') {
let parsed = parse_temporal_fraction(input)?;
(parsed.value, parsed.input)
} else {
(None, input)
};
let hms = HMS { hour, minute, second, fraction };
Ok(Parsed { input, value: hms })
}
#[cfg_attr(feature = "perf-inline", inline(always))]
fn parse_unit_value<'i>(
&self,
input: &'i [u8],
) -> Result<Parsed<'i, Option<u64>>, Error> {
let (value, input) = parse::u64_prefix(input)?;
Ok(Parsed { value, input })
}
#[cfg_attr(feature = "perf-inline", inline(always))]
fn parse_unit_designator<'i>(
&self,
input: &'i [u8],
) -> Result<Parsed<'i, Unit>, Error> {
let (unit, len) =
parser_label::find(input).ok_or(E::ExpectedUnitSuffix)?;
Ok(Parsed { value: unit, input: &input[len..] })
}
#[inline(never)]
fn parse_prefix_sign<'i>(
&self,
input: &'i [u8],
) -> Parsed<'i, Option<Sign>> {
let Some(sign) = input.first().copied() else {
return Parsed { value: None, input };
};
let sign = if sign == b'+' {
Sign::Positive
} else if sign == b'-' {
Sign::Negative
} else {
return Parsed { value: None, input };
};
Parsed { value: Some(sign), input: &input[1..] }
}
#[inline(never)]
fn parse_suffix_sign<'i>(
&self,
prefix_sign: Option<Sign>,
mut input: &'i [u8],
) -> Result<Parsed<'i, Sign>, Error> {
if !input.first().map_or(false, is_whitespace) {
let sign = prefix_sign.unwrap_or(Sign::Positive);
return Ok(Parsed { value: sign, input });
}
input = self.parse_optional_whitespace(&input[1..]).input;
let (suffix_sign, input) =
if let Some(tail) = input.strip_prefix(b"ago") {
(Some(Sign::Negative), tail)
} else {
(None, input)
};
let sign = match (prefix_sign, suffix_sign) {
(Some(_), Some(_)) => {
return Err(Error::from(E::ExpectedOneSign));
}
(Some(sign), None) => sign,
(None, Some(sign)) => sign,
(None, None) => Sign::Positive,
};
Ok(Parsed { value: sign, input })
}
#[inline(never)]
fn parse_optional_comma<'i>(
&self,
input: &'i [u8],
) -> Result<Parsed<'i, ()>, Error> {
let Some((&first, tail)) = input.split_first() else {
return Ok(Parsed { value: (), input });
};
if first != b',' {
return Ok(Parsed { value: (), input });
}
let (second, input) = tail
.split_first()
.ok_or(E::ExpectedWhitespaceAfterCommaEndOfInput)?;
if !is_whitespace(second) {
return Err(Error::from(E::ExpectedWhitespaceAfterComma {
byte: *second,
}));
}
Ok(Parsed { value: (), input })
}
#[cfg_attr(feature = "perf-inline", inline(always))]
fn parse_optional_whitespace<'i>(
&self,
mut input: &'i [u8],
) -> Parsed<'i, ()> {
while input.first().map_or(false, is_whitespace) {
input = &input[1..];
}
Parsed { value: (), input }
}
}
#[derive(Debug)]
struct HMS {
hour: u64,
minute: u64,
second: u64,
fraction: Option<u32>,
}
#[cfg_attr(feature = "perf-inline", inline(always))]
fn is_whitespace(byte: &u8) -> bool {
matches!(*byte, b' ' | b'\t' | b'\n' | b'\r' | b'\x0C')
}
#[cfg(feature = "alloc")]
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn parse_span_basic() {
let p = |s: &str| SpanParser::new().parse_span(s).unwrap();
insta::assert_snapshot!(p("5 years"), @"P5Y");
insta::assert_snapshot!(p("5 years 4 months"), @"P5Y4M");
insta::assert_snapshot!(p("5 years 4 months 3 hours"), @"P5Y4MT3H");
insta::assert_snapshot!(p("5 years, 4 months, 3 hours"), @"P5Y4MT3H");
insta::assert_snapshot!(p("01:02:03"), @"PT1H2M3S");
insta::assert_snapshot!(p("5 days 01:02:03"), @"P5DT1H2M3S");
insta::assert_snapshot!(p("5 days, 01:02:03"), @"P5DT1H2M3S");
insta::assert_snapshot!(p("3yrs 5 days 01:02:03"), @"P3Y5DT1H2M3S");
insta::assert_snapshot!(p("3yrs 5 days, 01:02:03"), @"P3Y5DT1H2M3S");
insta::assert_snapshot!(
p("3yrs 5 days, 01:02:03.123456789"),
@"P3Y5DT1H2M3.123456789S",
);
insta::assert_snapshot!(p("999:999:999"), @"PT999H999M999S");
}
#[test]
fn parse_span_fractional() {
let p = |s: &str| SpanParser::new().parse_span(s).unwrap();
insta::assert_snapshot!(p("1.5hrs"), @"PT1H30M");
insta::assert_snapshot!(p("1.5mins"), @"PT1M30S");
insta::assert_snapshot!(p("1.5secs"), @"PT1.5S");
insta::assert_snapshot!(p("1.5msecs"), @"PT0.0015S");
insta::assert_snapshot!(p("1.5µsecs"), @"PT0.0000015S");
insta::assert_snapshot!(p("1d 1.5hrs"), @"P1DT1H30M");
insta::assert_snapshot!(p("1h 1.5mins"), @"PT1H1M30S");
insta::assert_snapshot!(p("1m 1.5secs"), @"PT1M1.5S");
insta::assert_snapshot!(p("1s 1.5msecs"), @"PT1.0015S");
insta::assert_snapshot!(p("1ms 1.5µsecs"), @"PT0.0010015S");
insta::assert_snapshot!(p("1s2000ms"), @"PT3S");
}
#[test]
fn parse_span_boundaries() {
let p = |s: &str| SpanParser::new().parse_span(s).unwrap();
insta::assert_snapshot!(p("19998 years"), @"P19998Y");
insta::assert_snapshot!(p("19998 years ago"), @"-P19998Y");
insta::assert_snapshot!(p("239976 months"), @"P239976M");
insta::assert_snapshot!(p("239976 months ago"), @"-P239976M");
insta::assert_snapshot!(p("1043497 weeks"), @"P1043497W");
insta::assert_snapshot!(p("1043497 weeks ago"), @"-P1043497W");
insta::assert_snapshot!(p("7304484 days"), @"P7304484D");
insta::assert_snapshot!(p("7304484 days ago"), @"-P7304484D");
insta::assert_snapshot!(p("175307616 hours"), @"PT175307616H");
insta::assert_snapshot!(p("175307616 hours ago"), @"-PT175307616H");
insta::assert_snapshot!(p("10518456960 minutes"), @"PT10518456960M");
insta::assert_snapshot!(p("10518456960 minutes ago"), @"-PT10518456960M");
insta::assert_snapshot!(p("631107417600 seconds"), @"PT631107417600S");
insta::assert_snapshot!(p("631107417600 seconds ago"), @"-PT631107417600S");
insta::assert_snapshot!(p("631107417600000 milliseconds"), @"PT631107417600S");
insta::assert_snapshot!(p("631107417600000 milliseconds ago"), @"-PT631107417600S");
insta::assert_snapshot!(p("631107417600000000 microseconds"), @"PT631107417600S");
insta::assert_snapshot!(p("631107417600000000 microseconds ago"), @"-PT631107417600S");
insta::assert_snapshot!(p("9223372036854775807 nanoseconds"), @"PT9223372036.854775807S");
insta::assert_snapshot!(p("9223372036854775807 nanoseconds ago"), @"-PT9223372036.854775807S");
insta::assert_snapshot!(p("175307617 hours"), @"PT175307616H60M");
insta::assert_snapshot!(p("175307617 hours ago"), @"-PT175307616H60M");
insta::assert_snapshot!(p("10518456961 minutes"), @"PT10518456960M60S");
insta::assert_snapshot!(p("10518456961 minutes ago"), @"-PT10518456960M60S");
insta::assert_snapshot!(p("631107417601 seconds"), @"PT631107417601S");
insta::assert_snapshot!(p("631107417601 seconds ago"), @"-PT631107417601S");
insta::assert_snapshot!(p("631107417600001 milliseconds"), @"PT631107417600.001S");
insta::assert_snapshot!(p("631107417600001 milliseconds ago"), @"-PT631107417600.001S");
insta::assert_snapshot!(p("631107417600000001 microseconds"), @"PT631107417600.000001S");
insta::assert_snapshot!(p("631107417600000001 microseconds ago"), @"-PT631107417600.000001S");
}
#[test]
fn err_span_basic() {
let p = |s: &str| SpanParser::new().parse_span(s).unwrap_err();
insta::assert_snapshot!(
p(""),
@r#"failed to parse input in the "friendly" duration format: an empty string is not valid"#,
);
insta::assert_snapshot!(
p(" "),
@r#"failed to parse input in the "friendly" duration format: expected duration to start with a unit value (a decimal integer) after an optional sign, but no integer was found"#,
);
insta::assert_snapshot!(
p("a"),
@r#"failed to parse input in the "friendly" duration format: expected duration to start with a unit value (a decimal integer) after an optional sign, but no integer was found"#,
);
insta::assert_snapshot!(
p("2 months 1 year"),
@r#"failed to parse input in the "friendly" duration format: found value with unit year after unit month, but units must be written from largest to smallest (and they can't be repeated)"#,
);
insta::assert_snapshot!(
p("1 year 1 mont"),
@r#"failed to parse input in the "friendly" duration format: parsed value 'P1Y1M', but unparsed input "nt" remains (expected no unparsed input)"#,
);
insta::assert_snapshot!(
p("2 months,"),
@r#"failed to parse input in the "friendly" duration format: expected whitespace after comma, but found end of input"#,
);
insta::assert_snapshot!(
p("2 months, "),
@r#"failed to parse input in the "friendly" duration format: found comma at the end of duration, but a comma indicates at least one more unit follows"#,
);
insta::assert_snapshot!(
p("2 months ,"),
@r#"failed to parse input in the "friendly" duration format: parsed value 'P2M', but unparsed input "," remains (expected no unparsed input)"#,
);
}
#[test]
fn err_span_sign() {
let p = |s: &str| SpanParser::new().parse_span(s).unwrap_err();
insta::assert_snapshot!(
p("1yago"),
@r#"failed to parse input in the "friendly" duration format: parsed value 'P1Y', but unparsed input "ago" remains (expected no unparsed input)"#,
);
insta::assert_snapshot!(
p("1 year 1 monthago"),
@r#"failed to parse input in the "friendly" duration format: parsed value 'P1Y1M', but unparsed input "ago" remains (expected no unparsed input)"#,
);
insta::assert_snapshot!(
p("+1 year 1 month ago"),
@r#"failed to parse input in the "friendly" duration format: expected to find either a prefix sign (+/-) or a suffix sign (`ago`), but found both"#,
);
insta::assert_snapshot!(
p("-1 year 1 month ago"),
@r#"failed to parse input in the "friendly" duration format: expected to find either a prefix sign (+/-) or a suffix sign (`ago`), but found both"#,
);
}
#[test]
fn err_span_overflow_fraction() {
let p = |s: &str| SpanParser::new().parse_span(s).unwrap();
let pe = |s: &str| SpanParser::new().parse_span(s).unwrap_err();
insta::assert_snapshot!(
pe("640330789636854776 micros"),
@r#"failed to parse input in the "friendly" duration format: failed to set value for microsecond unit on span: failed to set nanosecond value from fractional component"#,
);
insta::assert_snapshot!(
p("640330789636854775 micros"),
@"PT640330789636.854775S"
);
insta::assert_snapshot!(
pe("640330789636854775.808 micros"),
@r#"failed to parse input in the "friendly" duration format: failed to set nanosecond value from fractional component"#,
);
insta::assert_snapshot!(
p("640330789636854775.807 micros"),
@"PT640330789636.854775807S"
);
}
#[test]
fn err_span_overflow_units() {
let p = |s: &str| SpanParser::new().parse_span(s).unwrap_err();
insta::assert_snapshot!(
p("19999 years"),
@r#"failed to parse input in the "friendly" duration format: failed to set value for year unit on span: parameter 'years' is not in the required range of -19998..=19998"#,
);
insta::assert_snapshot!(
p("19999 years ago"),
@r#"failed to parse input in the "friendly" duration format: failed to set value for year unit on span: parameter 'years' is not in the required range of -19998..=19998"#,
);
insta::assert_snapshot!(
p("239977 months"),
@r#"failed to parse input in the "friendly" duration format: failed to set value for month unit on span: parameter 'months' is not in the required range of -239976..=239976"#,
);
insta::assert_snapshot!(
p("239977 months ago"),
@r#"failed to parse input in the "friendly" duration format: failed to set value for month unit on span: parameter 'months' is not in the required range of -239976..=239976"#,
);
insta::assert_snapshot!(
p("1043498 weeks"),
@r#"failed to parse input in the "friendly" duration format: failed to set value for week unit on span: parameter 'weeks' is not in the required range of -1043497..=1043497"#,
);
insta::assert_snapshot!(
p("1043498 weeks ago"),
@r#"failed to parse input in the "friendly" duration format: failed to set value for week unit on span: parameter 'weeks' is not in the required range of -1043497..=1043497"#,
);
insta::assert_snapshot!(
p("7304485 days"),
@r#"failed to parse input in the "friendly" duration format: failed to set value for day unit on span: parameter 'days' is not in the required range of -7304484..=7304484"#,
);
insta::assert_snapshot!(
p("7304485 days ago"),
@r#"failed to parse input in the "friendly" duration format: failed to set value for day unit on span: parameter 'days' is not in the required range of -7304484..=7304484"#,
);
insta::assert_snapshot!(
p("9223372036854775808 nanoseconds"),
@r#"failed to parse input in the "friendly" duration format: value for nanoseconds is too big (or small) to fit into a signed 64-bit integer"#,
);
insta::assert_snapshot!(
p("9223372036854775808 nanoseconds ago"),
@r#"failed to parse input in the "friendly" duration format: failed to set value for nanosecond unit on span: parameter 'nanoseconds' is not in the required range of -9223372036854775807..=9223372036854775807"#,
);
}
#[test]
fn err_span_fraction() {
let p = |s: &str| SpanParser::new().parse_span(s).unwrap_err();
insta::assert_snapshot!(
p("1.5 years"),
@r#"failed to parse input in the "friendly" duration format: fractional years are not supported"#,
);
insta::assert_snapshot!(
p("1.5 nanos"),
@r#"failed to parse input in the "friendly" duration format: fractional nanoseconds are not supported"#,
);
}
#[test]
fn err_span_hms() {
let p = |s: &str| SpanParser::new().parse_span(s).unwrap_err();
insta::assert_snapshot!(
p("05:"),
@r#"failed to parse input in the "friendly" duration format: when parsing the `HH:MM:SS` format, expected to parse minute following hour"#,
);
insta::assert_snapshot!(
p("05:06"),
@r#"failed to parse input in the "friendly" duration format: when parsing the `HH:MM:SS` format, expected to parse `:` following minute"#,
);
insta::assert_snapshot!(
p("05:06:"),
@r#"failed to parse input in the "friendly" duration format: when parsing the `HH:MM:SS` format, expected to parse second following minute"#,
);
insta::assert_snapshot!(
p("2 hours, 05:06:07"),
@r#"failed to parse input in the "friendly" duration format: found `HH:MM:SS` after unit hour, but `HH:MM:SS` can only appear after years, months, weeks or days"#,
);
}
#[test]
fn parse_signed_duration_basic() {
let p = |s: &str| SpanParser::new().parse_duration(s).unwrap();
insta::assert_snapshot!(p("1 hour, 2 minutes, 3 seconds"), @"PT1H2M3S");
insta::assert_snapshot!(p("01:02:03"), @"PT1H2M3S");
insta::assert_snapshot!(p("999:999:999"), @"PT1015H55M39S");
}
#[test]
fn parse_signed_duration_negate() {
let p = |s: &str| SpanParser::new().parse_duration(s).unwrap();
let perr = |s: &str| SpanParser::new().parse_duration(s).unwrap_err();
insta::assert_snapshot!(
p("9223372036854775807s"),
@"PT2562047788015215H30M7S",
);
insta::assert_snapshot!(
perr("9223372036854775808s"),
@r#"failed to parse input in the "friendly" duration format: value for seconds is too big (or small) to fit into a signed 64-bit integer"#,
);
insta::assert_snapshot!(
p("-9223372036854775808s"),
@"-PT2562047788015215H30M8S",
);
}
#[test]
fn parse_signed_duration_fractional() {
let p = |s: &str| SpanParser::new().parse_duration(s).unwrap();
insta::assert_snapshot!(p("1.5hrs"), @"PT1H30M");
insta::assert_snapshot!(p("1.5mins"), @"PT1M30S");
insta::assert_snapshot!(p("1.5secs"), @"PT1.5S");
insta::assert_snapshot!(p("1.5msecs"), @"PT0.0015S");
insta::assert_snapshot!(p("1.5µsecs"), @"PT0.0000015S");
insta::assert_snapshot!(p("1h 1.5mins"), @"PT1H1M30S");
insta::assert_snapshot!(p("1m 1.5secs"), @"PT1M1.5S");
insta::assert_snapshot!(p("1s 1.5msecs"), @"PT1.0015S");
insta::assert_snapshot!(p("1ms 1.5µsecs"), @"PT0.0010015S");
insta::assert_snapshot!(p("1s2000ms"), @"PT3S");
}
#[test]
fn parse_signed_duration_boundaries() {
let p = |s: &str| SpanParser::new().parse_duration(s).unwrap();
let pe = |s: &str| SpanParser::new().parse_duration(s).unwrap_err();
insta::assert_snapshot!(p("175307616 hours"), @"PT175307616H");
insta::assert_snapshot!(p("175307616 hours ago"), @"-PT175307616H");
insta::assert_snapshot!(p("10518456960 minutes"), @"PT175307616H");
insta::assert_snapshot!(p("10518456960 minutes ago"), @"-PT175307616H");
insta::assert_snapshot!(p("631107417600 seconds"), @"PT175307616H");
insta::assert_snapshot!(p("631107417600 seconds ago"), @"-PT175307616H");
insta::assert_snapshot!(p("631107417600000 milliseconds"), @"PT175307616H");
insta::assert_snapshot!(p("631107417600000 milliseconds ago"), @"-PT175307616H");
insta::assert_snapshot!(p("631107417600000000 microseconds"), @"PT175307616H");
insta::assert_snapshot!(p("631107417600000000 microseconds ago"), @"-PT175307616H");
insta::assert_snapshot!(p("9223372036854775807 nanoseconds"), @"PT2562047H47M16.854775807S");
insta::assert_snapshot!(p("9223372036854775807 nanoseconds ago"), @"-PT2562047H47M16.854775807S");
insta::assert_snapshot!(p("175307617 hours"), @"PT175307617H");
insta::assert_snapshot!(p("175307617 hours ago"), @"-PT175307617H");
insta::assert_snapshot!(p("10518456961 minutes"), @"PT175307616H1M");
insta::assert_snapshot!(p("10518456961 minutes ago"), @"-PT175307616H1M");
insta::assert_snapshot!(p("631107417601 seconds"), @"PT175307616H1S");
insta::assert_snapshot!(p("631107417601 seconds ago"), @"-PT175307616H1S");
insta::assert_snapshot!(p("631107417600001 milliseconds"), @"PT175307616H0.001S");
insta::assert_snapshot!(p("631107417600001 milliseconds ago"), @"-PT175307616H0.001S");
insta::assert_snapshot!(p("631107417600000001 microseconds"), @"PT175307616H0.000001S");
insta::assert_snapshot!(p("631107417600000001 microseconds ago"), @"-PT175307616H0.000001S");
insta::assert_snapshot!(p("2562047788015215hours"), @"PT2562047788015215H");
insta::assert_snapshot!(p("-2562047788015215hours"), @"-PT2562047788015215H");
insta::assert_snapshot!(
pe("2562047788015216hrs"),
@r#"failed to parse input in the "friendly" duration format: accumulated duration overflowed when adding value to unit hour"#,
);
insta::assert_snapshot!(p("153722867280912930minutes"), @"PT2562047788015215H30M");
insta::assert_snapshot!(p("153722867280912930minutes ago"), @"-PT2562047788015215H30M");
insta::assert_snapshot!(
pe("153722867280912931mins"),
@r#"failed to parse input in the "friendly" duration format: accumulated duration overflowed when adding value to unit minute"#,
);
insta::assert_snapshot!(p("9223372036854775807seconds"), @"PT2562047788015215H30M7S");
insta::assert_snapshot!(p("-9223372036854775807seconds"), @"-PT2562047788015215H30M7S");
insta::assert_snapshot!(
pe("9223372036854775808s"),
@r#"failed to parse input in the "friendly" duration format: value for seconds is too big (or small) to fit into a signed 64-bit integer"#,
);
insta::assert_snapshot!(
p("-9223372036854775808s"),
@"-PT2562047788015215H30M8S",
);
}
#[test]
fn err_signed_duration_basic() {
let p = |s: &str| SpanParser::new().parse_duration(s).unwrap_err();
insta::assert_snapshot!(
p(""),
@r#"failed to parse input in the "friendly" duration format: an empty string is not valid"#,
);
insta::assert_snapshot!(
p(" "),
@r#"failed to parse input in the "friendly" duration format: expected duration to start with a unit value (a decimal integer) after an optional sign, but no integer was found"#,
);
insta::assert_snapshot!(
p("5"),
@r#"failed to parse input in the "friendly" duration format: expected to find unit designator suffix (e.g., `years` or `secs`) after parsing integer"#,
);
insta::assert_snapshot!(
p("a"),
@r#"failed to parse input in the "friendly" duration format: expected duration to start with a unit value (a decimal integer) after an optional sign, but no integer was found"#,
);
insta::assert_snapshot!(
p("2 minutes 1 hour"),
@r#"failed to parse input in the "friendly" duration format: found value with unit hour after unit minute, but units must be written from largest to smallest (and they can't be repeated)"#,
);
insta::assert_snapshot!(
p("1 hour 1 minut"),
@r#"failed to parse input in the "friendly" duration format: parsed value 'PT1H1M', but unparsed input "ut" remains (expected no unparsed input)"#,
);
insta::assert_snapshot!(
p("2 minutes,"),
@r#"failed to parse input in the "friendly" duration format: expected whitespace after comma, but found end of input"#,
);
insta::assert_snapshot!(
p("2 minutes, "),
@r#"failed to parse input in the "friendly" duration format: found comma at the end of duration, but a comma indicates at least one more unit follows"#,
);
insta::assert_snapshot!(
p("2 minutes ,"),
@r#"failed to parse input in the "friendly" duration format: parsed value 'PT2M', but unparsed input "," remains (expected no unparsed input)"#,
);
}
#[test]
fn err_signed_duration_sign() {
let p = |s: &str| SpanParser::new().parse_duration(s).unwrap_err();
insta::assert_snapshot!(
p("1hago"),
@r#"failed to parse input in the "friendly" duration format: parsed value 'PT1H', but unparsed input "ago" remains (expected no unparsed input)"#,
);
insta::assert_snapshot!(
p("1 hour 1 minuteago"),
@r#"failed to parse input in the "friendly" duration format: parsed value 'PT1H1M', but unparsed input "ago" remains (expected no unparsed input)"#,
);
insta::assert_snapshot!(
p("+1 hour 1 minute ago"),
@r#"failed to parse input in the "friendly" duration format: expected to find either a prefix sign (+/-) or a suffix sign (`ago`), but found both"#,
);
insta::assert_snapshot!(
p("-1 hour 1 minute ago"),
@r#"failed to parse input in the "friendly" duration format: expected to find either a prefix sign (+/-) or a suffix sign (`ago`), but found both"#,
);
}
#[test]
fn err_signed_duration_overflow_fraction() {
let p = |s: &str| SpanParser::new().parse_duration(s).unwrap();
let pe = |s: &str| SpanParser::new().parse_duration(s).unwrap_err();
insta::assert_snapshot!(
pe("9223372036854775808 micros"),
@r#"failed to parse input in the "friendly" duration format: value for microseconds is too big (or small) to fit into a signed 64-bit integer"#,
);
insta::assert_snapshot!(
p("9223372036854775807 micros"),
@"PT2562047788H54.775807S"
);
}
#[test]
fn err_signed_duration_fraction() {
let p = |s: &str| SpanParser::new().parse_duration(s).unwrap_err();
insta::assert_snapshot!(
p("1.5 nanos"),
@r#"failed to parse input in the "friendly" duration format: fractional nanoseconds are not supported"#,
);
}
#[test]
fn err_signed_duration_hms() {
let p = |s: &str| SpanParser::new().parse_duration(s).unwrap_err();
insta::assert_snapshot!(
p("05:"),
@r#"failed to parse input in the "friendly" duration format: when parsing the `HH:MM:SS` format, expected to parse minute following hour"#,
);
insta::assert_snapshot!(
p("05:06"),
@r#"failed to parse input in the "friendly" duration format: when parsing the `HH:MM:SS` format, expected to parse `:` following minute"#,
);
insta::assert_snapshot!(
p("05:06:"),
@r#"failed to parse input in the "friendly" duration format: when parsing the `HH:MM:SS` format, expected to parse second following minute"#,
);
insta::assert_snapshot!(
p("2 hours, 05:06:07"),
@r#"failed to parse input in the "friendly" duration format: found `HH:MM:SS` after unit hour, but `HH:MM:SS` can only appear after years, months, weeks or days"#,
);
}
#[test]
fn parse_unsigned_duration_basic() {
let p = |s: &str| {
let dur = SpanParser::new().parse_unsigned_duration(s).unwrap();
crate::fmt::temporal::SpanPrinter::new()
.unsigned_duration_to_string(&dur)
};
insta::assert_snapshot!(
p("1 hour, 2 minutes, 3 seconds"),
@"PT1H2M3S",
);
insta::assert_snapshot!(p("01:02:03"), @"PT1H2M3S");
insta::assert_snapshot!(p("999:999:999"), @"PT1015H55M39S");
insta::assert_snapshot!(
p("+1hr"),
@"PT1H",
);
}
#[test]
fn parse_unsigned_duration_negate() {
let p = |s: &str| {
let dur = SpanParser::new().parse_unsigned_duration(s).unwrap();
crate::fmt::temporal::SpanPrinter::new()
.unsigned_duration_to_string(&dur)
};
let perr = |s: &str| {
SpanParser::new().parse_unsigned_duration(s).unwrap_err()
};
insta::assert_snapshot!(
p("18446744073709551615s"),
@"PT5124095576030431H15S",
);
insta::assert_snapshot!(
perr("18446744073709551616s"),
@r#"failed to parse input in the "friendly" duration format: number too big to parse into 64-bit integer"#,
);
insta::assert_snapshot!(
perr("-1s"),
@r#"failed to parse input in the "friendly" duration format: cannot parse negative duration into unsigned `std::time::Duration`"#,
);
}
#[test]
fn parse_unsigned_duration_fractional() {
let p = |s: &str| {
let dur = SpanParser::new().parse_unsigned_duration(s).unwrap();
crate::fmt::temporal::SpanPrinter::new()
.unsigned_duration_to_string(&dur)
};
insta::assert_snapshot!(p("1.5hrs"), @"PT1H30M");
insta::assert_snapshot!(p("1.5mins"), @"PT1M30S");
insta::assert_snapshot!(p("1.5secs"), @"PT1.5S");
insta::assert_snapshot!(p("1.5msecs"), @"PT0.0015S");
insta::assert_snapshot!(p("1.5µsecs"), @"PT0.0000015S");
insta::assert_snapshot!(p("1h 1.5mins"), @"PT1H1M30S");
insta::assert_snapshot!(p("1m 1.5secs"), @"PT1M1.5S");
insta::assert_snapshot!(p("1s 1.5msecs"), @"PT1.0015S");
insta::assert_snapshot!(p("1ms 1.5µsecs"), @"PT0.0010015S");
insta::assert_snapshot!(p("1s2000ms"), @"PT3S");
}
#[test]
fn parse_unsigned_duration_boundaries() {
let p = |s: &str| {
let dur = SpanParser::new().parse_unsigned_duration(s).unwrap();
crate::fmt::temporal::SpanPrinter::new()
.unsigned_duration_to_string(&dur)
};
let pe = |s: &str| SpanParser::new().parse_duration(s).unwrap_err();
insta::assert_snapshot!(p("175307616 hours"), @"PT175307616H");
insta::assert_snapshot!(p("10518456960 minutes"), @"PT175307616H");
insta::assert_snapshot!(p("631107417600 seconds"), @"PT175307616H");
insta::assert_snapshot!(p("631107417600000 milliseconds"), @"PT175307616H");
insta::assert_snapshot!(p("631107417600000000 microseconds"), @"PT175307616H");
insta::assert_snapshot!(p("9223372036854775807 nanoseconds"), @"PT2562047H47M16.854775807S");
insta::assert_snapshot!(p("175307617 hours"), @"PT175307617H");
insta::assert_snapshot!(p("10518456961 minutes"), @"PT175307616H1M");
insta::assert_snapshot!(p("631107417601 seconds"), @"PT175307616H1S");
insta::assert_snapshot!(p("631107417600001 milliseconds"), @"PT175307616H0.001S");
insta::assert_snapshot!(p("631107417600000001 microseconds"), @"PT175307616H0.000001S");
insta::assert_snapshot!(p("5124095576030431hours"), @"PT5124095576030431H");
insta::assert_snapshot!(
pe("5124095576030432hrs"),
@r#"failed to parse input in the "friendly" duration format: accumulated duration overflowed when adding value to unit hour"#,
);
insta::assert_snapshot!(p("307445734561825860minutes"), @"PT5124095576030431H");
insta::assert_snapshot!(
pe("307445734561825861mins"),
@r#"failed to parse input in the "friendly" duration format: accumulated duration overflowed when adding value to unit minute"#,
);
insta::assert_snapshot!(p("18446744073709551615seconds"), @"PT5124095576030431H15S");
insta::assert_snapshot!(
pe("18446744073709551616s"),
@r#"failed to parse input in the "friendly" duration format: number too big to parse into 64-bit integer"#,
);
}
#[test]
fn err_unsigned_duration_basic() {
let p = |s: &str| {
SpanParser::new().parse_unsigned_duration(s).unwrap_err()
};
insta::assert_snapshot!(
p(""),
@r#"failed to parse input in the "friendly" duration format: an empty string is not valid"#,
);
insta::assert_snapshot!(
p(" "),
@r#"failed to parse input in the "friendly" duration format: expected duration to start with a unit value (a decimal integer) after an optional sign, but no integer was found"#,
);
insta::assert_snapshot!(
p("5"),
@r#"failed to parse input in the "friendly" duration format: expected to find unit designator suffix (e.g., `years` or `secs`) after parsing integer"#,
);
insta::assert_snapshot!(
p("a"),
@r#"failed to parse input in the "friendly" duration format: expected duration to start with a unit value (a decimal integer) after an optional sign, but no integer was found"#,
);
insta::assert_snapshot!(
p("2 minutes 1 hour"),
@r#"failed to parse input in the "friendly" duration format: found value with unit hour after unit minute, but units must be written from largest to smallest (and they can't be repeated)"#,
);
insta::assert_snapshot!(
p("1 hour 1 minut"),
@r#"failed to parse input in the "friendly" duration format: parsed value '3660s', but unparsed input "ut" remains (expected no unparsed input)"#,
);
insta::assert_snapshot!(
p("2 minutes,"),
@r#"failed to parse input in the "friendly" duration format: expected whitespace after comma, but found end of input"#,
);
insta::assert_snapshot!(
p("2 minutes, "),
@r#"failed to parse input in the "friendly" duration format: found comma at the end of duration, but a comma indicates at least one more unit follows"#,
);
insta::assert_snapshot!(
p("2 minutes ,"),
@r#"failed to parse input in the "friendly" duration format: parsed value '120s', but unparsed input "," remains (expected no unparsed input)"#,
);
}
#[test]
fn err_unsigned_duration_sign() {
let p = |s: &str| {
SpanParser::new().parse_unsigned_duration(s).unwrap_err()
};
insta::assert_snapshot!(
p("1hago"),
@r#"failed to parse input in the "friendly" duration format: parsed value '3600s', but unparsed input "ago" remains (expected no unparsed input)"#,
);
insta::assert_snapshot!(
p("1 hour 1 minuteago"),
@r#"failed to parse input in the "friendly" duration format: parsed value '3660s', but unparsed input "ago" remains (expected no unparsed input)"#,
);
insta::assert_snapshot!(
p("+1 hour 1 minute ago"),
@r#"failed to parse input in the "friendly" duration format: expected to find either a prefix sign (+/-) or a suffix sign (`ago`), but found both"#,
);
insta::assert_snapshot!(
p("-1 hour 1 minute ago"),
@r#"failed to parse input in the "friendly" duration format: expected to find either a prefix sign (+/-) or a suffix sign (`ago`), but found both"#,
);
}
#[test]
fn err_unsigned_duration_overflow_fraction() {
let p = |s: &str| {
let dur = SpanParser::new().parse_unsigned_duration(s).unwrap();
crate::fmt::temporal::SpanPrinter::new()
.unsigned_duration_to_string(&dur)
};
let pe = |s: &str| {
SpanParser::new().parse_unsigned_duration(s).unwrap_err()
};
insta::assert_snapshot!(
pe("18446744073709551616 micros"),
@r#"failed to parse input in the "friendly" duration format: number too big to parse into 64-bit integer"#,
);
insta::assert_snapshot!(
p("18446744073709551615 micros"),
@"PT5124095576H1M49.551615S"
);
}
#[test]
fn err_unsigned_duration_fraction() {
let p = |s: &str| {
SpanParser::new().parse_unsigned_duration(s).unwrap_err()
};
insta::assert_snapshot!(
p("1.5 nanos"),
@r#"failed to parse input in the "friendly" duration format: fractional nanoseconds are not supported"#,
);
}
#[test]
fn err_unsigned_duration_hms() {
let p = |s: &str| {
SpanParser::new().parse_unsigned_duration(s).unwrap_err()
};
insta::assert_snapshot!(
p("05:"),
@r#"failed to parse input in the "friendly" duration format: when parsing the `HH:MM:SS` format, expected to parse minute following hour"#,
);
insta::assert_snapshot!(
p("05:06"),
@r#"failed to parse input in the "friendly" duration format: when parsing the `HH:MM:SS` format, expected to parse `:` following minute"#,
);
insta::assert_snapshot!(
p("05:06:"),
@r#"failed to parse input in the "friendly" duration format: when parsing the `HH:MM:SS` format, expected to parse second following minute"#,
);
insta::assert_snapshot!(
p("2 hours, 05:06:07"),
@r#"failed to parse input in the "friendly" duration format: found `HH:MM:SS` after unit hour, but `HH:MM:SS` can only appear after years, months, weeks or days"#,
);
}
}