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.
Files changed (146) hide show
  1. package/LICENSE +21 -0
  2. package/dist/algorithm_catalog.d.ts +32 -0
  3. package/dist/algorithm_catalog.js +323 -0
  4. package/dist/algorithm_catalog.js.map +1 -0
  5. package/dist/ast.d.ts +244 -0
  6. package/dist/ast.js +8 -0
  7. package/dist/ast.js.map +1 -0
  8. package/dist/cli.d.ts +4 -0
  9. package/dist/cli.js +605 -0
  10. package/dist/cli.js.map +1 -0
  11. package/dist/emit_batch.d.ts +7 -0
  12. package/dist/emit_batch.js +133 -0
  13. package/dist/emit_batch.js.map +1 -0
  14. package/dist/emit_capability.d.ts +7 -0
  15. package/dist/emit_capability.js +376 -0
  16. package/dist/emit_capability.js.map +1 -0
  17. package/dist/emit_composition.d.ts +22 -0
  18. package/dist/emit_composition.js +184 -0
  19. package/dist/emit_composition.js.map +1 -0
  20. package/dist/emit_deploy.d.ts +9 -0
  21. package/dist/emit_deploy.js +191 -0
  22. package/dist/emit_deploy.js.map +1 -0
  23. package/dist/emit_events.d.ts +14 -0
  24. package/dist/emit_events.js +305 -0
  25. package/dist/emit_events.js.map +1 -0
  26. package/dist/emit_extras.d.ts +12 -0
  27. package/dist/emit_extras.js +234 -0
  28. package/dist/emit_extras.js.map +1 -0
  29. package/dist/emit_full.d.ts +13 -0
  30. package/dist/emit_full.js +273 -0
  31. package/dist/emit_full.js.map +1 -0
  32. package/dist/emit_maintenance.d.ts +16 -0
  33. package/dist/emit_maintenance.js +442 -0
  34. package/dist/emit_maintenance.js.map +1 -0
  35. package/dist/emit_runtime.d.ts +13 -0
  36. package/dist/emit_runtime.js +691 -0
  37. package/dist/emit_runtime.js.map +1 -0
  38. package/dist/emit_sourcemap.d.ts +29 -0
  39. package/dist/emit_sourcemap.js +123 -0
  40. package/dist/emit_sourcemap.js.map +1 -0
  41. package/dist/emit_tests.d.ts +15 -0
  42. package/dist/emit_tests.js +185 -0
  43. package/dist/emit_tests.js.map +1 -0
  44. package/dist/emit_websocket.d.ts +6 -0
  45. package/dist/emit_websocket.js +223 -0
  46. package/dist/emit_websocket.js.map +1 -0
  47. package/dist/emitter.d.ts +25 -0
  48. package/dist/emitter.js +511 -0
  49. package/dist/emitter.js.map +1 -0
  50. package/dist/extension_manager.d.ts +38 -0
  51. package/dist/extension_manager.js +170 -0
  52. package/dist/extension_manager.js.map +1 -0
  53. package/dist/formatter.d.ts +34 -0
  54. package/dist/formatter.js +317 -0
  55. package/dist/formatter.js.map +1 -0
  56. package/dist/index.d.ts +42 -0
  57. package/dist/index.js +113 -0
  58. package/dist/index.js.map +1 -0
  59. package/dist/ir.d.ts +168 -0
  60. package/dist/ir.js +10 -0
  61. package/dist/ir.js.map +1 -0
  62. package/dist/lexer.d.ts +195 -0
  63. package/dist/lexer.js +619 -0
  64. package/dist/lexer.js.map +1 -0
  65. package/dist/lowering.d.ts +25 -0
  66. package/dist/lowering.js +500 -0
  67. package/dist/lowering.js.map +1 -0
  68. package/dist/module_loader.d.ts +25 -0
  69. package/dist/module_loader.js +126 -0
  70. package/dist/module_loader.js.map +1 -0
  71. package/dist/optimizer.d.ts +26 -0
  72. package/dist/optimizer.js +158 -0
  73. package/dist/optimizer.js.map +1 -0
  74. package/dist/parse_decls.d.ts +13 -0
  75. package/dist/parse_decls.js +442 -0
  76. package/dist/parse_decls.js.map +1 -0
  77. package/dist/parse_decls2.d.ts +13 -0
  78. package/dist/parse_decls2.js +295 -0
  79. package/dist/parse_decls2.js.map +1 -0
  80. package/dist/parse_expr.d.ts +7 -0
  81. package/dist/parse_expr.js +197 -0
  82. package/dist/parse_expr.js.map +1 -0
  83. package/dist/parse_types.d.ts +6 -0
  84. package/dist/parse_types.js +51 -0
  85. package/dist/parse_types.js.map +1 -0
  86. package/dist/parser.d.ts +10 -0
  87. package/dist/parser.js +62 -0
  88. package/dist/parser.js.map +1 -0
  89. package/dist/parser_base.d.ts +19 -0
  90. package/dist/parser_base.js +50 -0
  91. package/dist/parser_base.js.map +1 -0
  92. package/dist/parser_recovery.d.ts +26 -0
  93. package/dist/parser_recovery.js +140 -0
  94. package/dist/parser_recovery.js.map +1 -0
  95. package/dist/scaffold.d.ts +13 -0
  96. package/dist/scaffold.js +376 -0
  97. package/dist/scaffold.js.map +1 -0
  98. package/dist/solver.d.ts +26 -0
  99. package/dist/solver.js +281 -0
  100. package/dist/solver.js.map +1 -0
  101. package/dist/typechecker.d.ts +52 -0
  102. package/dist/typechecker.js +534 -0
  103. package/dist/typechecker.js.map +1 -0
  104. package/dist/types.d.ts +38 -0
  105. package/dist/types.js +85 -0
  106. package/dist/types.js.map +1 -0
  107. package/dist/verifier.d.ts +46 -0
  108. package/dist/verifier.js +307 -0
  109. package/dist/verifier.js.map +1 -0
  110. package/package.json +52 -0
  111. package/src/algorithm_catalog.ts +345 -0
  112. package/src/ast.ts +334 -0
  113. package/src/cli.ts +624 -0
  114. package/src/emit_batch.ts +140 -0
  115. package/src/emit_capability.ts +436 -0
  116. package/src/emit_composition.ts +196 -0
  117. package/src/emit_deploy.ts +190 -0
  118. package/src/emit_events.ts +307 -0
  119. package/src/emit_extras.ts +240 -0
  120. package/src/emit_full.ts +309 -0
  121. package/src/emit_maintenance.ts +459 -0
  122. package/src/emit_runtime.ts +731 -0
  123. package/src/emit_sourcemap.ts +140 -0
  124. package/src/emit_tests.ts +205 -0
  125. package/src/emit_websocket.ts +229 -0
  126. package/src/emitter.ts +566 -0
  127. package/src/extension_manager.ts +187 -0
  128. package/src/formatter.ts +297 -0
  129. package/src/index.ts +88 -0
  130. package/src/ir.ts +215 -0
  131. package/src/lexer.ts +630 -0
  132. package/src/lowering.ts +556 -0
  133. package/src/module_loader.ts +114 -0
  134. package/src/optimizer.ts +196 -0
  135. package/src/parse_decls.ts +409 -0
  136. package/src/parse_decls2.ts +244 -0
  137. package/src/parse_expr.ts +197 -0
  138. package/src/parse_types.ts +54 -0
  139. package/src/parser.ts +64 -0
  140. package/src/parser_base.ts +57 -0
  141. package/src/parser_recovery.ts +153 -0
  142. package/src/scaffold.ts +375 -0
  143. package/src/solver.ts +330 -0
  144. package/src/typechecker.ts +591 -0
  145. package/src/types.ts +122 -0
  146. 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
+ }