-
Notifications
You must be signed in to change notification settings - Fork 46
/
grammar.js
125 lines (102 loc) · 2.32 KB
/
grammar.js
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
/**
* @file JSON grammar for tree-sitter
* @author Max Brunsfeld <[email protected]>
* @author Amaan Qureshi <[email protected]>
* @license MIT
*/
/// <reference types="tree-sitter-cli/dsl" />
// @ts-check
module.exports = grammar({
name: 'json',
extras: $ => [
/\s/,
$.comment,
],
supertypes: $ => [
$._value,
],
rules: {
document: $ => repeat($._value),
_value: $ => choice(
$.object,
$.array,
$.number,
$.string,
$.true,
$.false,
$.null,
),
object: $ => seq(
'{', commaSep($.pair), '}',
),
pair: $ => seq(
field('key', $.string),
':',
field('value', $._value),
),
array: $ => seq(
'[', commaSep($._value), ']',
),
string: $ => choice(
seq('"', '"'),
seq('"', $._string_content, '"'),
),
_string_content: $ => repeat1(choice(
$.string_content,
$.escape_sequence,
)),
string_content: _ => token.immediate(prec(1, /[^\\"\n]+/)),
escape_sequence: _ => token.immediate(seq(
'\\',
/(\"|\\|\/|b|f|n|r|t|u)/,
)),
number: _ => {
const decimalDigits = /\d+/;
const signedInteger = seq(optional('-'), decimalDigits);
const exponentPart = seq(choice('e', 'E'), signedInteger);
const decimalIntegerLiteral = seq(
optional('-'),
choice(
'0',
seq(/[1-9]/, optional(decimalDigits)),
),
);
const decimalLiteral = choice(
seq(decimalIntegerLiteral, '.', optional(decimalDigits), optional(exponentPart)),
seq(decimalIntegerLiteral, optional(exponentPart)),
);
return token(decimalLiteral);
},
true: _ => 'true',
false: _ => 'false',
null: _ => 'null',
comment: _ => token(choice(
seq('//', /.*/),
seq(
'/*',
/[^*]*\*+([^/*][^*]*\*+)*/,
'/',
),
)),
},
});
/**
* Creates a rule to match one or more of the rules separated by a comma
*
* @param {RuleOrLiteral} rule
*
* @returns {SeqRule}
*/
function commaSep1(rule) {
return seq(rule, repeat(seq(',', rule)));
}
/**
* Creates a rule to optionally match one or more of the rules separated by a comma
*
* @param {RuleOrLiteral} rule
*
* @returns {ChoiceRule}
*/
function commaSep(rule) {
return optional(commaSep1(rule));
}