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,
        });
    }
}