Skip to content

Commit

Permalink
Prevent the potential for a panic in the string_to_timestamp coercions
Browse files Browse the repository at this point in the history
The original code was turning nanos into micros anyways, so we can use
the more safe timestamp_micros() code.

For timestamp_nanos_opt() it can return None if the dates that cannot be
represented as nanoseconds between 1677-09-21T00:12:43.145224192 and
2262-04-11T23:47:16.854775807.

Signed-off-by: R. Tyler Croy <[email protected]>
  • Loading branch information
rtyler committed Dec 10, 2024
1 parent b7638ed commit d353ef5
Showing 1 changed file with 15 additions and 10 deletions.
25 changes: 15 additions & 10 deletions src/coercions.rs
Original file line number Diff line number Diff line change
Expand Up @@ -122,6 +122,9 @@ fn apply_coercion(value: &mut Value, node: &CoercionNode) {
}
}

/// Convert a given datetime looking string into microseconds using chrono's [DateTime] parsing
///
/// Since this will convert to microseconds, if the value is outside of the realm of conversion a None is returned
fn string_to_timestamp(string: &str) -> Option<Value> {
let parsed = DateTime::from_str(string);
if let Err(e) = parsed {
Expand All @@ -131,23 +134,25 @@ fn string_to_timestamp(string: &str) -> Option<Value> {
e
)
}
parsed.ok().map(|dt: DateTime<Utc>| {
Value::Number(
(dt.timestamp_nanos_opt()
.expect("Failed to turn timestamp nanoseconds")
/ 1000)
.into(),
)
})
parsed
.ok()
.map(|dt: DateTime<Utc>| Value::Number(dt.timestamp_micros().into()))
}

#[cfg(test)]
mod tests {

use super::*;
// use maplit::hashmap;
use serde_json::json;

#[test]
fn test_string_to_timestamp() {
let result = string_to_timestamp("2010-01-01T22:11:58Z");
assert!(result.is_some());
// exceeds nanos size
let result = string_to_timestamp("2400-01-01T22:11:58Z");
assert!(result.is_some());
}

lazy_static! {
static ref SCHEMA: Value = json!({
"type": "struct",
Expand Down

0 comments on commit d353ef5

Please sign in to comment.