bonescript-compiler 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/dist/algorithm_catalog.d.ts +32 -0
- package/dist/algorithm_catalog.js +323 -0
- package/dist/algorithm_catalog.js.map +1 -0
- package/dist/ast.d.ts +244 -0
- package/dist/ast.js +8 -0
- package/dist/ast.js.map +1 -0
- package/dist/cli.d.ts +4 -0
- package/dist/cli.js +605 -0
- package/dist/cli.js.map +1 -0
- package/dist/emit_batch.d.ts +7 -0
- package/dist/emit_batch.js +133 -0
- package/dist/emit_batch.js.map +1 -0
- package/dist/emit_capability.d.ts +7 -0
- package/dist/emit_capability.js +376 -0
- package/dist/emit_capability.js.map +1 -0
- package/dist/emit_composition.d.ts +22 -0
- package/dist/emit_composition.js +184 -0
- package/dist/emit_composition.js.map +1 -0
- package/dist/emit_deploy.d.ts +9 -0
- package/dist/emit_deploy.js +191 -0
- package/dist/emit_deploy.js.map +1 -0
- package/dist/emit_events.d.ts +14 -0
- package/dist/emit_events.js +305 -0
- package/dist/emit_events.js.map +1 -0
- package/dist/emit_extras.d.ts +12 -0
- package/dist/emit_extras.js +234 -0
- package/dist/emit_extras.js.map +1 -0
- package/dist/emit_full.d.ts +13 -0
- package/dist/emit_full.js +273 -0
- package/dist/emit_full.js.map +1 -0
- package/dist/emit_maintenance.d.ts +16 -0
- package/dist/emit_maintenance.js +442 -0
- package/dist/emit_maintenance.js.map +1 -0
- package/dist/emit_runtime.d.ts +13 -0
- package/dist/emit_runtime.js +691 -0
- package/dist/emit_runtime.js.map +1 -0
- package/dist/emit_sourcemap.d.ts +29 -0
- package/dist/emit_sourcemap.js +123 -0
- package/dist/emit_sourcemap.js.map +1 -0
- package/dist/emit_tests.d.ts +15 -0
- package/dist/emit_tests.js +185 -0
- package/dist/emit_tests.js.map +1 -0
- package/dist/emit_websocket.d.ts +6 -0
- package/dist/emit_websocket.js +223 -0
- package/dist/emit_websocket.js.map +1 -0
- package/dist/emitter.d.ts +25 -0
- package/dist/emitter.js +511 -0
- package/dist/emitter.js.map +1 -0
- package/dist/extension_manager.d.ts +38 -0
- package/dist/extension_manager.js +170 -0
- package/dist/extension_manager.js.map +1 -0
- package/dist/formatter.d.ts +34 -0
- package/dist/formatter.js +317 -0
- package/dist/formatter.js.map +1 -0
- package/dist/index.d.ts +42 -0
- package/dist/index.js +113 -0
- package/dist/index.js.map +1 -0
- package/dist/ir.d.ts +168 -0
- package/dist/ir.js +10 -0
- package/dist/ir.js.map +1 -0
- package/dist/lexer.d.ts +195 -0
- package/dist/lexer.js +619 -0
- package/dist/lexer.js.map +1 -0
- package/dist/lowering.d.ts +25 -0
- package/dist/lowering.js +500 -0
- package/dist/lowering.js.map +1 -0
- package/dist/module_loader.d.ts +25 -0
- package/dist/module_loader.js +126 -0
- package/dist/module_loader.js.map +1 -0
- package/dist/optimizer.d.ts +26 -0
- package/dist/optimizer.js +158 -0
- package/dist/optimizer.js.map +1 -0
- package/dist/parse_decls.d.ts +13 -0
- package/dist/parse_decls.js +442 -0
- package/dist/parse_decls.js.map +1 -0
- package/dist/parse_decls2.d.ts +13 -0
- package/dist/parse_decls2.js +295 -0
- package/dist/parse_decls2.js.map +1 -0
- package/dist/parse_expr.d.ts +7 -0
- package/dist/parse_expr.js +197 -0
- package/dist/parse_expr.js.map +1 -0
- package/dist/parse_types.d.ts +6 -0
- package/dist/parse_types.js +51 -0
- package/dist/parse_types.js.map +1 -0
- package/dist/parser.d.ts +10 -0
- package/dist/parser.js +62 -0
- package/dist/parser.js.map +1 -0
- package/dist/parser_base.d.ts +19 -0
- package/dist/parser_base.js +50 -0
- package/dist/parser_base.js.map +1 -0
- package/dist/parser_recovery.d.ts +26 -0
- package/dist/parser_recovery.js +140 -0
- package/dist/parser_recovery.js.map +1 -0
- package/dist/scaffold.d.ts +13 -0
- package/dist/scaffold.js +376 -0
- package/dist/scaffold.js.map +1 -0
- package/dist/solver.d.ts +26 -0
- package/dist/solver.js +281 -0
- package/dist/solver.js.map +1 -0
- package/dist/typechecker.d.ts +52 -0
- package/dist/typechecker.js +534 -0
- package/dist/typechecker.js.map +1 -0
- package/dist/types.d.ts +38 -0
- package/dist/types.js +85 -0
- package/dist/types.js.map +1 -0
- package/dist/verifier.d.ts +46 -0
- package/dist/verifier.js +307 -0
- package/dist/verifier.js.map +1 -0
- package/package.json +52 -0
- package/src/algorithm_catalog.ts +345 -0
- package/src/ast.ts +334 -0
- package/src/cli.ts +624 -0
- package/src/emit_batch.ts +140 -0
- package/src/emit_capability.ts +436 -0
- package/src/emit_composition.ts +196 -0
- package/src/emit_deploy.ts +190 -0
- package/src/emit_events.ts +307 -0
- package/src/emit_extras.ts +240 -0
- package/src/emit_full.ts +309 -0
- package/src/emit_maintenance.ts +459 -0
- package/src/emit_runtime.ts +731 -0
- package/src/emit_sourcemap.ts +140 -0
- package/src/emit_tests.ts +205 -0
- package/src/emit_websocket.ts +229 -0
- package/src/emitter.ts +566 -0
- package/src/extension_manager.ts +187 -0
- package/src/formatter.ts +297 -0
- package/src/index.ts +88 -0
- package/src/ir.ts +215 -0
- package/src/lexer.ts +630 -0
- package/src/lowering.ts +556 -0
- package/src/module_loader.ts +114 -0
- package/src/optimizer.ts +196 -0
- package/src/parse_decls.ts +409 -0
- package/src/parse_decls2.ts +244 -0
- package/src/parse_expr.ts +197 -0
- package/src/parse_types.ts +54 -0
- package/src/parser.ts +64 -0
- package/src/parser_base.ts +57 -0
- package/src/parser_recovery.ts +153 -0
- package/src/scaffold.ts +375 -0
- package/src/solver.ts +330 -0
- package/src/typechecker.ts +591 -0
- package/src/types.ts +122 -0
- package/src/verifier.ts +348 -0
|
@@ -0,0 +1,197 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* BoneScript Expression Parser — Pratt-style precedence climbing.
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import { TokenKind } from "./lexer";
|
|
6
|
+
import { TokenStream, ParseError } from "./parser_base";
|
|
7
|
+
import * as AST from "./ast";
|
|
8
|
+
|
|
9
|
+
export function parseExpr(s: TokenStream): AST.ExprNode {
|
|
10
|
+
return parseLogicalOr(s);
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export function parseExprList(s: TokenStream): AST.ExprNode[] {
|
|
14
|
+
const exprs: AST.ExprNode[] = [];
|
|
15
|
+
if (s.check(TokenKind.RBracket)) return exprs;
|
|
16
|
+
do {
|
|
17
|
+
exprs.push(parseExpr(s));
|
|
18
|
+
} while (s.match(TokenKind.Comma));
|
|
19
|
+
return exprs;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function parseLogicalOr(s: TokenStream): AST.ExprNode {
|
|
23
|
+
let left = parseLogicalAnd(s);
|
|
24
|
+
while (s.check(TokenKind.KwOr)) {
|
|
25
|
+
const loc = s.peek().loc;
|
|
26
|
+
s.advance();
|
|
27
|
+
const right = parseLogicalAnd(s);
|
|
28
|
+
left = { kind: "BinaryExpr", loc, op: "or", left, right };
|
|
29
|
+
}
|
|
30
|
+
return left;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function parseLogicalAnd(s: TokenStream): AST.ExprNode {
|
|
34
|
+
let left = parseComparison(s);
|
|
35
|
+
while (s.check(TokenKind.KwAnd)) {
|
|
36
|
+
const loc = s.peek().loc;
|
|
37
|
+
s.advance();
|
|
38
|
+
const right = parseComparison(s);
|
|
39
|
+
left = { kind: "BinaryExpr", loc, op: "and", left, right };
|
|
40
|
+
}
|
|
41
|
+
return left;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function parseComparison(s: TokenStream): AST.ExprNode {
|
|
45
|
+
let left = parseAdditive(s);
|
|
46
|
+
const compOps = [
|
|
47
|
+
TokenKind.EqEq, TokenKind.NotEq, TokenKind.LAngle, TokenKind.RAngle,
|
|
48
|
+
TokenKind.LtEq, TokenKind.GtEq, TokenKind.KwIn, TokenKind.KwContains,
|
|
49
|
+
];
|
|
50
|
+
if (compOps.includes(s.peek().kind)) {
|
|
51
|
+
const loc = s.peek().loc;
|
|
52
|
+
const op = s.advance().value;
|
|
53
|
+
const right = parseAdditive(s);
|
|
54
|
+
// Check for range: expr in expr..expr
|
|
55
|
+
if (op === "in" && s.check(TokenKind.DotDot)) {
|
|
56
|
+
const dotLoc = s.peek().loc;
|
|
57
|
+
s.advance(); // consume ..
|
|
58
|
+
const upper = parseAdditive(s);
|
|
59
|
+
// Desugar "x in low..high" into BinaryExpr with op "in_range"
|
|
60
|
+
const range: AST.BinaryExprNode = { kind: "BinaryExpr", loc: dotLoc, op: "..", left: right, right: upper };
|
|
61
|
+
left = { kind: "BinaryExpr", loc, op: "in", left, right: range };
|
|
62
|
+
} else {
|
|
63
|
+
left = { kind: "BinaryExpr", loc, op, left, right };
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
return left;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
function parseAdditive(s: TokenStream): AST.ExprNode {
|
|
70
|
+
let left = parseMultiplicative(s);
|
|
71
|
+
while (s.check(TokenKind.Plus) || s.check(TokenKind.Minus)) {
|
|
72
|
+
const loc = s.peek().loc;
|
|
73
|
+
const op = s.advance().value;
|
|
74
|
+
const right = parseMultiplicative(s);
|
|
75
|
+
left = { kind: "BinaryExpr", loc, op, left, right };
|
|
76
|
+
}
|
|
77
|
+
return left;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
function parseMultiplicative(s: TokenStream): AST.ExprNode {
|
|
81
|
+
let left = parseUnary(s);
|
|
82
|
+
while (s.check(TokenKind.Star) || s.check(TokenKind.Slash) || s.check(TokenKind.Percent)) {
|
|
83
|
+
const loc = s.peek().loc;
|
|
84
|
+
const op = s.advance().value;
|
|
85
|
+
const right = parseUnary(s);
|
|
86
|
+
left = { kind: "BinaryExpr", loc, op, left, right };
|
|
87
|
+
}
|
|
88
|
+
return left;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
function parseUnary(s: TokenStream): AST.ExprNode {
|
|
92
|
+
if (s.check(TokenKind.KwNot)) {
|
|
93
|
+
const loc = s.peek().loc;
|
|
94
|
+
s.advance();
|
|
95
|
+
const operand = parseUnary(s);
|
|
96
|
+
return { kind: "UnaryExpr", loc, op: "not", operand };
|
|
97
|
+
}
|
|
98
|
+
if (s.check(TokenKind.Minus)) {
|
|
99
|
+
const loc = s.peek().loc;
|
|
100
|
+
s.advance();
|
|
101
|
+
const operand = parseUnary(s);
|
|
102
|
+
return { kind: "UnaryExpr", loc, op: "-", operand };
|
|
103
|
+
}
|
|
104
|
+
return parsePrimary(s);
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
function parsePrimary(s: TokenStream): AST.ExprNode {
|
|
108
|
+
const tok = s.peek();
|
|
109
|
+
const loc = tok.loc;
|
|
110
|
+
|
|
111
|
+
// Parenthesized expression
|
|
112
|
+
if (tok.kind === TokenKind.LParen) {
|
|
113
|
+
s.advance();
|
|
114
|
+
const expr = parseExpr(s);
|
|
115
|
+
s.expect(TokenKind.RParen, "parenthesized expression");
|
|
116
|
+
return expr;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
// List literal
|
|
120
|
+
if (tok.kind === TokenKind.LBracket) {
|
|
121
|
+
s.advance();
|
|
122
|
+
const elements: AST.ExprNode[] = [];
|
|
123
|
+
if (!s.check(TokenKind.RBracket)) {
|
|
124
|
+
do { elements.push(parseExpr(s)); } while (s.match(TokenKind.Comma));
|
|
125
|
+
}
|
|
126
|
+
s.expect(TokenKind.RBracket, "list literal close");
|
|
127
|
+
return { kind: "Literal", loc, type: "list", value: elements };
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
// String literal
|
|
131
|
+
if (tok.kind === TokenKind.StringLiteral) {
|
|
132
|
+
s.advance();
|
|
133
|
+
return { kind: "Literal", loc, type: "string", value: tok.value };
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
// Int literal
|
|
137
|
+
if (tok.kind === TokenKind.IntLiteral) {
|
|
138
|
+
s.advance();
|
|
139
|
+
return { kind: "Literal", loc, type: "int", value: parseInt(tok.value, 10) };
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
// Float literal
|
|
143
|
+
if (tok.kind === TokenKind.FloatLiteral) {
|
|
144
|
+
s.advance();
|
|
145
|
+
return { kind: "Literal", loc, type: "float", value: parseFloat(tok.value) };
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
// Boolean
|
|
149
|
+
if (tok.kind === TokenKind.KwTrue) { s.advance(); return { kind: "Literal", loc, type: "bool", value: true }; }
|
|
150
|
+
if (tok.kind === TokenKind.KwFalse) { s.advance(); return { kind: "Literal", loc, type: "bool", value: false }; }
|
|
151
|
+
|
|
152
|
+
// None
|
|
153
|
+
if (tok.kind === TokenKind.KwNone) { s.advance(); return { kind: "Literal", loc, type: "none", value: null }; }
|
|
154
|
+
|
|
155
|
+
// now()
|
|
156
|
+
if (tok.kind === TokenKind.KwNow) {
|
|
157
|
+
s.advance();
|
|
158
|
+
if (s.match(TokenKind.LParen)) { s.expect(TokenKind.RParen, "now()"); }
|
|
159
|
+
return { kind: "CallExpr", loc, name: "now", args: [] };
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
// Identifier — field ref or function call
|
|
163
|
+
if (tok.kind === TokenKind.Identifier || isKeywordIdentifier(tok)) {
|
|
164
|
+
const path: string[] = [];
|
|
165
|
+
path.push(s.advance().value);
|
|
166
|
+
|
|
167
|
+
// Function call
|
|
168
|
+
if (s.check(TokenKind.LParen)) {
|
|
169
|
+
s.advance();
|
|
170
|
+
const args: AST.ExprNode[] = [];
|
|
171
|
+
if (!s.check(TokenKind.RParen)) {
|
|
172
|
+
do { args.push(parseExpr(s)); } while (s.match(TokenKind.Comma));
|
|
173
|
+
}
|
|
174
|
+
s.expect(TokenKind.RParen, "function call close");
|
|
175
|
+
return { kind: "CallExpr", loc, name: path[0], args };
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
// Dotted field reference
|
|
179
|
+
while (s.match(TokenKind.Dot)) {
|
|
180
|
+
const next = s.peek();
|
|
181
|
+
if (next.kind === TokenKind.Identifier || next.kind === TokenKind.KwUnique || isKeywordIdentifier(next)) {
|
|
182
|
+
path.push(s.advance().value);
|
|
183
|
+
} else {
|
|
184
|
+
break;
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
return { kind: "FieldRef", loc, path };
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
throw new ParseError(`Expected expression, got ${tok.kind} ('${tok.value}')`, tok.loc);
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
/** Check if a token is a keyword that can also serve as an identifier */
|
|
194
|
+
function isKeywordIdentifier(tok: { kind: TokenKind; value: string }): boolean {
|
|
195
|
+
// Keywords that are commonly used as variable/field names
|
|
196
|
+
return /^[a-zA-Z_][a-zA-Z0-9_]*$/.test(tok.value) && tok.kind !== TokenKind.EOF;
|
|
197
|
+
}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* BoneScript Type Expression Parser
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import { TokenKind } from "./lexer";
|
|
6
|
+
import { TokenStream, ParseError } from "./parser_base";
|
|
7
|
+
import * as AST from "./ast";
|
|
8
|
+
|
|
9
|
+
export function parseTypeExpr(s: TokenStream): AST.TypeExprNode {
|
|
10
|
+
const loc = s.peek().loc;
|
|
11
|
+
const tok = s.peek();
|
|
12
|
+
|
|
13
|
+
// Generic types
|
|
14
|
+
const generics = [TokenKind.KwSet, TokenKind.KwList, TokenKind.KwOptional, TokenKind.KwResult, TokenKind.KwMap];
|
|
15
|
+
if (generics.includes(tok.kind)) {
|
|
16
|
+
const name = s.advance().value;
|
|
17
|
+
s.expect(TokenKind.LAngle, "generic type arg");
|
|
18
|
+
const typeArgs: AST.TypeExprNode[] = [];
|
|
19
|
+
typeArgs.push(parseTypeExpr(s));
|
|
20
|
+
while (s.match(TokenKind.Comma)) {
|
|
21
|
+
typeArgs.push(parseTypeExpr(s));
|
|
22
|
+
}
|
|
23
|
+
s.expect(TokenKind.RAngle, "generic type arg close");
|
|
24
|
+
return { kind: "GenericType", loc, name, typeArgs };
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
// Primitive types
|
|
28
|
+
const primitives = [
|
|
29
|
+
TokenKind.KwString, TokenKind.KwUint, TokenKind.KwInt, TokenKind.KwFloat,
|
|
30
|
+
TokenKind.KwBool, TokenKind.KwTimestamp, TokenKind.KwUuid, TokenKind.KwBytes, TokenKind.KwJson,
|
|
31
|
+
];
|
|
32
|
+
if (primitives.includes(tok.kind)) {
|
|
33
|
+
return { kind: "PrimitiveType", loc, name: s.advance().value };
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
// Tuple type
|
|
37
|
+
if (tok.kind === TokenKind.LParen) {
|
|
38
|
+
s.advance();
|
|
39
|
+
const elements: AST.TypeExprNode[] = [];
|
|
40
|
+
elements.push(parseTypeExpr(s));
|
|
41
|
+
while (s.match(TokenKind.Comma)) {
|
|
42
|
+
elements.push(parseTypeExpr(s));
|
|
43
|
+
}
|
|
44
|
+
s.expect(TokenKind.RParen, "tuple type close");
|
|
45
|
+
return { kind: "TupleType", loc, elements };
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
// Entity reference
|
|
49
|
+
if (tok.kind === TokenKind.Identifier) {
|
|
50
|
+
return { kind: "EntityRefType", loc, name: s.advance().value };
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
throw new ParseError(`Expected type expression, got ${tok.kind}`, tok.loc);
|
|
54
|
+
}
|
package/src/parser.ts
ADDED
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import { Token, TokenKind } from "./lexer";
|
|
2
|
+
import { TokenStream, ParseError } from "./parser_base";
|
|
3
|
+
import * as AST from "./ast";
|
|
4
|
+
import { parseEntityDecl, parseCapabilityDecl } from "./parse_decls";
|
|
5
|
+
import { parseChannelDecl, parseStoreDecl, parseEventDecl, parseConstraintDecl, parsePolicyDecl, parseFlowDecl, parseImportDecl, parseExtensionPointDecl } from "./parse_decls2";
|
|
6
|
+
|
|
7
|
+
export { ParseError } from "./parser_base";
|
|
8
|
+
|
|
9
|
+
export class Parser {
|
|
10
|
+
private s: TokenStream;
|
|
11
|
+
|
|
12
|
+
constructor(tokens: Token[]) {
|
|
13
|
+
this.s = new TokenStream(tokens);
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
parse(): AST.ProgramNode {
|
|
17
|
+
const loc = this.s.peek().loc;
|
|
18
|
+
const systems: AST.SystemDeclNode[] = [];
|
|
19
|
+
while (!this.s.check(TokenKind.EOF)) {
|
|
20
|
+
systems.push(this.parseSystemDecl());
|
|
21
|
+
}
|
|
22
|
+
if (systems.length === 0) {
|
|
23
|
+
throw new ParseError("Program must contain at least one system declaration", loc);
|
|
24
|
+
}
|
|
25
|
+
return { kind: "Program", loc, systems };
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
private parseSystemDecl(): AST.SystemDeclNode {
|
|
29
|
+
const loc = this.s.peek().loc;
|
|
30
|
+
this.s.expect(TokenKind.KwSystem, "system declaration");
|
|
31
|
+
const name = this.s.expect(TokenKind.Identifier, "system name").value;
|
|
32
|
+
this.s.expect(TokenKind.LBrace, "system body");
|
|
33
|
+
let domain: string | null = null;
|
|
34
|
+
if (this.s.check(TokenKind.KwDomain)) {
|
|
35
|
+
this.s.advance();
|
|
36
|
+
this.s.expect(TokenKind.Colon, "domain");
|
|
37
|
+
domain = this.s.expect(TokenKind.Identifier, "domain name").value;
|
|
38
|
+
}
|
|
39
|
+
const declarations: AST.DeclarationNode[] = [];
|
|
40
|
+
while (!this.s.check(TokenKind.RBrace) && !this.s.check(TokenKind.EOF)) {
|
|
41
|
+
declarations.push(this.parseDeclaration());
|
|
42
|
+
}
|
|
43
|
+
this.s.expect(TokenKind.RBrace, "system body close");
|
|
44
|
+
return { kind: "SystemDecl", loc, name, domain, declarations };
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
private parseDeclaration(): AST.DeclarationNode {
|
|
48
|
+
const tok = this.s.peek();
|
|
49
|
+
switch (tok.kind) {
|
|
50
|
+
case TokenKind.KwEntity: return parseEntityDecl(this.s);
|
|
51
|
+
case TokenKind.KwCapability: return parseCapabilityDecl(this.s);
|
|
52
|
+
case TokenKind.KwChannel: return parseChannelDecl(this.s);
|
|
53
|
+
case TokenKind.KwStore: return parseStoreDecl(this.s);
|
|
54
|
+
case TokenKind.KwEvent: return parseEventDecl(this.s);
|
|
55
|
+
case TokenKind.KwConstraint: return parseConstraintDecl(this.s);
|
|
56
|
+
case TokenKind.KwPolicy: return parsePolicyDecl(this.s);
|
|
57
|
+
case TokenKind.KwFlow: return parseFlowDecl(this.s);
|
|
58
|
+
case TokenKind.KwImport: return parseImportDecl(this.s);
|
|
59
|
+
case TokenKind.KwExtensionPoint: return parseExtensionPointDecl(this.s);
|
|
60
|
+
default:
|
|
61
|
+
throw new ParseError("Expected declaration, got " + tok.kind, tok.loc);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
}
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* bone parser Base — Token stream utilities.
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import { Token, TokenKind, SourceLocation } from "./lexer";
|
|
6
|
+
|
|
7
|
+
export class ParseError extends Error {
|
|
8
|
+
constructor(message: string, public loc: SourceLocation) {
|
|
9
|
+
super(`Parse error at ${loc.line}:${loc.column}: ${message}`);
|
|
10
|
+
this.name = "ParseError";
|
|
11
|
+
}
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export class TokenStream {
|
|
15
|
+
private tokens: Token[];
|
|
16
|
+
private pos: number = 0;
|
|
17
|
+
|
|
18
|
+
constructor(tokens: Token[]) {
|
|
19
|
+
this.tokens = tokens;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
peek(offset: number = 0): Token {
|
|
23
|
+
return this.tokens[this.pos + offset] ?? this.tokens[this.tokens.length - 1];
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
check(kind: TokenKind): boolean {
|
|
27
|
+
return this.peek().kind === kind;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
checkAny(...kinds: TokenKind[]): boolean {
|
|
31
|
+
return kinds.includes(this.peek().kind);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
advance(): Token {
|
|
35
|
+
const t = this.tokens[this.pos];
|
|
36
|
+
if (this.pos < this.tokens.length - 1) this.pos++;
|
|
37
|
+
return t;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
expect(kind: TokenKind, context: string): Token {
|
|
41
|
+
if (!this.check(kind)) {
|
|
42
|
+
throw new ParseError(
|
|
43
|
+
`Expected ${kind} in ${context}, got ${this.peek().kind} ('${this.peek().value}')`,
|
|
44
|
+
this.peek().loc
|
|
45
|
+
);
|
|
46
|
+
}
|
|
47
|
+
return this.advance();
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
match(kind: TokenKind): boolean {
|
|
51
|
+
if (this.check(kind)) {
|
|
52
|
+
this.advance();
|
|
53
|
+
return true;
|
|
54
|
+
}
|
|
55
|
+
return false;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
@@ -0,0 +1,153 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* bone parser with Error Recovery
|
|
3
|
+
* Wraps the strict parser to collect multiple errors per file.
|
|
4
|
+
*
|
|
5
|
+
* Strategy:
|
|
6
|
+
* - On error, skip tokens until we hit a synchronization point
|
|
7
|
+
* - Synchronization points: declaration keywords (entity, capability, etc.) and closing braces
|
|
8
|
+
* - Each error is recorded with location, but parsing continues
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import { Token, TokenKind } from "./lexer";
|
|
12
|
+
import { TokenStream, ParseError } from "./parser_base";
|
|
13
|
+
import * as AST from "./ast";
|
|
14
|
+
import { parseEntityDecl, parseCapabilityDecl } from "./parse_decls";
|
|
15
|
+
import {
|
|
16
|
+
parseChannelDecl, parseStoreDecl, parseEventDecl,
|
|
17
|
+
parseConstraintDecl, parsePolicyDecl, parseFlowDecl, parseImportDecl,
|
|
18
|
+
parseExtensionPointDecl,
|
|
19
|
+
} from "./parse_decls2";
|
|
20
|
+
|
|
21
|
+
export interface RecoveredParseResult {
|
|
22
|
+
ast: AST.ProgramNode | null;
|
|
23
|
+
errors: ParseError[];
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
const SYNC_POINTS = new Set([
|
|
27
|
+
TokenKind.KwSystem,
|
|
28
|
+
TokenKind.KwEntity,
|
|
29
|
+
TokenKind.KwCapability,
|
|
30
|
+
TokenKind.KwChannel,
|
|
31
|
+
TokenKind.KwStore,
|
|
32
|
+
TokenKind.KwEvent,
|
|
33
|
+
TokenKind.KwConstraint,
|
|
34
|
+
TokenKind.KwPolicy,
|
|
35
|
+
TokenKind.KwFlow,
|
|
36
|
+
TokenKind.KwImport,
|
|
37
|
+
TokenKind.KwExtensionPoint,
|
|
38
|
+
TokenKind.RBrace,
|
|
39
|
+
]);
|
|
40
|
+
|
|
41
|
+
export class RecoveringParser {
|
|
42
|
+
private s: TokenStream;
|
|
43
|
+
private errors: ParseError[] = [];
|
|
44
|
+
|
|
45
|
+
constructor(tokens: Token[]) {
|
|
46
|
+
this.s = new TokenStream(tokens);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
parse(): RecoveredParseResult {
|
|
50
|
+
const loc = this.s.peek().loc;
|
|
51
|
+
const systems: AST.SystemDeclNode[] = [];
|
|
52
|
+
|
|
53
|
+
while (!this.s.check(TokenKind.EOF)) {
|
|
54
|
+
try {
|
|
55
|
+
systems.push(this.parseSystemDecl());
|
|
56
|
+
} catch (e) {
|
|
57
|
+
if (e instanceof ParseError) {
|
|
58
|
+
this.errors.push(e);
|
|
59
|
+
this.synchronize();
|
|
60
|
+
} else {
|
|
61
|
+
throw e;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
if (systems.length === 0 && this.errors.length === 0) {
|
|
67
|
+
this.errors.push(new ParseError("Program must contain at least one system declaration", loc));
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
return {
|
|
71
|
+
ast: systems.length > 0 ? { kind: "Program", loc, systems } : null,
|
|
72
|
+
errors: this.errors,
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
private synchronize() {
|
|
77
|
+
// Skip tokens until we find a synchronization point
|
|
78
|
+
while (!this.s.check(TokenKind.EOF)) {
|
|
79
|
+
const tok = this.s.peek();
|
|
80
|
+
if (SYNC_POINTS.has(tok.kind)) {
|
|
81
|
+
// Found sync point — if it's a closing brace, consume it; otherwise leave it for next parse
|
|
82
|
+
if (tok.kind === TokenKind.RBrace) this.s.advance();
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
this.s.advance();
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
private parseSystemDecl(): AST.SystemDeclNode {
|
|
90
|
+
const loc = this.s.peek().loc;
|
|
91
|
+
this.s.expect(TokenKind.KwSystem, "system declaration");
|
|
92
|
+
const name = this.s.expect(TokenKind.Identifier, "system name").value;
|
|
93
|
+
this.s.expect(TokenKind.LBrace, "system body");
|
|
94
|
+
|
|
95
|
+
let domain: string | null = null;
|
|
96
|
+
if (this.s.check(TokenKind.KwDomain)) {
|
|
97
|
+
this.s.advance();
|
|
98
|
+
this.s.expect(TokenKind.Colon, "domain");
|
|
99
|
+
domain = this.s.expect(TokenKind.Identifier, "domain name").value;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
const declarations: AST.DeclarationNode[] = [];
|
|
103
|
+
while (!this.s.check(TokenKind.RBrace) && !this.s.check(TokenKind.EOF)) {
|
|
104
|
+
try {
|
|
105
|
+
declarations.push(this.parseDeclaration());
|
|
106
|
+
} catch (e) {
|
|
107
|
+
if (e instanceof ParseError) {
|
|
108
|
+
this.errors.push(e);
|
|
109
|
+
this.synchronizeInBody();
|
|
110
|
+
} else {
|
|
111
|
+
throw e;
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
this.s.expect(TokenKind.RBrace, "system body close");
|
|
117
|
+
return { kind: "SystemDecl", loc, name, domain, declarations };
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
private synchronizeInBody() {
|
|
121
|
+
let depth = 0;
|
|
122
|
+
while (!this.s.check(TokenKind.EOF)) {
|
|
123
|
+
const tok = this.s.peek();
|
|
124
|
+
if (tok.kind === TokenKind.LBrace) depth++;
|
|
125
|
+
if (tok.kind === TokenKind.RBrace) {
|
|
126
|
+
if (depth === 0) return; // hit system close
|
|
127
|
+
depth--;
|
|
128
|
+
this.s.advance();
|
|
129
|
+
continue;
|
|
130
|
+
}
|
|
131
|
+
if (depth === 0 && SYNC_POINTS.has(tok.kind)) return;
|
|
132
|
+
this.s.advance();
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
private parseDeclaration(): AST.DeclarationNode {
|
|
137
|
+
const tok = this.s.peek();
|
|
138
|
+
switch (tok.kind) {
|
|
139
|
+
case TokenKind.KwEntity: return parseEntityDecl(this.s);
|
|
140
|
+
case TokenKind.KwCapability: return parseCapabilityDecl(this.s);
|
|
141
|
+
case TokenKind.KwChannel: return parseChannelDecl(this.s);
|
|
142
|
+
case TokenKind.KwStore: return parseStoreDecl(this.s);
|
|
143
|
+
case TokenKind.KwEvent: return parseEventDecl(this.s);
|
|
144
|
+
case TokenKind.KwConstraint: return parseConstraintDecl(this.s);
|
|
145
|
+
case TokenKind.KwPolicy: return parsePolicyDecl(this.s);
|
|
146
|
+
case TokenKind.KwFlow: return parseFlowDecl(this.s);
|
|
147
|
+
case TokenKind.KwImport: return parseImportDecl(this.s);
|
|
148
|
+
case TokenKind.KwExtensionPoint: return parseExtensionPointDecl(this.s);
|
|
149
|
+
default:
|
|
150
|
+
throw new ParseError(`Expected declaration, got ${tok.kind} ('${tok.value}')`, tok.loc);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
}
|