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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
#![feature(proc_macro)]
#![feature(nll)]
#![feature(rustc_private)]
#![crate_type = "proc-macro"]
extern crate proc_macro;
extern crate proc_macro2;
extern crate serde_json;
#[macro_use] extern crate lazy_static;
use serde_json::Value;
use proc_macro2::TokenTree;
use std::collections::HashMap;
use std::sync::Mutex;
lazy_static! {
static ref JSON_CACHE: Mutex<HashMap<String, Value>> = {
Mutex::new(HashMap::new())
};
}
fn extract_string_lit(text: &str) -> &str {
let start = text.find("\"").expect("How does a literal not have a \"?");
let end = text.rfind("\"").expect("How does a literal not have a \"?");
unsafe { text.get_unchecked(start+1..(end)) }
}
fn extract_json(input: proc_macro::TokenStream) -> Value {
let input: proc_macro2::TokenStream = input.into();
let mut trees = input.into_iter();
let token1 = trees.next().expect("json_token! expected 2 arguments");
let _token2 = trees.next().expect("json_token! expected 2 arguments");
let token3 = trees.next().expect("json_token! expected 2 arguments");
let lit1 = match token1 {
TokenTree::Literal(lit) => lit.to_string(),
_ => panic!("macro only accepts a string literal as first argument"),
};
let lit2 = match token3 {
TokenTree::Literal(lit) => lit.to_string(),
_ => panic!("macro only accepts a string literal as second argument"),
};
let file_path = extract_string_lit(&lit1);
let pointer = extract_string_lit(&lit2);
let mut cache = JSON_CACHE.lock().unwrap();
let json_val = match cache.get(file_path) {
Some(val) => val,
None => {
let json = ::std::fs::read(extract_string_lit(&lit1)).expect("failed to read JSON file specified by json_token! macro");
let val: Value = serde_json::from_slice(&json).expect("json_token! file is not valid JSON");
let _ = cache.insert(file_path.to_owned(), val);
cache.get(file_path).unwrap()
}
};
json_val.pointer(&pointer).expect("json_token! macro did not find a value for this JSON pointer").clone()
}
#[proc_macro]
pub fn json_token(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
let json_token = extract_json(input);
let json_token_str = json_token.as_str().expect("json_token! macro expected to find a string at this JSON pointer");
let output: proc_macro2::TokenStream = json_token_str.parse().expect("JSON schema type isn't a valid Rust token");
output.into()
}
#[proc_macro]
pub fn json_literal(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
let json_token = extract_json(input);
let json_token_str = serde_json::to_string(&json_token).unwrap();
let output: proc_macro2::TokenStream = json_token_str.parse().expect("JSON schema type isn't a valid Rust token");
output.into()
}