bonescript-compiler 0.5.3 → 0.5.5

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 (201) 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 +617 -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_nakama.d.ts +23 -0
  22. package/dist/emit_nakama.js +510 -0
  23. package/dist/emit_nakama.js.map +1 -0
  24. package/dist/emit_runtime.d.ts +11 -17
  25. package/dist/emit_runtime.js +688 -29
  26. package/dist/emit_runtime.js.map +1 -1
  27. package/dist/emit_sourcemap.js +66 -66
  28. package/dist/emit_tests.js +12 -47
  29. package/dist/emit_tests.js.map +1 -1
  30. package/dist/emit_websocket.js +3 -0
  31. package/dist/emit_websocket.js.map +1 -1
  32. package/dist/emitter.js +49 -94
  33. package/dist/emitter.js.map +1 -1
  34. package/dist/extension_manager.d.ts +2 -2
  35. package/dist/extension_manager.js +20 -9
  36. package/dist/extension_manager.js.map +1 -1
  37. package/dist/index.d.ts +2 -0
  38. package/dist/index.js +3 -1
  39. package/dist/index.js.map +1 -1
  40. package/dist/ir.d.ts +0 -4
  41. package/dist/lowering.d.ts +14 -5
  42. package/dist/lowering.js +417 -66
  43. package/dist/lowering.js.map +1 -1
  44. package/dist/module_loader.d.ts +2 -2
  45. package/dist/module_loader.js +23 -20
  46. package/dist/module_loader.js.map +1 -1
  47. package/dist/optimizer.js +3 -6
  48. package/dist/optimizer.js.map +1 -1
  49. package/dist/scaffold.d.ts +2 -2
  50. package/dist/scaffold.js +319 -315
  51. package/dist/scaffold.js.map +1 -1
  52. package/dist/solver.js +1 -1
  53. package/dist/solver.js.map +1 -1
  54. package/dist/source_map.js.map +1 -0
  55. package/dist/test.js.map +1 -0
  56. package/dist/test_typechecker.d.ts +5 -0
  57. package/dist/test_typechecker.js +126 -0
  58. package/dist/test_typechecker.js.map +1 -0
  59. package/dist/typechecker.d.ts +0 -7
  60. package/dist/typechecker.js +16 -103
  61. package/dist/typechecker.js.map +1 -1
  62. package/dist/verifier.d.ts +1 -5
  63. package/dist/verifier.js +38 -142
  64. package/dist/verifier.js.map +1 -1
  65. package/package.json +53 -62
  66. package/src/algorithm_catalog.ts +345 -345
  67. package/src/ast.d.ts +244 -0
  68. package/src/ast.ts +334 -334
  69. package/src/cli.ts +704 -98
  70. package/src/emit_batch.ts +140 -140
  71. package/src/emit_capability.ts +436 -613
  72. package/src/emit_composition.ts +196 -229
  73. package/src/emit_deploy.ts +190 -187
  74. package/src/emit_events.ts +307 -362
  75. package/src/emit_extras.ts +240 -237
  76. package/src/emit_full.ts +309 -472
  77. package/src/emit_maintenance.ts +459 -459
  78. package/src/emit_nakama.ts +576 -0
  79. package/src/emit_runtime.ts +730 -17
  80. package/src/emit_sourcemap.ts +140 -140
  81. package/src/emit_tests.ts +205 -243
  82. package/src/emit_websocket.ts +229 -226
  83. package/src/emitter.ts +578 -626
  84. package/src/extension_manager.ts +187 -177
  85. package/src/formatter.ts +297 -297
  86. package/src/index.ts +90 -88
  87. package/src/ir.ts +215 -216
  88. package/src/lexer.d.ts +195 -0
  89. package/src/lexer.ts +630 -630
  90. package/src/lowering.ts +556 -168
  91. package/src/module_loader.ts +114 -112
  92. package/src/optimizer.ts +196 -199
  93. package/src/parse_decls.d.ts +13 -0
  94. package/src/parse_decls.ts +409 -409
  95. package/src/parse_decls2.d.ts +13 -0
  96. package/src/parse_decls2.ts +244 -244
  97. package/src/parse_expr.d.ts +7 -0
  98. package/src/parse_expr.ts +197 -197
  99. package/src/parse_types.d.ts +6 -0
  100. package/src/parse_types.ts +54 -54
  101. package/src/parser.d.ts +10 -0
  102. package/src/parser.ts +1 -1
  103. package/src/parser_base.d.ts +19 -0
  104. package/src/parser_base.ts +57 -57
  105. package/src/parser_recovery.ts +153 -153
  106. package/src/scaffold.ts +375 -371
  107. package/src/solver.ts +330 -330
  108. package/src/typechecker.d.ts +52 -0
  109. package/src/typechecker.ts +591 -700
  110. package/src/types.d.ts +38 -0
  111. package/src/types.ts +122 -122
  112. package/src/verifier.ts +49 -154
  113. package/README.md +0 -382
  114. package/dist/commands/check.d.ts +0 -5
  115. package/dist/commands/check.js +0 -34
  116. package/dist/commands/check.js.map +0 -1
  117. package/dist/commands/compile.d.ts +0 -5
  118. package/dist/commands/compile.js +0 -215
  119. package/dist/commands/compile.js.map +0 -1
  120. package/dist/commands/debug.d.ts +0 -5
  121. package/dist/commands/debug.js +0 -59
  122. package/dist/commands/debug.js.map +0 -1
  123. package/dist/commands/diff.d.ts +0 -5
  124. package/dist/commands/diff.js +0 -123
  125. package/dist/commands/diff.js.map +0 -1
  126. package/dist/commands/fmt.d.ts +0 -5
  127. package/dist/commands/fmt.js +0 -49
  128. package/dist/commands/fmt.js.map +0 -1
  129. package/dist/commands/init.d.ts +0 -5
  130. package/dist/commands/init.js +0 -96
  131. package/dist/commands/init.js.map +0 -1
  132. package/dist/commands/ir.d.ts +0 -5
  133. package/dist/commands/ir.js +0 -27
  134. package/dist/commands/ir.js.map +0 -1
  135. package/dist/commands/lex.d.ts +0 -5
  136. package/dist/commands/lex.js +0 -21
  137. package/dist/commands/lex.js.map +0 -1
  138. package/dist/commands/parse.d.ts +0 -5
  139. package/dist/commands/parse.js +0 -30
  140. package/dist/commands/parse.js.map +0 -1
  141. package/dist/commands/test.d.ts +0 -5
  142. package/dist/commands/test.js +0 -61
  143. package/dist/commands/test.js.map +0 -1
  144. package/dist/commands/verify_determinism.d.ts +0 -5
  145. package/dist/commands/verify_determinism.js +0 -64
  146. package/dist/commands/verify_determinism.js.map +0 -1
  147. package/dist/commands/watch.d.ts +0 -5
  148. package/dist/commands/watch.js +0 -50
  149. package/dist/commands/watch.js.map +0 -1
  150. package/dist/emit_auth.d.ts +0 -18
  151. package/dist/emit_auth.js +0 -507
  152. package/dist/emit_auth.js.map +0 -1
  153. package/dist/emit_database.d.ts +0 -7
  154. package/dist/emit_database.js +0 -72
  155. package/dist/emit_database.js.map +0 -1
  156. package/dist/emit_index.d.ts +0 -6
  157. package/dist/emit_index.js +0 -202
  158. package/dist/emit_index.js.map +0 -1
  159. package/dist/emit_models.d.ts +0 -12
  160. package/dist/emit_models.js +0 -171
  161. package/dist/emit_models.js.map +0 -1
  162. package/dist/emit_openapi.d.ts +0 -9
  163. package/dist/emit_openapi.js +0 -306
  164. package/dist/emit_openapi.js.map +0 -1
  165. package/dist/emit_package.d.ts +0 -7
  166. package/dist/emit_package.js +0 -68
  167. package/dist/emit_package.js.map +0 -1
  168. package/dist/emit_router.d.ts +0 -12
  169. package/dist/emit_router.js +0 -389
  170. package/dist/emit_router.js.map +0 -1
  171. package/dist/lowering_channels.d.ts +0 -11
  172. package/dist/lowering_channels.js +0 -103
  173. package/dist/lowering_channels.js.map +0 -1
  174. package/dist/lowering_entities.d.ts +0 -11
  175. package/dist/lowering_entities.js +0 -232
  176. package/dist/lowering_entities.js.map +0 -1
  177. package/dist/lowering_helpers.d.ts +0 -13
  178. package/dist/lowering_helpers.js +0 -76
  179. package/dist/lowering_helpers.js.map +0 -1
  180. package/src/commands/check.ts +0 -33
  181. package/src/commands/compile.ts +0 -191
  182. package/src/commands/debug.ts +0 -33
  183. package/src/commands/diff.ts +0 -105
  184. package/src/commands/fmt.ts +0 -22
  185. package/src/commands/init.ts +0 -72
  186. package/src/commands/ir.ts +0 -23
  187. package/src/commands/lex.ts +0 -17
  188. package/src/commands/parse.ts +0 -24
  189. package/src/commands/test.ts +0 -36
  190. package/src/commands/verify_determinism.ts +0 -66
  191. package/src/commands/watch.ts +0 -25
  192. package/src/emit_auth.ts +0 -513
  193. package/src/emit_database.ts +0 -72
  194. package/src/emit_index.ts +0 -210
  195. package/src/emit_models.ts +0 -176
  196. package/src/emit_openapi.ts +0 -315
  197. package/src/emit_package.ts +0 -66
  198. package/src/emit_router.ts +0 -408
  199. package/src/lowering_channels.ts +0 -108
  200. package/src/lowering_entities.ts +0 -258
  201. package/src/lowering_helpers.ts +0 -75
