use crate::{
error::{err, ErrorContext},
fmt::{
friendly::parser_label,
util::{
fractional_time_to_duration, fractional_time_to_span,
parse_temporal_fraction,
},
Parsed,
},
util::{escape, t},
Error, SignedDuration, Span, Unit,
};
#[derive(Clone, Debug, Default)]
pub struct SpanParser {
_private: (),
}
impl SpanParser {
#[inline]
pub const fn new() -> SpanParser {
SpanParser { _private: () }
}
pub fn parse_span<I: AsRef<[u8]>>(&self, input: I) -> Result<Span, Error> {
let input = input.as_ref();
let parsed = self.parse_to_span(input).with_context(|| {
err!(
"failed to parse {input:?} in the \"friendly\" format",
input = escape::Bytes(input)
)
})?;
let span = parsed.into_full().with_context(|| {
err!(
"failed to parse {input:?} in the \"friendly\" format",
input = escape::Bytes(input)
)
})?;
Ok(span)
}
pub fn parse_duration<I: AsRef<[u8]>>(
&self,
input: I,
) -> Result<SignedDuration, Error> {
let input = input.as_ref();
let parsed = self.parse_to_duration(input).with_context(|| {
err!(
"failed to parse {input:?} in the \"friendly\" format",
input = escape::Bytes(input)
)
})?;
let sdur = parsed.into_full().with_context(|| {
err!(
"failed to parse {input:?} in the \"friendly\" format",
input = escape::Bytes(input)
)
})?;
Ok(sdur)
}
#[inline(always)]
fn parse_to_span<'i>(
&self,
input: &'i [u8],
) -> Result<Parsed<'i, Span>, 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 { value: span, input } =
self.parse_units_to_span(input, first_unit_value)?;
let (sign, input) = if !input.first().map_or(false, is_whitespace) {
(sign.unwrap_or(t::Sign::N::<1>()), input)
} else {
let parsed = self.parse_suffix_sign(sign, input)?;
(parsed.value, parsed.input)
};
Ok(Parsed { value: span * i64::from(sign.get()), input })
}
#[inline(always)]
fn parse_to_duration<'i>(
&self,
input: &'i [u8],
) -> Result<Parsed<'i, SignedDuration>, 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 { value: mut sdur, input } =
self.parse_units_to_duration(input, first_unit_value)?;
let (sign, input) = if !input.first().map_or(false, is_whitespace) {
(sign.unwrap_or(t::Sign::N::<1>()), input)
} else {
let parsed = self.parse_suffix_sign(sign, input)?;
(parsed.value, parsed.input)
};
if sign < 0 {
sdur = -sdur;
}
Ok(Parsed { value: sdur, input })
}
#[inline(always)]
fn parse_units_to_span<'i>(
&self,
mut input: &'i [u8],
first_unit_value: t::NoUnits,
) -> Result<Parsed<'i, Span>, Error> {
let mut parsed_any_after_comma = true;
let mut prev_unit: Option<Unit> = None;
let mut value = first_unit_value;
let mut span = Span::new();
loop {
let parsed = self.parse_hms_maybe(input, value)?;
input = parsed.input;
if let Some(hms) = parsed.value {
if let Some(prev_unit) = prev_unit {
if prev_unit <= Unit::Hour {
return Err(err!(
"found 'HH:MM:SS' after unit {prev_unit}, \
but 'HH:MM:SS' can only appear after \
years, months, weeks or days",
prev_unit = prev_unit.singular(),
));
}
}
span = set_span_unit_value(Unit::Hour, hms.hour, span)?;
span = set_span_unit_value(Unit::Minute, hms.minute, span)?;
span = if let Some(fraction) = hms.fraction {
fractional_time_to_span(
Unit::Second,
hms.second,
fraction,
span,
)?
} else {
set_span_unit_value(Unit::Second, hms.second, span)?
};
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;
}
if let Some(prev_unit) = prev_unit {
if prev_unit <= unit {
return Err(err!(
"found value {value:?} with unit {unit} \
after unit {prev_unit}, but units must be \
written from largest to smallest \
(and they can't be repeated)",
unit = unit.singular(),
prev_unit = prev_unit.singular(),
));
}
}
prev_unit = Some(unit);
if let Some(fraction) = fraction {
span = fractional_time_to_span(unit, value, fraction, span)?;
break;
} else {
span = set_span_unit_value(unit, value, span)?;
}
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 and none were found after \
{prev_unit}",
prev_unit = prev_unit.unwrap().plural(),
));
}
Ok(Parsed { value: span, input })
}
#[inline(always)]
fn parse_units_to_duration<'i>(
&self,
mut input: &'i [u8],
first_unit_value: t::NoUnits,
) -> Result<Parsed<'i, SignedDuration>, Error> {
let mut parsed_any_after_comma = true;
let mut prev_unit: Option<Unit> = None;
let mut value = first_unit_value;
let mut sdur = SignedDuration::ZERO;
loop {
let parsed = self.parse_hms_maybe(input, value)?;
input = parsed.input;
if let Some(hms) = parsed.value {
if let Some(prev_unit) = prev_unit {
if prev_unit <= Unit::Hour {
return Err(err!(
"found 'HH:MM:SS' after unit {prev_unit}, \
but 'HH:MM:SS' can only appear after \
years, months, weeks or days",
prev_unit = prev_unit.singular(),
));
}
}
sdur = sdur
.checked_add(duration_unit_value(Unit::Hour, hms.hour)?)
.ok_or_else(|| {
err!(
"accumulated `SignedDuration` overflowed when \
adding {value} of unit hour",
)
})?;
sdur = sdur
.checked_add(duration_unit_value(
Unit::Minute,
hms.minute,
)?)
.ok_or_else(|| {
err!(
"accumulated `SignedDuration` overflowed when \
adding {value} of unit minute",
)
})?;
sdur = sdur
.checked_add(duration_unit_value(
Unit::Second,
hms.second,
)?)
.ok_or_else(|| {
err!(
"accumulated `SignedDuration` overflowed when \
adding {value} of unit second",
)
})?;
if let Some(f) = hms.fraction {
let f = fractional_time_to_duration(Unit::Second, f)?;
sdur = sdur.checked_add(f).ok_or_else(|| err!(""))?;
};
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;
}
if let Some(prev_unit) = prev_unit {
if prev_unit <= unit {
return Err(err!(
"found value {value:?} with unit {unit} \
after unit {prev_unit}, but units must be \
written from largest to smallest \
(and they can't be repeated)",
unit = unit.singular(),
prev_unit = prev_unit.singular(),
));
}
}
prev_unit = Some(unit);
sdur = sdur
.checked_add(duration_unit_value(unit, value)?)
.ok_or_else(|| {
err!(
"accumulated `SignedDuration` overflowed when adding \
{value} of unit {unit}",
unit = unit.singular(),
)
})?;
if let Some(f) = fraction {
let f = fractional_time_to_duration(unit, f)?;
sdur = sdur.checked_add(f).ok_or_else(|| err!(""))?;
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 and none were found after \
{prev_unit}",
prev_unit = prev_unit.unwrap().plural(),
));
}
Ok(Parsed { value: sdur, input })
}
#[inline(always)]
fn parse_hms_maybe<'i>(
&self,
input: &'i [u8],
hour: t::NoUnits,
) -> 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: t::NoUnits,
) -> 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 })
}
#[inline(always)]
fn parse_unit_value<'i>(
&self,
mut input: &'i [u8],
) -> Result<Parsed<'i, Option<t::NoUnits>>, Error> {
const MAX_I64_DIGITS: usize = 19;
let mut digit_count = 0;
let mut n: i64 = 0;
while digit_count <= MAX_I64_DIGITS
&& input.get(digit_count).map_or(false, u8::is_ascii_digit)
{
let byte = input[digit_count];
digit_count += 1;
let digit = match byte.checked_sub(b'0') {
None => {
return Err(err!(
"invalid digit, expected 0-9 but got {}",
escape::Byte(byte),
));
}
Some(digit) if digit > 9 => {
return Err(err!(
"invalid digit, expected 0-9 but got {}",
escape::Byte(byte),
))
}
Some(digit) => {
debug_assert!((0..=9).contains(&digit));
i64::from(digit)
}
};
n = n
.checked_mul(10)
.and_then(|n| n.checked_add(digit))
.ok_or_else(|| {
err!(
"number '{}' too big to parse into 64-bit integer",
escape::Bytes(&input[..digit_count]),
)
})?;
}
if digit_count == 0 {
return Ok(Parsed { value: None, input });
}
input = &input[digit_count..];
let value = t::NoUnits::new(n).unwrap();
Ok(Parsed { value: Some(value), input })
}
#[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<t::Sign>> {
let Some(sign) = input.first().copied() else {
return Parsed { value: None, input };
};
let sign = if sign == b'+' {
t::Sign::N::<1>()
} else if sign == b'-' {
t::Sign::N::<-1>()
} else {
return Parsed { value: None, input };
};
Parsed { value: Some(sign), input: &input[1..] }
}
#[inline(never)]
fn parse_suffix_sign<'i>(
&self,
prefix_sign: Option<t::Sign>,
mut input: &'i [u8],
) -> Result<Parsed<'i, t::Sign>, Error> {
if !input.first().map_or(false, is_whitespace) {
let sign = prefix_sign.unwrap_or(t::Sign::N::<1>());
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(t::Sign::N::<-1>()), &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) => t::Sign::N::<1>(),
};
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..] })
}
#[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: t::NoUnits,
minute: t::NoUnits,
second: t::NoUnits,
fraction: Option<t::SubsecNanosecond>,
}
#[inline(always)]
fn set_span_unit_value(
unit: Unit,
value: t::NoUnits,
mut span: Span,
) -> Result<Span, Error> {
if unit <= Unit::Hour {
let result = span.try_units_ranged(unit, value).with_context(|| {
err!(
"failed to set value {value:?} \
as {unit} unit on span",
unit = Unit::from(unit).singular(),
)
});
span = match result {
Ok(span) => span,
Err(_) => fractional_time_to_span(
unit,
value,
t::SubsecNanosecond::N::<0>(),
span,
)?,
};
} else {
span = span.try_units_ranged(unit, value).with_context(|| {
err!(
"failed to set value {value:?} \
as {unit} unit on span",
unit = Unit::from(unit).singular(),
)
})?;
}
Ok(span)
}
#[inline(always)]
fn duration_unit_value(
unit: Unit,
value: t::NoUnits,
) -> Result<SignedDuration, Error> {
let sdur = match unit {
Unit::Hour => {
let seconds =
value.checked_mul(t::SECONDS_PER_HOUR).ok_or_else(|| {
err!("converting {value} hours to seconds overflows i64")
})?;
SignedDuration::from_secs(seconds.get())
}
Unit::Minute => {
let seconds = value.try_checked_mul(
"minutes-to-seconds",
t::SECONDS_PER_MINUTE,
)?;
SignedDuration::from_secs(seconds.get())
}
Unit::Second => SignedDuration::from_secs(value.get()),
Unit::Millisecond => SignedDuration::from_millis(value.get()),
Unit::Microsecond => SignedDuration::from_micros(value.get()),
Unit::Nanosecond => SignedDuration::from_nanos(value.get()),
unsupported => {
return Err(err!(
"parsing {unit} units into a `SignedDuration` is not supported \
(perhaps try parsing into a `Span` instead)",
unit = unsupported.singular(),
));
}
};
Ok(sdur)
}
#[inline(always)]
fn is_whitespace(byte: &u8) -> bool {
matches!(*byte, b' ' | b'\t' | b'\n' | b'\r' | b'\x0C')
}
#[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 and none were found after months"###,
);
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 nanosecond value 9223372036854776000 on span determined from 640330789636854776.0: parameter 'nanoseconds' with value 9223372036854776000 is not in the required range of -9223372036854775807..=9223372036854775807"###,
);
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 on span determined from 640330789636854775.808000000: parameter 'nanoseconds' with value 9223372036854775808 is not in the required range of -9223372036854775807..=9223372036854775807"###,
);
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: number '9223372036854775808' too big to parse into 64-bit integer"###,
);
insta::assert_snapshot!(
p("9223372036854775808 nanoseconds ago"),
@r###"failed to parse "9223372036854775808 nanoseconds ago" in the "friendly" format: number '9223372036854775808' too big to parse into 64-bit integer"###,
);
}
#[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 year units are not allowed"###,
);
insta::assert_snapshot!(
p("1.5 nanos"),
@r###"failed to parse "1.5 nanos" in the "friendly" format: fractional nanosecond units are not allowed"###,
);
}
#[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_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_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: number '9223372036854775808' too big to parse into 64-bit integer"###,
);
insta::assert_snapshot!(
perr("-9223372036854775808s"),
@r###"failed to parse "-9223372036854775808s" in the "friendly" format: number '9223372036854775808' too big to parse into 64-bit integer"###,
);
}
#[test]
fn parse_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_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: converting 2562047788015216 hours to seconds overflows i64"###,
);
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: parameter 'minutes-to-seconds' with value 60 is not in the required range of -9223372036854775808..=9223372036854775807"###,
);
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: number '9223372036854775808' too big to parse into 64-bit integer"###,
);
insta::assert_snapshot!(
pe("-9223372036854775808s"),
@r###"failed to parse "-9223372036854775808s" in the "friendly" format: number '9223372036854775808' too big to parse into 64-bit integer"###,
);
}
#[test]
fn err_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 and none were found after minutes"###,
);
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_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_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: number '9223372036854775808' too big to parse into 64-bit integer"###,
);
insta::assert_snapshot!(
p("9223372036854775807 micros"),
@"PT2562047788H54.775807S"
);
}
#[test]
fn err_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 nanosecond units are not allowed"###,
);
}
#[test]
fn err_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"###,
);
}
}