bonescript-compiler 0.5.3 → 0.5.4

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 (194) hide show
  1. package/LICENSE +21 -21
  2. package/dist/algorithm_catalog.js +166 -166
  3. package/dist/cli.d.ts +1 -2
  4. package/dist/cli.js +543 -75
  5. package/dist/cli.js.map +1 -1
  6. package/dist/emit_capability.d.ts +0 -13
  7. package/dist/emit_capability.js +134 -296
  8. package/dist/emit_capability.js.map +1 -1
  9. package/dist/emit_composition.js +3 -37
  10. package/dist/emit_composition.js.map +1 -1
  11. package/dist/emit_deploy.js +167 -165
  12. package/dist/emit_deploy.js.map +1 -1
  13. package/dist/emit_events.d.ts +0 -1
  14. package/dist/emit_events.js +275 -325
  15. package/dist/emit_events.js.map +1 -1
  16. package/dist/emit_extras.js +5 -3
  17. package/dist/emit_extras.js.map +1 -1
  18. package/dist/emit_full.js +112 -272
  19. package/dist/emit_full.js.map +1 -1
  20. package/dist/emit_maintenance.js +249 -249
  21. package/dist/emit_runtime.d.ts +11 -17
  22. package/dist/emit_runtime.js +688 -29
  23. package/dist/emit_runtime.js.map +1 -1
  24. package/dist/emit_sourcemap.js +66 -66
  25. package/dist/emit_tests.js +12 -47
  26. package/dist/emit_tests.js.map +1 -1
  27. package/dist/emit_websocket.js +3 -0
  28. package/dist/emit_websocket.js.map +1 -1
  29. package/dist/emitter.js +49 -94
  30. package/dist/emitter.js.map +1 -1
  31. package/dist/extension_manager.d.ts +2 -2
  32. package/dist/extension_manager.js +20 -9
  33. package/dist/extension_manager.js.map +1 -1
  34. package/dist/ir.d.ts +0 -4
  35. package/dist/lowering.d.ts +14 -5
  36. package/dist/lowering.js +417 -66
  37. package/dist/lowering.js.map +1 -1
  38. package/dist/module_loader.d.ts +2 -2
  39. package/dist/module_loader.js +23 -20
  40. package/dist/module_loader.js.map +1 -1
  41. package/dist/optimizer.js +3 -6
  42. package/dist/optimizer.js.map +1 -1
  43. package/dist/scaffold.d.ts +2 -2
  44. package/dist/scaffold.js +319 -315
  45. package/dist/scaffold.js.map +1 -1
  46. package/dist/solver.js +1 -1
  47. package/dist/solver.js.map +1 -1
  48. package/dist/source_map.js.map +1 -0
  49. package/dist/test.js.map +1 -0
  50. package/dist/test_typechecker.d.ts +5 -0
  51. package/dist/test_typechecker.js +126 -0
  52. package/dist/test_typechecker.js.map +1 -0
  53. package/dist/typechecker.d.ts +0 -7
  54. package/dist/typechecker.js +16 -103
  55. package/dist/typechecker.js.map +1 -1
  56. package/dist/verifier.d.ts +1 -5
  57. package/dist/verifier.js +38 -142
  58. package/dist/verifier.js.map +1 -1
  59. package/package.json +52 -62
  60. package/src/algorithm_catalog.ts +345 -345
  61. package/src/ast.d.ts +244 -0
  62. package/src/ast.ts +334 -334
  63. package/src/cli.ts +624 -98
  64. package/src/emit_batch.ts +140 -140
  65. package/src/emit_capability.ts +436 -613
  66. package/src/emit_composition.ts +196 -229
  67. package/src/emit_deploy.ts +190 -187
  68. package/src/emit_events.ts +307 -362
  69. package/src/emit_extras.ts +240 -237
  70. package/src/emit_full.ts +309 -472
  71. package/src/emit_maintenance.ts +459 -459
  72. package/src/emit_runtime.ts +730 -17
  73. package/src/emit_sourcemap.ts +140 -140
  74. package/src/emit_tests.ts +205 -243
  75. package/src/emit_websocket.ts +229 -226
  76. package/src/emitter.ts +578 -626
  77. package/src/extension_manager.ts +187 -177
  78. package/src/formatter.ts +297 -297
  79. package/src/index.ts +88 -88
  80. package/src/ir.ts +215 -216
  81. package/src/lexer.d.ts +195 -0
  82. package/src/lexer.ts +630 -630
  83. package/src/lowering.ts +556 -168
  84. package/src/module_loader.ts +114 -112
  85. package/src/optimizer.ts +196 -199
  86. package/src/parse_decls.d.ts +13 -0
  87. package/src/parse_decls.ts +409 -409
  88. package/src/parse_decls2.d.ts +13 -0
  89. package/src/parse_decls2.ts +244 -244
  90. package/src/parse_expr.d.ts +7 -0
  91. package/src/parse_expr.ts +197 -197
  92. package/src/parse_types.d.ts +6 -0
  93. package/src/parse_types.ts +54 -54
  94. package/src/parser.d.ts +10 -0
  95. package/src/parser.ts +1 -1
  96. package/src/parser_base.d.ts +19 -0
  97. package/src/parser_base.ts +57 -57
  98. package/src/parser_recovery.ts +153 -153
  99. package/src/scaffold.ts +375 -371
  100. package/src/solver.ts +330 -330
  101. package/src/typechecker.d.ts +52 -0
  102. package/src/typechecker.ts +591 -700
  103. package/src/types.d.ts +38 -0
  104. package/src/types.ts +122 -122
  105. package/src/verifier.ts +49 -154
  106. package/README.md +0 -382
  107. package/dist/commands/check.d.ts +0 -5
  108. package/dist/commands/check.js +0 -34
  109. package/dist/commands/check.js.map +0 -1
  110. package/dist/commands/compile.d.ts +0 -5
  111. package/dist/commands/compile.js +0 -215
  112. package/dist/commands/compile.js.map +0 -1
  113. package/dist/commands/debug.d.ts +0 -5
  114. package/dist/commands/debug.js +0 -59
  115. package/dist/commands/debug.js.map +0 -1
  116. package/dist/commands/diff.d.ts +0 -5
  117. package/dist/commands/diff.js +0 -123
  118. package/dist/commands/diff.js.map +0 -1
  119. package/dist/commands/fmt.d.ts +0 -5
  120. package/dist/commands/fmt.js +0 -49
  121. package/dist/commands/fmt.js.map +0 -1
  122. package/dist/commands/init.d.ts +0 -5
  123. package/dist/commands/init.js +0 -96
  124. package/dist/commands/init.js.map +0 -1
  125. package/dist/commands/ir.d.ts +0 -5
  126. package/dist/commands/ir.js +0 -27
  127. package/dist/commands/ir.js.map +0 -1
  128. package/dist/commands/lex.d.ts +0 -5
  129. package/dist/commands/lex.js +0 -21
  130. package/dist/commands/lex.js.map +0 -1
  131. package/dist/commands/parse.d.ts +0 -5
  132. package/dist/commands/parse.js +0 -30
  133. package/dist/commands/parse.js.map +0 -1
  134. package/dist/commands/test.d.ts +0 -5
  135. package/dist/commands/test.js +0 -61
  136. package/dist/commands/test.js.map +0 -1
  137. package/dist/commands/verify_determinism.d.ts +0 -5
  138. package/dist/commands/verify_determinism.js +0 -64
  139. package/dist/commands/verify_determinism.js.map +0 -1
  140. package/dist/commands/watch.d.ts +0 -5
  141. package/dist/commands/watch.js +0 -50
  142. package/dist/commands/watch.js.map +0 -1
  143. package/dist/emit_auth.d.ts +0 -18
  144. package/dist/emit_auth.js +0 -507
  145. package/dist/emit_auth.js.map +0 -1
  146. package/dist/emit_database.d.ts +0 -7
  147. package/dist/emit_database.js +0 -72
  148. package/dist/emit_database.js.map +0 -1
  149. package/dist/emit_index.d.ts +0 -6
  150. package/dist/emit_index.js +0 -202
  151. package/dist/emit_index.js.map +0 -1
  152. package/dist/emit_models.d.ts +0 -12
  153. package/dist/emit_models.js +0 -171
  154. package/dist/emit_models.js.map +0 -1
  155. package/dist/emit_openapi.d.ts +0 -9
  156. package/dist/emit_openapi.js +0 -306
  157. package/dist/emit_openapi.js.map +0 -1
  158. package/dist/emit_package.d.ts +0 -7
  159. package/dist/emit_package.js +0 -68
  160. package/dist/emit_package.js.map +0 -1
  161. package/dist/emit_router.d.ts +0 -12
  162. package/dist/emit_router.js +0 -389
  163. package/dist/emit_router.js.map +0 -1
  164. package/dist/lowering_channels.d.ts +0 -11
  165. package/dist/lowering_channels.js +0 -103
  166. package/dist/lowering_channels.js.map +0 -1
  167. package/dist/lowering_entities.d.ts +0 -11
  168. package/dist/lowering_entities.js +0 -232
  169. package/dist/lowering_entities.js.map +0 -1
  170. package/dist/lowering_helpers.d.ts +0 -13
  171. package/dist/lowering_helpers.js +0 -76
  172. package/dist/lowering_helpers.js.map +0 -1
  173. package/src/commands/check.ts +0 -33
  174. package/src/commands/compile.ts +0 -191
  175. package/src/commands/debug.ts +0 -33
  176. package/src/commands/diff.ts +0 -105
  177. package/src/commands/fmt.ts +0 -22
  178. package/src/commands/init.ts +0 -72
  179. package/src/commands/ir.ts +0 -23
  180. package/src/commands/lex.ts +0 -17
  181. package/src/commands/parse.ts +0 -24
  182. package/src/commands/test.ts +0 -36
  183. package/src/commands/verify_determinism.ts +0 -66
  184. package/src/commands/watch.ts +0 -25
  185. package/src/emit_auth.ts +0 -513
  186. package/src/emit_database.ts +0 -72
  187. package/src/emit_index.ts +0 -210
  188. package/src/emit_models.ts +0 -176
  189. package/src/emit_openapi.ts +0 -315
  190. package/src/emit_package.ts +0 -66
  191. package/src/emit_router.ts +0 -408
  192. package/src/lowering_channels.ts +0 -108
  193. package/src/lowering_entities.ts +0 -258
  194. package/src/lowering_helpers.ts +0 -75