@@ -1,613 +1,436 @@
1
- /**
2
- * BoneScript Capability Body Emitter
3
- *
4
- * Translates IR effects and preconditions into real TypeScript + SQL.
5
- *
6
- * Performance strategies applied (PERF-003):
7
- *
8
- * 1. Entity fetches — same-table fetches batched into WHERE id = ANY($1::uuid[])
9
- * and resolved with a Map lookup. Different-table fetches run in parallel via
10
- * Promise.all rather than sequentially.
11
- *
12
- * 2. Effect batching — multiple effects targeting the same entity+table are
13
- * collapsed into a single UPDATE ... SET a=$1, b=$2 ... WHERE id = $n
14
- * instead of one UPDATE per field.
15
- *
16
- * 3. LIST queries combined with COUNT(*) OVER() window function to avoid a
17
- * separate COUNT(*) round-trip (handled in emit_router.ts).
18
- */
19
-
20
- import * as IR from "./ir";
21
- import { toSnakeCase } from "./lowering_helpers";
22
-
23
-
24
- type ExprKind =
25
- | { kind: "literal"; value: string; raw: string }
26
- | { kind: "field"; path: string[] }
27
- | { kind: "binop"; op: string; left: Expr; right: Expr }
28
- | { kind: "call"; name: string; args: Expr[] };
29
-
30
- type Expr = ExprKind;
31
-
32
- function parseExprStr(s: string): Expr {
33
- s = s.trim();
34
- if (s.startsWith("(") && s.endsWith(")")) return parseExprStr(s.slice(1, -1));
35
- if (s.startsWith('"') && s.endsWith('"')) return { kind: "literal", value: s.slice(1, -1), raw: s };
36
- if (/^-?\d+(\.\d+)?$/.test(s)) return { kind: "literal", value: s, raw: s };
37
- if (s === "true" || s === "false") return { kind: "literal", value: s, raw: s };
38
-
39
- const binOps = [" or ", " and ", " == ", " != ", " >= ", " <= ", " > ", " < ", " in ", " contains ", " + ", " - ", " * ", " / "];
40
- for (const op of binOps) {
41
- const idx = findBinOp(s, op);
42
- if (idx !== -1) {
43
- return { kind: "binop", op: op.trim(), left: parseExprStr(s.slice(0, idx)), right: parseExprStr(s.slice(idx + op.length)) };
44
- }
45
- }
46
-
47
- const callMatch = s.match(/^(\w+)\((.*)?\)$/);
48
- if (callMatch) {
49
- const args = callMatch[2] ? splitArgs(callMatch[2]).map(parseExprStr) : [];
50
- return { kind: "call", name: callMatch[1], args };
51
- }
52
-
53
- if (/^[\w.]+$/.test(s)) return { kind: "field", path: s.split(".") };
54
- return { kind: "literal", value: s, raw: s };
55
- }
56
-
57
- function findBinOp(s: string, op: string): number {
58
- let depth = 0;
59
- for (let i = 0; i <= s.length - op.length; i++) {
60
- const ch = s[i];
61
- if (ch === "(" || ch === "[") depth++;
62
- else if (ch === ")" || ch === "]") depth--;
63
- else if (depth === 0 && s.slice(i, i + op.length) === op) return i;
64
- }
65
- return -1;
66
- }
67
-
68
- function splitArgs(s: string): string[] {
69
- const args: string[] = [];
70
- let depth = 0;
71
- let current = "";
72
- for (const ch of s) {
73
- if (ch === "(" || ch === "[") depth++;
74
- else if (ch === ")" || ch === "]") depth--;
75
- else if (ch === "," && depth === 0) { args.push(current.trim()); current = ""; continue; }
76
- current += ch;
77
- }
78
- if (current.trim()) args.push(current.trim());
79
- return args;
80
- }
81
-
82
- // ─── Entity Resolution ────────────────────────────────────────────────────────
83
-
84
- interface EntityFetch {
85
- paramName: string;
86
- entityType: string;
87
- tableName: string;
88
- idField: string;
89
- }
90
-
91
- function getEntityFetches(method: IR.IRMethod, mod: IR.IRModule, system: IR.IRSystem): EntityFetch[] {
92
- const fetches: EntityFetch[] = [];
93
- const seen = new Set<string>();
94
-
95
- const allModels = new Map<string, string>();
96
- for (const m of system.modules) {
97
- for (const model of m.models) {
98
- allModels.set(model.name, toSnakeCase(model.name) + "s");
99
- allModels.set(model.name.toLowerCase(), toSnakeCase(model.name) + "s");
100
- }
101
- }
102
-
103
- for (const param of method.input) {
104
- const tableName = allModels.get(param.type) || allModels.get(param.type.toLowerCase());
105
- if (tableName && !seen.has(param.name)) {
106
- seen.add(param.name);
107
- fetches.push({ paramName: param.name, entityType: param.type, tableName, idField: param.name + "_id" });
108
- }
109
- }
110
-
111
- return fetches;
112
- }
113
-
114
- // ─── Precondition Compiler ────────────────────────────────────────────────────
115
-
116
- function compilePrecondition(expr: Expr, indent: string): string {
117
- const condition = exprToTs(expr, true);
118
- const description = exprToDescription(expr).replace(/"/g, '\\"');
119
- return [
120
- `${indent}if (${condition}) {`,
121
- `${indent} return res.status(422).json({ error: { code: "PRECONDITION_FAILED", message: ${JSON.stringify(description)} } });`,
122
- `${indent}}`,
123
- ].join("\n");
124
- }
125
-
126
- function exprToTs(expr: Expr, negate = false): string {
127
- const inner = exprToTsInner(expr);
128
- return negate ? `!(${inner})` : inner;
129
- }
130
-
131
- function exprToTsInner(expr: Expr): string {
132
- switch (expr.kind) {
133
- case "literal":
134
- if (expr.value === "true") return "true";
135
- if (expr.value === "false") return "false";
136
- if (/^"/.test(expr.raw)) return expr.raw;
137
- return expr.value;
138
- case "field":
139
- return expr.path.join("?.");
140
- case "binop": {
141
- const l = exprToTsInner(expr.left);
142
- const r = exprToTsInner(expr.right);
143
- switch (expr.op) {
144
- case "==": return `${l} === ${r}`;
145
- case "!=": return `${l} !== ${r}`;
146
- case "and": return `(${l} && ${r})`;
147
- case "or": return `(${l} || ${r})`;
148
- case "in": return `[${r}].flat().includes(${l})`;
149
- case "contains": return `${l}?.includes(${r})`;
150
- default: return `${l} ${expr.op} ${r}`;
151
- }
152
- }
153
- case "call":
154
- if (expr.name === "now") return "new Date()";
155
- return `${expr.name}(${expr.args.map(exprToTsInner).join(", ")})`;
156
- }
157
- }
158
-
159
- function exprToDescription(expr: Expr): string {
160
- switch (expr.kind) {
161
- case "literal": return expr.raw;
162
- case "field": return expr.path.join(".");
163
- case "binop": return `${exprToDescription(expr.left)} ${expr.op} ${exprToDescription(expr.right)}`;
164
- case "call": return `${expr.name}(${expr.args.map(exprToDescription).join(", ")})`;
165
- }
166
- }
167
-
168
- // ─── Effect Compiler ──────────────────────────────────────────────────────────
169
-
170
- interface CompiledEffect {
171
- tableName: string;
172
- entityParam: string;
173
- idParam: string;
174
- // For batched UPDATE: list of (column, paramPlaceholder, tsValue) tuples
175
- assignments: { column: string; placeholder: string; tsValue: string }[];
176
- description: string;
177
- // For non-batchable effects (JSONB, array ops) — emitted as standalone query
178
- standalone?: { sql: string; params: string[] };
179
- }
180
-
181
- /**
182
- * Compile a single effect into a structured form.
183
- * Returns null if the effect target can't be resolved.
184
- */
185
- function compileEffect(
186
- effect: IR.IREffect,
187
- mod: IR.IRModule,
188
- system: IR.IRSystem,
189
- paramIdx: { n: number },
190
- method?: IR.IRMethod,
191
- ): CompiledEffect | null {
192
- const targetParts = effect.target.split(".");
193
- if (targetParts.length < 2) return null;
194
-
195
- const entityParam = targetParts[0];
196
- const fieldName = targetParts[1];
197
- const nestedPath = targetParts.slice(2);
198
-
199
- // Resolve the model: first try matching by param name → entity type via method inputs,
200
- // then fall back to matching by entity name directly.
201
- const model = (() => {
202
- // If we have method context, resolve the param name to its entity type
203
- if (method) {
204
- const param = method.input.find(p => p.name === entityParam);
205
- if (param) {
206
- for (const m of system.modules) {
207
- const found = m.models.find(mdl => mdl.name === param.type || mdl.name.toLowerCase() === param.type.toLowerCase());
208
- if (found) return found;
209
- }
210
- }
211
- }
212
- // Fall back to matching by entity name directly
213
- for (const m of system.modules) {
214
- const found = m.models.find(mdl =>
215
- toSnakeCase(mdl.name) === entityParam || mdl.name.toLowerCase() === entityParam.toLowerCase()
216
- );
217
- if (found) return found;
218
- }
219
- return mod.models.find(m =>
220
- toSnakeCase(m.name) === entityParam || m.name.toLowerCase() === entityParam.toLowerCase()
221
- );
222
- })();
223
- if (!model) return null;
224
-
225
- const tableName = toSnakeCase(model.name) + "s";
226
- const valueTs = exprToTsInner(parseExprStr(effect.value));
227
- const idParam = `req.body.${entityParam}_id || req.params.id`;
228
-
229
- // JSONB nested path — must be standalone (jsonb_set can't be batched cleanly)
230
- if (nestedPath.length > 0) {
231
- const p1 = `$${paramIdx.n++}`;
232
- const p2 = `$${paramIdx.n++}`;
233
- const jsonbPathLiteral = `'{${nestedPath.join(",")}}'`;
234
- // Use to_jsonb($1) directly casting via ::text loses type information for
235
- // non-string values (numbers, booleans, objects). to_jsonb() handles all
236
- // PostgreSQL types correctly without an intermediate text cast.
237
- return {
238
- tableName, entityParam, idParam,
239
- assignments: [],
240
- description: `${effect.target} = ${effect.value}`,
241
- standalone: {
242
- sql: `UPDATE ${tableName} SET ${fieldName} = jsonb_set(COALESCE(${fieldName}, '{}'), ${jsonbPathLiteral}, to_jsonb(${p1}), true), updated_at = NOW() WHERE id = ${p2} RETURNING *`,
243
- params: [valueTs, idParam],
244
- },
245
- };
246
- }
247
-
248
- const fieldType = model.fields.find(f => f.name === fieldName)?.type || "";
249
- const isNumeric = ["uint", "int", "float"].includes(fieldType);
250
-
251
- switch (effect.op) {
252
- case "assign": {
253
- const p1 = `$${paramIdx.n++}`;
254
- return {
255
- tableName, entityParam, idParam,
256
- assignments: [{ column: fieldName, placeholder: p1, tsValue: valueTs }],
257
- description: `${effect.target} = ${effect.value}`,
258
- };
259
- }
260
- case "add": {
261
- const p1 = `$${paramIdx.n++}`;
262
- if (isNumeric) {
263
- return {
264
- tableName, entityParam, idParam,
265
- assignments: [{ column: `${fieldName} = ${fieldName} + `, placeholder: p1, tsValue: valueTs }],
266
- description: `${effect.target} += ${effect.value}`,
267
- };
268
- }
269
- // Array append — standalone
270
- const p2 = `$${paramIdx.n++}`;
271
- return {
272
- tableName, entityParam, idParam,
273
- assignments: [],
274
- description: `${effect.target} += ${effect.value}`,
275
- standalone: {
276
- sql: `UPDATE ${tableName} SET ${fieldName} = ${fieldName} || jsonb_build_array(${p1}::jsonb), updated_at = NOW() WHERE id = ${p2} RETURNING *`,
277
- params: [valueTs, idParam],
278
- },
279
- };
280
- }
281
- case "remove": {
282
- const p1 = `$${paramIdx.n++}`;
283
- if (isNumeric) {
284
- return {
285
- tableName, entityParam, idParam,
286
- assignments: [{ column: `${fieldName} = ${fieldName} - `, placeholder: p1, tsValue: valueTs }],
287
- description: `${effect.target} -= ${effect.value}`,
288
- };
289
- }
290
- // Array remove — standalone
291
- const p2 = `$${paramIdx.n++}`;
292
- return {
293
- tableName, entityParam, idParam,
294
- assignments: [],
295
- description: `${effect.target} -= ${effect.value}`,
296
- standalone: {
297
- sql: `UPDATE ${tableName} SET ${fieldName} = (SELECT jsonb_agg(elem) FROM jsonb_array_elements(${fieldName}) elem WHERE elem != ${p1}::jsonb), updated_at = NOW() WHERE id = ${p2} RETURNING *`,
298
- params: [valueTs, idParam],
299
- },
300
- };
301
- }
302
- }
303
- }
304
-
305
- // ─── Effect Batching ──────────────────────────────────────────────────────────
306
- // Groups effects targeting the same (tableName, entityParam) into a single UPDATE.
307
-
308
- interface BatchedUpdate {
309
- tableName: string;
310
- entityParam: string;
311
- idParam: string;
312
- setClauses: string[]; // e.g. ["hp = $1", "xp = xp + $2"]
313
- paramValues: string[]; // TypeScript expressions for each $n
314
- descriptions: string[];
315
- }
316
-
317
- function batchEffects(compiled: (CompiledEffect | null)[]): {
318
- batches: BatchedUpdate[];
319
- standalones: { sql: string; params: string[]; description: string }[];
320
- } {
321
- const batches = new Map<string, BatchedUpdate>(); // key: `${tableName}::${entityParam}::${idParam}`
322
- const standalones: { sql: string; params: string[]; description: string }[] = [];
323
-
324
- // Re-number parameters globally across all batches
325
- let globalParamN = 1;
326
-
327
- for (const effect of compiled) {
328
- if (!effect) continue;
329
-
330
- if (effect.standalone) {
331
- // Re-number the standalone params
332
- let sql = effect.standalone.sql;
333
- const params = effect.standalone.params;
334
- const renumbered: string[] = [];
335
- let localN = 1;
336
- for (const p of params) {
337
- sql = sql.replace(`$${localN}`, `$${globalParamN}`);
338
- renumbered.push(p);
339
- globalParamN++;
340
- localN++;
341
- }
342
- standalones.push({ sql, params: renumbered, description: effect.description });
343
- continue;
344
- }
345
-
346
- if (effect.assignments.length === 0) continue;
347
-
348
- const key = `${effect.tableName}::${effect.entityParam}::${effect.idParam}`;
349
- if (!batches.has(key)) {
350
- batches.set(key, {
351
- tableName: effect.tableName,
352
- entityParam: effect.entityParam,
353
- idParam: effect.idParam,
354
- setClauses: [],
355
- paramValues: [],
356
- descriptions: [],
357
- });
358
- }
359
- const batch = batches.get(key)!;
360
-
361
- for (const { column, placeholder, tsValue } of effect.assignments) {
362
- // column may be "fieldName" (assign) or "fieldName = fieldName + " (numeric add/remove)
363
- if (column.includes(" = ")) {
364
- // Numeric add/remove: column is already "field = field + " — append placeholder
365
- batch.setClauses.push(`${column}$${globalParamN}`);
366
- } else {
367
- batch.setClauses.push(`${column} = $${globalParamN}`);
368
- }
369
- batch.paramValues.push(tsValue);
370
- globalParamN++;
371
- }
372
- batch.descriptions.push(effect.description);
373
- }
374
-
375
- return { batches: Array.from(batches.values()), standalones };
376
- }
377
-
378
- // ─── Main Capability Body Emitter ─────────────────────────────────────────────
379
-
380
- export function emitCapabilityBody(
381
- method: IR.IRMethod,
382
- mod: IR.IRModule,
383
- system: IR.IRSystem,
384
- indent: string = " ",
385
- ): string {
386
- const lines: string[] = [];
387
- const fetches = getEntityFetches(method, mod, system);
388
-
389
- // 0. Destructure primitive params
390
- const primitiveParams = method.input.filter(p => {
391
- const isPrimitive = ["string", "uint", "int", "float", "bool", "timestamp", "uuid", "bytes", "json"].includes(p.type);
392
- const isListOrSet = p.type.startsWith("list<") || p.type.startsWith("set<");
393
- return (isPrimitive || isListOrSet) && !fetches.some(f => f.paramName === p.name);
394
- });
395
- if (primitiveParams.length > 0) {
396
- lines.push(`${indent}const { ${primitiveParams.map(p => p.name).join(", ")} } = req.body;`);
397
- lines.push(``);
398
- }
399
-
400
- // 1. Fetch entities — batch same-table fetches, parallelize different-table fetches
401
- if (fetches.length > 0) {
402
- lines.push(`${indent}// Fetch entities`);
403
-
404
- // Group fetches by table
405
- const byTable = new Map<string, EntityFetch[]>();
406
- for (const f of fetches) {
407
- if (!byTable.has(f.tableName)) byTable.set(f.tableName, []);
408
- byTable.get(f.tableName)!.push(f);
409
- }
410
-
411
- const fetchGroups = Array.from(byTable.entries());
412
-
413
- if (fetchGroups.length === 1 && fetchGroups[0][1].length === 1) {
414
- // Single fetch — simple queryOne
415
- const f = fetchGroups[0][1][0];
416
- const idExpr = `req.body.${f.idField} || req.params.id`;
417
- lines.push(`${indent}const ${f.paramName} = await queryOne(\`SELECT * FROM ${f.tableName} WHERE id = $1\`, [${idExpr}]);`);
418
- lines.push(`${indent}if (!${f.paramName}) {`);
419
- lines.push(`${indent} return res.status(404).json({ error: { code: "NOT_FOUND", message: "${f.paramName} not found" } });`);
420
- lines.push(`${indent}}`);
421
-
422
- } else if (fetchGroups.length === 1 && fetchGroups[0][1].length > 1) {
423
- // Multiple fetches from the SAME table — batch into WHERE id = ANY($1::uuid[])
424
- const [tableName, group] = fetchGroups[0];
425
- const idExprs = group.map(f => `req.body.${f.idField} || req.params.id`);
426
- lines.push(`${indent}// Batch fetch: ${group.map(f => f.paramName).join(", ")} from ${tableName} in one query`);
427
- lines.push(`${indent}const __ids_${tableName} = [${idExprs.join(", ")}];`);
428
- lines.push(`${indent}const __rows_${tableName} = await query(\`SELECT * FROM ${tableName} WHERE id = ANY($1::uuid[])\`, [__ids_${tableName}]);`);
429
- lines.push(`${indent}const __map_${tableName} = new Map(__rows_${tableName}.map((r: any) => [r.id, r]));`);
430
- for (const f of group) {
431
- const idExpr = `req.body.${f.idField} || req.params.id`;
432
- lines.push(`${indent}const ${f.paramName} = __map_${tableName}.get(${idExpr}) ?? null;`);
433
- lines.push(`${indent}if (!${f.paramName}) {`);
434
- lines.push(`${indent} return res.status(404).json({ error: { code: "NOT_FOUND", message: "${f.paramName} not found" } });`);
435
- lines.push(`${indent}}`);
436
- }
437
-
438
- } else {
439
- // Multiple fetches from DIFFERENT tables — run in parallel with Promise.all
440
- lines.push(`${indent}// Parallel fetch from ${fetchGroups.length} tables`);
441
- const resultVars: string[] = [];
442
- const fetchExprs: string[] = [];
443
-
444
- for (const [tableName, group] of fetchGroups) {
445
- if (group.length === 1) {
446
- const f = group[0];
447
- const idExpr = `req.body.${f.idField} || req.params.id`;
448
- resultVars.push(`__r_${f.paramName}`);
449
- fetchExprs.push(`queryOne(\`SELECT * FROM ${tableName} WHERE id = $1\`, [${idExpr}])`);
450
- } else {
451
- // Same-table batch within a multi-table parallel fetch
452
- const idExprs = group.map(f => `req.body.${f.idField} || req.params.id`);
453
- resultVars.push(`__rows_${tableName}`);
454
- fetchExprs.push(`query(\`SELECT * FROM ${tableName} WHERE id = ANY($1::uuid[])\`, [[${idExprs.join(", ")}]])`);
455
- }
456
- }
457
-
458
- lines.push(`${indent}const [${resultVars.join(", ")}] = await Promise.all([`);
459
- for (const expr of fetchExprs) lines.push(`${indent} ${expr},`);
460
- lines.push(`${indent}]);`);
461
-
462
- // Unpack results
463
- let resultIdx = 0;
464
- for (const [tableName, group] of fetchGroups) {
465
- if (group.length === 1) {
466
- const f = group[0];
467
- lines.push(`${indent}const ${f.paramName} = ${resultVars[resultIdx]};`);
468
- lines.push(`${indent}if (!${f.paramName}) {`);
469
- lines.push(`${indent} return res.status(404).json({ error: { code: "NOT_FOUND", message: "${f.paramName} not found" } });`);
470
- lines.push(`${indent}}`);
471
- } else {
472
- const mapVar = `__map_${tableName}`;
473
- lines.push(`${indent}const ${mapVar} = new Map((${resultVars[resultIdx]} as any[]).map((r: any) => [r.id, r]));`);
474
- for (const f of group) {
475
- const idExpr = `req.body.${f.idField} || req.params.id`;
476
- lines.push(`${indent}const ${f.paramName} = ${mapVar}.get(${idExpr}) ?? null;`);
477
- lines.push(`${indent}if (!${f.paramName}) {`);
478
- lines.push(`${indent} return res.status(404).json({ error: { code: "NOT_FOUND", message: "${f.paramName} not found" } });`);
479
- lines.push(`${indent}}`);
480
- }
481
- }
482
- resultIdx++;
483
- }
484
- }
485
- lines.push(``);
486
- }
487
-
488
- // 2. Precondition checks
489
- if (method.preconditions.length > 0) {
490
- lines.push(`${indent}// Preconditions`);
491
- for (const pre of method.preconditions) {
492
- try {
493
- lines.push(compilePrecondition(parseExprStr(pre.expression), indent));
494
- } catch {
495
- lines.push(`${indent}// CHECK: ${pre.description}`);
496
- }
497
- }
498
- lines.push(``);
499
- }
500
-
501
- // 3. Effects — batch same-entity updates into single UPDATEs
502
- if (method.effects.length > 0) {
503
- lines.push(`${indent}// Effects (batched by entity to minimise round-trips)`);
504
-
505
- const paramIdx = { n: 1 };
506
- const compiled = method.effects.map(e => compileEffect(e, mod, system, paramIdx, method));
507
- const { batches, standalones } = batchEffects(compiled);
508
-
509
- // Emit batched UPDATEs
510
- for (const batch of batches) {
511
- if (batch.setClauses.length === 0) continue;
512
- const idParamN = paramIdx.n++;
513
- const setClauses = batch.setClauses.join(", ");
514
- const sql = `UPDATE ${batch.tableName} SET ${setClauses}, updated_at = NOW() WHERE id = $${idParamN} RETURNING *`;
515
- const params = [...batch.paramValues, batch.idParam].join(", ");
516
- const resultVar = `__upd_${batch.entityParam}`;
517
- lines.push(`${indent}// ${batch.descriptions.join("; ")}`);
518
- lines.push(`${indent}const ${resultVar} = await query(\`${sql}\`, [${params}]);`);
519
- lines.push(`${indent}if (!${resultVar} || ${resultVar}.length === 0) {`);
520
- lines.push(`${indent} throw new Error("Update failed for ${batch.entityParam}");`);
521
- lines.push(`${indent}}`);
522
- }
523
-
524
- // Emit standalone effects (JSONB, array ops)
525
- for (const s of standalones) {
526
- const resultVar = `__eff_${standalones.indexOf(s)}`;
527
- lines.push(`${indent}// ${s.description}`);
528
- lines.push(`${indent}const ${resultVar} = await query(\`${s.sql}\`, [${s.params.join(", ")}]);`);
529
- lines.push(`${indent}if (!${resultVar} || ${resultVar}.length === 0) {`);
530
- lines.push(`${indent} throw new Error("Effect failed: ${s.description.replace(/"/g, '\\"')}");`);
531
- lines.push(`${indent}}`);
532
- }
533
-
534
- // Fallback for effects that couldn't be compiled.
535
- // Collection-level effects (e.g. list<T>.field = value) are not yet supported
536
- // by the batch compiler — emit a clear TODO rather than silently dropping them.
537
- // Effects that reference a completely unknown model are a hard error.
538
- for (const effect of method.effects) {
539
- const paramIdx2 = { n: 1 };
540
- if (!compileEffect(effect, mod, system, paramIdx2, method)) {
541
- const targetParts = effect.target.split(".");
542
- const paramName = targetParts[0];
543
- const param = method.input.find(p => p.name === paramName);
544
- const isCollectionEffect = param && (param.type.startsWith("list<") || param.type.startsWith("set<"));
545
-
546
- if (isCollectionEffect) {
547
- // Collection-level effects: apply the field update to all items in the collection
548
- // using a single batched UPDATE ... WHERE id = ANY($ids::uuid[])
549
- const innerType = param.type.replace(/^(list|set)<(.+)>$/, "$2");
550
- // Find the model for the inner element type
551
- let elemModel: IR.IRModel | undefined;
552
- for (const m of system.modules) {
553
- elemModel = m.models.find(mdl => mdl.name === innerType || mdl.name.toLowerCase() === innerType.toLowerCase());
554
- if (elemModel) break;
555
- }
556
- const tableName = elemModel ? (elemModel.name.replace(/([a-z])([A-Z])/g, "$1_$2").toLowerCase() + "s") : (innerType.toLowerCase() + "s");
557
- const fieldName = targetParts[1];
558
- const valueTs = targetParts[1] ? effect.value : "null";
559
- const opSql = effect.op === "add"
560
- ? `${fieldName} = ${fieldName} + $1`
561
- : effect.op === "remove"
562
- ? `${fieldName} = ${fieldName} - $1`
563
- : `${fieldName} = $1`;
564
- lines.push(`${indent}// Collection effect: ${effect.target} ${effect.op === "assign" ? "=" : effect.op === "add" ? "+=" : "-="} ${effect.value}`);
565
- lines.push(`${indent}if (${paramName} && ${paramName}.length > 0) {`);
566
- lines.push(`${indent} const __ids_${paramName} = ${paramName}.map((x: any) => x.id ?? x);`);
567
- lines.push(`${indent} await query(`);
568
- lines.push(`${indent} \`UPDATE ${tableName} SET ${opSql}, updated_at = NOW() WHERE id = ANY($2::uuid[])\`,`);
569
- lines.push(`${indent} [${effect.value}, __ids_${paramName}],`);
570
- lines.push(`${indent} );`);
571
- lines.push(`${indent}}`);
572
- } else {
573
- throw new Error(
574
- `Unsupported effect in method '${method.name}': target '${effect.target}' could not be resolved to a known model field. ` +
575
- `Ensure the effect target matches a declared entity field (e.g. 'entityName.fieldName').`
576
- );
577
- }
578
- }
579
- }
580
- lines.push(``);
581
- }
582
-
583
- // 4. Event emissions
584
- if (method.emissions.length > 0) {
585
- lines.push(`${indent}// Emit events`);
586
- for (const ev of method.emissions) {
587
- const payload = buildEventPayload(method, fetches);
588
- if (method.sync === "transactional") {
589
- lines.push(`${indent}await eventBus.publish("${ev}", ${payload}, "${mod.name}", auth.trace_id, __client);`);
590
- } else {
591
- lines.push(`${indent}await eventBus.publish("${ev}", ${payload}, "${mod.name}", auth.trace_id);`);
592
- }
593
- }
594
- lines.push(``);
595
- }
596
-
597
- // 5. Return result
598
- const resultEntity = fetches[0];
599
- if (resultEntity) {
600
- lines.push(`${indent}res.json({ ok: true, action: "${method.name}", entity: ${resultEntity.paramName} });`);
601
- } else {
602
- lines.push(`${indent}res.json({ ok: true, action: "${method.name}" });`);
603
- }
604
-
605
- return lines.join("\n");
606
- }
607
-
608
- function buildEventPayload(method: IR.IRMethod, fetches: EntityFetch[]): string {
609
- const fields: string[] = fetches.map(f => `${f.paramName}_id: ${f.paramName}?.id`);
610
- fields.push(`timestamp: new Date().toISOString()`);
611
- fields.push(`actor_id: auth.actor_id`);
612
- return `{ ${fields.join(", ")} }`;
613
- }
1
+ /**
2
+ * BoneScript Capability Body Emitter
3
+ *
4
+ * Translates IR effects and preconditions into real TypeScript + SQL.
5
+ */
6
+
7
+ import * as IR from "./ir";
8
+
9
+ // ─── Expression Parser ────────────────────────────────────────────────────────
10
+ // Parses the serialized expression strings from the IR back into a structured form.
11
+
12
+ type ExprKind =
13
+ | { kind: "literal"; value: string; raw: string }
14
+ | { kind: "field"; path: string[] }
15
+ | { kind: "binop"; op: string; left: Expr; right: Expr }
16
+ | { kind: "call"; name: string; args: Expr[] };
17
+
18
+ type Expr = ExprKind;
19
+
20
+ function parseExprStr(s: string): Expr {
21
+ s = s.trim();
22
+
23
+ // Strip outer parens
24
+ if (s.startsWith("(") && s.endsWith(")")) {
25
+ return parseExprStr(s.slice(1, -1));
26
+ }
27
+
28
+ // String literal
29
+ if (s.startsWith('"') && s.endsWith('"')) {
30
+ return { kind: "literal", value: s.slice(1, -1), raw: s };
31
+ }
32
+
33
+ // Number literal
34
+ if (/^-?\d+(\.\d+)?$/.test(s)) {
35
+ return { kind: "literal", value: s, raw: s };
36
+ }
37
+
38
+ // Boolean
39
+ if (s === "true" || s === "false") {
40
+ return { kind: "literal", value: s, raw: s };
41
+ }
42
+
43
+ // Binary operators (check in precedence order, right-to-left to handle left-assoc)
44
+ const binOps = [" or ", " and ", " == ", " != ", " >= ", " <= ", " > ", " < ", " in ", " contains ", " + ", " - ", " * ", " / "];
45
+ for (const op of binOps) {
46
+ const idx = findBinOp(s, op);
47
+ if (idx !== -1) {
48
+ const left = parseExprStr(s.slice(0, idx));
49
+ const right = parseExprStr(s.slice(idx + op.length));
50
+ return { kind: "binop", op: op.trim(), left, right };
51
+ }
52
+ }
53
+
54
+ // Function call: name(args)
55
+ const callMatch = s.match(/^(\w+)\((.*)?\)$/);
56
+ if (callMatch) {
57
+ const args = callMatch[2] ? splitArgs(callMatch[2]).map(parseExprStr) : [];
58
+ return { kind: "call", name: callMatch[1], args };
59
+ }
60
+
61
+ // Field reference: a.b.c
62
+ if (/^[\w.]+$/.test(s)) {
63
+ return { kind: "field", path: s.split(".") };
64
+ }
65
+
66
+ // Fallback: treat as opaque literal
67
+ return { kind: "literal", value: s, raw: s };
68
+ }
69
+
70
+ function findBinOp(s: string, op: string): number {
71
+ let depth = 0;
72
+ for (let i = 0; i <= s.length - op.length; i++) {
73
+ const ch = s[i];
74
+ if (ch === "(" || ch === "[") depth++;
75
+ else if (ch === ")" || ch === "]") depth--;
76
+ else if (depth === 0 && s.slice(i, i + op.length) === op) {
77
+ return i;
78
+ }
79
+ }
80
+ return -1;
81
+ }
82
+
83
+ function splitArgs(s: string): string[] {
84
+ const args: string[] = [];
85
+ let depth = 0;
86
+ let current = "";
87
+ for (const ch of s) {
88
+ if (ch === "(" || ch === "[") depth++;
89
+ else if (ch === ")" || ch === "]") depth--;
90
+ else if (ch === "," && depth === 0) {
91
+ args.push(current.trim());
92
+ current = "";
93
+ continue;
94
+ }
95
+ current += ch;
96
+ }
97
+ if (current.trim()) args.push(current.trim());
98
+ return args;
99
+ }
100
+
101
+ // ─── Entity Resolution ────────────────────────────────────────────────────────
102
+ // Determines which entities need to be fetched from the DB for a capability.
103
+
104
+ interface EntityFetch {
105
+ paramName: string; // capability parameter name (e.g., "item")
106
+ entityType: string; // entity type name (e.g., "Item")
107
+ tableName: string; // SQL table name (e.g., "items")
108
+ idField: string; // request body field for the ID (e.g., "item_id")
109
+ }
110
+
111
+ function toSnakeCase(s: string): string {
112
+ return s.replace(/([a-z])([A-Z])/g, "$1_$2").toLowerCase();
113
+ }
114
+
115
+ function getEntityFetches(method: IR.IRMethod, mod: IR.IRModule, system: IR.IRSystem): EntityFetch[] {
116
+ const fetches: EntityFetch[] = [];
117
+ const seen = new Set<string>();
118
+
119
+ // Build a map of all entity names → table names across the whole system
120
+ const allModels = new Map<string, string>(); // entityName → tableName
121
+ for (const m of system.modules) {
122
+ for (const model of m.models) {
123
+ allModels.set(model.name, toSnakeCase(model.name) + "s");
124
+ allModels.set(model.name.toLowerCase(), toSnakeCase(model.name) + "s");
125
+ }
126
+ }
127
+
128
+ for (const param of method.input) {
129
+ const tableName = allModels.get(param.type) || allModels.get(param.type.toLowerCase());
130
+ if (tableName && !seen.has(param.name)) {
131
+ seen.add(param.name);
132
+ fetches.push({
133
+ paramName: param.name,
134
+ entityType: param.type,
135
+ tableName,
136
+ idField: param.name + "_id",
137
+ });
138
+ }
139
+ }
140
+
141
+ return fetches;
142
+ }
143
+
144
+ // ─── Precondition Compiler ────────────────────────────────────────────────────
145
+
146
+ interface CompiledPrecondition {
147
+ code: string; // TypeScript guard clause
148
+ description: string;
149
+ }
150
+
151
+ function compilePrecondition(expr: Expr, indent: string): string {
152
+ const condition = exprToTs(expr, true);
153
+ const description = exprToDescription(expr).replace(/"/g, '\\"');
154
+ return [
155
+ `${indent}if (${condition}) {`,
156
+ `${indent} return res.status(422).json({ error: { code: "PRECONDITION_FAILED", message: ${JSON.stringify(description)} } });`,
157
+ `${indent}}`,
158
+ ].join("\n");
159
+ }
160
+
161
+ function exprToTs(expr: Expr, negate: boolean = false): string {
162
+ const inner = exprToTsInner(expr);
163
+ return negate ? `!(${inner})` : inner;
164
+ }
165
+
166
+ function exprToTsInner(expr: Expr): string {
167
+ switch (expr.kind) {
168
+ case "literal":
169
+ if (expr.value === "true") return "true";
170
+ if (expr.value === "false") return "false";
171
+ if (/^"/.test(expr.raw)) return expr.raw;
172
+ return expr.value;
173
+
174
+ case "field":
175
+ // Convert field path to JS property access
176
+ return expr.path.join("?.");
177
+
178
+ case "binop": {
179
+ const l = exprToTsInner(expr.left);
180
+ const r = exprToTsInner(expr.right);
181
+ switch (expr.op) {
182
+ case "==": return `${l} === ${r}`;
183
+ case "!=": return `${l} !== ${r}`;
184
+ case "and": return `(${l} && ${r})`;
185
+ case "or": return `(${l} || ${r})`;
186
+ case "in": return `[${r}].flat().includes(${l})`;
187
+ case "contains": return `${l}?.includes(${r})`;
188
+ case ">": case "<": case ">=": case "<=":
189
+ case "+": case "-": case "*": case "/":
190
+ return `${l} ${expr.op} ${r}`;
191
+ default: return `${l} ${expr.op} ${r}`;
192
+ }
193
+ }
194
+
195
+ case "call":
196
+ if (expr.name === "now") return "new Date()";
197
+ return `${expr.name}(${expr.args.map(exprToTsInner).join(", ")})`;
198
+ }
199
+ }
200
+
201
+ function exprToDescription(expr: Expr): string {
202
+ switch (expr.kind) {
203
+ case "literal": return expr.raw;
204
+ case "field": return expr.path.join(".");
205
+ case "binop": {
206
+ const l = exprToDescription(expr.left);
207
+ const r = exprToDescription(expr.right);
208
+ return `${l} ${expr.op} ${r}`;
209
+ }
210
+ case "call": return `${expr.name}(${expr.args.map(exprToDescription).join(", ")})`;
211
+ }
212
+ }
213
+
214
+ // ─── Effect Compiler ──────────────────────────────────────────────────────────
215
+
216
+ interface CompiledEffect {
217
+ sql: string;
218
+ params: string[];
219
+ description: string;
220
+ }
221
+
222
+ function compileEffect(effect: IR.IREffect, mod: IR.IRModule, system: IR.IRSystem, paramIdx: { n: number }): CompiledEffect | null {
223
+ const targetParts = effect.target.split(".");
224
+ if (targetParts.length < 2) return null;
225
+
226
+ const entityParam = targetParts[0]; // e.g., "item" or "trade"
227
+ const fieldName = targetParts[1]; // e.g., "quantity" or "offered_items"
228
+ const nestedPath = targetParts.slice(2); // e.g., ["owner_id"] for nested JSONB
229
+
230
+ // Find the model for this entity param — search across all modules
231
+ const model = (() => {
232
+ for (const m of system.modules) {
233
+ const found = m.models.find(mdl =>
234
+ toSnakeCase(mdl.name) === entityParam ||
235
+ mdl.name.toLowerCase() === entityParam.toLowerCase()
236
+ );
237
+ if (found) return found;
238
+ }
239
+ return mod.models.find(m =>
240
+ toSnakeCase(m.name) === entityParam ||
241
+ m.name.toLowerCase() === entityParam.toLowerCase()
242
+ );
243
+ })();
244
+ if (!model) return null;
245
+
246
+ const tableName = toSnakeCase(model.name) + "s";
247
+ const valueExpr = parseExprStr(effect.value);
248
+ const valueTs = exprToTsInner(valueExpr);
249
+ const idParam = `req.body.${entityParam}_id || req.params.id`;
250
+
251
+ // Detect if the param is a list type (bulk operation)
252
+ const isBulk = effect.target.includes("[]") ||
253
+ (entityParam.endsWith("s") && !model.name.toLowerCase().endsWith("s"));
254
+ const bulkIdParam = `req.body.${entityParam}_ids || req.body.${entityParam}?.map((x: any) => x.id)`;
255
+ const whereClause = isBulk
256
+ ? `WHERE id = ANY($2::uuid[])`
257
+ : `WHERE id = ${`$${2}`}`;
258
+
259
+ // Handle nested JSONB path: trade.offered_items.owner_id
260
+ if (nestedPath.length > 0) {
261
+ const jsonbField = fieldName;
262
+ const jsonbPath = nestedPath.join(".");
263
+ const p1 = `$${paramIdx.n++}`;
264
+ const p2 = `$${paramIdx.n++}`;
265
+ // Use jsonb_set to update nested path
266
+ const jsonbPathLiteral = `'{${nestedPath.join(",")}}'`;
267
+ return {
268
+ sql: `UPDATE ${tableName} SET ${jsonbField} = jsonb_set(COALESCE(${jsonbField}, '{}'), ${jsonbPathLiteral}, to_jsonb(${p1}::text), true), updated_at = NOW() WHERE id = ${p2} RETURNING *`,
269
+ params: [valueTs, idParam],
270
+ description: `${effect.target} = ${effect.value}`,
271
+ };
272
+ }
273
+
274
+ switch (effect.op) {
275
+ case "assign": {
276
+ const p1 = `$${paramIdx.n++}`;
277
+ const p2 = `$${paramIdx.n++}`;
278
+ return {
279
+ sql: `UPDATE ${tableName} SET ${fieldName} = ${p1}, updated_at = NOW() WHERE id = ${p2} RETURNING *`,
280
+ params: [valueTs, idParam],
281
+ description: `${effect.target} = ${effect.value}`,
282
+ };
283
+ }
284
+ case "add": {
285
+ const p1 = `$${paramIdx.n++}`;
286
+ const p2 = `$${paramIdx.n++}`;
287
+ const fieldType = model.fields.find(f => f.name === fieldName)?.type || "";
288
+ const isNumeric = ["uint", "int", "float"].includes(fieldType);
289
+ if (isNumeric) {
290
+ return {
291
+ sql: `UPDATE ${tableName} SET ${fieldName} = ${fieldName} + ${p1}, updated_at = NOW() WHERE id = ${p2} RETURNING *`,
292
+ params: [valueTs, idParam],
293
+ description: `${effect.target} += ${effect.value}`,
294
+ };
295
+ } else {
296
+ return {
297
+ sql: `UPDATE ${tableName} SET ${fieldName} = ${fieldName} || jsonb_build_array(${p1}::jsonb), updated_at = NOW() WHERE id = ${p2} RETURNING *`,
298
+ params: [valueTs, idParam],
299
+ description: `${effect.target} += ${effect.value}`,
300
+ };
301
+ }
302
+ }
303
+ case "remove": {
304
+ const p1 = `$${paramIdx.n++}`;
305
+ const p2 = `$${paramIdx.n++}`;
306
+ const fieldType = model.fields.find(f => f.name === fieldName)?.type || "";
307
+ const isNumeric = ["uint", "int", "float"].includes(fieldType);
308
+ if (isNumeric) {
309
+ return {
310
+ sql: `UPDATE ${tableName} SET ${fieldName} = ${fieldName} - ${p1}, updated_at = NOW() WHERE id = ${p2} RETURNING *`,
311
+ params: [valueTs, idParam],
312
+ description: `${effect.target} -= ${effect.value}`,
313
+ };
314
+ } else {
315
+ return {
316
+ sql: `UPDATE ${tableName} SET ${fieldName} = (SELECT jsonb_agg(elem) FROM jsonb_array_elements(${fieldName}) elem WHERE elem != ${p1}::jsonb), updated_at = NOW() WHERE id = ${p2} RETURNING *`,
317
+ params: [valueTs, idParam],
318
+ description: `${effect.target} -= ${effect.value}`,
319
+ };
320
+ }
321
+ }
322
+ }
323
+ }
324
+
325
+ // ─── Main Capability Body Emitter ─────────────────────────────────────────────
326
+
327
+ export function emitCapabilityBody(
328
+ method: IR.IRMethod,
329
+ mod: IR.IRModule,
330
+ system: IR.IRSystem,
331
+ indent: string = " "
332
+ ): string {
333
+ const lines: string[] = [];
334
+ const fetches = getEntityFetches(method, mod, system);
335
+
336
+ // 0. Destructure primitive params from req.body
337
+ const primitiveParams = method.input.filter(p => {
338
+ const isPrimitive = ["string", "uint", "int", "float", "bool", "timestamp", "uuid", "bytes", "json"].includes(p.type);
339
+ const isListOrSet = p.type.startsWith("list<") || p.type.startsWith("set<");
340
+ const isEntityFetch = fetches.some(f => f.paramName === p.name);
341
+ return (isPrimitive || isListOrSet) && !isEntityFetch;
342
+ });
343
+
344
+ if (primitiveParams.length > 0) {
345
+ const destructured = primitiveParams.map(p => p.name).join(", ");
346
+ lines.push(`${indent}const { ${destructured} } = req.body;`);
347
+ lines.push(``);
348
+ }
349
+
350
+ // 1. Fetch entities referenced in preconditions/effects
351
+ if (fetches.length > 0) {
352
+ lines.push(`${indent}// Fetch entities`);
353
+ for (const fetch of fetches) {
354
+ const idExpr = `req.body.${fetch.idField} || req.params.id`;
355
+ lines.push(`${indent}const ${fetch.paramName} = await queryOne(\`SELECT * FROM ${fetch.tableName} WHERE id = $1\`, [${idExpr}]);`);
356
+ lines.push(`${indent}if (!${fetch.paramName}) {`);
357
+ lines.push(`${indent} return res.status(404).json({ error: { code: "NOT_FOUND", message: "${fetch.paramName} not found" } });`);
358
+ lines.push(`${indent}}`);
359
+ }
360
+ lines.push(``);
361
+ }
362
+
363
+ // 2. Precondition checks
364
+ if (method.preconditions.length > 0) {
365
+ lines.push(`${indent}// Preconditions`);
366
+ for (const pre of method.preconditions) {
367
+ try {
368
+ const expr = parseExprStr(pre.expression);
369
+ lines.push(compilePrecondition(expr, indent));
370
+ } catch {
371
+ // Fallback: emit as comment if parsing fails
372
+ lines.push(`${indent}// CHECK: ${pre.description}`);
373
+ }
374
+ }
375
+ lines.push(``);
376
+ }
377
+
378
+ // 3. Effects (applied in declaration order, each in its own query)
379
+ if (method.effects.length > 0) {
380
+ lines.push(`${indent}// Effects (applied in declaration order)`);
381
+ const effectResults: string[] = [];
382
+
383
+ for (const effect of method.effects) {
384
+ // Each effect gets its own parameter numbering starting at 1
385
+ const paramIdx = { n: 1 };
386
+ const compiled = compileEffect(effect, mod, system, paramIdx);
387
+ if (compiled) {
388
+ const resultVar = `__effect_${effectResults.length}`;
389
+ effectResults.push(resultVar);
390
+ lines.push(`${indent}const ${resultVar} = await query(\`${compiled.sql}\`, [${compiled.params.join(", ")}]);`);
391
+ lines.push(`${indent}if (!${resultVar} || ${resultVar}.length === 0) {`);
392
+ lines.push(`${indent} throw new Error("Effect failed: ${compiled.description.replace(/"/g, '\\"')}");`);
393
+ lines.push(`${indent}}`);
394
+ } else {
395
+ // Fallback for complex effects we can't compile
396
+ lines.push(`${indent}// EFFECT: ${effect.target} ${effect.op === "assign" ? "=" : effect.op === "add" ? "+=" : "-="} ${effect.value}`);
397
+ lines.push(`${indent}// TODO: Implement this effect manually`);
398
+ }
399
+ }
400
+ lines.push(``);
401
+ }
402
+
403
+ // 4. Event emissions
404
+ if (method.emissions.length > 0) {
405
+ lines.push(`${indent}// Emit events`);
406
+ for (const ev of method.emissions) {
407
+ const payload = buildEventPayload(method, fetches);
408
+ if (method.sync === "transactional") {
409
+ lines.push(`${indent}await eventBus.publish("${ev}", ${payload}, "${mod.name}", auth.trace_id, __client);`);
410
+ } else {
411
+ lines.push(`${indent}await eventBus.publish("${ev}", ${payload}, "${mod.name}", auth.trace_id);`);
412
+ }
413
+ }
414
+ lines.push(``);
415
+ }
416
+
417
+ // 5. Return result
418
+ const resultEntity = fetches[0];
419
+ if (resultEntity) {
420
+ lines.push(`${indent}res.json({ ok: true, action: "${method.name}", entity: ${resultEntity.paramName} });`);
421
+ } else {
422
+ lines.push(`${indent}res.json({ ok: true, action: "${method.name}" });`);
423
+ }
424
+
425
+ return lines.join("\n");
426
+ }
427
+
428
+ function buildEventPayload(method: IR.IRMethod, fetches: EntityFetch[]): string {
429
+ const fields: string[] = [];
430
+ for (const fetch of fetches) {
431
+ fields.push(`${fetch.paramName}_id: ${fetch.paramName}?.id`);
432
+ }
433
+ fields.push(`timestamp: new Date().toISOString()`);
434
+ fields.push(`actor_id: auth.actor_id`);
435
+ return `{ ${fields.join(", ")} }`;
436
+ }