Как десериализовать подполе структуры из JSON исходной структуры с помощью Serde?

Я хочу, чтобы атрибут Test::team_size десериализовался из данных самого объекта Test:

#[derive(Debug, Serialize, Deserialize)]
struct TeamSize {
    pub min: i64,
    pub max: i64,
}

#[derive(Debug, Serialize, Deserialize)]
struct Test {
    pub i: i64,
    pub team_size: TeamSize,
}

fn main() {
    let t: Test = serde_json::from_str(r#"{"i": -2, "min": 2, "max": 5}"#).unwrap();
    assert_eq!(t.i, -2);
    assert_eq!(t.team_size.min, 2);
    assert_eq!(t.team_size.max, 5);
}

Этот код не компилируется, и я не знаю, как заставить Серде делать то, что я хочу. Есть ли способ десериализации team_size в этом примере из JSON исходной структуры, где это подполе?

Похоже, мне нужно что-то вроде #[serde(untagged)], но для структуры и для поля, а не всей структуры.


person Victor Polevoy    schedule 01.06.2017    source источник
comment
Соответствующая проблема: serde # 119   -  person E_net4 the curator    schedule 01.06.2017


Ответы (2)


Начиная с Serde 1.0.34, теперь вы можете использовать #[serde(flatten)]:

#[derive(Debug, Serialize, Deserialize)]
struct Test {
    pub i: i64,
    #[serde(flatten)]
    pub team_size: TeamSize,
}

Перед этим самым простым обходным решением было бы десериализовать частный вспомогательный тип и при необходимости реструктурировать его:

#[macro_use]
extern crate serde_derive;

extern crate serde;
extern crate serde_json;

use serde::{Serialize, Serializer, Deserialize, Deserializer};

#[derive(Debug)]
pub struct TeamSize {
    pub min: i64,
    pub max: i64,
}

#[derive(Debug)]
pub struct Test {
    pub i: i64,
    pub team_size: TeamSize,
}

// Private helper
#[derive(Serialize, Deserialize)]
struct Helper {
    pub i: i64,
    pub min: i64,
    pub max: i64,
}

impl Serialize for Test {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
        where S: Serializer
    {
        let helper = Helper {
            i: self.i,
            min: self.team_size.min,
            max: self.team_size.max,
        };
        helper.serialize(serializer)
    }
}

impl<'de> Deserialize<'de> for Test {
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
        where D: Deserializer<'de>
    {
        let helper = Helper::deserialize(deserializer)?;
        Ok(Test {
            i: helper.i,
            team_size: TeamSize {
                min: helper.min,
                max: helper.max,
            },
        })
    }
}

fn main() {
    let j = r#" { "i": -2, "min": 2, "max": 5 } "#;

    let de: Test = serde_json::from_str(j).unwrap();
    println!("{:#?}", de);

    let ser = serde_json::to_string_pretty(&de).unwrap();
    println!("{}", ser);
}
person dtolnay    schedule 01.06.2017

В документации Serde есть глава о том, как реализовать настраиваемую десериализацию, когда автоматически сгенерированные атрибуты не в полной мере выполняют то, что вы хотите. Начиная с этого момента, это не слишком сложно, просто утомительно:

extern crate serde;
extern crate serde_json;

use std::fmt;

use serde::de::{self, Deserialize, Deserializer, Visitor, MapAccess};

#[derive(Debug)]
struct TeamSize {
    pub min: i64,
    pub max: i64,
}

#[derive(Debug)]
struct Test {
    pub i: i64,
    pub team_size: TeamSize,
}

impl<'de> Deserialize<'de> for Test {
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
        where D: Deserializer<'de>
    {
        struct TestVisitor;

        impl<'de> Visitor<'de> for TestVisitor {
            type Value = Test;

            fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
                formatter.write_str("struct Test")
            }

            fn visit_map<V>(self, mut map: V) -> Result<Test, V::Error>
                where V: MapAccess<'de>
            {
                let mut min = None;
                let mut max = None;
                let mut i = None;

                while let Some(key) = map.next_key()? {
                    match key {
                        "min" => {
                            if min.is_some() {
                                return Err(de::Error::duplicate_field("min"));
                            }
                            min = Some(map.next_value()?);
                        }
                        "max" => {
                            if max.is_some() {
                                return Err(de::Error::duplicate_field("max"));
                            }
                            max = Some(map.next_value()?);
                        }
                        "i" => {
                            if i.is_some() {
                                return Err(de::Error::duplicate_field("i"));
                            }
                            i = Some(map.next_value()?);
                        }
                        _ => {
                            /* Ignore extra fields */
                        }
                    }
                }

                let min = min.ok_or_else(|| de::Error::missing_field("min"))?;
                let max = max.ok_or_else(|| de::Error::missing_field("max"))?;
                let i = i.ok_or_else(|| de::Error::missing_field("i"))?;

                Ok(Test { i, team_size: TeamSize { min, max }})
            }
        }

        const FIELDS: &'static [&'static str] = &["min", "max", "i"];
        deserializer.deserialize_struct("Test", FIELDS, TestVisitor)
    }
}

fn main() {
    let t: Test = serde_json::from_str(r#"{"i": -2, "min": 2, "max": 5}"#).unwrap();
    assert_eq!(t.i, -2);
    assert_eq!(t.team_size.min, 2);
    assert_eq!(t.team_size.max, 5);
}
person Shepmaster    schedule 01.06.2017