1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
use Opaque;
use core;
use core::errors::Result;
use format;
use linked_hash_map::LinkedHashMap;
use serde_json as json;
use sir::{FieldSir, Sir};
use utils::is_datetime;
#[derive(Debug)]
pub struct Json;
impl format::Format for Json {
fn decode(&self, object: &core::Source) -> Result<Sir> {
let mut der = json::Deserializer::from_reader(object.read()?).into_iter::<json::Value>();
let value: Result<json::Value> = der.next()
.ok_or_else(|| format!("Expected at least one JSON value").into())
.and_then(|v| v.map_err(|e| format!("Bad JSON: {}", e).into()));
let value = value?;
let sir = from_json(&value)?;
Ok(sir)
}
}
impl format::Object for json::Map<String, json::Value> {
type Value = json::Value;
fn get(&self, key: &str) -> Option<&Self::Value> {
self.get(key)
}
}
impl format::Value for json::Value {
fn as_object(&self) -> Option<&format::Object<Value = Self>> {
match *self {
json::Value::Object(ref object) => Some(object as &format::Object<Value = Self>),
_ => None,
}
}
fn as_str(&self) -> Option<&str> {
match *self {
json::Value::String(ref string) => Some(string),
_ => None,
}
}
}
fn from_json(value: &json::Value) -> Result<Sir> {
let f = match *value {
json::Value::Number(ref number) if number.is_u64() => {
let number = number
.as_u64()
.ok_or_else(|| format!("Expected u64, got: {}", number))?;
Sir::U64(Opaque::new(vec![number]))
}
json::Value::Number(ref number) if number.is_i64() => {
let number = number
.as_i64()
.ok_or_else(|| format!("Expected i64, got: {}", number))?;
Sir::I64(Opaque::new(vec![number]))
}
json::Value::Number(ref number) => {
let number = number
.as_f64()
.ok_or_else(|| format!("Expected f64, got: {}", number))?;
let diff = (((number as f32) as f64) - number).abs();
if diff != 0f64 {
Sir::Double
} else {
Sir::Float
}
}
json::Value::Bool(_) => Sir::Boolean,
json::Value::String(ref string) => {
if is_datetime(string) {
Sir::DateTime(Opaque::new(vec![string.to_string()]))
} else {
Sir::String(Opaque::new(vec![string.to_string()]))
}
}
json::Value::Null => Sir::Any,
json::Value::Array(ref array) => Sir::process_array(&array, from_json)?,
json::Value::Object(ref map) => {
let mut entries = LinkedHashMap::new();
for (key, value) in map {
let value = from_json(value)?;
let field = FieldSir {
optional: value == Sir::Any,
field: value,
};
entries.insert(key.to_string(), field);
}
Sir::Object(entries)
}
};
return Ok(f);
}