1
0
mirror of https://github.com/bspeice/dtparse synced 2025-04-04 13:01:29 -04:00

fix: invalid timezone should return err not panic

This commit is contained in:
taichong 2024-07-16 22:18:00 +08:00
parent 61d2aa60c6
commit 317a854b9f
4 changed files with 2678 additions and 747 deletions

View File

@ -715,7 +715,8 @@ impl Parser {
) -> ParseResult<(NaiveDateTime, Option<FixedOffset>, Option<Vec<String>>)> { ) -> ParseResult<(NaiveDateTime, Option<FixedOffset>, Option<Vec<String>>)> {
let default_date = default.unwrap_or(&Local::now().naive_local()).date(); let default_date = default.unwrap_or(&Local::now().naive_local()).date();
let default_ts = NaiveDateTime::new(default_date, NaiveTime::from_hms_opt(0, 0, 0).unwrap()); let default_ts =
NaiveDateTime::new(default_date, NaiveTime::from_hms_opt(0, 0, 0).unwrap());
let (res, tokens) = let (res, tokens) =
self.parse_with_tokens(timestr, dayfirst, yearfirst, fuzzy, fuzzy_with_tokens)?; self.parse_with_tokens(timestr, dayfirst, yearfirst, fuzzy, fuzzy_with_tokens)?;
@ -853,7 +854,12 @@ impl Parser {
} else if i + 2 < len_l && l[i + 2] == ":" { } else if i + 2 < len_l && l[i + 2] == ":" {
// -03:00 // -03:00
hour_offset = Some(l[i + 1].parse::<i32>()?); hour_offset = Some(l[i + 1].parse::<i32>()?);
min_offset = Some(l[i + 3].parse::<i32>()?); // if timezone is wrong format like "-03:" just return a Err, should not panic.
min_offset = if i + 3 > l.len() - 1 {
return Err(ParseError::TimezoneUnsupported);
} else {
Some(l[i + 3].parse::<i32>()?)
};
i += 2; i += 2;
} else if len_li <= 2 { } else if len_li <= 2 {
// -[0]3 // -[0]3
@ -979,9 +985,9 @@ impl Parser {
let hour = res.hour.unwrap_or(default.hour() as i32) as u32; let hour = res.hour.unwrap_or(default.hour() as i32) as u32;
let minute = res.minute.unwrap_or(default.minute() as i32) as u32; let minute = res.minute.unwrap_or(default.minute() as i32) as u32;
let second = res.second.unwrap_or(default.second() as i32) as u32; let second = res.second.unwrap_or(default.second() as i32) as u32;
let nanosecond = res let nanosecond =
.nanosecond res.nanosecond
.unwrap_or(default.timestamp_subsec_nanos() as i64) as u32; .unwrap_or(default.and_utc().timestamp_subsec_nanos() as i64) as u32;
let t = let t =
NaiveTime::from_hms_nano_opt(hour, minute, second, nanosecond).ok_or_else(|| { NaiveTime::from_hms_nano_opt(hour, minute, second, nanosecond).ok_or_else(|| {
if hour >= 24 { if hour >= 24 {
@ -1155,7 +1161,10 @@ impl Parser {
idx += 1; idx += 1;
} else { } else {
//let value = value.floor().to_i32().ok_or(Err(ParseError::InvalidNumeric())) //let value = value.floor().to_i32().ok_or(Err(ParseError::InvalidNumeric()))
let value = value.floor().to_i32().ok_or_else(|| ParseError::InvalidNumeric(value_repr.to_owned()))?; let value = value
.floor()
.to_i32()
.ok_or_else(|| ParseError::InvalidNumeric(value_repr.to_owned()))?;
ymd.append(value, &value_repr, None)?; ymd.append(value, &value_repr, None)?;
} }

View File

@ -1,5 +1,5 @@
use base64::Engine;
use base64::engine::general_purpose::STANDARD; use base64::engine::general_purpose::STANDARD;
use base64::Engine;
use chrono::NaiveDate; use chrono::NaiveDate;
use std::collections::HashMap; use std::collections::HashMap;
use std::str; use std::str;
@ -21,7 +21,10 @@ fn test_fuzz() {
Err(ParseError::UnrecognizedFormat) Err(ParseError::UnrecognizedFormat)
); );
let default = NaiveDate::from_ymd_opt(2016, 6, 29).unwrap().and_hms_opt(0, 0, 0).unwrap(); let default = NaiveDate::from_ymd_opt(2016, 6, 29)
.unwrap()
.and_hms_opt(0, 0, 0)
.unwrap();
let p = Parser::default(); let p = Parser::default();
let res = p.parse( let res = p.parse(
"\x0D\x31", "\x0D\x31",
@ -66,12 +69,20 @@ fn empty_string() {
#[test] #[test]
fn github_33() { fn github_33() {
assert_eq!(parse("66:'"), Err(ParseError::InvalidNumeric("'".to_owned()))) assert_eq!(
parse("66:'"),
Err(ParseError::InvalidNumeric("'".to_owned()))
)
} }
#[test] #[test]
fn github_32() { fn github_32() {
assert_eq!(parse("99999999999999999999999"), Err(ParseError::InvalidNumeric("99999999999999999999999".to_owned()))) assert_eq!(
parse("99999999999999999999999"),
Err(ParseError::InvalidNumeric(
"99999999999999999999999".to_owned()
))
)
} }
#[test] #[test]
@ -97,3 +108,9 @@ fn github_36() {
let parse_result = parse(parse_str); let parse_result = parse(parse_str);
assert!(parse_result.is_err()); assert!(parse_result.is_err());
} }
#[test]
fn github_46() {
let parse_result = parse("2000-01-01 12:00:00+00:");
assert!(parse_result.is_err());
}

View File

@ -2,13 +2,19 @@ mod fuzzing;
mod pycompat_parser; mod pycompat_parser;
mod pycompat_tokenizer; mod pycompat_tokenizer;
use chrono::NaiveDate;
use crate::parse; use crate::parse;
use chrono::NaiveDate;
#[test] #[test]
fn nanosecond_precision() { fn nanosecond_precision() {
assert_eq!( assert_eq!(
parse("2008.12.29T08:09:10.123456789").unwrap(), parse("2008.12.29T08:09:10.123456789").unwrap(),
(NaiveDate::from_ymd_opt(2008, 12, 29).unwrap().and_hms_nano_opt(8, 9, 10, 123_456_789).unwrap(), None) (
NaiveDate::from_ymd_opt(2008, 12, 29)
.unwrap()
.and_hms_nano_opt(8, 9, 10, 123_456_789)
.unwrap(),
None
)
) )
} }

File diff suppressed because it is too large Load Diff