1use super::{INVALID, OUT_OF_RANGE, ParseResult, TOO_SHORT};
9use crate::Weekday;
10
11#[inline]
17pub(super) fn number(s: &str, min: usize, max: usize) -> ParseResult<(&str, i64)> {
18 if !(min <= max) { ::core::panicking::panic("assertion failed: min <= max") };assert!(min <= max);
19
20 let bytes = s.as_bytes();
24 if bytes.len() < min {
25 return Err(TOO_SHORT);
26 }
27
28 let mut n = 0i64;
29 for (i, c) in bytes.iter().take(max).cloned().enumerate() {
30 if !c.is_ascii_digit() {
32 if i < min {
33 return Err(INVALID);
34 } else {
35 return Ok((&s[i..], n));
36 }
37 }
38
39 n = match n.checked_mul(10).and_then(|n| n.checked_add((c - b'0') as i64)) {
40 Some(n) => n,
41 None => return Err(OUT_OF_RANGE),
42 };
43 }
44
45 Ok((&s[core::cmp::min(max, bytes.len())..], n))
46}
47
48pub(super) fn nanosecond(s: &str) -> ParseResult<(&str, u32)> {
51 let origlen = s.len();
53 let (s, v) = number(s, 1, 9)?;
54 let v = u32::try_from(v).expect("999,999,999 should fit u32");
55 let consumed = origlen - s.len();
56
57 const SCALE: [u32; 10] =
59 [0, 100_000_000, 10_000_000, 1_000_000, 100_000, 10_000, 1_000, 100, 10, 1];
60 let v = v.checked_mul(SCALE[consumed]).ok_or(OUT_OF_RANGE)?;
61
62 let s = s.trim_start_matches(|c: char| c.is_ascii_digit());
64
65 Ok((s, v))
66}
67
68pub(super) fn nanosecond_fixed(s: &str, digits: usize) -> ParseResult<(&str, i64)> {
71 let (s, v) = number(s, digits, digits)?;
73
74 static SCALE: [i64; 10] =
76 [0, 100_000_000, 10_000_000, 1_000_000, 100_000, 10_000, 1_000, 100, 10, 1];
77 let v = v.checked_mul(SCALE[digits]).ok_or(OUT_OF_RANGE)?;
78
79 Ok((s, v))
80}
81
82pub(super) fn short_month0(s: &str) -> ParseResult<(&str, u8)> {
84 if s.len() < 3 {
85 return Err(TOO_SHORT);
86 }
87 let buf = s.as_bytes();
88 let month0 = match (buf[0] | 32, buf[1] | 32, buf[2] | 32) {
89 (b'j', b'a', b'n') => 0,
90 (b'f', b'e', b'b') => 1,
91 (b'm', b'a', b'r') => 2,
92 (b'a', b'p', b'r') => 3,
93 (b'm', b'a', b'y') => 4,
94 (b'j', b'u', b'n') => 5,
95 (b'j', b'u', b'l') => 6,
96 (b'a', b'u', b'g') => 7,
97 (b's', b'e', b'p') => 8,
98 (b'o', b'c', b't') => 9,
99 (b'n', b'o', b'v') => 10,
100 (b'd', b'e', b'c') => 11,
101 _ => return Err(INVALID),
102 };
103 Ok((&s[3..], month0))
104}
105
106pub(super) fn short_weekday(s: &str) -> ParseResult<(&str, Weekday)> {
108 if s.len() < 3 {
109 return Err(TOO_SHORT);
110 }
111 let buf = s.as_bytes();
112 let weekday = match (buf[0] | 32, buf[1] | 32, buf[2] | 32) {
113 (b'm', b'o', b'n') => Weekday::Mon,
114 (b't', b'u', b'e') => Weekday::Tue,
115 (b'w', b'e', b'd') => Weekday::Wed,
116 (b't', b'h', b'u') => Weekday::Thu,
117 (b'f', b'r', b'i') => Weekday::Fri,
118 (b's', b'a', b't') => Weekday::Sat,
119 (b's', b'u', b'n') => Weekday::Sun,
120 _ => return Err(INVALID),
121 };
122 Ok((&s[3..], weekday))
123}
124
125pub(super) fn short_or_long_month0(s: &str) -> ParseResult<(&str, u8)> {
128 static LONG_MONTH_SUFFIXES: [&[u8]; 12] = [
130 b"uary", b"ruary", b"ch", b"il", b"", b"e", b"y", b"ust", b"tember", b"ober", b"ember",
131 b"ember",
132 ];
133
134 let (mut s, month0) = short_month0(s)?;
135
136 let suffix = LONG_MONTH_SUFFIXES[month0 as usize];
138 if s.len() >= suffix.len() && s.as_bytes()[..suffix.len()].eq_ignore_ascii_case(suffix) {
139 s = &s[suffix.len()..];
140 }
141
142 Ok((s, month0))
143}
144
145pub(super) fn short_or_long_weekday(s: &str) -> ParseResult<(&str, Weekday)> {
148 static LONG_WEEKDAY_SUFFIXES: [&[u8]; 7] =
150 [b"day", b"sday", b"nesday", b"rsday", b"day", b"urday", b"day"];
151
152 let (mut s, weekday) = short_weekday(s)?;
153
154 let suffix = LONG_WEEKDAY_SUFFIXES[weekday.num_days_from_monday() as usize];
156 if s.len() >= suffix.len() && s.as_bytes()[..suffix.len()].eq_ignore_ascii_case(suffix) {
157 s = &s[suffix.len()..];
158 }
159
160 Ok((s, weekday))
161}
162
163pub(super) fn char(s: &str, c1: u8) -> ParseResult<&str> {
165 match s.as_bytes().first() {
166 Some(&c) if c == c1 => Ok(&s[1..]),
167 Some(_) => Err(INVALID),
168 None => Err(TOO_SHORT),
169 }
170}
171
172pub(super) fn space(s: &str) -> ParseResult<&str> {
174 let s_ = s.trim_start();
175 if s_.len() < s.len() {
176 Ok(s_)
177 } else if s.is_empty() {
178 Err(TOO_SHORT)
179 } else {
180 Err(INVALID)
181 }
182}
183
184pub(crate) fn colon_or_space(s: &str) -> ParseResult<&str> {
186 Ok(s.trim_start_matches(|c: char| c == ':' || c.is_whitespace()))
187}
188
189pub(crate) fn timezone_offset<F>(
204 mut s: &str,
205 mut consume_colon: F,
206 allow_zulu: bool,
207 allow_missing_minutes: bool,
208 allow_tz_minus_sign: bool,
209) -> ParseResult<(&str, i32)>
210where
211 F: FnMut(&str) -> ParseResult<&str>,
212{
213 if allow_zulu {
214 if let Some(&b'Z' | &b'z') = s.as_bytes().first() {
215 return Ok((&s[1..], 0));
216 }
217 }
218
219 const fn digits(s: &str) -> ParseResult<(u8, u8)> {
220 let b = s.as_bytes();
221 if b.len() < 2 { Err(TOO_SHORT) } else { Ok((b[0], b[1])) }
222 }
223 let negative = match s.chars().next() {
224 Some('+') => {
225 s = &s['+'.len_utf8()..];
227
228 false
229 }
230 Some('-') => {
231 s = &s['-'.len_utf8()..];
233
234 true
235 }
236 Some('−') => {
237 if !allow_tz_minus_sign {
239 return Err(INVALID);
240 }
241 s = &s['−'.len_utf8()..];
242
243 true
244 }
245 Some(_) => return Err(INVALID),
246 None => return Err(TOO_SHORT),
247 };
248
249 let hours = match digits(s)? {
251 (h1 @ b'0'..=b'9', h2 @ b'0'..=b'9') => i32::from((h1 - b'0') * 10 + (h2 - b'0')),
252 _ => return Err(INVALID),
253 };
254 s = &s[2..];
255
256 s = consume_colon(s)?;
258
259 let minutes = if let Ok(ds) = digits(s) {
262 match ds {
263 (m1 @ b'0'..=b'5', m2 @ b'0'..=b'9') => i32::from((m1 - b'0') * 10 + (m2 - b'0')),
264 (b'6'..=b'9', b'0'..=b'9') => return Err(OUT_OF_RANGE),
265 _ => return Err(INVALID),
266 }
267 } else if allow_missing_minutes {
268 0
269 } else {
270 return Err(TOO_SHORT);
271 };
272 s = match s.len() {
273 len if len >= 2 => &s[2..],
274 0 => s,
275 _ => return Err(TOO_SHORT),
276 };
277
278 let seconds = hours * 3600 + minutes * 60;
279 Ok((s, if negative { -seconds } else { seconds }))
280}
281
282pub(super) fn timezone_offset_2822(s: &str) -> ParseResult<(&str, i32)> {
288 let upto = s.as_bytes().iter().position(|&c| !c.is_ascii_alphabetic()).unwrap_or(s.len());
290 if upto > 0 {
291 let name = &s.as_bytes()[..upto];
292 let s = &s[upto..];
293 let offset_hours = |o| Ok((s, o * 3600));
294 if name.eq_ignore_ascii_case(b"gmt")
297 || name.eq_ignore_ascii_case(b"ut")
298 || name.eq_ignore_ascii_case(b"z")
299 {
300 return offset_hours(0);
301 } else if name.eq_ignore_ascii_case(b"edt") {
302 return offset_hours(-4);
303 } else if name.eq_ignore_ascii_case(b"est") || name.eq_ignore_ascii_case(b"cdt") {
304 return offset_hours(-5);
305 } else if name.eq_ignore_ascii_case(b"cst") || name.eq_ignore_ascii_case(b"mdt") {
306 return offset_hours(-6);
307 } else if name.eq_ignore_ascii_case(b"mst") || name.eq_ignore_ascii_case(b"pdt") {
308 return offset_hours(-7);
309 } else if name.eq_ignore_ascii_case(b"pst") {
310 return offset_hours(-8);
311 } else if name.len() == 1 {
312 if let b'a'..=b'i' | b'k'..=b'y' | b'A'..=b'I' | b'K'..=b'Y' = name[0] {
313 return Ok((s, 0));
315 }
316 }
317 Err(INVALID)
318 } else {
319 timezone_offset(s, |s| Ok(s), false, false, false)
320 }
321}
322
323pub(super) fn comment_2822(s: &str) -> ParseResult<(&str, ())> {
327 use CommentState::*;
328
329 let s = s.trim_start();
330
331 let mut state = Start;
332 for (i, c) in s.bytes().enumerate() {
333 state = match (state, c) {
334 (Start, b'(') => Next(1),
335 (Next(1), b')') => return Ok((&s[i + 1..], ())),
336 (Next(depth), b'\\') => Escape(depth),
337 (Next(depth), b'(') => Next(depth + 1),
338 (Next(depth), b')') => Next(depth - 1),
339 (Next(depth), _) | (Escape(depth), _) => Next(depth),
340 _ => return Err(INVALID),
341 };
342 }
343
344 Err(TOO_SHORT)
345}
346
347enum CommentState {
348 Start,
349 Next(usize),
350 Escape(usize),
351}
352
353#[cfg(test)]
354mod tests {
355 use super::{
356 comment_2822, nanosecond, nanosecond_fixed, short_or_long_month0, short_or_long_weekday,
357 timezone_offset_2822,
358 };
359 use crate::Weekday;
360 use crate::format::{INVALID, TOO_SHORT};
361
362 #[test]
363 fn test_rfc2822_comments() {
364 let testdata = [
365 ("", Err(TOO_SHORT)),
366 (" ", Err(TOO_SHORT)),
367 ("x", Err(INVALID)),
368 ("(", Err(TOO_SHORT)),
369 ("()", Ok("")),
370 (" \r\n\t()", Ok("")),
371 ("() ", Ok(" ")),
372 ("()z", Ok("z")),
373 ("(x)", Ok("")),
374 ("(())", Ok("")),
375 ("((()))", Ok("")),
376 ("(x(x(x)x)x)", Ok("")),
377 ("( x ( x ( x ) x ) x )", Ok("")),
378 (r"(\)", Err(TOO_SHORT)),
379 (r"(\()", Ok("")),
380 (r"(\))", Ok("")),
381 (r"(\\)", Ok("")),
382 ("(()())", Ok("")),
383 ("( x ( x ) x ( x ) x )", Ok("")),
384 ];
385
386 for (test_in, expected) in testdata.iter() {
387 let actual = comment_2822(test_in).map(|(s, _)| s);
388 assert_eq!(
389 *expected, actual,
390 "{test_in:?} expected to produce {expected:?}, but produced {actual:?}."
391 );
392 }
393 }
394
395 #[test]
396 fn test_timezone_offset_2822() {
397 assert_eq!(timezone_offset_2822("cSt").unwrap(), ("", -21600));
398 assert_eq!(timezone_offset_2822("pSt").unwrap(), ("", -28800));
399 assert_eq!(timezone_offset_2822("mSt").unwrap(), ("", -25200));
400 assert_eq!(timezone_offset_2822("-1551").unwrap(), ("", -57060));
401 assert_eq!(timezone_offset_2822("Gp"), Err(INVALID));
402 }
403
404 #[test]
405 fn test_short_or_long_month0() {
406 assert_eq!(short_or_long_month0("JUn").unwrap(), ("", 5));
407 assert_eq!(short_or_long_month0("mAy").unwrap(), ("", 4));
408 assert_eq!(short_or_long_month0("AuG").unwrap(), ("", 7));
409 assert_eq!(short_or_long_month0("Aprâ").unwrap(), ("â", 3));
410 assert_eq!(short_or_long_month0("JUl").unwrap(), ("", 6));
411 assert_eq!(short_or_long_month0("mAr").unwrap(), ("", 2));
412 assert_eq!(short_or_long_month0("Jan").unwrap(), ("", 0));
413 }
414
415 #[test]
416 fn test_short_or_long_weekday() {
417 assert_eq!(short_or_long_weekday("sAtu").unwrap(), ("u", Weekday::Sat));
418 assert_eq!(short_or_long_weekday("thu").unwrap(), ("", Weekday::Thu));
419 }
420
421 #[test]
422 fn test_nanosecond_fixed() {
423 assert_eq!(nanosecond_fixed("", 0usize).unwrap(), ("", 0));
424 assert!(nanosecond_fixed("", 1usize).is_err());
425 }
426
427 #[test]
428 fn test_nanosecond() {
429 assert_eq!(nanosecond("2Ù").unwrap(), ("Ù", 200000000));
430 assert_eq!(nanosecond("8").unwrap(), ("", 800000000));
431 }
432}