Serde nested JSON
Summary and basic use
This is a small utility crate to help deal with nested JSON in structs.
Here's a basic example:
use serde::{Serialize, Deserialize};
use serde_json::{json, from_value, to_string_pretty};
use serde_nested_json;
let as_json = json!({
"someData": {
"foo": "bar",
"baz": 123
},
"nestedJson": "{\"foo\":\"bar\",\"baz\":123}"
});
#[derive(Serialize, Deserialize)]
struct SomeData {
foo: String,
baz: u32
}
#[derive(Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
struct MyStruct {
some_data: SomeData,
#[serde(with = "serde_nested_json")]
nested_json: SomeData
}
# println!("{}", to_string_pretty(&from_value::<MyStruct>(as_json).unwrap()).unwrap());
Optional values
Optional values deserialize just fine from "null"
, but if the value may be undefined please add default
field attribute:
use serde::{Serialize, Deserialize};
use serde_json::{json, from_value, to_string_pretty};
use serde_nested_json;
let as_json = json!({
"here": "\"hello!\"",
"null": "null",
});
#[derive(Serialize, Deserialize)]
struct MyStruct {
#[serde(with = "serde_nested_json")]
here: Option<String>,
#[serde(with = "serde_nested_json")]
null: Option<String>,
#[serde(with = "serde_nested_json", default)] undefined: Option<String>,
}
# println!("{}", to_string_pretty(&from_value::<MyStruct>(as_json).unwrap()).unwrap());
Vecs
There's also a helper module for vecs containing nested items.
To use it, Just add ::vec
to the with
field attribute:
use serde::{Serialize, Deserialize};
use serde_json::{json, from_value, to_string_pretty};
use serde_nested_json;
let as_json = json!({
"array": [
"{\"foo\":\"bar\",\"baz\":123}",
"{\"foo\":\"baz\",\"baz\":54321}"
]
});
#[derive(Serialize, Deserialize)]
struct SomeData {
foo: String,
baz: u32
}
#[derive(Serialize, Deserialize)]
struct MyStruct {
#[serde(with = "serde_nested_json::vec")]
array: Vec<SomeData>,
}
# println!("{}", to_string_pretty(&from_value::<MyStruct>(as_json).unwrap()).unwrap());
NestedJson<T>
The main helper type of this crate is NestedJson<T>
which
can be used without field annotation. You will however need
to extract and insert the inner value on your own:
use serde::{Serialize, Deserialize};
use serde_json::{json, from_value, to_string_pretty};
use serde_nested_json::NestedJson;
let as_json = json!({
"nestedJson": "{\"baz\":123}"
});
#[derive(Serialize, Deserialize)]
struct SomeData {
baz: u32
}
#[derive(Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
struct MyStruct {
nested_json: NestedJson<SomeData>
}
impl From<SomeData> for MyStruct {
fn from(value: SomeData) -> MyStruct {
MyStruct {
nested_json: value.into()
}
}
}
impl From<MyStruct> for SomeData {
fn from(value: MyStruct) -> SomeData {
value.nested_json.into()
}
}
impl AsRef<SomeData> for MyStruct {
fn as_ref(&self) -> &SomeData {
self.nested_json.as_ref()
}
}
let a = SomeData { baz: 32 };
let b: MyStruct = a.into();
let c: SomeData = b.into();
let d: NestedJson<SomeData> = c.into();
let e: SomeData = d.into();
let a = SomeData { baz: 32 };
let b = MyStruct::from(a);
let c = SomeData::from(b);
let d = NestedJson::<SomeData>::from(c);
# println!("{}", to_string_pretty(&from_value::<MyStruct>(as_json).unwrap()).unwrap());