Можно ли сгладить поля подобъекта при разборе с serde_json?

#[serde(rename)] кажется правильным вариантом, но в документации не указано, возможно ли это или как это сделать.

Этот объект JSON:

{
   "name" : "myobject"
   "info" : 
   {
      "counter" : "3"
      "foo" : "bar"
   }
}

Соответствующая плоская структура Rust должна быть:

#[derive(Deserialize)]
struct Object {
    name: String,
    #[serde(rename="info.counter")] // wrong syntax here !!
    count: i32,
    #[serde(rename="info::foo")] // neither this works
    foo: String,
}

2 ответа

Решение

Нет встроенного способа сделать это с атрибутами, но вы можете написать свой собственный Deserialize для вашего Object type, который десериализует сначала до некоторого промежуточного вспомогательного представления, а затем переставляет данные в желаемую структуру.

#[macro_use]
extern crate serde_derive;

extern crate serde;
extern crate serde_json;

use serde::{Deserialize, Deserializer};

#[derive(Debug)]
struct Object {
    name: String,
    count: i32,
    foo: String,
}

impl<'de> Deserialize<'de> for Object {
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
        where D: Deserializer<'de>
    {
        #[derive(Deserialize)]
        struct Outer {
            name: String,
            info: Inner,
        }

        #[derive(Deserialize)]
        struct Inner {
            count: i32,
            foo: String,
        }

        let helper = Outer::deserialize(deserializer)?;
        Ok(Object {
            name: helper.name,
            count: helper.info.count,
            foo: helper.info.foo,
        })
    }
}

fn main() {
    let j = r#"{
                 "name": "myobject",
                 "info": {
                   "count": 3,
                   "foo": "bar"
                 }
               }"#;

    println!("{:#?}", serde_json::from_str::<Object>(j).unwrap());
}

Выход:

Object {
    name: "myobject",
    count: 3,
    foo: "bar"
}

Теперь вы можете #[serde(flatten)] следующее:

      #[derive(Serialize, Deserialize)]
struct Pagination {
    limit: u64,
    offset: u64,
    total: u64,
}

#[derive(Serialize, Deserialize)]
struct Users {
    users: Vec<User>,

    #[serde(flatten)]
    pagination: Pagination,
}

От: https://serde.rs/attr-flatten.html

Другие вопросы по тегам