bonescript-compiler 0.2.0 → 0.3.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 -21
  2. package/dist/algorithm_catalog.js +166 -166
  3. package/dist/cli.d.ts +2 -1
  4. package/dist/cli.js +75 -543
  5. package/dist/cli.js.map +1 -1
  6. package/dist/commands/check.d.ts +5 -0
  7. package/dist/commands/check.js +34 -0
  8. package/dist/commands/check.js.map +1 -0
  9. package/dist/commands/compile.d.ts +5 -0
  10. package/dist/commands/compile.js +183 -0
  11. package/dist/commands/compile.js.map +1 -0
  12. package/dist/commands/debug.d.ts +5 -0
  13. package/dist/commands/debug.js +59 -0
  14. package/dist/commands/debug.js.map +1 -0
  15. package/dist/commands/diff.d.ts +5 -0
  16. package/dist/commands/diff.js +125 -0
  17. package/dist/commands/diff.js.map +1 -0
  18. package/dist/commands/fmt.d.ts +5 -0
  19. package/dist/commands/fmt.js +49 -0
  20. package/dist/commands/fmt.js.map +1 -0
  21. package/dist/commands/init.d.ts +5 -0
  22. package/dist/commands/init.js +69 -0
  23. package/dist/commands/init.js.map +1 -0
  24. package/dist/commands/ir.d.ts +5 -0
  25. package/dist/commands/ir.js +27 -0
  26. package/dist/commands/ir.js.map +1 -0
  27. package/dist/commands/lex.d.ts +5 -0
  28. package/dist/commands/lex.js +21 -0
  29. package/dist/commands/lex.js.map +1 -0
  30. package/dist/commands/parse.d.ts +5 -0
  31. package/dist/commands/parse.js +30 -0
  32. package/dist/commands/parse.js.map +1 -0
  33. package/dist/commands/test.d.ts +5 -0
  34. package/dist/commands/test.js +61 -0
  35. package/dist/commands/test.js.map +1 -0
  36. package/dist/commands/verify_determinism.d.ts +5 -0
  37. package/dist/commands/verify_determinism.js +64 -0
  38. package/dist/commands/verify_determinism.js.map +1 -0
  39. package/dist/commands/watch.d.ts +5 -0
  40. package/dist/commands/watch.js +50 -0
  41. package/dist/commands/watch.js.map +1 -0
  42. package/dist/emit_auth.d.ts +6 -0
  43. package/dist/emit_auth.js +69 -0
  44. package/dist/emit_auth.js.map +1 -0
  45. package/dist/emit_capability.d.ts +13 -0
  46. package/dist/emit_capability.js +235 -125
  47. package/dist/emit_capability.js.map +1 -1
  48. package/dist/emit_database.d.ts +7 -0
  49. package/dist/emit_database.js +74 -0
  50. package/dist/emit_database.js.map +1 -0
  51. package/dist/emit_deploy.js +162 -162
  52. package/dist/emit_events.js +274 -274
  53. package/dist/emit_full.js +102 -95
  54. package/dist/emit_full.js.map +1 -1
  55. package/dist/emit_index.d.ts +6 -0
  56. package/dist/emit_index.js +157 -0
  57. package/dist/emit_index.js.map +1 -0
  58. package/dist/emit_maintenance.js +249 -249
  59. package/dist/emit_package.d.ts +7 -0
  60. package/dist/emit_package.js +70 -0
  61. package/dist/emit_package.js.map +1 -0
  62. package/dist/emit_router.d.ts +12 -0
  63. package/dist/emit_router.js +375 -0
  64. package/dist/emit_router.js.map +1 -0
  65. package/dist/emit_runtime.d.ts +17 -11
  66. package/dist/emit_runtime.js +29 -686
  67. package/dist/emit_runtime.js.map +1 -1
  68. package/dist/emit_sourcemap.js +66 -66
  69. package/dist/extension_manager.d.ts +2 -2
  70. package/dist/extension_manager.js +6 -3
  71. package/dist/extension_manager.js.map +1 -1
  72. package/dist/lowering.d.ts +5 -14
  73. package/dist/lowering.js +32 -417
  74. package/dist/lowering.js.map +1 -1
  75. package/dist/lowering_channels.d.ts +11 -0
  76. package/dist/lowering_channels.js +102 -0
  77. package/dist/lowering_channels.js.map +1 -0
  78. package/dist/lowering_entities.d.ts +11 -0
  79. package/dist/lowering_entities.js +222 -0
  80. package/dist/lowering_entities.js.map +1 -0
  81. package/dist/lowering_helpers.d.ts +13 -0
  82. package/dist/lowering_helpers.js +76 -0
  83. package/dist/lowering_helpers.js.map +1 -0
  84. package/dist/module_loader.d.ts +2 -2
  85. package/dist/module_loader.js +20 -23
  86. package/dist/module_loader.js.map +1 -1
  87. package/dist/scaffold.d.ts +2 -2
  88. package/dist/scaffold.js +316 -319
  89. package/dist/scaffold.js.map +1 -1
  90. package/package.json +62 -52
  91. package/src/algorithm_catalog.ts +345 -345
  92. package/src/ast.ts +334 -334
  93. package/src/cli.ts +98 -624
  94. package/src/commands/check.ts +33 -0
  95. package/src/commands/compile.ts +160 -0
  96. package/src/commands/debug.ts +33 -0
  97. package/src/commands/diff.ts +108 -0
  98. package/src/commands/fmt.ts +22 -0
  99. package/src/commands/init.ts +46 -0
  100. package/src/commands/ir.ts +23 -0
  101. package/src/commands/lex.ts +17 -0
  102. package/src/commands/parse.ts +24 -0
  103. package/src/commands/test.ts +36 -0
  104. package/src/commands/verify_determinism.ts +66 -0
  105. package/src/commands/watch.ts +25 -0
  106. package/src/emit_auth.ts +67 -0
  107. package/src/emit_batch.ts +140 -140
  108. package/src/emit_capability.ts +562 -436
  109. package/src/emit_composition.ts +196 -196
  110. package/src/emit_database.ts +75 -0
  111. package/src/emit_deploy.ts +190 -190
  112. package/src/emit_events.ts +307 -307
  113. package/src/emit_extras.ts +240 -240
  114. package/src/emit_full.ts +316 -309
  115. package/src/emit_index.ts +161 -0
  116. package/src/emit_maintenance.ts +459 -459
  117. package/src/emit_package.ts +69 -0
  118. package/src/emit_router.ts +395 -0
  119. package/src/emit_runtime.ts +17 -728
  120. package/src/emit_sourcemap.ts +140 -140
  121. package/src/emit_tests.ts +205 -205
  122. package/src/emit_websocket.ts +229 -229
  123. package/src/emitter.ts +566 -566
  124. package/src/extension_manager.ts +189 -187
  125. package/src/formatter.ts +297 -297
  126. package/src/index.ts +88 -88
  127. package/src/ir.ts +215 -215
  128. package/src/lexer.ts +630 -630
  129. package/src/lowering.ts +124 -556
  130. package/src/lowering_channels.ts +107 -0
  131. package/src/lowering_entities.ts +248 -0
  132. package/src/lowering_helpers.ts +75 -0
  133. package/src/module_loader.ts +112 -114
  134. package/src/optimizer.ts +196 -196
  135. package/src/parse_decls.ts +409 -409
  136. package/src/parse_decls2.ts +244 -244
  137. package/src/parse_expr.ts +197 -197
  138. package/src/parse_types.ts +54 -54
  139. package/src/parser.ts +1 -1
  140. package/src/parser_base.ts +57 -57
  141. package/src/parser_recovery.ts +153 -153
  142. package/src/scaffold.ts +372 -375
  143. package/src/solver.ts +330 -330
  144. package/src/typechecker.ts +591 -591
  145. package/src/types.ts +122 -122
  146. package/src/verifier.ts +348 -348
