@contractspec/lib.surface-runtime 0.3.0 → 0.3.3

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 (152) hide show
  1. package/dist/adapters/ai-sdk-stub.js +1 -6
  2. package/dist/adapters/blocknote-stub.js +3 -8
  3. package/dist/adapters/dnd-kit-adapter.js +6 -11
  4. package/dist/adapters/dnd-kit-stub.js +1 -6
  5. package/dist/adapters/floating-ui-stub.js +4 -9
  6. package/dist/adapters/index.js +7 -176
  7. package/dist/adapters/interfaces.js +0 -1
  8. package/dist/adapters/motion-stub.js +2 -7
  9. package/dist/adapters/resizable-panels-stub.js +7 -12
  10. package/dist/browser/adapters/ai-sdk-stub.js +1 -5
  11. package/dist/browser/adapters/blocknote-stub.js +3 -7
  12. package/dist/browser/adapters/dnd-kit-adapter.js +6 -10
  13. package/dist/browser/adapters/dnd-kit-stub.js +1 -5
  14. package/dist/browser/adapters/floating-ui-stub.js +4 -8
  15. package/dist/browser/adapters/index.js +7 -175
  16. package/dist/browser/adapters/motion-stub.js +2 -6
  17. package/dist/browser/adapters/resizable-panels-stub.js +5 -9
  18. package/dist/browser/evals/golden-harness.js +4 -825
  19. package/dist/browser/examples/pm-workbench.bundle.js +2 -316
  20. package/dist/browser/i18n/catalogs/en.js +1 -5
  21. package/dist/browser/i18n/catalogs/es.js +1 -5
  22. package/dist/browser/i18n/catalogs/fr.js +1 -5
  23. package/dist/browser/i18n/catalogs/index.js +3 -133
  24. package/dist/browser/i18n/index.js +3 -168
  25. package/dist/browser/i18n/keys.js +1 -5
  26. package/dist/browser/i18n/messages.js +6 -137
  27. package/dist/browser/index.js +4 -2470
  28. package/dist/browser/react/BundleProvider.js +12 -19
  29. package/dist/browser/react/BundleRenderer.js +15 -662
  30. package/dist/browser/react/OverlayConflictResolver.js +12 -180
  31. package/dist/browser/react/PatchProposalCard.js +9 -177
  32. package/dist/browser/react/RegionRenderer.js +14 -60
  33. package/dist/browser/react/SlotRenderer.js +15 -58
  34. package/dist/browser/react/WidgetPalette.js +6 -10
  35. package/dist/browser/react/index.js +14 -794
  36. package/dist/browser/runtime/apply-surface-patch.js +3 -208
  37. package/dist/browser/runtime/audit-events.js +8 -19
  38. package/dist/browser/runtime/build-context.js +2 -6
  39. package/dist/browser/runtime/extension-registry.js +3 -9
  40. package/dist/browser/runtime/field-renderer-registry.js +5 -11
  41. package/dist/browser/runtime/index.js +54 -1487
  42. package/dist/browser/runtime/overlay-alignment.js +8 -19
  43. package/dist/browser/runtime/overlay-signer.js +2 -7
  44. package/dist/browser/runtime/override-store.js +3 -9
  45. package/dist/browser/runtime/planner-prompt.js +2 -6
  46. package/dist/browser/runtime/planner-tools.js +3 -9
  47. package/dist/browser/runtime/policy-eval.js +3 -135
  48. package/dist/browser/runtime/preference-adapter.js +2 -62
  49. package/dist/browser/runtime/resolve-bundle.js +10 -447
  50. package/dist/browser/runtime/resolve-preferences.js +3 -7
  51. package/dist/browser/runtime/rollback.js +2 -325
  52. package/dist/browser/runtime/widget-registry.js +1 -5
  53. package/dist/browser/spec/define-module-bundle.js +2 -67
  54. package/dist/browser/spec/index.js +9 -317
  55. package/dist/browser/spec/validate-bundle.js +3 -8
  56. package/dist/browser/spec/validate-surface-patch.js +5 -11
  57. package/dist/browser/telemetry/index.js +4 -16
  58. package/dist/browser/telemetry/surface-metrics.js +7 -16
  59. package/dist/evals/golden-context.js +0 -1
  60. package/dist/evals/golden-harness.js +4 -826
  61. package/dist/examples/pm-workbench.bundle.js +2 -317
  62. package/dist/i18n/catalogs/en.js +1 -6
  63. package/dist/i18n/catalogs/es.js +1 -6
  64. package/dist/i18n/catalogs/fr.js +1 -6
  65. package/dist/i18n/catalogs/index.js +3 -134
  66. package/dist/i18n/index.js +3 -169
  67. package/dist/i18n/keys.js +1 -6
  68. package/dist/i18n/messages.js +6 -138
  69. package/dist/index.js +4 -2471
  70. package/dist/node/adapters/ai-sdk-stub.js +1 -5
  71. package/dist/node/adapters/blocknote-stub.js +3 -7
  72. package/dist/node/adapters/dnd-kit-adapter.js +6 -10
  73. package/dist/node/adapters/dnd-kit-stub.js +1 -5
  74. package/dist/node/adapters/floating-ui-stub.js +4 -8
  75. package/dist/node/adapters/index.js +7 -175
  76. package/dist/node/adapters/motion-stub.js +2 -6
  77. package/dist/node/adapters/resizable-panels-stub.js +5 -9
  78. package/dist/node/evals/golden-harness.js +4 -825
  79. package/dist/node/examples/pm-workbench.bundle.js +2 -316
  80. package/dist/node/i18n/catalogs/en.js +1 -5
  81. package/dist/node/i18n/catalogs/es.js +1 -5
  82. package/dist/node/i18n/catalogs/fr.js +1 -5
  83. package/dist/node/i18n/catalogs/index.js +3 -133
  84. package/dist/node/i18n/index.js +3 -168
  85. package/dist/node/i18n/keys.js +1 -5
  86. package/dist/node/i18n/messages.js +6 -137
  87. package/dist/node/index.js +4 -2470
  88. package/dist/node/react/BundleProvider.js +12 -19
  89. package/dist/node/react/BundleRenderer.js +15 -662
  90. package/dist/node/react/OverlayConflictResolver.js +12 -180
  91. package/dist/node/react/PatchProposalCard.js +9 -177
  92. package/dist/node/react/RegionRenderer.js +14 -60
  93. package/dist/node/react/SlotRenderer.js +15 -58
  94. package/dist/node/react/WidgetPalette.js +6 -10
  95. package/dist/node/react/index.js +14 -794
  96. package/dist/node/runtime/apply-surface-patch.js +3 -208
  97. package/dist/node/runtime/audit-events.js +8 -19
  98. package/dist/node/runtime/build-context.js +2 -6
  99. package/dist/node/runtime/extension-registry.js +3 -9
  100. package/dist/node/runtime/field-renderer-registry.js +5 -11
  101. package/dist/node/runtime/index.js +54 -1487
  102. package/dist/node/runtime/overlay-alignment.js +8 -19
  103. package/dist/node/runtime/overlay-signer.js +2 -7
  104. package/dist/node/runtime/override-store.js +3 -9
  105. package/dist/node/runtime/planner-prompt.js +2 -6
  106. package/dist/node/runtime/planner-tools.js +3 -9
  107. package/dist/node/runtime/policy-eval.js +3 -135
  108. package/dist/node/runtime/preference-adapter.js +2 -62
  109. package/dist/node/runtime/resolve-bundle.js +10 -447
  110. package/dist/node/runtime/resolve-preferences.js +3 -7
  111. package/dist/node/runtime/rollback.js +2 -325
  112. package/dist/node/runtime/widget-registry.js +1 -5
  113. package/dist/node/spec/define-module-bundle.js +2 -67
  114. package/dist/node/spec/index.js +9 -317
  115. package/dist/node/spec/validate-bundle.js +3 -8
  116. package/dist/node/spec/validate-surface-patch.js +5 -11
  117. package/dist/node/telemetry/index.js +4 -16
  118. package/dist/node/telemetry/surface-metrics.js +7 -16
  119. package/dist/react/BundleProvider.js +12 -20
  120. package/dist/react/BundleRenderer.js +15 -663
  121. package/dist/react/OverlayConflictResolver.js +12 -181
  122. package/dist/react/PatchProposalCard.js +9 -178
  123. package/dist/react/RegionRenderer.js +14 -61
  124. package/dist/react/SlotRenderer.js +15 -59
  125. package/dist/react/WidgetPalette.js +6 -11
  126. package/dist/react/index.js +14 -795
  127. package/dist/runtime/apply-surface-patch.js +3 -209
  128. package/dist/runtime/audit-events.js +8 -20
  129. package/dist/runtime/build-context.js +2 -7
  130. package/dist/runtime/extension-registry.js +3 -10
  131. package/dist/runtime/field-renderer-registry.js +9 -16
  132. package/dist/runtime/index.js +54 -1488
  133. package/dist/runtime/overlay-alignment.js +8 -20
  134. package/dist/runtime/overlay-signer.js +2 -8
  135. package/dist/runtime/override-store.js +3 -10
  136. package/dist/runtime/planner-prompt.js +11 -9
  137. package/dist/runtime/planner-tools.js +4 -11
  138. package/dist/runtime/policy-eval.js +3 -136
  139. package/dist/runtime/preference-adapter.js +2 -63
  140. package/dist/runtime/resolve-bundle.js +10 -448
  141. package/dist/runtime/resolve-preferences.js +3 -8
  142. package/dist/runtime/rollback.js +2 -326
  143. package/dist/runtime/widget-registry.js +1 -6
  144. package/dist/spec/define-module-bundle.js +2 -68
  145. package/dist/spec/index.js +9 -318
  146. package/dist/spec/types.js +0 -1
  147. package/dist/spec/validate-bundle.js +3 -9
  148. package/dist/spec/validate-surface-patch.js +5 -12
  149. package/dist/spec/verification-snapshot-types.js +0 -1
  150. package/dist/telemetry/index.js +4 -17
  151. package/dist/telemetry/surface-metrics.js +7 -17
  152. package/package.json +7 -7
