2022-04-20 17:44:08 +02:00
|
|
|
|
#![doc = include_str!("../README.md")]
|
|
|
|
|
|
|
|
|
|
use std::collections::HashSet;
|
|
|
|
|
|
|
|
|
|
use serde_json::*;
|
|
|
|
|
|
|
|
|
|
type Document = Map<String, Value>;
|
|
|
|
|
|
|
|
|
|
const SPLIT_SYMBOL: char = '.';
|
|
|
|
|
|
|
|
|
|
/// Returns `true` if the `selector` match the `key`.
|
|
|
|
|
///
|
|
|
|
|
/// ```text
|
|
|
|
|
/// Example:
|
|
|
|
|
/// `animaux` match `animaux`
|
|
|
|
|
/// `animaux.chien` match `animaux`
|
|
|
|
|
/// `animaux.chien` match `animaux`
|
|
|
|
|
/// `animaux.chien.nom` match `animaux`
|
|
|
|
|
/// `animaux.chien.nom` match `animaux.chien`
|
|
|
|
|
/// -----------------------------------------
|
|
|
|
|
/// `animaux` doesn't match `animaux.chien`
|
|
|
|
|
/// `animaux.` doesn't match `animaux`
|
|
|
|
|
/// `animaux.ch` doesn't match `animaux.chien`
|
|
|
|
|
/// `animau` doesn't match `animaux`
|
|
|
|
|
/// ```
|
|
|
|
|
fn contained_in(selector: &str, key: &str) -> bool {
|
|
|
|
|
selector.starts_with(key)
|
2022-10-20 18:00:07 +02:00
|
|
|
|
&& selector[key.len()..].chars().next().map(|c| c == SPLIT_SYMBOL).unwrap_or(true)
|
2022-04-20 17:44:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Map the selected leaf values of a json allowing you to update only the fields that were selected.
|
|
|
|
|
/// ```
|
|
|
|
|
/// use serde_json::{Value, json};
|
|
|
|
|
/// use permissive_json_pointer::map_leaf_values;
|
|
|
|
|
///
|
|
|
|
|
/// let mut value: Value = json!({
|
|
|
|
|
/// "jean": {
|
|
|
|
|
/// "age": 8,
|
|
|
|
|
/// "race": {
|
|
|
|
|
/// "name": "bernese mountain",
|
|
|
|
|
/// "size": "80cm",
|
|
|
|
|
/// }
|
|
|
|
|
/// }
|
|
|
|
|
/// });
|
|
|
|
|
/// map_leaf_values(
|
|
|
|
|
/// value.as_object_mut().unwrap(),
|
|
|
|
|
/// ["jean.race.name"],
|
2024-11-20 01:00:43 +01:00
|
|
|
|
/// |key, _array_indices, value| match (value, key) {
|
2022-04-20 17:44:08 +02:00
|
|
|
|
/// (Value::String(name), "jean.race.name") => *name = "patou".to_string(),
|
|
|
|
|
/// _ => unreachable!(),
|
|
|
|
|
/// },
|
|
|
|
|
/// );
|
|
|
|
|
/// assert_eq!(
|
|
|
|
|
/// value,
|
|
|
|
|
/// json!({
|
|
|
|
|
/// "jean": {
|
|
|
|
|
/// "age": 8,
|
|
|
|
|
/// "race": {
|
|
|
|
|
/// "name": "patou",
|
|
|
|
|
/// "size": "80cm",
|
|
|
|
|
/// }
|
|
|
|
|
/// }
|
|
|
|
|
/// })
|
|
|
|
|
/// );
|
|
|
|
|
/// ```
|
|
|
|
|
pub fn map_leaf_values<'a>(
|
|
|
|
|
value: &mut Map<String, Value>,
|
|
|
|
|
selectors: impl IntoIterator<Item = &'a str>,
|
2024-11-20 01:00:43 +01:00
|
|
|
|
mut mapper: impl FnMut(&str, &[usize], &mut Value),
|
2022-04-20 17:44:08 +02:00
|
|
|
|
) {
|
|
|
|
|
let selectors: Vec<_> = selectors.into_iter().collect();
|
2024-11-20 01:00:43 +01:00
|
|
|
|
map_leaf_values_in_object(value, &selectors, "", &[], &mut mapper);
|
2022-04-20 17:44:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-01-17 18:01:26 +01:00
|
|
|
|
pub fn map_leaf_values_in_object(
|
2022-04-20 17:44:08 +02:00
|
|
|
|
value: &mut Map<String, Value>,
|
2023-01-17 18:01:26 +01:00
|
|
|
|
selectors: &[&str],
|
2022-04-20 17:44:08 +02:00
|
|
|
|
base_key: &str,
|
2024-11-20 01:00:43 +01:00
|
|
|
|
array_indices: &[usize],
|
|
|
|
|
mapper: &mut impl FnMut(&str, &[usize], &mut Value),
|
2022-04-20 17:44:08 +02:00
|
|
|
|
) {
|
|
|
|
|
for (key, value) in value.iter_mut() {
|
|
|
|
|
let base_key = if base_key.is_empty() {
|
|
|
|
|
key.to_string()
|
|
|
|
|
} else {
|
|
|
|
|
format!("{}{}{}", base_key, SPLIT_SYMBOL, key)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// here if the user only specified `doggo` we need to iterate in all the fields of `doggo`
|
|
|
|
|
// so we check the contained_in on both side
|
|
|
|
|
let should_continue = selectors
|
|
|
|
|
.iter()
|
|
|
|
|
.any(|selector| contained_in(selector, &base_key) || contained_in(&base_key, selector));
|
|
|
|
|
|
|
|
|
|
if should_continue {
|
|
|
|
|
match value {
|
|
|
|
|
Value::Object(object) => {
|
2024-11-20 01:00:43 +01:00
|
|
|
|
map_leaf_values_in_object(object, selectors, &base_key, array_indices, mapper)
|
2022-04-20 17:44:08 +02:00
|
|
|
|
}
|
|
|
|
|
Value::Array(array) => {
|
2024-11-20 01:00:43 +01:00
|
|
|
|
map_leaf_values_in_array(array, selectors, &base_key, array_indices, mapper)
|
2022-04-20 17:44:08 +02:00
|
|
|
|
}
|
2024-11-20 01:00:43 +01:00
|
|
|
|
value => mapper(&base_key, array_indices, value),
|
2022-04-20 17:44:08 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn map_leaf_values_in_array(
|
|
|
|
|
values: &mut [Value],
|
|
|
|
|
selectors: &[&str],
|
|
|
|
|
base_key: &str,
|
2024-11-20 01:00:43 +01:00
|
|
|
|
base_array_indices: &[usize],
|
|
|
|
|
mapper: &mut impl FnMut(&str, &[usize], &mut Value),
|
2022-04-20 17:44:08 +02:00
|
|
|
|
) {
|
2024-11-20 01:00:43 +01:00
|
|
|
|
// This avoids allocating twice
|
|
|
|
|
let mut array_indices = Vec::with_capacity(base_array_indices.len() + 1);
|
|
|
|
|
array_indices.extend_from_slice(base_array_indices);
|
|
|
|
|
array_indices.push(0);
|
|
|
|
|
|
|
|
|
|
for (i, value) in values.iter_mut().enumerate() {
|
|
|
|
|
*array_indices.last_mut().unwrap() = i;
|
2022-04-20 17:44:08 +02:00
|
|
|
|
match value {
|
2024-11-20 01:00:43 +01:00
|
|
|
|
Value::Object(object) => {
|
|
|
|
|
map_leaf_values_in_object(object, selectors, base_key, &array_indices, mapper)
|
|
|
|
|
}
|
|
|
|
|
Value::Array(array) => {
|
|
|
|
|
map_leaf_values_in_array(array, selectors, base_key, &array_indices, mapper)
|
|
|
|
|
}
|
|
|
|
|
value => mapper(base_key, &array_indices, value),
|
2022-04-20 17:44:08 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Permissively selects values in a json with a list of selectors.
|
|
|
|
|
/// Returns a new json containing all the selected fields.
|
|
|
|
|
/// ```
|
|
|
|
|
/// use serde_json::*;
|
|
|
|
|
/// use permissive_json_pointer::select_values;
|
|
|
|
|
///
|
|
|
|
|
/// let value: Value = json!({
|
|
|
|
|
/// "name": "peanut",
|
|
|
|
|
/// "age": 8,
|
|
|
|
|
/// "race": {
|
|
|
|
|
/// "name": "bernese mountain",
|
|
|
|
|
/// "avg_age": 12,
|
|
|
|
|
/// "size": "80cm",
|
|
|
|
|
/// },
|
|
|
|
|
/// });
|
|
|
|
|
/// let value: &Map<String, Value> = value.as_object().unwrap();
|
|
|
|
|
///
|
|
|
|
|
/// let res: Value = select_values(value, vec!["name", "race.name"]).into();
|
|
|
|
|
/// assert_eq!(
|
|
|
|
|
/// res,
|
|
|
|
|
/// json!({
|
|
|
|
|
/// "name": "peanut",
|
|
|
|
|
/// "race": {
|
|
|
|
|
/// "name": "bernese mountain",
|
|
|
|
|
/// },
|
|
|
|
|
/// })
|
|
|
|
|
/// );
|
|
|
|
|
/// ```
|
|
|
|
|
pub fn select_values<'a>(
|
|
|
|
|
value: &Map<String, Value>,
|
|
|
|
|
selectors: impl IntoIterator<Item = &'a str>,
|
|
|
|
|
) -> Map<String, Value> {
|
|
|
|
|
let selectors = selectors.into_iter().collect();
|
|
|
|
|
create_value(value, selectors)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn create_value(value: &Document, mut selectors: HashSet<&str>) -> Document {
|
|
|
|
|
let mut new_value: Document = Map::new();
|
|
|
|
|
|
|
|
|
|
for (key, value) in value.iter() {
|
|
|
|
|
// first we insert all the key at the root level
|
|
|
|
|
if selectors.contains(key as &str) {
|
|
|
|
|
new_value.insert(key.to_string(), value.clone());
|
|
|
|
|
// if the key was simple we can delete it and move to
|
|
|
|
|
// the next key
|
|
|
|
|
if is_simple(key) {
|
|
|
|
|
selectors.remove(key as &str);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// we extract all the sub selectors matching the current field
|
|
|
|
|
// if there was [person.name, person.age] and if we are on the field
|
|
|
|
|
// `person`. Then we generate the following sub selectors: [name, age].
|
|
|
|
|
let sub_selectors: HashSet<&str> = selectors
|
|
|
|
|
.iter()
|
|
|
|
|
.filter(|s| contained_in(s, key))
|
|
|
|
|
.filter_map(|s| s.trim_start_matches(key).get(SPLIT_SYMBOL.len_utf8()..))
|
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
|
|
if !sub_selectors.is_empty() {
|
|
|
|
|
match value {
|
|
|
|
|
Value::Array(array) => {
|
|
|
|
|
let array = create_array(array, &sub_selectors);
|
|
|
|
|
if !array.is_empty() {
|
|
|
|
|
new_value.insert(key.to_string(), array.into());
|
2023-08-14 00:22:14 +02:00
|
|
|
|
} else {
|
|
|
|
|
new_value.insert(key.to_string(), Value::Array(vec![]));
|
2022-04-20 17:44:08 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Value::Object(object) => {
|
|
|
|
|
let object = create_value(object, sub_selectors);
|
|
|
|
|
if !object.is_empty() {
|
|
|
|
|
new_value.insert(key.to_string(), object.into());
|
2023-08-14 00:22:14 +02:00
|
|
|
|
} else {
|
|
|
|
|
new_value.insert(key.to_string(), Value::Object(Map::new()));
|
2022-04-20 17:44:08 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
_ => (),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
new_value
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-06 12:45:52 +02:00
|
|
|
|
fn create_array(array: &[Value], selectors: &HashSet<&str>) -> Vec<Value> {
|
2022-04-20 17:44:08 +02:00
|
|
|
|
let mut res = Vec::new();
|
|
|
|
|
|
|
|
|
|
for value in array {
|
|
|
|
|
match value {
|
|
|
|
|
Value::Array(array) => {
|
|
|
|
|
let array = create_array(array, selectors);
|
|
|
|
|
if !array.is_empty() {
|
|
|
|
|
res.push(array.into());
|
2023-08-14 00:22:14 +02:00
|
|
|
|
} else {
|
|
|
|
|
res.push(Value::Array(vec![]));
|
2022-04-20 17:44:08 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Value::Object(object) => {
|
|
|
|
|
let object = create_value(object, selectors.clone());
|
|
|
|
|
if !object.is_empty() {
|
|
|
|
|
res.push(object.into());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
_ => (),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
res
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn is_simple(key: impl AsRef<str>) -> bool {
|
|
|
|
|
!key.as_ref().contains(SPLIT_SYMBOL)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
|
mod tests {
|
|
|
|
|
use big_s::S;
|
|
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_contained_in() {
|
|
|
|
|
assert!(contained_in("animaux", "animaux"));
|
|
|
|
|
assert!(contained_in("animaux.chien", "animaux"));
|
2022-10-20 18:00:07 +02:00
|
|
|
|
assert!(contained_in("animaux.chien.race.bouvier bernois.fourrure.couleur", "animaux"));
|
2022-04-20 17:44:08 +02:00
|
|
|
|
assert!(contained_in(
|
|
|
|
|
"animaux.chien.race.bouvier bernois.fourrure.couleur",
|
|
|
|
|
"animaux.chien"
|
|
|
|
|
));
|
|
|
|
|
assert!(contained_in(
|
|
|
|
|
"animaux.chien.race.bouvier bernois.fourrure.couleur",
|
|
|
|
|
"animaux.chien.race.bouvier bernois"
|
|
|
|
|
));
|
|
|
|
|
assert!(contained_in(
|
|
|
|
|
"animaux.chien.race.bouvier bernois.fourrure.couleur",
|
|
|
|
|
"animaux.chien.race.bouvier bernois.fourrure"
|
|
|
|
|
));
|
|
|
|
|
assert!(contained_in(
|
|
|
|
|
"animaux.chien.race.bouvier bernois.fourrure.couleur",
|
|
|
|
|
"animaux.chien.race.bouvier bernois.fourrure.couleur"
|
|
|
|
|
));
|
|
|
|
|
|
|
|
|
|
// -- the wrongs
|
|
|
|
|
assert!(!contained_in("chien", "chat"));
|
|
|
|
|
assert!(!contained_in("animaux", "animaux.chien"));
|
|
|
|
|
assert!(!contained_in("animaux.chien", "animaux.chat"));
|
|
|
|
|
|
|
|
|
|
// -- the strange edge cases
|
|
|
|
|
assert!(!contained_in("animaux.chien", "anima"));
|
|
|
|
|
assert!(!contained_in("animaux.chien", "animau"));
|
|
|
|
|
assert!(!contained_in("animaux.chien", "animaux."));
|
|
|
|
|
assert!(!contained_in("animaux.chien", "animaux.c"));
|
|
|
|
|
assert!(!contained_in("animaux.chien", "animaux.ch"));
|
|
|
|
|
assert!(!contained_in("animaux.chien", "animaux.chi"));
|
|
|
|
|
assert!(!contained_in("animaux.chien", "animaux.chie"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn simple_key() {
|
|
|
|
|
let value: Value = json!({
|
|
|
|
|
"name": "peanut",
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"avg_age": 12,
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
let value: &Document = value.as_object().unwrap();
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["name"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"name": "peanut",
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["age"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"age": 8,
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["name", "age"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"name": "peanut",
|
|
|
|
|
"age": 8,
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["race"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"avg_age": 12,
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["name", "age", "race"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"name": "peanut",
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"avg_age": 12,
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn complex_key() {
|
|
|
|
|
let value: Value = json!({
|
|
|
|
|
"name": "peanut",
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"avg_age": 12,
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
let value: &Document = value.as_object().unwrap();
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["race"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"avg_age": 12,
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
println!("RIGHT BEFORE");
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["race.name"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["race.name", "race.size"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(
|
|
|
|
|
value,
|
|
|
|
|
vec!["race.name", "race.size", "race.avg_age", "race.size", "age"],
|
|
|
|
|
)
|
|
|
|
|
.into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"avg_age": 12,
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["race.name", "race"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"avg_age": 12,
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["race", "race.name"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"avg_age": 12,
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn multi_level_nested() {
|
|
|
|
|
let value: Value = json!({
|
|
|
|
|
"jean": {
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
let value: &Document = value.as_object().unwrap();
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["jean"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"jean": {
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["jean.age"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"jean": {
|
|
|
|
|
"age": 8,
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["jean.race.size"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"jean": {
|
|
|
|
|
"race": {
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["jean.race.name", "jean.age"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"jean": {
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["jean.race"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"jean": {
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn array_and_deep_nested() {
|
|
|
|
|
let value: Value = json!({
|
|
|
|
|
"doggos": [
|
|
|
|
|
{
|
|
|
|
|
"jean": {
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"marc": {
|
|
|
|
|
"age": 4,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "golden retriever",
|
|
|
|
|
"size": "60cm",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
]
|
|
|
|
|
});
|
|
|
|
|
let value: &Document = value.as_object().unwrap();
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["doggos.jean"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"doggos": [
|
|
|
|
|
{
|
|
|
|
|
"jean": {
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
]
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["doggos.marc"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"doggos": [
|
|
|
|
|
{
|
|
|
|
|
"marc": {
|
|
|
|
|
"age": 4,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "golden retriever",
|
|
|
|
|
"size": "60cm",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
]
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["doggos.marc.race"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"doggos": [
|
|
|
|
|
{
|
|
|
|
|
"marc": {
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "golden retriever",
|
|
|
|
|
"size": "60cm",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
]
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value =
|
|
|
|
|
select_values(value, vec!["doggos.marc.race.name", "doggos.marc.age"]).into();
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"doggos": [
|
|
|
|
|
{
|
|
|
|
|
"marc": {
|
|
|
|
|
"age": 4,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "golden retriever",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
]
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(
|
|
|
|
|
value,
|
|
|
|
|
vec![
|
|
|
|
|
"doggos.marc.race.name",
|
|
|
|
|
"doggos.marc.age",
|
|
|
|
|
"doggos.jean.race.name",
|
|
|
|
|
"other.field",
|
|
|
|
|
],
|
|
|
|
|
)
|
|
|
|
|
.into();
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"doggos": [
|
|
|
|
|
{
|
|
|
|
|
"jean": {
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"marc": {
|
|
|
|
|
"age": 4,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "golden retriever",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
]
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-14 00:22:14 +02:00
|
|
|
|
#[test]
|
|
|
|
|
fn empty_array_object_return_empty() {
|
|
|
|
|
let value: Value = json!({
|
|
|
|
|
"array": [],
|
|
|
|
|
"object": {},
|
|
|
|
|
});
|
|
|
|
|
let value: &Document = value.as_object().unwrap();
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["array.name", "object.name"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"array": [],
|
|
|
|
|
"object": {},
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-20 17:44:08 +02:00
|
|
|
|
#[test]
|
|
|
|
|
fn all_conflict_variation() {
|
|
|
|
|
let value: Value = json!({
|
|
|
|
|
"pet.dog.name": "jean",
|
|
|
|
|
"pet.dog": {
|
|
|
|
|
"name": "bob"
|
|
|
|
|
},
|
|
|
|
|
"pet": {
|
|
|
|
|
"dog.name": "michel"
|
|
|
|
|
},
|
|
|
|
|
"pet": {
|
|
|
|
|
"dog": {
|
|
|
|
|
"name": "milan"
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
let value: &Document = value.as_object().unwrap();
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["pet.dog.name"]).into();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"pet.dog.name": "jean",
|
|
|
|
|
"pet.dog": {
|
|
|
|
|
"name": "bob"
|
|
|
|
|
},
|
|
|
|
|
"pet": {
|
|
|
|
|
"dog.name": "michel"
|
|
|
|
|
},
|
|
|
|
|
"pet": {
|
|
|
|
|
"dog": {
|
|
|
|
|
"name": "milan"
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let value: Value = json!({
|
|
|
|
|
"pet.dog.name": "jean",
|
|
|
|
|
"pet.dog": {
|
|
|
|
|
"name": "bob",
|
|
|
|
|
},
|
|
|
|
|
"pet": {
|
|
|
|
|
"dog.name": "michel",
|
|
|
|
|
"dog": {
|
|
|
|
|
"name": "milan",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
let value: &Document = value.as_object().unwrap();
|
|
|
|
|
|
|
|
|
|
let res: Value = select_values(value, vec!["pet.dog.name", "pet.dog", "pet"]).into();
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
res,
|
|
|
|
|
json!({
|
|
|
|
|
"pet.dog.name": "jean",
|
|
|
|
|
"pet.dog": {
|
|
|
|
|
"name": "bob",
|
|
|
|
|
},
|
|
|
|
|
"pet": {
|
|
|
|
|
"dog.name": "michel",
|
|
|
|
|
"dog": {
|
|
|
|
|
"name": "milan",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn map_object() {
|
|
|
|
|
let mut value: Value = json!({
|
|
|
|
|
"jean": {
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2024-11-20 01:00:43 +01:00
|
|
|
|
map_leaf_values(
|
|
|
|
|
value.as_object_mut().unwrap(),
|
|
|
|
|
["jean.race.name"],
|
|
|
|
|
|key, _, value| match (value, key) {
|
2022-04-20 17:44:08 +02:00
|
|
|
|
(Value::String(name), "jean.race.name") => *name = S("patou"),
|
|
|
|
|
_ => unreachable!(),
|
2024-11-20 01:00:43 +01:00
|
|
|
|
},
|
|
|
|
|
);
|
2022-04-20 17:44:08 +02:00
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
value,
|
|
|
|
|
json!({
|
|
|
|
|
"jean": {
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "patou",
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let mut value: Value = json!({
|
|
|
|
|
"jean": {
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "bernese mountain",
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
"bob": "lolpied",
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
let mut calls = 0;
|
2024-11-20 01:00:43 +01:00
|
|
|
|
map_leaf_values(value.as_object_mut().unwrap(), ["jean"], |key, _, value| {
|
2022-04-20 17:44:08 +02:00
|
|
|
|
calls += 1;
|
|
|
|
|
match (value, key) {
|
|
|
|
|
(Value::String(name), "jean.race.name") => *name = S("patou"),
|
|
|
|
|
_ => println!("Called with {key}"),
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
assert_eq!(calls, 3);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
value,
|
|
|
|
|
json!({
|
|
|
|
|
"jean": {
|
|
|
|
|
"age": 8,
|
|
|
|
|
"race": {
|
|
|
|
|
"name": "patou",
|
|
|
|
|
"size": "80cm",
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
"bob": "lolpied",
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
}
|
2024-11-20 01:00:43 +01:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn map_array() {
|
|
|
|
|
let mut value: Value = json!({
|
|
|
|
|
"no_array": "peter",
|
|
|
|
|
"simple": ["foo", "bar"],
|
|
|
|
|
"nested": [
|
|
|
|
|
{
|
|
|
|
|
"a": [
|
|
|
|
|
["cat", "dog"],
|
|
|
|
|
["fox", "bear"],
|
|
|
|
|
],
|
|
|
|
|
"b": "hi",
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"a": ["green", "blue"],
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
map_leaf_values(
|
|
|
|
|
value.as_object_mut().unwrap(),
|
|
|
|
|
["no_array", "simple", "nested"],
|
|
|
|
|
|_key, array_indices, value| {
|
|
|
|
|
*value = format!("{array_indices:?}").into();
|
|
|
|
|
},
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
value,
|
|
|
|
|
json!({
|
|
|
|
|
"no_array": "[]",
|
|
|
|
|
"simple": ["[0]", "[1]"],
|
|
|
|
|
"nested": [
|
|
|
|
|
{
|
|
|
|
|
"a": [
|
|
|
|
|
["[0, 0, 0]", "[0, 0, 1]"],
|
|
|
|
|
["[0, 1, 0]", "[0, 1, 1]"],
|
|
|
|
|
],
|
|
|
|
|
"b": "[0]",
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"a": ["[1, 0]", "[1, 1]"],
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
}
|
2022-04-20 17:44:08 +02:00
|
|
|
|
}
|