@@ -1,436 +1,562 @@
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
- }
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
+
22
+ // ─── Expression Parser ────────────────────────────────────────────────────────
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 toSnakeCase(s: string): string {
92
+ return s.replace(/([a-z])([A-Z])/g, "$1_$2").toLowerCase();
93
+ }
94
+
95
+ function getEntityFetches(method: IR.IRMethod, mod: IR.IRModule, system: IR.IRSystem): EntityFetch[] {
96
+ const fetches: EntityFetch[] = [];
97
+ const seen = new Set<string>();
98
+
99
+ const allModels = new Map<string, string>();
100
+ for (const m of system.modules) {
101
+ for (const model of m.models) {
102
+ allModels.set(model.name, toSnakeCase(model.name) + "s");
103
+ allModels.set(model.name.toLowerCase(), toSnakeCase(model.name) + "s");
104
+ }
105
+ }
106
+
107
+ for (const param of method.input) {
108
+ const tableName = allModels.get(param.type) || allModels.get(param.type.toLowerCase());
109
+ if (tableName && !seen.has(param.name)) {
110
+ seen.add(param.name);
111
+ fetches.push({ paramName: param.name, entityType: param.type, tableName, idField: param.name + "_id" });
112
+ }
113
+ }
114
+
115
+ return fetches;
116
+ }
117
+
118
+ // ─── Precondition Compiler ────────────────────────────────────────────────────
119
+
120
+ function compilePrecondition(expr: Expr, indent: string): string {
121
+ const condition = exprToTs(expr, true);
122
+ const description = exprToDescription(expr).replace(/"/g, '\\"');
123
+ return [
124
+ `${indent}if (${condition}) {`,
125
+ `${indent} return res.status(422).json({ error: { code: "PRECONDITION_FAILED", message: ${JSON.stringify(description)} } });`,
126
+ `${indent}}`,
127
+ ].join("\n");
128
+ }
129
+
130
+ function exprToTs(expr: Expr, negate = false): string {
131
+ const inner = exprToTsInner(expr);
132
+ return negate ? `!(${inner})` : inner;
133
+ }
134
+
135
+ function exprToTsInner(expr: Expr): string {
136
+ switch (expr.kind) {
137
+ case "literal":
138
+ if (expr.value === "true") return "true";
139
+ if (expr.value === "false") return "false";
140
+ if (/^"/.test(expr.raw)) return expr.raw;
141
+ return expr.value;
142
+ case "field":
143
+ return expr.path.join("?.");
144
+ case "binop": {
145
+ const l = exprToTsInner(expr.left);
146
+ const r = exprToTsInner(expr.right);
147
+ switch (expr.op) {
148
+ case "==": return `${l} === ${r}`;
149
+ case "!=": return `${l} !== ${r}`;
150
+ case "and": return `(${l} && ${r})`;
151
+ case "or": return `(${l} || ${r})`;
152
+ case "in": return `[${r}].flat().includes(${l})`;
153
+ case "contains": return `${l}?.includes(${r})`;
154
+ default: return `${l} ${expr.op} ${r}`;
155
+ }
156
+ }
157
+ case "call":
158
+ if (expr.name === "now") return "new Date()";
159
+ return `${expr.name}(${expr.args.map(exprToTsInner).join(", ")})`;
160
+ }
161
+ }
162
+
163
+ function exprToDescription(expr: Expr): string {
164
+ switch (expr.kind) {
165
+ case "literal": return expr.raw;
166
+ case "field": return expr.path.join(".");
167
+ case "binop": return `${exprToDescription(expr.left)} ${expr.op} ${exprToDescription(expr.right)}`;
168
+ case "call": return `${expr.name}(${expr.args.map(exprToDescription).join(", ")})`;
169
+ }
170
+ }
171
+
172
+ // ─── Effect Compiler ──────────────────────────────────────────────────────────
173
+
174
+ interface CompiledEffect {
175
+ tableName: string;
176
+ entityParam: string;
177
+ idParam: string;
178
+ // For batched UPDATE: list of (column, paramPlaceholder, tsValue) tuples
179
+ assignments: { column: string; placeholder: string; tsValue: string }[];
180
+ description: string;
181
+ // For non-batchable effects (JSONB, array ops) — emitted as standalone query
182
+ standalone?: { sql: string; params: string[] };
183
+ }
184
+
185
+ /**
186
+ * Compile a single effect into a structured form.
187
+ * Returns null if the effect target can't be resolved.
188
+ */
189
+ function compileEffect(
190
+ effect: IR.IREffect,
191
+ mod: IR.IRModule,
192
+ system: IR.IRSystem,
193
+ paramIdx: { n: number },
194
+ ): CompiledEffect | null {
195
+ const targetParts = effect.target.split(".");
196
+ if (targetParts.length < 2) return null;
197
+
198
+ const entityParam = targetParts[0];
199
+ const fieldName = targetParts[1];
200
+ const nestedPath = targetParts.slice(2);
201
+
202
+ const model = (() => {
203
+ for (const m of system.modules) {
204
+ const found = m.models.find(mdl =>
205
+ toSnakeCase(mdl.name) === entityParam || mdl.name.toLowerCase() === entityParam.toLowerCase()
206
+ );
207
+ if (found) return found;
208
+ }
209
+ return mod.models.find(m =>
210
+ toSnakeCase(m.name) === entityParam || m.name.toLowerCase() === entityParam.toLowerCase()
211
+ );
212
+ })();
213
+ if (!model) return null;
214
+
215
+ const tableName = toSnakeCase(model.name) + "s";
216
+ const valueTs = exprToTsInner(parseExprStr(effect.value));
217
+ const idParam = `req.body.${entityParam}_id || req.params.id`;
218
+
219
+ // JSONB nested path — must be standalone (jsonb_set can't be batched cleanly)
220
+ if (nestedPath.length > 0) {
221
+ const p1 = `$${paramIdx.n++}`;
222
+ const p2 = `$${paramIdx.n++}`;
223
+ const jsonbPathLiteral = `'{${nestedPath.join(",")}}'`;
224
+ return {
225
+ tableName, entityParam, idParam,
226
+ assignments: [],
227
+ description: `${effect.target} = ${effect.value}`,
228
+ standalone: {
229
+ sql: `UPDATE ${tableName} SET ${fieldName} = jsonb_set(COALESCE(${fieldName}, '{}'), ${jsonbPathLiteral}, to_jsonb(${p1}::text), true), updated_at = NOW() WHERE id = ${p2} RETURNING *`,
230
+ params: [valueTs, idParam],
231
+ },
232
+ };
233
+ }
234
+
235
+ const fieldType = model.fields.find(f => f.name === fieldName)?.type || "";
236
+ const isNumeric = ["uint", "int", "float"].includes(fieldType);
237
+
238
+ switch (effect.op) {
239
+ case "assign": {
240
+ const p1 = `$${paramIdx.n++}`;
241
+ return {
242
+ tableName, entityParam, idParam,
243
+ assignments: [{ column: fieldName, placeholder: p1, tsValue: valueTs }],
244
+ description: `${effect.target} = ${effect.value}`,
245
+ };
246
+ }
247
+ case "add": {
248
+ const p1 = `$${paramIdx.n++}`;
249
+ if (isNumeric) {
250
+ return {
251
+ tableName, entityParam, idParam,
252
+ assignments: [{ column: `${fieldName} = ${fieldName} + `, placeholder: p1, tsValue: valueTs }],
253
+ description: `${effect.target} += ${effect.value}`,
254
+ };
255
+ }
256
+ // Array append standalone
257
+ const p2 = `$${paramIdx.n++}`;
258
+ return {
259
+ tableName, entityParam, idParam,
260
+ assignments: [],
261
+ description: `${effect.target} += ${effect.value}`,
262
+ standalone: {
263
+ sql: `UPDATE ${tableName} SET ${fieldName} = ${fieldName} || jsonb_build_array(${p1}::jsonb), updated_at = NOW() WHERE id = ${p2} RETURNING *`,
264
+ params: [valueTs, idParam],
265
+ },
266
+ };
267
+ }
268
+ case "remove": {
269
+ const p1 = `$${paramIdx.n++}`;
270
+ if (isNumeric) {
271
+ return {
272
+ tableName, entityParam, idParam,
273
+ assignments: [{ column: `${fieldName} = ${fieldName} - `, placeholder: p1, tsValue: valueTs }],
274
+ description: `${effect.target} -= ${effect.value}`,
275
+ };
276
+ }
277
+ // Array remove — standalone
278
+ const p2 = `$${paramIdx.n++}`;
279
+ return {
280
+ tableName, entityParam, idParam,
281
+ assignments: [],
282
+ description: `${effect.target} -= ${effect.value}`,
283
+ standalone: {
284
+ 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 *`,
285
+ params: [valueTs, idParam],
286
+ },
287
+ };
288
+ }
289
+ }
290
+ }
291
+
292
+ // ─── Effect Batching ──────────────────────────────────────────────────────────
293
+ // Groups effects targeting the same (tableName, entityParam) into a single UPDATE.
294
+
295
+ interface BatchedUpdate {
296
+ tableName: string;
297
+ entityParam: string;
298
+ idParam: string;
299
+ setClauses: string[]; // e.g. ["hp = $1", "xp = xp + $2"]
300
+ paramValues: string[]; // TypeScript expressions for each $n
301
+ descriptions: string[];
302
+ }
303
+
304
+ function batchEffects(compiled: (CompiledEffect | null)[]): {
305
+ batches: BatchedUpdate[];
306
+ standalones: { sql: string; params: string[]; description: string }[];
307
+ } {
308
+ const batches = new Map<string, BatchedUpdate>(); // key: `${tableName}::${entityParam}::${idParam}`
309
+ const standalones: { sql: string; params: string[]; description: string }[] = [];
310
+
311
+ // Re-number parameters globally across all batches
312
+ let globalParamN = 1;
313
+
314
+ for (const effect of compiled) {
315
+ if (!effect) continue;
316
+
317
+ if (effect.standalone) {
318
+ // Re-number the standalone params
319
+ let sql = effect.standalone.sql;
320
+ const params = effect.standalone.params;
321
+ const renumbered: string[] = [];
322
+ let localN = 1;
323
+ for (const p of params) {
324
+ sql = sql.replace(`$${localN}`, `$${globalParamN}`);
325
+ renumbered.push(p);
326
+ globalParamN++;
327
+ localN++;
328
+ }
329
+ standalones.push({ sql, params: renumbered, description: effect.description });
330
+ continue;
331
+ }
332
+
333
+ if (effect.assignments.length === 0) continue;
334
+
335
+ const key = `${effect.tableName}::${effect.entityParam}::${effect.idParam}`;
336
+ if (!batches.has(key)) {
337
+ batches.set(key, {
338
+ tableName: effect.tableName,
339
+ entityParam: effect.entityParam,
340
+ idParam: effect.idParam,
341
+ setClauses: [],
342
+ paramValues: [],
343
+ descriptions: [],
344
+ });
345
+ }
346
+ const batch = batches.get(key)!;
347
+
348
+ for (const { column, placeholder, tsValue } of effect.assignments) {
349
+ // column may be "fieldName" (assign) or "fieldName = fieldName + " (numeric add/remove)
350
+ if (column.includes(" = ")) {
351
+ // Numeric add/remove: column is already "field = field + " — append placeholder
352
+ batch.setClauses.push(`${column}$${globalParamN}`);
353
+ } else {
354
+ batch.setClauses.push(`${column} = $${globalParamN}`);
355
+ }
356
+ batch.paramValues.push(tsValue);
357
+ globalParamN++;
358
+ }
359
+ batch.descriptions.push(effect.description);
360
+ }
361
+
362
+ return { batches: Array.from(batches.values()), standalones };
363
+ }
364
+
365
+ // ─── Main Capability Body Emitter ─────────────────────────────────────────────
366
+
367
+ export function emitCapabilityBody(
368
+ method: IR.IRMethod,
369
+ mod: IR.IRModule,
370
+ system: IR.IRSystem,
371
+ indent: string = " ",
372
+ ): string {
373
+ const lines: string[] = [];
374
+ const fetches = getEntityFetches(method, mod, system);
375
+
376
+ // 0. Destructure primitive params
377
+ const primitiveParams = method.input.filter(p => {
378
+ const isPrimitive = ["string", "uint", "int", "float", "bool", "timestamp", "uuid", "bytes", "json"].includes(p.type);
379
+ const isListOrSet = p.type.startsWith("list<") || p.type.startsWith("set<");
380
+ return (isPrimitive || isListOrSet) && !fetches.some(f => f.paramName === p.name);
381
+ });
382
+ if (primitiveParams.length > 0) {
383
+ lines.push(`${indent}const { ${primitiveParams.map(p => p.name).join(", ")} } = req.body;`);
384
+ lines.push(``);
385
+ }
386
+
387
+ // 1. Fetch entities — batch same-table fetches, parallelize different-table fetches
388
+ if (fetches.length > 0) {
389
+ lines.push(`${indent}// Fetch entities`);
390
+
391
+ // Group fetches by table
392
+ const byTable = new Map<string, EntityFetch[]>();
393
+ for (const f of fetches) {
394
+ if (!byTable.has(f.tableName)) byTable.set(f.tableName, []);
395
+ byTable.get(f.tableName)!.push(f);
396
+ }
397
+
398
+ const fetchGroups = Array.from(byTable.entries());
399
+
400
+ if (fetchGroups.length === 1 && fetchGroups[0][1].length === 1) {
401
+ // Single fetch — simple queryOne
402
+ const f = fetchGroups[0][1][0];
403
+ const idExpr = `req.body.${f.idField} || req.params.id`;
404
+ lines.push(`${indent}const ${f.paramName} = await queryOne(\`SELECT * FROM ${f.tableName} WHERE id = $1\`, [${idExpr}]);`);
405
+ lines.push(`${indent}if (!${f.paramName}) {`);
406
+ lines.push(`${indent} return res.status(404).json({ error: { code: "NOT_FOUND", message: "${f.paramName} not found" } });`);
407
+ lines.push(`${indent}}`);
408
+
409
+ } else if (fetchGroups.length === 1 && fetchGroups[0][1].length > 1) {
410
+ // Multiple fetches from the SAME table — batch into WHERE id = ANY($1::uuid[])
411
+ const [tableName, group] = fetchGroups[0];
412
+ const idExprs = group.map(f => `req.body.${f.idField} || req.params.id`);
413
+ lines.push(`${indent}// Batch fetch: ${group.map(f => f.paramName).join(", ")} from ${tableName} in one query`);
414
+ lines.push(`${indent}const __ids_${tableName} = [${idExprs.join(", ")}];`);
415
+ lines.push(`${indent}const __rows_${tableName} = await query(\`SELECT * FROM ${tableName} WHERE id = ANY($1::uuid[])\`, [__ids_${tableName}]);`);
416
+ lines.push(`${indent}const __map_${tableName} = new Map(__rows_${tableName}.map((r: any) => [r.id, r]));`);
417
+ for (const f of group) {
418
+ const idExpr = `req.body.${f.idField} || req.params.id`;
419
+ lines.push(`${indent}const ${f.paramName} = __map_${tableName}.get(${idExpr}) ?? null;`);
420
+ lines.push(`${indent}if (!${f.paramName}) {`);
421
+ lines.push(`${indent} return res.status(404).json({ error: { code: "NOT_FOUND", message: "${f.paramName} not found" } });`);
422
+ lines.push(`${indent}}`);
423
+ }
424
+
425
+ } else {
426
+ // Multiple fetches from DIFFERENT tables — run in parallel with Promise.all
427
+ lines.push(`${indent}// Parallel fetch from ${fetchGroups.length} tables`);
428
+ const resultVars: string[] = [];
429
+ const fetchExprs: string[] = [];
430
+
431
+ for (const [tableName, group] of fetchGroups) {
432
+ if (group.length === 1) {
433
+ const f = group[0];
434
+ const idExpr = `req.body.${f.idField} || req.params.id`;
435
+ resultVars.push(`__r_${f.paramName}`);
436
+ fetchExprs.push(`queryOne(\`SELECT * FROM ${tableName} WHERE id = $1\`, [${idExpr}])`);
437
+ } else {
438
+ // Same-table batch within a multi-table parallel fetch
439
+ const idExprs = group.map(f => `req.body.${f.idField} || req.params.id`);
440
+ resultVars.push(`__rows_${tableName}`);
441
+ fetchExprs.push(`query(\`SELECT * FROM ${tableName} WHERE id = ANY($1::uuid[])\`, [[${idExprs.join(", ")}]])`);
442
+ }
443
+ }
444
+
445
+ lines.push(`${indent}const [${resultVars.join(", ")}] = await Promise.all([`);
446
+ for (const expr of fetchExprs) lines.push(`${indent} ${expr},`);
447
+ lines.push(`${indent}]);`);
448
+
449
+ // Unpack results
450
+ let resultIdx = 0;
451
+ for (const [tableName, group] of fetchGroups) {
452
+ if (group.length === 1) {
453
+ const f = group[0];
454
+ lines.push(`${indent}const ${f.paramName} = ${resultVars[resultIdx]};`);
455
+ lines.push(`${indent}if (!${f.paramName}) {`);
456
+ lines.push(`${indent} return res.status(404).json({ error: { code: "NOT_FOUND", message: "${f.paramName} not found" } });`);
457
+ lines.push(`${indent}}`);
458
+ } else {
459
+ const mapVar = `__map_${tableName}`;
460
+ lines.push(`${indent}const ${mapVar} = new Map((${resultVars[resultIdx]} as any[]).map((r: any) => [r.id, r]));`);
461
+ for (const f of group) {
462
+ const idExpr = `req.body.${f.idField} || req.params.id`;
463
+ lines.push(`${indent}const ${f.paramName} = ${mapVar}.get(${idExpr}) ?? null;`);
464
+ lines.push(`${indent}if (!${f.paramName}) {`);
465
+ lines.push(`${indent} return res.status(404).json({ error: { code: "NOT_FOUND", message: "${f.paramName} not found" } });`);
466
+ lines.push(`${indent}}`);
467
+ }
468
+ }
469
+ resultIdx++;
470
+ }
471
+ }
472
+ lines.push(``);
473
+ }
474
+
475
+ // 2. Precondition checks
476
+ if (method.preconditions.length > 0) {
477
+ lines.push(`${indent}// Preconditions`);
478
+ for (const pre of method.preconditions) {
479
+ try {
480
+ lines.push(compilePrecondition(parseExprStr(pre.expression), indent));
481
+ } catch {
482
+ lines.push(`${indent}// CHECK: ${pre.description}`);
483
+ }
484
+ }
485
+ lines.push(``);
486
+ }
487
+
488
+ // 3. Effects — batch same-entity updates into single UPDATEs
489
+ if (method.effects.length > 0) {
490
+ lines.push(`${indent}// Effects (batched by entity to minimise round-trips)`);
491
+
492
+ const paramIdx = { n: 1 };
493
+ const compiled = method.effects.map(e => compileEffect(e, mod, system, paramIdx));
494
+ const { batches, standalones } = batchEffects(compiled);
495
+
496
+ // Emit batched UPDATEs
497
+ for (const batch of batches) {
498
+ if (batch.setClauses.length === 0) continue;
499
+ const idParamN = paramIdx.n++;
500
+ const setClauses = batch.setClauses.join(", ");
501
+ const sql = `UPDATE ${batch.tableName} SET ${setClauses}, updated_at = NOW() WHERE id = $${idParamN} RETURNING *`;
502
+ const params = [...batch.paramValues, batch.idParam].join(", ");
503
+ const resultVar = `__upd_${batch.entityParam}`;
504
+ lines.push(`${indent}// ${batch.descriptions.join("; ")}`);
505
+ lines.push(`${indent}const ${resultVar} = await query(\`${sql}\`, [${params}]);`);
506
+ lines.push(`${indent}if (!${resultVar} || ${resultVar}.length === 0) {`);
507
+ lines.push(`${indent} throw new Error("Update failed for ${batch.entityParam}");`);
508
+ lines.push(`${indent}}`);
509
+ }
510
+
511
+ // Emit standalone effects (JSONB, array ops)
512
+ for (const s of standalones) {
513
+ const resultVar = `__eff_${standalones.indexOf(s)}`;
514
+ lines.push(`${indent}// ${s.description}`);
515
+ lines.push(`${indent}const ${resultVar} = await query(\`${s.sql}\`, [${s.params.join(", ")}]);`);
516
+ lines.push(`${indent}if (!${resultVar} || ${resultVar}.length === 0) {`);
517
+ lines.push(`${indent} throw new Error("Effect failed: ${s.description.replace(/"/g, '\\"')}");`);
518
+ lines.push(`${indent}}`);
519
+ }
520
+
521
+ // Fallback for effects that couldn't be compiled
522
+ for (const effect of method.effects) {
523
+ const paramIdx2 = { n: 1 };
524
+ if (!compileEffect(effect, mod, system, paramIdx2)) {
525
+ lines.push(`${indent}// EFFECT: ${effect.target} ${effect.op === "assign" ? "=" : effect.op === "add" ? "+=" : "-="} ${effect.value}`);
526
+ lines.push(`${indent}// TODO: Implement this effect manually`);
527
+ }
528
+ }
529
+ lines.push(``);
530
+ }
531
+
532
+ // 4. Event emissions
533
+ if (method.emissions.length > 0) {
534
+ lines.push(`${indent}// Emit events`);
535
+ for (const ev of method.emissions) {
536
+ const payload = buildEventPayload(method, fetches);
537
+ if (method.sync === "transactional") {
538
+ lines.push(`${indent}await eventBus.publish("${ev}", ${payload}, "${mod.name}", auth.trace_id, __client);`);
539
+ } else {
540
+ lines.push(`${indent}await eventBus.publish("${ev}", ${payload}, "${mod.name}", auth.trace_id);`);
541
+ }
542
+ }
543
+ lines.push(``);
544
+ }
545
+
546
+ // 5. Return result
547
+ const resultEntity = fetches[0];
548
+ if (resultEntity) {
549
+ lines.push(`${indent}res.json({ ok: true, action: "${method.name}", entity: ${resultEntity.paramName} });`);
550
+ } else {
551
+ lines.push(`${indent}res.json({ ok: true, action: "${method.name}" });`);
552
+ }
553
+
554
+ return lines.join("\n");
555
+ }
556
+
557
+ function buildEventPayload(method: IR.IRMethod, fetches: EntityFetch[]): string {
558
+ const fields: string[] = fetches.map(f => `${f.paramName}_id: ${f.paramName}?.id`);
559
+ fields.push(`timestamp: new Date().toISOString()`);
560
+ fields.push(`actor_id: auth.actor_id`);
561
+ return `{ ${fields.join(", ")} }`;
562
+ }