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
#[macro_use]
mod macros;
mod statement;
mod expression;
mod function;
mod value;
use serde::ser::{Serialize, Serializer, SerializeStruct};
use crate::ast::{Loc, Node};
use crate::module::Module;
pub trait SerializeInLoc {
#[inline]
fn in_loc<S, F>(&self, serializer: S, name: &'static str, length: usize, build: F) -> Result<S::SerializeStruct, S::Error>
where
S: Serializer,
F: FnOnce(&mut S::SerializeStruct) -> Result<(), S::Error>
{
let mut state = serializer.serialize_struct(name, length + 3)?;
state.serialize_field("type", name)?;
build(&mut state).map(move |_| state)
}
fn serialize<S>(&self, serializer: S) -> Result<S::SerializeStruct, S::Error>
where S: Serializer;
}
impl<'ast, T: SerializeInLoc> Serialize for Loc<T> {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer
{
let mut state = self.item.serialize(serializer)?;
state.serialize_field("start", &self.start)?;
state.serialize_field("end", &self.end)?;
state.end()
}
}
impl<'ast, T: SerializeInLoc> Serialize for Node<'ast, T> {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer
{
Loc::<T>::serialize(&*self, serializer)
}
}
impl<'ast> Serialize for Module<'ast> {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer
{
let body = self.body();
let mut start = 0;
let mut end = 0;
let mut iter = body.iter();
if let Some(node) = iter.next() {
start = node.start;
end = node.end;
}
if let Some(node) = iter.last() {
end = node.end;
}
let name = "Program";
let mut state = serializer.serialize_struct(name, 4)?;
state.serialize_field("type", &name)?;
state.serialize_field("body", &body)?;
state.serialize_field("start", &start)?;
state.serialize_field("end", &end)?;
state.end()
}
}
#[cfg(test)]
mod test {
#[test]
fn test_generate_ast_empty() {
expect_parse!("", {
"type": "Program",
"body": [],
"start": 0,
"end": 0,
});
}
#[test]
fn test_generate_ast_expression() {
expect_parse!("this;", {
"type": "Program",
"body": [
{
"type": "ExpressionStatement",
"expression": {
"type": "ThisExpression",
"start": 0,
"end": 4,
},
"start": 0,
"end": 4,
}
],
"start": 0,
"end": 4,
});
}
}