use crate::{
error::{err, ErrorContext},
fmt::{
friendly::parser_label,
util::{parse_temporal_fraction, DurationUnits},
Parsed,
},
util::{c::Sign, escape, 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).with_context(|| {
err!(
"failed to parse {input:?} in the \"friendly\" format",
input = escape::Bytes(input)
)
})
}
#[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).with_context(|| {
err!(
"failed to parse {input:?} in the \"friendly\" format",
input = escape::Bytes(input)
)
})
}
#[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).with_context(|| {
err!(
"failed to parse {input:?} in the \"friendly\" format",
input = escape::Bytes(input)
)
})
}
#[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(err!("an empty string is not a valid duration"));
}
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(err!(
"parsing a friendly duration requires it to start \
with a unit value (a decimal integer) after an \
optional sign, but no integer was found",
));
};
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(err!(
"found comma at the end of duration, \
but a comma indicates at least one more \
unit follows",
));
}
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> {
if !input.first().map_or(false, |&b| b == b':') {
return Ok(Parsed { input, value: None });
}
let Parsed { input, value } = self.parse_hms(&input[1..], 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 Some(minute) = value else {
return Err(err!(
"expected to parse minute in 'HH:MM:SS' format \
following parsed hour of {hour}",
));
};
if !input.first().map_or(false, |&b| b == b':') {
return Err(err!(
"when parsing 'HH:MM:SS' format, expected to \
see a ':' after the parsed minute of {minute}",
));
}
let input = &input[1..];
let Parsed { input, value } = self.parse_unit_value(input)?;
let Some(second) = value else {
return Err(err!(
"expected to parse second in 'HH:MM:SS' format \
following parsed minute of {minute}",
));
};
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 Some((unit, len)) = parser_label::find(input) else {
if input.is_empty() {
return Err(err!(
"expected to find unit designator suffix \
(e.g., 'years' or 'secs'), \
but found end of input",
));
} else {
return Err(err!(
"expected to find unit designator suffix \
(e.g., 'years' or 'secs'), \
but found input beginning with {found:?} instead",
found = escape::Bytes(&input[..input.len().min(20)]),
));
}
};
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 input.starts_with(b"ago") {
(Some(Sign::Negative), &input[3..])
} else {
(None, input)
};
let sign = match (prefix_sign, suffix_sign) {
(Some(_), Some(_)) => {
return Err(err!(
"expected to find either a prefix sign (+/-) or \
a suffix sign (ago), but found both",
))
}
(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,
mut input: &'i [u8],
) -> Result<Parsed<'i, ()>, Error> {
if !input.first().map_or(false, |&b| b == b',') {
return Ok(Parsed { value: (), input });
}
input = &input[1..];
if input.is_empty() {
return Err(err!(
"expected whitespace after comma, but found end of input"
));
}
if !is_whitespace(&input[0]) {
return Err(err!(
"expected whitespace after comma, but found {found:?}",
found = escape::Byte(input[0]),
));
}
Ok(Parsed { value: (), input: &input[1..] })
}
#[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 "" in the "friendly" format: an empty string is not a valid duration"###,
);
insta::assert_snapshot!(
p(" "),
@r###"failed to parse " " in the "friendly" format: parsing a friendly duration requires it 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 "a" in the "friendly" format: parsing a friendly duration requires it 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 "2 months 1 year" in the "friendly" format: found value 1 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 "1 year 1 mont" in the "friendly" format: parsed value 'P1Y1M', but unparsed input "nt" remains (expected no unparsed input)"###,
);
insta::assert_snapshot!(
p("2 months,"),
@r###"failed to parse "2 months," in the "friendly" format: expected whitespace after comma, but found end of input"###,
);
insta::assert_snapshot!(
p("2 months, "),
@r#"failed to parse "2 months, " in the "friendly" 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 "2 months ," in the "friendly" 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 "1yago" in the "friendly" 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 "1 year 1 monthago" in the "friendly" 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 "+1 year 1 month ago" in the "friendly" 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 "-1 year 1 month ago" in the "friendly" 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 "640330789636854776 micros" in the "friendly" format: failed to set value 640330789636854776 as microsecond unit on span: failed to set nanosecond value 9223372036854776000 (it overflows `i64`) on span determined from 640330789636854776.0"#,
);
insta::assert_snapshot!(
p("640330789636854775 micros"),
@"PT640330789636.854775S"
);
insta::assert_snapshot!(
pe("640330789636854775.808 micros"),
@r#"failed to parse "640330789636854775.808 micros" in the "friendly" format: failed to set nanosecond value 9223372036854775808 (it overflows `i64`) on span determined from 640330789636854775.808000000"#,
);
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 "19999 years" in the "friendly" format: failed to set value 19999 as year unit on span: parameter 'years' with value 19999 is not in the required range of -19998..=19998"###,
);
insta::assert_snapshot!(
p("19999 years ago"),
@r#"failed to parse "19999 years ago" in the "friendly" format: failed to set value -19999 as year unit on span: parameter 'years' with value -19999 is not in the required range of -19998..=19998"#,
);
insta::assert_snapshot!(
p("239977 months"),
@r###"failed to parse "239977 months" in the "friendly" format: failed to set value 239977 as month unit on span: parameter 'months' with value 239977 is not in the required range of -239976..=239976"###,
);
insta::assert_snapshot!(
p("239977 months ago"),
@r#"failed to parse "239977 months ago" in the "friendly" format: failed to set value -239977 as month unit on span: parameter 'months' with value -239977 is not in the required range of -239976..=239976"#,
);
insta::assert_snapshot!(
p("1043498 weeks"),
@r###"failed to parse "1043498 weeks" in the "friendly" format: failed to set value 1043498 as week unit on span: parameter 'weeks' with value 1043498 is not in the required range of -1043497..=1043497"###,
);
insta::assert_snapshot!(
p("1043498 weeks ago"),
@r#"failed to parse "1043498 weeks ago" in the "friendly" format: failed to set value -1043498 as week unit on span: parameter 'weeks' with value -1043498 is not in the required range of -1043497..=1043497"#,
);
insta::assert_snapshot!(
p("7304485 days"),
@r###"failed to parse "7304485 days" in the "friendly" format: failed to set value 7304485 as day unit on span: parameter 'days' with value 7304485 is not in the required range of -7304484..=7304484"###,
);
insta::assert_snapshot!(
p("7304485 days ago"),
@r#"failed to parse "7304485 days ago" in the "friendly" format: failed to set value -7304485 as day unit on span: parameter 'days' with value -7304485 is not in the required range of -7304484..=7304484"#,
);
insta::assert_snapshot!(
p("9223372036854775808 nanoseconds"),
@r#"failed to parse "9223372036854775808 nanoseconds" in the "friendly" format: `9223372036854775808` 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 "9223372036854775808 nanoseconds ago" in the "friendly" format: failed to set value -9223372036854775808 as nanosecond unit on span: parameter 'nanoseconds' with value -9223372036854775808 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 "1.5 years" in the "friendly" format: fractional years are not supported"#,
);
insta::assert_snapshot!(
p("1.5 nanos"),
@r#"failed to parse "1.5 nanos" in the "friendly" 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 "05:" in the "friendly" format: expected to parse minute in 'HH:MM:SS' format following parsed hour of 5"###,
);
insta::assert_snapshot!(
p("05:06"),
@r###"failed to parse "05:06" in the "friendly" format: when parsing 'HH:MM:SS' format, expected to see a ':' after the parsed minute of 6"###,
);
insta::assert_snapshot!(
p("05:06:"),
@r###"failed to parse "05:06:" in the "friendly" format: expected to parse second in 'HH:MM:SS' format following parsed minute of 6"###,
);
insta::assert_snapshot!(
p("2 hours, 05:06:07"),
@r#"failed to parse "2 hours, 05:06:07" in the "friendly" 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 "9223372036854775808s" in the "friendly" format: `9223372036854775808` 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 "2562047788015216hrs" in the "friendly" format: accumulated `SignedDuration` of `0s` overflowed when adding 2562047788015216 of unit hour"#,
);
insta::assert_snapshot!(p("153722867280912930minutes"), @"PT2562047788015215H30M");
insta::assert_snapshot!(p("153722867280912930minutes ago"), @"-PT2562047788015215H30M");
insta::assert_snapshot!(
pe("153722867280912931mins"),
@r#"failed to parse "153722867280912931mins" in the "friendly" format: accumulated `SignedDuration` of `0s` overflowed when adding 153722867280912931 of unit minute"#,
);
insta::assert_snapshot!(p("9223372036854775807seconds"), @"PT2562047788015215H30M7S");
insta::assert_snapshot!(p("-9223372036854775807seconds"), @"-PT2562047788015215H30M7S");
insta::assert_snapshot!(
pe("9223372036854775808s"),
@r#"failed to parse "9223372036854775808s" in the "friendly" format: `9223372036854775808` 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 "" in the "friendly" format: an empty string is not a valid duration"###,
);
insta::assert_snapshot!(
p(" "),
@r###"failed to parse " " in the "friendly" format: parsing a friendly duration requires it 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 "5" in the "friendly" format: expected to find unit designator suffix (e.g., 'years' or 'secs'), but found end of input"###,
);
insta::assert_snapshot!(
p("a"),
@r###"failed to parse "a" in the "friendly" format: parsing a friendly duration requires it 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 "2 minutes 1 hour" in the "friendly" format: found value 1 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 "1 hour 1 minut" in the "friendly" format: parsed value 'PT1H1M', but unparsed input "ut" remains (expected no unparsed input)"###,
);
insta::assert_snapshot!(
p("2 minutes,"),
@r###"failed to parse "2 minutes," in the "friendly" format: expected whitespace after comma, but found end of input"###,
);
insta::assert_snapshot!(
p("2 minutes, "),
@r#"failed to parse "2 minutes, " in the "friendly" 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 "2 minutes ," in the "friendly" 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 "1hago" in the "friendly" 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 "1 hour 1 minuteago" in the "friendly" 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 "+1 hour 1 minute ago" in the "friendly" 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 "-1 hour 1 minute ago" in the "friendly" 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 "9223372036854775808 micros" in the "friendly" format: `9223372036854775808` 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 "1.5 nanos" in the "friendly" 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 "05:" in the "friendly" format: expected to parse minute in 'HH:MM:SS' format following parsed hour of 5"###,
);
insta::assert_snapshot!(
p("05:06"),
@r###"failed to parse "05:06" in the "friendly" format: when parsing 'HH:MM:SS' format, expected to see a ':' after the parsed minute of 6"###,
);
insta::assert_snapshot!(
p("05:06:"),
@r###"failed to parse "05:06:" in the "friendly" format: expected to parse second in 'HH:MM:SS' format following parsed minute of 6"###,
);
insta::assert_snapshot!(
p("2 hours, 05:06:07"),
@r#"failed to parse "2 hours, 05:06:07" in the "friendly" 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 "18446744073709551616s" in the "friendly" format: number `18446744073709551616` too big to parse into 64-bit integer"#,
);
insta::assert_snapshot!(
perr("-1s"),
@r#"failed to parse "-1s" in the "friendly" 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 "5124095576030432hrs" in the "friendly" format: accumulated `SignedDuration` of `0s` overflowed when adding 5124095576030432 of unit hour"#,
);
insta::assert_snapshot!(p("307445734561825860minutes"), @"PT5124095576030431H");
insta::assert_snapshot!(
pe("307445734561825861mins"),
@r#"failed to parse "307445734561825861mins" in the "friendly" format: accumulated `SignedDuration` of `0s` overflowed when adding 307445734561825861 of unit minute"#,
);
insta::assert_snapshot!(p("18446744073709551615seconds"), @"PT5124095576030431H15S");
insta::assert_snapshot!(
pe("18446744073709551616s"),
@r#"failed to parse "18446744073709551616s" in the "friendly" format: number `18446744073709551616` 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 "" in the "friendly" format: an empty string is not a valid duration"###,
);
insta::assert_snapshot!(
p(" "),
@r###"failed to parse " " in the "friendly" format: parsing a friendly duration requires it 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 "5" in the "friendly" format: expected to find unit designator suffix (e.g., 'years' or 'secs'), but found end of input"###,
);
insta::assert_snapshot!(
p("a"),
@r###"failed to parse "a" in the "friendly" format: parsing a friendly duration requires it 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 "2 minutes 1 hour" in the "friendly" format: found value 1 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 "1 hour 1 minut" in the "friendly" format: parsed value '3660s', but unparsed input "ut" remains (expected no unparsed input)"#,
);
insta::assert_snapshot!(
p("2 minutes,"),
@r###"failed to parse "2 minutes," in the "friendly" format: expected whitespace after comma, but found end of input"###,
);
insta::assert_snapshot!(
p("2 minutes, "),
@r#"failed to parse "2 minutes, " in the "friendly" 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 "2 minutes ," in the "friendly" 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 "1hago" in the "friendly" 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 "1 hour 1 minuteago" in the "friendly" 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 "+1 hour 1 minute ago" in the "friendly" 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 "-1 hour 1 minute ago" in the "friendly" 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 "18446744073709551616 micros" in the "friendly" format: number `18446744073709551616` 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 "1.5 nanos" in the "friendly" 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 "05:" in the "friendly" format: expected to parse minute in 'HH:MM:SS' format following parsed hour of 5"###,
);
insta::assert_snapshot!(
p("05:06"),
@r###"failed to parse "05:06" in the "friendly" format: when parsing 'HH:MM:SS' format, expected to see a ':' after the parsed minute of 6"###,
);
insta::assert_snapshot!(
p("05:06:"),
@r###"failed to parse "05:06:" in the "friendly" format: expected to parse second in 'HH:MM:SS' format following parsed minute of 6"###,
);
insta::assert_snapshot!(
p("2 hours, 05:06:07"),
@r#"failed to parse "2 hours, 05:06:07" in the "friendly" format: found `HH:MM:SS` after unit hour, but `HH:MM:SS` can only appear after years, months, weeks or days"#,
);
}
}