Skip to content

Commit

Permalink
doc mapper: convert number handling to deserialization
Browse files Browse the repository at this point in the history
change number deserialization in docmapper from json to generic deserialization.
This improves codes reuse between different code paths, e.g.
serialization and validation.
  • Loading branch information
PSeitz committed Oct 28, 2024
1 parent a91d2e7 commit 9d883c5
Show file tree
Hide file tree
Showing 6 changed files with 232 additions and 122 deletions.
4 changes: 2 additions & 2 deletions quickwit/Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion quickwit/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -212,7 +212,7 @@ sea-query-binder = { version = "0.5", features = [
# ^1.0.184 due to serde-rs/serde#2538
serde = { version = "1.0.184", features = ["derive", "rc"] }
serde_json = "1.0"
serde_json_borrow = "0.5"
serde_json_borrow = "0.7"
serde_qs = { version = "0.12", features = ["warp"] }
serde_with = "3.9.0"
serde_yaml = "0.9"
Expand Down
206 changes: 206 additions & 0 deletions quickwit/quickwit-doc-mapper/src/doc_mapper/deser_num.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,206 @@
use std::fmt;

use serde::de::{self, Deserializer, IntoDeserializer, Visitor};
use serde::Deserialize;
use serde_json::Value;

/// Deserialize a number from a string or number, with optional coercion.
fn deserialize_num_with_coerce<'de, T, D>(deserializer: D, coerce: bool) -> Result<T, String>
where
T: std::str::FromStr + Deserialize<'de>,
T::Err: fmt::Display,
D: Deserializer<'de>,
{
struct CoerceVisitor<T> {
coerce: bool,
marker: std::marker::PhantomData<T>,
}

impl<'de, T> Visitor<'de> for CoerceVisitor<T>
where
T: std::str::FromStr + Deserialize<'de>,
T::Err: fmt::Display,
{
type Value = T;

fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("a number (i64, u64, or f64) or a string that can be coerced")
}

fn visit_str<E>(self, v: &str) -> Result<T, E>
where
E: de::Error,
{
if self.coerce {
v.parse::<T>().map_err(|_e| {
de::Error::custom(format!(
"failed to coerce JSON string `\"{}\"` to {}",
v,
std::any::type_name::<T>(),
))
})
} else {
Err(de::Error::custom(format!(
"expected JSON number, got string `\"{}\"`. enable coercion to {} with the \
`coerce` parameter in the field mapping",
v,
std::any::type_name::<T>()
)))
}
}

fn visit_i64<E>(self, v: i64) -> Result<T, E>
where
E: de::Error,
{
T::deserialize(v.into_deserializer()).map_err(|_: E| {
de::Error::custom(format!(
"expected {}, got inconvertible JSON number `{}`",
std::any::type_name::<T>(),
v
))
})
}

fn visit_u64<E>(self, v: u64) -> Result<T, E>
where
E: de::Error,
{
T::deserialize(v.into_deserializer()).map_err(|_: E| {
de::Error::custom(format!(
"expected {}, got inconvertible JSON number `{}`",
std::any::type_name::<T>(),
v
))
})
}

fn visit_f64<E>(self, v: f64) -> Result<T, E>
where
E: de::Error,
{
T::deserialize(v.into_deserializer()).map_err(|_: E| {
de::Error::custom(format!(
"expected {}, got inconvertible JSON number `{}`",
std::any::type_name::<T>(),
v
))
})
}

fn visit_map<M>(self, mut map: M) -> Result<T, M::Error>
where
M: de::MapAccess<'de>,
{
let json_value: Value =
Deserialize::deserialize(de::value::MapAccessDeserializer::new(&mut map))?;
Err(de::Error::custom(format!(
"expected JSON number or string, got `{}`",
json_value
)))
}

fn visit_seq<S>(self, mut seq: S) -> Result<T, S::Error>
where
S: de::SeqAccess<'de>,
{
let json_value: Value =
Deserialize::deserialize(de::value::SeqAccessDeserializer::new(&mut seq))?;
Err(de::Error::custom(format!(
"expected JSON number or string, got `{}`",
json_value
)))
}
}

deserializer
.deserialize_any(CoerceVisitor {
coerce,
marker: std::marker::PhantomData,
})
.map_err(|err| err.to_string())
}

pub fn deserialize_i64<'de, D>(deserializer: D, coerce: bool) -> Result<i64, String>
where
D: Deserializer<'de>,
{
deserialize_num_with_coerce(deserializer, coerce)
}

pub fn deserialize_u64<'de, D>(deserializer: D, coerce: bool) -> Result<u64, String>
where
D: Deserializer<'de>,
{
deserialize_num_with_coerce(deserializer, coerce)
}

pub fn deserialize_f64<'de, D>(deserializer: D, coerce: bool) -> Result<f64, String>
where
D: Deserializer<'de>,
{
deserialize_num_with_coerce(deserializer, coerce)
}

#[cfg(test)]
mod tests {
use serde_json::json;

use super::*;

#[test]
fn test_deserialize_i64_with_coercion() {
let json_data = json!("-123");
let result: i64 = deserialize_i64(json_data.into_deserializer(), true).unwrap();
assert_eq!(result, -123);

let json_data = json!("456");
let result: i64 = deserialize_i64(json_data.into_deserializer(), true).unwrap();
assert_eq!(result, 456);
}

#[test]
fn test_deserialize_u64_with_coercion() {
let json_data = json!("789");
let result: u64 = deserialize_u64(json_data.into_deserializer(), true).unwrap();
assert_eq!(result, 789);

let json_data = json!(123);
let result: u64 = deserialize_u64(json_data.into_deserializer(), false).unwrap();
assert_eq!(result, 123);
}

#[test]
fn test_deserialize_f64_with_coercion() {
let json_data = json!("78.9");
let result: f64 = deserialize_f64(json_data.into_deserializer(), true).unwrap();
assert_eq!(result, 78.9);

let json_data = json!(45.6);
let result: f64 = deserialize_f64(json_data.into_deserializer(), false).unwrap();
assert_eq!(result, 45.6);
}

#[test]
fn test_deserialize_invalid_string_coercion() {
let json_data = json!("abc");
let result: Result<i64, _> = deserialize_i64(json_data.into_deserializer(), true);
assert!(result.is_err());

let err_msg = result.unwrap_err().to_string();
assert_eq!(err_msg, "failed to coerce JSON string `\"abc\"` to i64");
}

#[test]
fn test_deserialize_json_object() {
let json_data = json!({ "key": "value" });
let result: Result<i64, _> = deserialize_i64(json_data.into_deserializer(), true);
assert!(result.is_err());

let err_msg = result.unwrap_err().to_string();
assert_eq!(
err_msg,
"expected JSON number or string, got `{\"key\":\"value\"}`"
);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -1846,7 +1846,7 @@ mod tests {
}"#,
"concat",
r#"{"some_int": 25}"#,
vec![25_u64.into()],
vec![25_i64.into()],
);
}

Expand Down
Loading

0 comments on commit 9d883c5

Please sign in to comment.