Как получить один конкретный элемент в файле JSON, используя serde_json, не получая структуры? - PullRequest
2 голосов
/ 23 мая 2019

У меня сложный файл JSON, и я хотел бы извлечь из него только одно значение. Я мог бы определить все struct и получить Deserialize для всех из них, но я хотел бы просто написать небольшой ручной код, чтобы вытащить это одно значение. Документация Serde , откровенно говоря, просто смутила меня.

Мой контент JSON имеет следующую компоновку:

{
  "data": [
    {
      "hostname": "a hostname"
    }
  ]
}

Я ищу значение, к которому нужно перейти, войдя в data, затем взяв первый элемент массива и получив значение hostname.

В Хаскеле я бы сделал это так:

newtype Host = Host Text

instance FromJSON Host where
    parseJSON (Object o) = (return . Host) <=< (.: "hostname") <=< (fmap (!! 0) . parseJSON) <=< (.: "data") $ o
    parseJSON _ = mzero

Что эквивалентно Серде?

Ответы [ 3 ]

2 голосов
/ 24 мая 2019

Я бы использовал синтаксис Index ([...])

fn main() {
    let json_value = serde_json::json!({
      "data": [
        {
          "hostname": "a hostname"
        }
      ]
    });

    let host = &json_value["data"][0]["hostname"];
    println!("Host: {:?}", host);
}

Вы также можете использовать JSON Pointer через pointer:

let host = json_value.pointer("/data/0/hostname");
println!("Host: {:?}", host);
1 голос
/ 24 мая 2019

Я бы связал сплющенные структуры

use serde::{Serialize, Deserialize};
use serde_json::Value;

#[derive(Serialize, Deserialize)]
struct Payload {
    data: Vec<Data>,

    #[serde(flatten)]
    _: HashMap<String, Value>,
}

#[derive(Serialize, Deserialize)]
struct Data {
    hostname: String,

    #[serde(flatten)]
    _: HashMap<String, Value>,
}

let payload: Payload = serde_json::from_str(your_string)?;
assert_eq!(payload.data.0.hostname, "a hostname");
0 голосов
/ 23 мая 2019

serde_json предоставляет типы для общих значений JSON: serde_json::Value:

use serde_json::Value;

// input variable
let input: &str = "{...}";

// parse into generic JSON value
let root: Value = serde_json::from_str(input)?;

// access element using .get()
let hostname: Option<&str> = root.get("data")
    .and_then(|value| value.get(0))
    .and_then(|value| value.get("hostname"))
    .and_then(|value| value.as_str());

// hostname is Some(string_value) if .data[0].hostname is a string,
// and None if it was not found
println!("hostname = {:?}", hostname); // = Some("a hostname")

( пример полной игровой площадки )

...