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
|
use serde_json;
use source::Source;
use std::collections::HashMap;
use std::error::Error;
use value::{Value, ValueKind};
pub fn parse(uri: Option<&String>,
text: &str,
namespace: Option<&String>)
-> Result<HashMap<String, Value>, Box<Error>> {
// Parse a JSON object value from the text
let mut root: serde_json::Value = serde_json::from_str(text)?;
// Limit to namespace
if let Some(namespace) = namespace {
root = serde_json::Value::Object(match root {
serde_json::Value::Object(ref mut table) => {
if let Some(serde_json::Value::Object(table)) =
table.remove(namespace) {
table
} else {
serde_json::Map::new()
}
}
_ => serde_json::Map::new(),
});
};
// TODO: Have a proper error fire if the root of a file is ever not a Table
let value = from_json_value(uri, &root);
match value.kind {
ValueKind::Table(map) => Ok(map),
_ => Ok(HashMap::new()),
}
}
fn from_json_value(uri: Option<&String>, value: &serde_json::Value) -> Value {
match *value {
serde_json::Value::String(ref value) => Value::new(uri, ValueKind::String(value.clone())),
serde_json::Value::Number(ref value) => {
if let Some(value) = value.as_i64() {
Value::new(uri, ValueKind::Integer(value))
} else if let Some(value) = value.as_f64() {
Value::new(uri, ValueKind::Float(value))
} else {
unreachable!();
}
}
serde_json::Value::Bool(value) => Value::new(uri, ValueKind::Boolean(value)),
serde_json::Value::Object(ref table) => {
let mut m = HashMap::new();
for (key, value) in table {
m.insert(key.to_lowercase().clone(), from_json_value(uri, value));
}
Value::new(uri, ValueKind::Table(m))
}
serde_json::Value::Array(ref array) => {
let mut l = Vec::new();
for value in array {
l.push(from_json_value(uri, value));
}
Value::new(uri, ValueKind::Array(l))
}
serde_json::Value::Null => Value::new(uri, ValueKind::Nil),
}
}
|