@@ -0,0 +1,13 @@
1
+ /**
2
+ * BoneScript Declaration Parsers — Channel, Store, Event, Constraint, Policy, Flow, Import
3
+ */
4
+ import { TokenStream } from "./parser_base";
5
+ import * as AST from "./ast";
6
+ export declare function parseChannelDecl(s: TokenStream): AST.ChannelDeclNode;
7
+ export declare function parseStoreDecl(s: TokenStream): AST.StoreDeclNode;
8
+ export declare function parseEventDecl(s: TokenStream): AST.EventDeclNode;
9
+ export declare function parseConstraintDecl(s: TokenStream): AST.ConstraintDeclNode;
10
+ export declare function parsePolicyDecl(s: TokenStream): AST.PolicyDeclNode;
11
+ export declare function parseFlowDecl(s: TokenStream): AST.FlowDeclNode;
12
+ export declare function parseImportDecl(s: TokenStream): AST.ImportDeclNode;
13
+ export declare function parseExtensionPointDecl(s: TokenStream): AST.ExtensionPointDeclNode;
@@ -1,244 +1,244 @@
1
- /**
2
- * BoneScript Declaration Parsers — Channel, Store, Event, Constraint, Policy, Flow, Import
3
- */
4
-
5
- import { TokenKind } from "./lexer";
6
- import { TokenStream, ParseError } from "./parser_base";
7
- import * as AST from "./ast";
8
- import { parseExpr } from "./parse_expr";
9
- import { parseTypeExpr } from "./parse_types";
10
- import { parseFieldList, parseDuration, parseIdentList } from "./parse_decls";
11
-
12
- // ─── Channel ─────────────────────────────────────────────────────────────────
13
-
14
- export function parseChannelDecl(s: TokenStream): AST.ChannelDeclNode {
15
- const loc = s.peek().loc;
16
- s.expect(TokenKind.KwChannel, "channel");
17
- const name = s.expect(TokenKind.Identifier, "name").value;
18
- s.expect(TokenKind.LBrace, "{");
19
-
20
- const node: AST.ChannelDeclNode = {
21
- kind: "ChannelDecl", loc, name,
22
- transport: null, ordering: null, participants: null,
23
- persistence: null, filter: null, maxSize: null,
24
- };
25
-
26
- while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
27
- const tok = s.peek();
28
- switch (tok.kind) {
29
- case TokenKind.KwTransport: s.advance(); s.expect(TokenKind.Colon, ":"); node.transport = s.advance().value; break;
30
- case TokenKind.KwOrdering: s.advance(); s.expect(TokenKind.Colon, ":"); node.ordering = s.advance().value; break;
31
- case TokenKind.KwParticipants: s.advance(); s.expect(TokenKind.Colon, ":"); node.participants = parseTypeExpr(s); break;
32
- case TokenKind.KwPersistence: s.advance(); s.expect(TokenKind.Colon, ":"); node.persistence = s.advance().value; break;
33
- case TokenKind.KwFilter: s.advance(); s.expect(TokenKind.Colon, ":"); node.filter = parseExpr(s); break;
34
- case TokenKind.KwMaxSize: s.advance(); s.expect(TokenKind.Colon, ":"); node.maxSize = parseInt(s.expect(TokenKind.IntLiteral, "n").value, 10); break;
35
- default: throw new ParseError(`Unexpected in channel: ${tok.kind}`, tok.loc);
36
- }
37
- }
38
- s.expect(TokenKind.RBrace, "}");
39
- return node;
40
- }
41
-
42
- // ─── Store ───────────────────────────────────────────────────────────────────
43
-
44
- export function parseStoreDecl(s: TokenStream): AST.StoreDeclNode {
45
- const loc = s.peek().loc;
46
- s.expect(TokenKind.KwStore, "store");
47
- const name = s.expect(TokenKind.Identifier, "name").value;
48
- s.expect(TokenKind.LBrace, "{");
49
-
50
- const node: AST.StoreDeclNode = {
51
- kind: "StoreDecl", loc, name,
52
- engine: null, schema: [], retention: null, partition: null, replicas: null,
53
- };
54
-
55
- while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
56
- const tok = s.peek();
57
- switch (tok.kind) {
58
- case TokenKind.KwEngine: s.advance(); s.expect(TokenKind.Colon, ":"); node.engine = s.advance().value; break;
59
- case TokenKind.KwSchema:
60
- s.advance(); s.expect(TokenKind.Colon, ":");
61
- s.expect(TokenKind.LBrace, "{");
62
- node.schema = parseFieldList(s);
63
- s.expect(TokenKind.RBrace, "}");
64
- break;
65
- case TokenKind.KwRetention: s.advance(); s.expect(TokenKind.Colon, ":"); node.retention = parseDuration(s); break;
66
- case TokenKind.KwPartition: s.advance(); s.expect(TokenKind.Colon, ":"); node.partition = s.expect(TokenKind.Identifier, "field").value; break;
67
- case TokenKind.KwReplicas: s.advance(); s.expect(TokenKind.Colon, ":"); node.replicas = parseInt(s.expect(TokenKind.IntLiteral, "n").value, 10); break;
68
- default: throw new ParseError(`Unexpected in store: ${tok.kind}`, tok.loc);
69
- }
70
- }
71
- s.expect(TokenKind.RBrace, "}");
72
- return node;
73
- }
74
-
75
- // ─── Event ───────────────────────────────────────────────────────────────────
76
-
77
- export function parseEventDecl(s: TokenStream): AST.EventDeclNode {
78
- const loc = s.peek().loc;
79
- s.expect(TokenKind.KwEvent, "event");
80
- const name = s.expect(TokenKind.Identifier, "name").value;
81
- s.expect(TokenKind.LBrace, "{");
82
-
83
- const node: AST.EventDeclNode = {
84
- kind: "EventDecl", loc, name, payload: [], delivery: null, ttl: null,
85
- };
86
-
87
- while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
88
- const tok = s.peek();
89
- switch (tok.kind) {
90
- case TokenKind.KwPayload:
91
- s.advance(); s.expect(TokenKind.Colon, ":");
92
- s.expect(TokenKind.LBrace, "{");
93
- node.payload = parseFieldList(s);
94
- s.expect(TokenKind.RBrace, "}");
95
- break;
96
- case TokenKind.KwDelivery: s.advance(); s.expect(TokenKind.Colon, ":"); node.delivery = s.advance().value; break;
97
- case TokenKind.KwTtl: s.advance(); s.expect(TokenKind.Colon, ":"); node.ttl = parseDuration(s); break;
98
- default: throw new ParseError(`Unexpected in event: ${tok.kind}`, tok.loc);
99
- }
100
- }
101
- s.expect(TokenKind.RBrace, "}");
102
- return node;
103
- }
104
-
105
- // ─── Constraint ──────────────────────────────────────────────────────────────
106
-
107
- export function parseConstraintDecl(s: TokenStream): AST.ConstraintDeclNode {
108
- const loc = s.peek().loc;
109
- s.expect(TokenKind.KwConstraint, "constraint");
110
- const name = s.expect(TokenKind.Identifier, "name").value;
111
- s.expect(TokenKind.Colon, ":");
112
- const expr = parseExpr(s);
113
- return { kind: "ConstraintDecl", loc, name, expr };
114
- }
115
-
116
- // ─── Policy ──────────────────────────────────────────────────────────────────
117
-
118
- export function parsePolicyDecl(s: TokenStream): AST.PolicyDeclNode {
119
- const loc = s.peek().loc;
120
- s.expect(TokenKind.KwPolicy, "policy");
121
- const name = s.expect(TokenKind.Identifier, "name").value;
122
- s.expect(TokenKind.LBrace, "{");
123
-
124
- const node: AST.PolicyDeclNode = {
125
- kind: "PolicyDecl", loc, name, rateLimit: null, access: [], audit: null, encryption: null,
126
- };
127
-
128
- while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
129
- const tok = s.peek();
130
- switch (tok.kind) {
131
- case TokenKind.KwRateLimit:
132
- s.advance(); s.expect(TokenKind.Colon, ":");
133
- const count = parseInt(s.expect(TokenKind.IntLiteral, "count").value, 10);
134
- s.expect(TokenKind.KwPer, "per");
135
- const per = parseDuration(s);
136
- node.rateLimit = { count, per };
137
- break;
138
- case TokenKind.KwAccess:
139
- s.advance(); s.expect(TokenKind.Colon, ":");
140
- s.expect(TokenKind.LBracket, "[");
141
- node.access = parseIdentList(s);
142
- s.expect(TokenKind.RBracket, "]");
143
- break;
144
- case TokenKind.KwAudit: s.advance(); s.expect(TokenKind.Colon, ":"); node.audit = s.advance().kind === TokenKind.KwTrue; break;
145
- case TokenKind.KwEncryption: s.advance(); s.expect(TokenKind.Colon, ":"); node.encryption = s.advance().value; break;
146
- default: throw new ParseError(`Unexpected in policy: ${tok.kind}`, tok.loc);
147
- }
148
- }
149
- s.expect(TokenKind.RBrace, "}");
150
- return node;
151
- }
152
-
153
- // ─── Flow ────────────────────────────────────────────────────────────────────
154
-
155
- export function parseFlowDecl(s: TokenStream): AST.FlowDeclNode {
156
- const loc = s.peek().loc;
157
- s.expect(TokenKind.KwFlow, "flow");
158
- const name = s.expect(TokenKind.Identifier, "name").value;
159
- s.expect(TokenKind.LBrace, "{");
160
-
161
- const steps: AST.FlowStepNode[] = [];
162
- while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
163
- const sloc = s.peek().loc;
164
- s.expect(TokenKind.KwStep, "step");
165
- const stepName = s.expect(TokenKind.Identifier, "step name").value;
166
- s.expect(TokenKind.Colon, ":");
167
- const action = parseCallExpr(s);
168
- let compensate: AST.CallExprNode | null = null;
169
- if (s.check(TokenKind.KwCompensate)) {
170
- s.advance(); s.expect(TokenKind.Colon, ":");
171
- compensate = parseCallExpr(s);
172
- }
173
- steps.push({ kind: "FlowStep", loc: sloc, name: stepName, action, compensate });
174
- }
175
- s.expect(TokenKind.RBrace, "}");
176
- return { kind: "FlowDecl", loc, name, steps };
177
- }
178
-
179
- function parseCallExpr(s: TokenStream): AST.CallExprNode {
180
- const loc = s.peek().loc;
181
- const name = s.expect(TokenKind.Identifier, "call name").value;
182
- s.expect(TokenKind.LParen, "(");
183
- const args: AST.ExprNode[] = [];
184
- if (!s.check(TokenKind.RParen)) {
185
- do { args.push(parseExpr(s)); } while (s.match(TokenKind.Comma));
186
- }
187
- s.expect(TokenKind.RParen, ")");
188
- return { kind: "CallExpr", loc, name, args };
189
- }
190
-
191
- // ─── Import ──────────────────────────────────────────────────────────────────
192
-
193
- export function parseImportDecl(s: TokenStream): AST.ImportDeclNode {
194
- const loc = s.peek().loc;
195
- s.expect(TokenKind.KwImport, "import");
196
- const name = s.expect(TokenKind.Identifier, "name").value;
197
- s.expect(TokenKind.KwFrom, "from");
198
- const from = s.expect(TokenKind.StringLiteral, "path").value;
199
- return { kind: "ImportDecl", loc, name, from };
200
- }
201
-
202
- // ─── Extension Point ─────────────────────────────────────────────────────────
203
-
204
- export function parseExtensionPointDecl(s: TokenStream): AST.ExtensionPointDeclNode {
205
- const loc = s.peek().loc;
206
- s.expect(TokenKind.KwExtensionPoint, "extension_point");
207
- const name = s.expect(TokenKind.Identifier, "extension point name").value;
208
- s.expect(TokenKind.LParen, "(");
209
- const params: AST.ParamNode[] = [];
210
- if (!s.check(TokenKind.RParen)) {
211
- do {
212
- const ploc = s.peek().loc;
213
- // Allow keywords as param names
214
- const pname = s.peek().kind === TokenKind.Identifier ? s.advance().value : s.advance().value;
215
- s.expect(TokenKind.Colon, ":");
216
- const ptype = parseTypeExpr(s);
217
- params.push({ kind: "Param", loc: ploc, name: pname, type: ptype });
218
- } while (s.match(TokenKind.Comma));
219
- }
220
- s.expect(TokenKind.RParen, ")");
221
-
222
- let returns: AST.TypeExprNode | null = null;
223
- let stable = false;
224
-
225
- s.expect(TokenKind.LBrace, "{");
226
- while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
227
- const tok = s.peek();
228
- if (tok.kind === TokenKind.KwReturns) {
229
- s.advance(); s.expect(TokenKind.Colon, ":");
230
- returns = parseTypeExpr(s);
231
- } else if (tok.kind === TokenKind.KwStable) {
232
- s.advance(); s.expect(TokenKind.Colon, ":");
233
- stable = s.advance().kind === TokenKind.KwTrue;
234
- } else if (tok.kind === TokenKind.KwLanguage) {
235
- // language: typescript — consume and ignore (only TS supported)
236
- s.advance(); s.expect(TokenKind.Colon, ":"); s.advance();
237
- } else {
238
- throw new ParseError(`Unexpected in extension_point: ${tok.kind}`, tok.loc);
239
- }
240
- }
241
- s.expect(TokenKind.RBrace, "}");
242
-
243
- return { kind: "ExtensionPointDecl", loc, name, params, returns, stable };
244
- }
1
+ /**
2
+ * BoneScript Declaration Parsers — Channel, Store, Event, Constraint, Policy, Flow, Import
3
+ */
4
+
5
+ import { TokenKind } from "./lexer";
6
+ import { TokenStream, ParseError } from "./parser_base";
7
+ import * as AST from "./ast";
8
+ import { parseExpr } from "./parse_expr";
9
+ import { parseTypeExpr } from "./parse_types";
10
+ import { parseFieldList, parseDuration, parseIdentList } from "./parse_decls";
11
+
12
+ // ─── Channel ─────────────────────────────────────────────────────────────────
13
+
14
+ export function parseChannelDecl(s: TokenStream): AST.ChannelDeclNode {
15
+ const loc = s.peek().loc;
16
+ s.expect(TokenKind.KwChannel, "channel");
17
+ const name = s.expect(TokenKind.Identifier, "name").value;
18
+ s.expect(TokenKind.LBrace, "{");
19
+
20
+ const node: AST.ChannelDeclNode = {
21
+ kind: "ChannelDecl", loc, name,
22
+ transport: null, ordering: null, participants: null,
23
+ persistence: null, filter: null, maxSize: null,
24
+ };
25
+
26
+ while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
27
+ const tok = s.peek();
28
+ switch (tok.kind) {
29
+ case TokenKind.KwTransport: s.advance(); s.expect(TokenKind.Colon, ":"); node.transport = s.advance().value; break;
30
+ case TokenKind.KwOrdering: s.advance(); s.expect(TokenKind.Colon, ":"); node.ordering = s.advance().value; break;
31
+ case TokenKind.KwParticipants: s.advance(); s.expect(TokenKind.Colon, ":"); node.participants = parseTypeExpr(s); break;
32
+ case TokenKind.KwPersistence: s.advance(); s.expect(TokenKind.Colon, ":"); node.persistence = s.advance().value; break;
33
+ case TokenKind.KwFilter: s.advance(); s.expect(TokenKind.Colon, ":"); node.filter = parseExpr(s); break;
34
+ case TokenKind.KwMaxSize: s.advance(); s.expect(TokenKind.Colon, ":"); node.maxSize = parseInt(s.expect(TokenKind.IntLiteral, "n").value, 10); break;
35
+ default: throw new ParseError(`Unexpected in channel: ${tok.kind}`, tok.loc);
36
+ }
37
+ }
38
+ s.expect(TokenKind.RBrace, "}");
39
+ return node;
40
+ }
41
+
42
+ // ─── Store ───────────────────────────────────────────────────────────────────
43
+
44
+ export function parseStoreDecl(s: TokenStream): AST.StoreDeclNode {
45
+ const loc = s.peek().loc;
46
+ s.expect(TokenKind.KwStore, "store");
47
+ const name = s.expect(TokenKind.Identifier, "name").value;
48
+ s.expect(TokenKind.LBrace, "{");
49
+
50
+ const node: AST.StoreDeclNode = {
51
+ kind: "StoreDecl", loc, name,
52
+ engine: null, schema: [], retention: null, partition: null, replicas: null,
53
+ };
54
+
55
+ while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
56
+ const tok = s.peek();
57
+ switch (tok.kind) {
58
+ case TokenKind.KwEngine: s.advance(); s.expect(TokenKind.Colon, ":"); node.engine = s.advance().value; break;
59
+ case TokenKind.KwSchema:
60
+ s.advance(); s.expect(TokenKind.Colon, ":");
61
+ s.expect(TokenKind.LBrace, "{");
62
+ node.schema = parseFieldList(s);
63
+ s.expect(TokenKind.RBrace, "}");
64
+ break;
65
+ case TokenKind.KwRetention: s.advance(); s.expect(TokenKind.Colon, ":"); node.retention = parseDuration(s); break;
66
+ case TokenKind.KwPartition: s.advance(); s.expect(TokenKind.Colon, ":"); node.partition = s.expect(TokenKind.Identifier, "field").value; break;
67
+ case TokenKind.KwReplicas: s.advance(); s.expect(TokenKind.Colon, ":"); node.replicas = parseInt(s.expect(TokenKind.IntLiteral, "n").value, 10); break;
68
+ default: throw new ParseError(`Unexpected in store: ${tok.kind}`, tok.loc);
69
+ }
70
+ }
71
+ s.expect(TokenKind.RBrace, "}");
72
+ return node;
73
+ }
74
+
75
+ // ─── Event ───────────────────────────────────────────────────────────────────
76
+
77
+ export function parseEventDecl(s: TokenStream): AST.EventDeclNode {
78
+ const loc = s.peek().loc;
79
+ s.expect(TokenKind.KwEvent, "event");
80
+ const name = s.expect(TokenKind.Identifier, "name").value;
81
+ s.expect(TokenKind.LBrace, "{");
82
+
83
+ const node: AST.EventDeclNode = {
84
+ kind: "EventDecl", loc, name, payload: [], delivery: null, ttl: null,
85
+ };
86
+
87
+ while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
88
+ const tok = s.peek();
89
+ switch (tok.kind) {
90
+ case TokenKind.KwPayload:
91
+ s.advance(); s.expect(TokenKind.Colon, ":");
92
+ s.expect(TokenKind.LBrace, "{");
93
+ node.payload = parseFieldList(s);
94
+ s.expect(TokenKind.RBrace, "}");
95
+ break;
96
+ case TokenKind.KwDelivery: s.advance(); s.expect(TokenKind.Colon, ":"); node.delivery = s.advance().value; break;
97
+ case TokenKind.KwTtl: s.advance(); s.expect(TokenKind.Colon, ":"); node.ttl = parseDuration(s); break;
98
+ default: throw new ParseError(`Unexpected in event: ${tok.kind}`, tok.loc);
99
+ }
100
+ }
101
+ s.expect(TokenKind.RBrace, "}");
102
+ return node;
103
+ }
104
+
105
+ // ─── Constraint ──────────────────────────────────────────────────────────────
106
+
107
+ export function parseConstraintDecl(s: TokenStream): AST.ConstraintDeclNode {
108
+ const loc = s.peek().loc;
109
+ s.expect(TokenKind.KwConstraint, "constraint");
110
+ const name = s.expect(TokenKind.Identifier, "name").value;
111
+ s.expect(TokenKind.Colon, ":");
112
+ const expr = parseExpr(s);
113
+ return { kind: "ConstraintDecl", loc, name, expr };
114
+ }
115
+
116
+ // ─── Policy ──────────────────────────────────────────────────────────────────
117
+
118
+ export function parsePolicyDecl(s: TokenStream): AST.PolicyDeclNode {
119
+ const loc = s.peek().loc;
120
+ s.expect(TokenKind.KwPolicy, "policy");
121
+ const name = s.expect(TokenKind.Identifier, "name").value;
122
+ s.expect(TokenKind.LBrace, "{");
123
+
124
+ const node: AST.PolicyDeclNode = {
125
+ kind: "PolicyDecl", loc, name, rateLimit: null, access: [], audit: null, encryption: null,
126
+ };
127
+
128
+ while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
129
+ const tok = s.peek();
130
+ switch (tok.kind) {
131
+ case TokenKind.KwRateLimit:
132
+ s.advance(); s.expect(TokenKind.Colon, ":");
133
+ const count = parseInt(s.expect(TokenKind.IntLiteral, "count").value, 10);
134
+ s.expect(TokenKind.KwPer, "per");
135
+ const per = parseDuration(s);
136
+ node.rateLimit = { count, per };
137
+ break;
138
+ case TokenKind.KwAccess:
139
+ s.advance(); s.expect(TokenKind.Colon, ":");
140
+ s.expect(TokenKind.LBracket, "[");
141
+ node.access = parseIdentList(s);
142
+ s.expect(TokenKind.RBracket, "]");
143
+ break;
144
+ case TokenKind.KwAudit: s.advance(); s.expect(TokenKind.Colon, ":"); node.audit = s.advance().kind === TokenKind.KwTrue; break;
145
+ case TokenKind.KwEncryption: s.advance(); s.expect(TokenKind.Colon, ":"); node.encryption = s.advance().value; break;
146
+ default: throw new ParseError(`Unexpected in policy: ${tok.kind}`, tok.loc);
147
+ }
148
+ }
149
+ s.expect(TokenKind.RBrace, "}");
150
+ return node;
151
+ }
152
+
153
+ // ─── Flow ────────────────────────────────────────────────────────────────────
154
+
155
+ export function parseFlowDecl(s: TokenStream): AST.FlowDeclNode {
156
+ const loc = s.peek().loc;
157
+ s.expect(TokenKind.KwFlow, "flow");
158
+ const name = s.expect(TokenKind.Identifier, "name").value;
159
+ s.expect(TokenKind.LBrace, "{");
160
+
161
+ const steps: AST.FlowStepNode[] = [];
162
+ while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
163
+ const sloc = s.peek().loc;
164
+ s.expect(TokenKind.KwStep, "step");
165
+ const stepName = s.expect(TokenKind.Identifier, "step name").value;
166
+ s.expect(TokenKind.Colon, ":");
167
+ const action = parseCallExpr(s);
168
+ let compensate: AST.CallExprNode | null = null;
169
+ if (s.check(TokenKind.KwCompensate)) {
170
+ s.advance(); s.expect(TokenKind.Colon, ":");
171
+ compensate = parseCallExpr(s);
172
+ }
173
+ steps.push({ kind: "FlowStep", loc: sloc, name: stepName, action, compensate });
174
+ }
175
+ s.expect(TokenKind.RBrace, "}");
176
+ return { kind: "FlowDecl", loc, name, steps };
177
+ }
178
+
179
+ function parseCallExpr(s: TokenStream): AST.CallExprNode {
180
+ const loc = s.peek().loc;
181
+ const name = s.expect(TokenKind.Identifier, "call name").value;
182
+ s.expect(TokenKind.LParen, "(");
183
+ const args: AST.ExprNode[] = [];
184
+ if (!s.check(TokenKind.RParen)) {
185
+ do { args.push(parseExpr(s)); } while (s.match(TokenKind.Comma));
186
+ }
187
+ s.expect(TokenKind.RParen, ")");
188
+ return { kind: "CallExpr", loc, name, args };
189
+ }
190
+
191
+ // ─── Import ──────────────────────────────────────────────────────────────────
192
+
193
+ export function parseImportDecl(s: TokenStream): AST.ImportDeclNode {
194
+ const loc = s.peek().loc;
195
+ s.expect(TokenKind.KwImport, "import");
196
+ const name = s.expect(TokenKind.Identifier, "name").value;
197
+ s.expect(TokenKind.KwFrom, "from");
198
+ const from = s.expect(TokenKind.StringLiteral, "path").value;
199
+ return { kind: "ImportDecl", loc, name, from };
200
+ }
201
+
202
+ // ─── Extension Point ─────────────────────────────────────────────────────────
203
+
204
+ export function parseExtensionPointDecl(s: TokenStream): AST.ExtensionPointDeclNode {
205
+ const loc = s.peek().loc;
206
+ s.expect(TokenKind.KwExtensionPoint, "extension_point");
207
+ const name = s.expect(TokenKind.Identifier, "extension point name").value;
208
+ s.expect(TokenKind.LParen, "(");
209
+ const params: AST.ParamNode[] = [];
210
+ if (!s.check(TokenKind.RParen)) {
211
+ do {
212
+ const ploc = s.peek().loc;
213
+ // Allow keywords as param names
214
+ const pname = s.peek().kind === TokenKind.Identifier ? s.advance().value : s.advance().value;
215
+ s.expect(TokenKind.Colon, ":");
216
+ const ptype = parseTypeExpr(s);
217
+ params.push({ kind: "Param", loc: ploc, name: pname, type: ptype });
218
+ } while (s.match(TokenKind.Comma));
219
+ }
220
+ s.expect(TokenKind.RParen, ")");
221
+
222
+ let returns: AST.TypeExprNode | null = null;
223
+ let stable = false;
224
+
225
+ s.expect(TokenKind.LBrace, "{");
226
+ while (!s.check(TokenKind.RBrace) && !s.check(TokenKind.EOF)) {
227
+ const tok = s.peek();
228
+ if (tok.kind === TokenKind.KwReturns) {
229
+ s.advance(); s.expect(TokenKind.Colon, ":");
230
+ returns = parseTypeExpr(s);
231
+ } else if (tok.kind === TokenKind.KwStable) {
232
+ s.advance(); s.expect(TokenKind.Colon, ":");
233
+ stable = s.advance().kind === TokenKind.KwTrue;
234
+ } else if (tok.kind === TokenKind.KwLanguage) {
235
+ // language: typescript — consume and ignore (only TS supported)
236
+ s.advance(); s.expect(TokenKind.Colon, ":"); s.advance();
237
+ } else {
238
+ throw new ParseError(`Unexpected in extension_point: ${tok.kind}`, tok.loc);
239
+ }
240
+ }
241
+ s.expect(TokenKind.RBrace, "}");
242
+
243
+ return { kind: "ExtensionPointDecl", loc, name, params, returns, stable };
244
+ }
@@ -0,0 +1,7 @@
1
+ /**
2
+ * BoneScript Expression Parser — Pratt-style precedence climbing.
3
+ */
4
+ import { TokenStream } from "./parser_base";
5
+ import * as AST from "./ast";
6
+ export declare function parseExpr(s: TokenStream): AST.ExprNode;
7
+ export declare function parseExprList(s: TokenStream): AST.ExprNode[];