@@ -1,325 +1,5 @@
1
- // src/spec/validate-surface-patch.ts
2
- var VALID_OPS = [
3
- "insert-node",
4
- "replace-node",
5
- "remove-node",
6
- "move-node",
7
- "resize-panel",
8
- "set-layout",
9
- "reveal-field",
10
- "hide-field",
11
- "promote-action",
12
- "set-focus"
13
- ];
14
- var VALID_NODE_KINDS = [
15
- "metric-strip",
16
- "data-view",
17
- "entity-card",
18
- "entity-header",
19
- "entity-summary",
20
- "entity-section",
21
- "entity-field",
22
- "entity-activity",
23
- "entity-relations",
24
- "entity-timeline",
25
- "entity-comments",
26
- "entity-attachments",
27
- "entity-view-switcher",
28
- "entity-automation-panel",
29
- "rich-doc",
30
- "chat-thread",
31
- "assistant-panel",
32
- "action-bar",
33
- "timeline",
34
- "board",
35
- "table",
36
- "calendar",
37
- "form",
38
- "chart",
39
- "relation-graph",
40
- "custom-widget"
41
- ];
42
- function validateSurfaceNode(node, path) {
43
- if (!node.nodeId || typeof node.nodeId !== "string") {
44
- throw new Error(`${path}: nodeId must be a non-empty string`);
45
- }
46
- if (!node.kind || !VALID_NODE_KINDS.includes(node.kind)) {
47
- throw new Error(`${path}: kind must be one of ${VALID_NODE_KINDS.join(", ")}`);
48
- }
49
- if (node.children) {
50
- for (let i = 0;i < node.children.length; i++) {
51
- const child = node.children[i];
52
- if (child)
53
- validateSurfaceNode(child, `${path}.children[${i}]`);
54
- }
55
- }
56
- }
57
- function validateSurfacePatchOp(op, index) {
58
- const path = `ops[${index}]`;
59
- if (!op || typeof op !== "object" || !("op" in op)) {
60
- throw new Error(`${path}: must be an object with op field`);
61
- }
62
- const opType = op.op;
63
- if (!VALID_OPS.includes(opType)) {
64
- throw new Error(`${path}: op must be one of ${VALID_OPS.join(", ")}`);
65
- }
66
- switch (op.op) {
67
- case "insert-node":
68
- if (!op.slotId || typeof op.slotId !== "string") {
69
- throw new Error(`${path}: insert-node requires slotId string`);
70
- }
71
- if (!op.node) {
72
- throw new Error(`${path}: insert-node requires node`);
73
- }
74
- validateSurfaceNode(op.node, `${path}.node`);
75
- if (op.index !== undefined && typeof op.index !== "number") {
76
- throw new Error(`${path}: insert-node index must be number if present`);
77
- }
78
- break;
79
- case "replace-node":
80
- if (!op.nodeId || typeof op.nodeId !== "string") {
81
- throw new Error(`${path}: replace-node requires nodeId string`);
82
- }
83
- if (!op.node) {
84
- throw new Error(`${path}: replace-node requires node`);
85
- }
86
- validateSurfaceNode(op.node, `${path}.node`);
87
- break;
88
- case "remove-node":
89
- if (!op.nodeId || typeof op.nodeId !== "string") {
90
- throw new Error(`${path}: remove-node requires nodeId string`);
91
- }
92
- break;
93
- case "move-node":
94
- if (!op.nodeId || typeof op.nodeId !== "string") {
95
- throw new Error(`${path}: move-node requires nodeId string`);
96
- }
97
- if (!op.toSlotId || typeof op.toSlotId !== "string") {
98
- throw new Error(`${path}: move-node requires toSlotId string`);
99
- }
100
- if (op.index !== undefined && typeof op.index !== "number") {
101
- throw new Error(`${path}: move-node index must be number if present`);
102
- }
103
- break;
104
- case "resize-panel":
105
- if (!op.persistKey || typeof op.persistKey !== "string") {
106
- throw new Error(`${path}: resize-panel requires persistKey string`);
107
- }
108
- if (!Array.isArray(op.sizes) || op.sizes.some((s) => typeof s !== "number")) {
109
- throw new Error(`${path}: resize-panel requires sizes number[]`);
110
- }
111
- break;
112
- case "set-layout":
113
- if (!op.layoutId || typeof op.layoutId !== "string") {
114
- throw new Error(`${path}: set-layout requires layoutId string`);
115
- }
116
- break;
117
- case "reveal-field":
118
- case "hide-field":
119
- if (!op.fieldId || typeof op.fieldId !== "string") {
120
- throw new Error(`${path}: ${op.op} requires fieldId string`);
121
- }
122
- break;
123
- case "promote-action": {
124
- if (!op.actionId || typeof op.actionId !== "string") {
125
- throw new Error(`${path}: promote-action requires actionId string`);
126
- }
127
- const validPlacements = ["header", "inline", "context", "assistant"];
128
- if (!op.placement || !validPlacements.includes(op.placement)) {
129
- throw new Error(`${path}: promote-action placement must be one of ${validPlacements.join(", ")}`);
130
- }
131
- break;
132
- }
133
- case "set-focus":
134
- if (!op.targetId || typeof op.targetId !== "string") {
135
- throw new Error(`${path}: set-focus requires targetId string`);
136
- }
137
- break;
138
- default:
139
- throw new Error(`${path}: unknown op "${opType}"`);
140
- }
141
- }
142
- function validateSurfacePatch(ops) {
143
- if (!Array.isArray(ops)) {
144
- throw new Error("Patch ops must be an array");
145
- }
146
- for (let i = 0;i < ops.length; i++) {
147
- const op = ops[i];
148
- if (op)
149
- validateSurfacePatchOp(op, i);
150
- }
151
- }
152
- function validateSurfaceNodeAgainstKinds(node, allowedKinds, path) {
153
- if (!allowedKinds.includes(node.kind)) {
154
- throw new Error(`${path}: kind "${node.kind}" not in allowed list [${allowedKinds.join(", ")}]`);
155
- }
156
- if (node.children) {
157
- for (let i = 0;i < node.children.length; i++) {
158
- const child = node.children[i];
159
- if (child)
160
- validateSurfaceNodeAgainstKinds(child, allowedKinds, `${path}.children[${i}]`);
161
- }
162
- }
163
- }
164
- function validatePatchProposal(ops, constraints) {
165
- if (!Array.isArray(ops)) {
166
- throw new Error("Patch ops must be an array");
167
- }
168
- const { allowedOps, allowedSlots, allowedNodeKinds } = constraints;
169
- for (let i = 0;i < ops.length; i++) {
170
- const op = ops[i];
171
- if (!op)
172
- continue;
173
- const path = `ops[${i}]`;
174
- if (!allowedOps.includes(op.op)) {
175
- throw new Error(`${path}: op "${op.op}" not in allowed list [${allowedOps.join(", ")}]`);
176
- }
177
- switch (op.op) {
178
- case "insert-node":
179
- if (!allowedSlots.includes(op.slotId)) {
180
- throw new Error(`${path}: slotId "${op.slotId}" not in allowed slots [${allowedSlots.join(", ")}]`);
181
- }
182
- if (op.node) {
183
- validateSurfaceNodeAgainstKinds(op.node, allowedNodeKinds, `${path}.node`);
184
- }
185
- break;
186
- case "move-node":
187
- if (!allowedSlots.includes(op.toSlotId)) {
188
- throw new Error(`${path}: toSlotId "${op.toSlotId}" not in allowed slots [${allowedSlots.join(", ")}]`);
189
- }
190
- break;
191
- case "replace-node":
192
- if (op.node) {
193
- validateSurfaceNodeAgainstKinds(op.node, allowedNodeKinds, `${path}.node`);
194
- }
195
- break;
196
- default:
197
- break;
198
- }
199
- }
200
- validateSurfacePatch(ops);
201
- }
202
-
203
- // src/runtime/apply-surface-patch.ts
204
- import { Logger } from "@contractspec/lib.observability";
205
- var logger = new Logger("@contractspec/lib.surface-runtime");
206
- function findNode(nodes, nodeId) {
207
- for (const n of nodes) {
208
- if (n.nodeId === nodeId)
209
- return n;
210
- if (n.children) {
211
- const found = findNode(n.children, nodeId);
212
- if (found)
213
- return found;
214
- }
215
- }
216
- return;
217
- }
218
- function collectNodeIds(nodes) {
219
- const ids = new Set;
220
- for (const n of nodes) {
221
- ids.add(n.nodeId);
222
- if (n.children)
223
- collectNodeIds(n.children).forEach((id) => ids.add(id));
224
- }
225
- return ids;
226
- }
227
- function validateOp(op, nodeIds) {
228
- if (op.op === "remove-node" || op.op === "replace-node" || op.op === "move-node") {
229
- if (!nodeIds.has(op.nodeId)) {
230
- throw new Error(`Patch op references unknown nodeId: ${op.nodeId}`);
231
- }
232
- }
233
- if (op.op === "insert-node" && !op.node?.nodeId) {
234
- throw new Error("insert-node requires node with nodeId");
235
- }
236
- }
237
- function produceInverse(op, plan) {
238
- switch (op.op) {
239
- case "insert-node":
240
- return op.node ? { op: "remove-node", nodeId: op.node.nodeId } : null;
241
- case "remove-node": {
242
- const node = findNode(plan.nodes, op.nodeId);
243
- return node ? { op: "insert-node", slotId: "primary", node } : null;
244
- }
245
- case "replace-node": {
246
- const prev = findNode(plan.nodes, op.nodeId);
247
- return prev ? { op: "replace-node", nodeId: op.nodeId, node: prev } : null;
248
- }
249
- case "set-layout":
250
- return { op: "set-layout", layoutId: plan.layoutId };
251
- case "reveal-field":
252
- return { op: "hide-field", fieldId: op.fieldId };
253
- case "hide-field":
254
- return { op: "reveal-field", fieldId: op.fieldId };
255
- case "move-node":
256
- case "resize-panel":
257
- case "set-focus":
258
- case "promote-action":
259
- return null;
260
- default:
261
- return null;
262
- }
263
- }
264
- function applySurfacePatch(plan, ops) {
265
- if (ops.length === 0)
266
- return { plan, inverseOps: [] };
267
- validateSurfacePatch(ops);
268
- const nodeIds = collectNodeIds(plan.nodes);
269
- for (const op of ops) {
270
- validateOp(op, nodeIds);
271
- }
272
- let nextNodes = [...plan.nodes];
273
- let nextLayoutId = plan.layoutId;
274
- const inverseOps = [];
275
- for (const op of ops) {
276
- const inv = produceInverse(op, {
277
- ...plan,
278
- nodes: nextNodes,
279
- layoutId: nextLayoutId
280
- });
281
- if (inv)
282
- inverseOps.unshift(inv);
283
- switch (op.op) {
284
- case "insert-node":
285
- if (op.node)
286
- nextNodes = [...nextNodes, op.node];
287
- break;
288
- case "remove-node":
289
- nextNodes = nextNodes.filter((n) => n.nodeId !== op.nodeId);
290
- break;
291
- case "replace-node": {
292
- const replace = (nodes) => nodes.map((n) => n.nodeId === op.nodeId ? op.node : { ...n, children: n.children ? replace(n.children) : undefined });
293
- nextNodes = replace(nextNodes);
294
- break;
295
- }
296
- case "set-layout":
297
- nextLayoutId = op.layoutId;
298
- break;
299
- case "move-node":
300
- case "resize-panel":
301
- case "set-focus":
302
- case "reveal-field":
303
- case "hide-field":
304
- case "promote-action":
305
- break;
306
- }
307
- }
308
- const result = {
309
- plan: { ...plan, nodes: nextNodes, layoutId: nextLayoutId },
310
- inverseOps
311
- };
312
- logger.info("bundle.surface.patch.applied", {
313
- bundleKey: plan.bundleKey,
314
- surfaceId: plan.surfaceId,
315
- opCount: ops.length,
316
- opTypes: [...new Set(ops.map((o) => o.op))]
317
- });
318
- return result;
319
- }
320
-
321
- // src/runtime/rollback.ts
322
- function rollbackSurfacePatches(plan, approvalStack, count = 1) {
1
+ import { applySurfacePatch } from "./apply-surface-patch";
2
+ export function rollbackSurfacePatches(plan, approvalStack, count = 1) {
323
3
  if (count <= 0 || approvalStack.length === 0) {
324
4
  return { plan, revertedCount: 0, remainingStack: approvalStack };
325
5
  }
@@ -342,6 +22,3 @@ function rollbackSurfacePatches(plan, approvalStack, count = 1) {
342
22
  remainingStack: remaining
343
23
  };
344
24
  }
345
- export {
346
- rollbackSurfacePatches
347
- };
@@ -1,4 +1,3 @@
1
- // src/runtime/widget-registry.ts
2
1
  function validateTrust(trust) {
3
2
  if (trust === "ephemeral-ai") {
4
3
  throw new Error("Widgets cannot be registered with ephemeral-ai trust. Registration is a code/package concern.");
@@ -7,7 +6,7 @@ function validateTrust(trust) {
7
6
  throw new Error(`Invalid widget trust: ${trust}`);
8
7
  }
9
8
  }
10
- function createWidgetRegistry() {
9
+ export function createWidgetRegistry() {
11
10
  const entries = new Map;
12
11
  return {
13
12
  register(entry) {
@@ -31,6 +30,3 @@ function createWidgetRegistry() {
31
30
  }
32
31
  };
33
32
  }
34
- export {
35
- createWidgetRegistry
36
- };
@@ -1,67 +1,5 @@
1
- // src/spec/validate-bundle.ts
2
- var KNOWN_NODE_KIND_RENDERERS = new Set([
3
- "entity-section",
4
- "entity-field",
5
- "action-bar",
6
- "table",
7
- "timeline",
8
- "rich-doc",
9
- "chat-thread",
10
- "assistant-panel",
11
- "entity-card",
12
- "entity-header",
13
- "entity-summary",
14
- "entity-activity",
15
- "entity-relations",
16
- "relation-graph",
17
- "custom-widget"
18
- ]);
19
- function collectSlotIdsFromRegion(node) {
20
- const ids = [];
21
- if (node.type === "slot") {
22
- ids.push(node.slotId);
23
- }
24
- if (node.type === "panel-group" || node.type === "stack") {
25
- for (const child of node.children) {
26
- ids.push(...collectSlotIdsFromRegion(child));
27
- }
28
- }
29
- if (node.type === "tabs") {
30
- for (const tab of node.tabs) {
31
- ids.push(...collectSlotIdsFromRegion(tab.child));
32
- }
33
- }
34
- if (node.type === "floating") {
35
- ids.push(node.anchorSlotId);
36
- ids.push(...collectSlotIdsFromRegion(node.child));
37
- }
38
- return ids;
39
- }
40
- function validateLayoutSlots(surface) {
41
- const declaredSlotIds = new Set(surface.slots.map((s) => s.slotId));
42
- for (const layout of surface.layouts) {
43
- const layoutSlotIds = collectSlotIdsFromRegion(layout.root);
44
- for (const slotId of layoutSlotIds) {
45
- if (!declaredSlotIds.has(slotId)) {
46
- throw new Error(`Surface "${surface.surfaceId}" layout "${layout.layoutId}" references undeclared slot "${slotId}". Declared slots: ${[...declaredSlotIds].join(", ")}`);
47
- }
48
- }
49
- }
50
- }
51
- function validateBundleNodeKinds(surface) {
52
- const warnings = [];
53
- for (const slot of surface.slots) {
54
- for (const kind of slot.accepts) {
55
- if (!KNOWN_NODE_KIND_RENDERERS.has(kind)) {
56
- warnings.push(`Surface "${surface.surfaceId}" slot "${slot.slotId}" accepts "${kind}" which has no dedicated renderer (generic fallback used)`);
57
- }
58
- }
59
- }
60
- return { warnings };
61
- }
62
-
63
- // src/spec/define-module-bundle.ts
64
- function defineModuleBundle(spec) {
1
+ import { validateLayoutSlots } from "./validate-bundle";
2
+ export function defineModuleBundle(spec) {
65
3
  if (!spec.meta?.key || !spec.meta?.version || !spec.meta?.title) {
66
4
  throw new Error("ModuleBundleSpec must have meta.key, meta.version, and meta.title");
67
5
  }
@@ -108,6 +46,3 @@ function defineModuleBundle(spec) {
108
46
  }
109
47
  return spec;
110
48
  }
111
- export {
112
- defineModuleBundle
113
- };