@contractspec/lib.surface-runtime 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (232) hide show
  1. package/README.md +164 -0
  2. package/dist/adapters/ai-sdk-stub.d.ts +5 -0
  3. package/dist/adapters/ai-sdk-stub.js +13 -0
  4. package/dist/adapters/blocknote-stub.d.ts +6 -0
  5. package/dist/adapters/blocknote-stub.js +31 -0
  6. package/dist/adapters/dnd-kit-adapter.d.ts +13 -0
  7. package/dist/adapters/dnd-kit-adapter.js +44 -0
  8. package/dist/adapters/dnd-kit-stub.d.ts +6 -0
  9. package/dist/adapters/dnd-kit-stub.js +8 -0
  10. package/dist/adapters/floating-ui-stub.d.ts +6 -0
  11. package/dist/adapters/floating-ui-stub.js +19 -0
  12. package/dist/adapters/index.d.ts +11 -0
  13. package/dist/adapters/index.js +176 -0
  14. package/dist/adapters/interfaces.d.ts +75 -0
  15. package/dist/adapters/interfaces.js +1 -0
  16. package/dist/adapters/motion-stub.d.ts +7 -0
  17. package/dist/adapters/motion-stub.js +27 -0
  18. package/dist/adapters/motion-stub.test.d.ts +1 -0
  19. package/dist/adapters/resizable-panels-stub.d.ts +6 -0
  20. package/dist/adapters/resizable-panels-stub.js +46 -0
  21. package/dist/adapters/resizable-panels-stub.test.d.ts +1 -0
  22. package/dist/browser/adapters/ai-sdk-stub.js +12 -0
  23. package/dist/browser/adapters/blocknote-stub.js +30 -0
  24. package/dist/browser/adapters/dnd-kit-adapter.js +43 -0
  25. package/dist/browser/adapters/dnd-kit-stub.js +7 -0
  26. package/dist/browser/adapters/floating-ui-stub.js +18 -0
  27. package/dist/browser/adapters/index.js +175 -0
  28. package/dist/browser/adapters/interfaces.js +0 -0
  29. package/dist/browser/adapters/motion-stub.js +26 -0
  30. package/dist/browser/adapters/resizable-panels-stub.js +45 -0
  31. package/dist/browser/evals/golden-context.js +0 -0
  32. package/dist/browser/evals/golden-harness.js +848 -0
  33. package/dist/browser/examples/pm-workbench.bundle.js +476 -0
  34. package/dist/browser/i18n/catalogs/en.js +71 -0
  35. package/dist/browser/i18n/catalogs/es.js +32 -0
  36. package/dist/browser/i18n/catalogs/fr.js +32 -0
  37. package/dist/browser/i18n/catalogs/index.js +133 -0
  38. package/dist/browser/i18n/index.js +173 -0
  39. package/dist/browser/i18n/keys.js +19 -0
  40. package/dist/browser/i18n/messages.js +143 -0
  41. package/dist/browser/index.js +2466 -0
  42. package/dist/browser/react/BundleProvider.js +47 -0
  43. package/dist/browser/react/BundleRenderer.js +726 -0
  44. package/dist/browser/react/OverlayConflictResolver.js +255 -0
  45. package/dist/browser/react/PatchProposalCard.js +255 -0
  46. package/dist/browser/react/RegionRenderer.js +128 -0
  47. package/dist/browser/react/SlotRenderer.js +118 -0
  48. package/dist/browser/react/WidgetPalette.js +59 -0
  49. package/dist/browser/react/index.js +792 -0
  50. package/dist/browser/runtime/apply-surface-patch.js +322 -0
  51. package/dist/browser/runtime/audit-events.js +137 -0
  52. package/dist/browser/runtime/build-context.js +55 -0
  53. package/dist/browser/runtime/extension-registry.js +58 -0
  54. package/dist/browser/runtime/field-renderer-registry.js +145 -0
  55. package/dist/browser/runtime/index.js +1496 -0
  56. package/dist/browser/runtime/overlay-alignment.js +83 -0
  57. package/dist/browser/runtime/overlay-signer.js +15 -0
  58. package/dist/browser/runtime/override-store.js +52 -0
  59. package/dist/browser/runtime/planner-prompt.js +67 -0
  60. package/dist/browser/runtime/planner-tools.js +77 -0
  61. package/dist/browser/runtime/policy-eval.js +155 -0
  62. package/dist/browser/runtime/preference-adapter.js +67 -0
  63. package/dist/browser/runtime/resolve-bundle.js +767 -0
  64. package/dist/browser/runtime/resolve-preferences.js +59 -0
  65. package/dist/browser/runtime/rollback.js +347 -0
  66. package/dist/browser/runtime/widget-registry.js +36 -0
  67. package/dist/browser/spec/define-module-bundle.js +113 -0
  68. package/dist/browser/spec/index.js +319 -0
  69. package/dist/browser/spec/types.js +0 -0
  70. package/dist/browser/spec/validate-bundle.js +65 -0
  71. package/dist/browser/spec/validate-surface-patch.js +206 -0
  72. package/dist/browser/spec/verification-snapshot-types.js +0 -0
  73. package/dist/browser/telemetry/index.js +20 -0
  74. package/dist/browser/telemetry/surface-metrics.js +20 -0
  75. package/dist/evals/golden-context.d.ts +24 -0
  76. package/dist/evals/golden-context.js +1 -0
  77. package/dist/evals/golden-harness.d.ts +29 -0
  78. package/dist/evals/golden-harness.js +849 -0
  79. package/dist/evals/golden-harness.test.d.ts +1 -0
  80. package/dist/examples/pm-workbench.bundle.d.ts +177 -0
  81. package/dist/examples/pm-workbench.bundle.js +477 -0
  82. package/dist/i18n/catalogs/en.d.ts +1 -0
  83. package/dist/i18n/catalogs/en.js +72 -0
  84. package/dist/i18n/catalogs/es.d.ts +1 -0
  85. package/dist/i18n/catalogs/es.js +33 -0
  86. package/dist/i18n/catalogs/fr.d.ts +1 -0
  87. package/dist/i18n/catalogs/fr.js +33 -0
  88. package/dist/i18n/catalogs/index.d.ts +3 -0
  89. package/dist/i18n/catalogs/index.js +134 -0
  90. package/dist/i18n/index.d.ts +5 -0
  91. package/dist/i18n/index.js +174 -0
  92. package/dist/i18n/keys.d.ts +20 -0
  93. package/dist/i18n/keys.js +20 -0
  94. package/dist/i18n/messages.d.ts +5 -0
  95. package/dist/i18n/messages.js +144 -0
  96. package/dist/index.d.ts +4 -0
  97. package/dist/index.js +2467 -0
  98. package/dist/node/adapters/ai-sdk-stub.js +12 -0
  99. package/dist/node/adapters/blocknote-stub.js +30 -0
  100. package/dist/node/adapters/dnd-kit-adapter.js +43 -0
  101. package/dist/node/adapters/dnd-kit-stub.js +7 -0
  102. package/dist/node/adapters/floating-ui-stub.js +18 -0
  103. package/dist/node/adapters/index.js +175 -0
  104. package/dist/node/adapters/interfaces.js +0 -0
  105. package/dist/node/adapters/motion-stub.js +26 -0
  106. package/dist/node/adapters/resizable-panels-stub.js +45 -0
  107. package/dist/node/evals/golden-context.js +0 -0
  108. package/dist/node/evals/golden-harness.js +848 -0
  109. package/dist/node/examples/pm-workbench.bundle.js +476 -0
  110. package/dist/node/i18n/catalogs/en.js +71 -0
  111. package/dist/node/i18n/catalogs/es.js +32 -0
  112. package/dist/node/i18n/catalogs/fr.js +32 -0
  113. package/dist/node/i18n/catalogs/index.js +133 -0
  114. package/dist/node/i18n/index.js +173 -0
  115. package/dist/node/i18n/keys.js +19 -0
  116. package/dist/node/i18n/messages.js +143 -0
  117. package/dist/node/index.js +2466 -0
  118. package/dist/node/react/BundleProvider.js +47 -0
  119. package/dist/node/react/BundleRenderer.js +726 -0
  120. package/dist/node/react/OverlayConflictResolver.js +255 -0
  121. package/dist/node/react/PatchProposalCard.js +255 -0
  122. package/dist/node/react/RegionRenderer.js +128 -0
  123. package/dist/node/react/SlotRenderer.js +118 -0
  124. package/dist/node/react/WidgetPalette.js +59 -0
  125. package/dist/node/react/index.js +792 -0
  126. package/dist/node/runtime/apply-surface-patch.js +322 -0
  127. package/dist/node/runtime/audit-events.js +137 -0
  128. package/dist/node/runtime/build-context.js +55 -0
  129. package/dist/node/runtime/extension-registry.js +58 -0
  130. package/dist/node/runtime/field-renderer-registry.js +145 -0
  131. package/dist/node/runtime/index.js +1496 -0
  132. package/dist/node/runtime/overlay-alignment.js +83 -0
  133. package/dist/node/runtime/overlay-signer.js +15 -0
  134. package/dist/node/runtime/override-store.js +52 -0
  135. package/dist/node/runtime/planner-prompt.js +67 -0
  136. package/dist/node/runtime/planner-tools.js +77 -0
  137. package/dist/node/runtime/policy-eval.js +155 -0
  138. package/dist/node/runtime/preference-adapter.js +67 -0
  139. package/dist/node/runtime/resolve-bundle.js +767 -0
  140. package/dist/node/runtime/resolve-preferences.js +59 -0
  141. package/dist/node/runtime/rollback.js +347 -0
  142. package/dist/node/runtime/widget-registry.js +36 -0
  143. package/dist/node/spec/define-module-bundle.js +113 -0
  144. package/dist/node/spec/index.js +319 -0
  145. package/dist/node/spec/types.js +0 -0
  146. package/dist/node/spec/validate-bundle.js +65 -0
  147. package/dist/node/spec/validate-surface-patch.js +206 -0
  148. package/dist/node/spec/verification-snapshot-types.js +0 -0
  149. package/dist/node/telemetry/index.js +20 -0
  150. package/dist/node/telemetry/surface-metrics.js +20 -0
  151. package/dist/react/BundleProvider.d.ts +13 -0
  152. package/dist/react/BundleProvider.js +48 -0
  153. package/dist/react/BundleRenderer.d.ts +22 -0
  154. package/dist/react/BundleRenderer.js +727 -0
  155. package/dist/react/OverlayConflictResolver.d.ts +15 -0
  156. package/dist/react/OverlayConflictResolver.js +256 -0
  157. package/dist/react/PatchProposalCard.d.ts +13 -0
  158. package/dist/react/PatchProposalCard.js +256 -0
  159. package/dist/react/RegionRenderer.d.ts +13 -0
  160. package/dist/react/RegionRenderer.js +129 -0
  161. package/dist/react/SlotRenderer.d.ts +13 -0
  162. package/dist/react/SlotRenderer.js +119 -0
  163. package/dist/react/WidgetPalette.d.ts +12 -0
  164. package/dist/react/WidgetPalette.js +60 -0
  165. package/dist/react/index.d.ts +7 -0
  166. package/dist/react/index.js +793 -0
  167. package/dist/runtime/apply-surface-patch.d.ts +15 -0
  168. package/dist/runtime/apply-surface-patch.js +323 -0
  169. package/dist/runtime/apply-surface-patch.test.d.ts +1 -0
  170. package/dist/runtime/audit-events.d.ts +70 -0
  171. package/dist/runtime/audit-events.js +138 -0
  172. package/dist/runtime/audit-events.test.d.ts +1 -0
  173. package/dist/runtime/build-context.d.ts +9 -0
  174. package/dist/runtime/build-context.js +56 -0
  175. package/dist/runtime/extension-registry.d.ts +39 -0
  176. package/dist/runtime/extension-registry.js +59 -0
  177. package/dist/runtime/field-renderer-registry.d.ts +23 -0
  178. package/dist/runtime/field-renderer-registry.js +146 -0
  179. package/dist/runtime/field-renderer-registry.test.d.ts +1 -0
  180. package/dist/runtime/index.d.ts +16 -0
  181. package/dist/runtime/index.js +1497 -0
  182. package/dist/runtime/overlay-alignment.d.ts +49 -0
  183. package/dist/runtime/overlay-alignment.js +84 -0
  184. package/dist/runtime/overlay-alignment.test.d.ts +1 -0
  185. package/dist/runtime/overlay-signer.d.ts +15 -0
  186. package/dist/runtime/overlay-signer.js +16 -0
  187. package/dist/runtime/override-store.d.ts +44 -0
  188. package/dist/runtime/override-store.js +53 -0
  189. package/dist/runtime/override-store.test.d.ts +1 -0
  190. package/dist/runtime/planner-prompt.d.ts +39 -0
  191. package/dist/runtime/planner-prompt.js +68 -0
  192. package/dist/runtime/planner-prompt.test.d.ts +1 -0
  193. package/dist/runtime/planner-tools.d.ts +106 -0
  194. package/dist/runtime/planner-tools.js +78 -0
  195. package/dist/runtime/planner-tools.test.d.ts +1 -0
  196. package/dist/runtime/policy-eval.d.ts +23 -0
  197. package/dist/runtime/policy-eval.js +156 -0
  198. package/dist/runtime/preference-adapter.d.ts +6 -0
  199. package/dist/runtime/preference-adapter.js +68 -0
  200. package/dist/runtime/resolve-bundle.d.ts +68 -0
  201. package/dist/runtime/resolve-bundle.js +768 -0
  202. package/dist/runtime/resolve-bundle.test.d.ts +1 -0
  203. package/dist/runtime/resolve-preferences.d.ts +9 -0
  204. package/dist/runtime/resolve-preferences.js +60 -0
  205. package/dist/runtime/resolve-preferences.test.d.ts +1 -0
  206. package/dist/runtime/rollback.d.ts +21 -0
  207. package/dist/runtime/rollback.js +348 -0
  208. package/dist/runtime/rollback.test.d.ts +1 -0
  209. package/dist/runtime/widget-registry.d.ts +26 -0
  210. package/dist/runtime/widget-registry.js +37 -0
  211. package/dist/runtime/widget-registry.test.d.ts +1 -0
  212. package/dist/spec/define-module-bundle.d.ts +17 -0
  213. package/dist/spec/define-module-bundle.js +114 -0
  214. package/dist/spec/define-module-bundle.test.d.ts +1 -0
  215. package/dist/spec/index.d.ts +5 -0
  216. package/dist/spec/index.js +320 -0
  217. package/dist/spec/types.d.ts +494 -0
  218. package/dist/spec/types.js +1 -0
  219. package/dist/spec/validate-bundle.d.ts +23 -0
  220. package/dist/spec/validate-bundle.js +66 -0
  221. package/dist/spec/validate-bundle.test.d.ts +1 -0
  222. package/dist/spec/validate-surface-patch.d.ts +39 -0
  223. package/dist/spec/validate-surface-patch.js +207 -0
  224. package/dist/spec/validate-surface-patch.test.d.ts +1 -0
  225. package/dist/spec/verification-snapshot-types.d.ts +23 -0
  226. package/dist/spec/verification-snapshot-types.js +1 -0
  227. package/dist/spec/verification-snapshot.test.d.ts +5 -0
  228. package/dist/telemetry/index.d.ts +5 -0
  229. package/dist/telemetry/index.js +21 -0
  230. package/dist/telemetry/surface-metrics.d.ts +17 -0
  231. package/dist/telemetry/surface-metrics.js +21 -0
  232. package/package.json +920 -0
@@ -0,0 +1,476 @@
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
+ // src/spec/validate-bundle.ts
203
+ var KNOWN_NODE_KIND_RENDERERS = new Set([
204
+ "entity-section",
205
+ "entity-field",
206
+ "action-bar",
207
+ "table",
208
+ "timeline",
209
+ "rich-doc",
210
+ "chat-thread",
211
+ "assistant-panel",
212
+ "entity-card",
213
+ "entity-header",
214
+ "entity-summary",
215
+ "entity-activity",
216
+ "entity-relations",
217
+ "relation-graph",
218
+ "custom-widget"
219
+ ]);
220
+ function collectSlotIdsFromRegion(node) {
221
+ const ids = [];
222
+ if (node.type === "slot") {
223
+ ids.push(node.slotId);
224
+ }
225
+ if (node.type === "panel-group" || node.type === "stack") {
226
+ for (const child of node.children) {
227
+ ids.push(...collectSlotIdsFromRegion(child));
228
+ }
229
+ }
230
+ if (node.type === "tabs") {
231
+ for (const tab of node.tabs) {
232
+ ids.push(...collectSlotIdsFromRegion(tab.child));
233
+ }
234
+ }
235
+ if (node.type === "floating") {
236
+ ids.push(node.anchorSlotId);
237
+ ids.push(...collectSlotIdsFromRegion(node.child));
238
+ }
239
+ return ids;
240
+ }
241
+ function validateLayoutSlots(surface) {
242
+ const declaredSlotIds = new Set(surface.slots.map((s) => s.slotId));
243
+ for (const layout of surface.layouts) {
244
+ const layoutSlotIds = collectSlotIdsFromRegion(layout.root);
245
+ for (const slotId of layoutSlotIds) {
246
+ if (!declaredSlotIds.has(slotId)) {
247
+ throw new Error(`Surface "${surface.surfaceId}" layout "${layout.layoutId}" references undeclared slot "${slotId}". Declared slots: ${[...declaredSlotIds].join(", ")}`);
248
+ }
249
+ }
250
+ }
251
+ }
252
+ function validateBundleNodeKinds(surface) {
253
+ const warnings = [];
254
+ for (const slot of surface.slots) {
255
+ for (const kind of slot.accepts) {
256
+ if (!KNOWN_NODE_KIND_RENDERERS.has(kind)) {
257
+ warnings.push(`Surface "${surface.surfaceId}" slot "${slot.slotId}" accepts "${kind}" which has no dedicated renderer (generic fallback used)`);
258
+ }
259
+ }
260
+ }
261
+ return { warnings };
262
+ }
263
+
264
+ // src/spec/define-module-bundle.ts
265
+ function defineModuleBundle(spec) {
266
+ if (!spec.meta?.key || !spec.meta?.version || !spec.meta?.title) {
267
+ throw new Error("ModuleBundleSpec must have meta.key, meta.version, and meta.title");
268
+ }
269
+ if (!spec.routes?.length) {
270
+ throw new Error("ModuleBundleSpec must have at least one route");
271
+ }
272
+ if (!spec.surfaces || Object.keys(spec.surfaces).length === 0) {
273
+ throw new Error("ModuleBundleSpec must have at least one surface");
274
+ }
275
+ if (spec.entities) {
276
+ if (!spec.entities.entityTypes || typeof spec.entities.entityTypes !== "object") {
277
+ throw new Error("ModuleBundleSpec.entities must have entityTypes object when present");
278
+ }
279
+ if (!spec.entities.fieldKinds || typeof spec.entities.fieldKinds !== "object") {
280
+ throw new Error("ModuleBundleSpec.entities must have fieldKinds object when present");
281
+ }
282
+ }
283
+ const REQUIRED_DIMENSIONS = [
284
+ "guidance",
285
+ "density",
286
+ "dataDepth",
287
+ "control",
288
+ "media",
289
+ "pace",
290
+ "narrative"
291
+ ];
292
+ const MIN_DESCRIPTION_LENGTH = 10;
293
+ for (const surface of Object.values(spec.surfaces)) {
294
+ if (!surface.verification?.dimensions) {
295
+ throw new Error(`Surface "${surface.surfaceId}" must have verification.dimensions for all 7 preference dimensions`);
296
+ }
297
+ const dims = surface.verification.dimensions;
298
+ for (const d of REQUIRED_DIMENSIONS) {
299
+ const val = dims[d];
300
+ if (!val || typeof val !== "string") {
301
+ throw new Error(`Surface "${surface.surfaceId}" verification.dimensions.${d} must be a non-empty string`);
302
+ }
303
+ const trimmed = val.trim();
304
+ if (trimmed.length < MIN_DESCRIPTION_LENGTH) {
305
+ throw new Error(`Surface "${surface.surfaceId}" verification.dimensions.${d} must be at least ${MIN_DESCRIPTION_LENGTH} characters (got ${trimmed.length})`);
306
+ }
307
+ }
308
+ validateLayoutSlots(surface);
309
+ }
310
+ return spec;
311
+ }
312
+ // src/examples/pm-workbench.bundle.ts
313
+ var PmWorkbenchBundle = defineModuleBundle({
314
+ meta: {
315
+ key: "pm.workbench",
316
+ version: "0.1.0",
317
+ title: "PM Workbench",
318
+ description: "AI-native PM workbench for relation-rich issues and saved views",
319
+ owners: ["team-pm-platform"],
320
+ tags: ["pm", "issues", "workbench", "ai-native"],
321
+ stability: "experimental"
322
+ },
323
+ routes: [
324
+ {
325
+ routeId: "pm-issue",
326
+ path: "/operate/pm/issues/:issueId",
327
+ defaultSurface: "issue-workbench"
328
+ }
329
+ ],
330
+ entities: {
331
+ entityTypes: {
332
+ "pm.issue": {
333
+ entityType: "pm.issue",
334
+ defaultSurfaceId: "issue-workbench",
335
+ detailBlueprints: ["balanced-three-pane", "dense-ops-mode"],
336
+ supportedViews: [
337
+ "minimal-summary",
338
+ "balanced-detail",
339
+ "dense-workbench"
340
+ ],
341
+ sectionsFromSchema: true,
342
+ fieldsFromSchema: true,
343
+ relationPanels: ["relations", "meeting-evidence", "decision-trail"]
344
+ }
345
+ },
346
+ fieldKinds: {
347
+ text: { fieldKind: "text", viewer: "text-viewer", editor: "text-editor" },
348
+ number: {
349
+ fieldKind: "number",
350
+ viewer: "number-viewer",
351
+ editor: "number-editor"
352
+ },
353
+ date: { fieldKind: "date", viewer: "date-viewer", editor: "date-editor" },
354
+ select: {
355
+ fieldKind: "select",
356
+ viewer: "select-viewer",
357
+ editor: "select-editor"
358
+ }
359
+ },
360
+ sectionKinds: {
361
+ overview: { sectionKind: "overview", renderer: "section-overview" },
362
+ details: { sectionKind: "details", renderer: "section-details" },
363
+ relations: { sectionKind: "relations", renderer: "section-relations" },
364
+ activity: { sectionKind: "activity", renderer: "section-activity" }
365
+ },
366
+ viewKinds: {
367
+ "minimal-summary": {
368
+ viewKind: "minimal-summary",
369
+ renderer: "view-minimal"
370
+ },
371
+ "balanced-detail": {
372
+ viewKind: "balanced-detail",
373
+ renderer: "view-balanced",
374
+ defaultLayoutId: "balanced-three-pane"
375
+ },
376
+ "dense-workbench": {
377
+ viewKind: "dense-workbench",
378
+ renderer: "view-dense",
379
+ defaultLayoutId: "dense-ops-mode"
380
+ }
381
+ }
382
+ },
383
+ surfaces: {
384
+ "issue-workbench": {
385
+ surfaceId: "issue-workbench",
386
+ kind: "workbench",
387
+ title: "Issue Workbench",
388
+ slots: [
389
+ {
390
+ slotId: "header",
391
+ role: "header",
392
+ accepts: ["action-bar"],
393
+ cardinality: "many"
394
+ },
395
+ {
396
+ slotId: "primary",
397
+ role: "primary",
398
+ accepts: ["entity-section", "table", "rich-doc"],
399
+ cardinality: "many",
400
+ mutableByAi: true,
401
+ mutableByUser: true
402
+ },
403
+ {
404
+ slotId: "secondary",
405
+ role: "secondary",
406
+ accepts: ["entity-section", "table", "timeline"],
407
+ cardinality: "many",
408
+ mutableByAi: true,
409
+ mutableByUser: true
410
+ },
411
+ {
412
+ slotId: "assistant",
413
+ role: "assistant",
414
+ accepts: ["assistant-panel", "chat-thread"],
415
+ cardinality: "many",
416
+ mutableByAi: true,
417
+ mutableByUser: true
418
+ },
419
+ {
420
+ slotId: "inspector",
421
+ role: "inspector",
422
+ accepts: ["entity-field", "relation-graph", "custom-widget"],
423
+ cardinality: "many",
424
+ mutableByAi: true,
425
+ mutableByUser: true
426
+ }
427
+ ],
428
+ layouts: [
429
+ {
430
+ layoutId: "balanced-three-pane",
431
+ title: "Balanced 3-pane",
432
+ root: {
433
+ type: "panel-group",
434
+ direction: "horizontal",
435
+ persistKey: "pm.issue.balanced-three-pane",
436
+ children: [
437
+ { type: "slot", slotId: "primary" },
438
+ {
439
+ type: "panel-group",
440
+ direction: "vertical",
441
+ persistKey: "pm.issue.right-stack",
442
+ children: [
443
+ { type: "slot", slotId: "secondary" },
444
+ { type: "slot", slotId: "assistant" }
445
+ ]
446
+ },
447
+ { type: "slot", slotId: "inspector" }
448
+ ]
449
+ }
450
+ }
451
+ ],
452
+ data: [
453
+ {
454
+ recipeId: "issue-core",
455
+ source: { kind: "entity", entityType: "pm.issue" },
456
+ requestedDepth: "detailed",
457
+ hydrateInto: "issue"
458
+ }
459
+ ],
460
+ verification: {
461
+ dimensions: {
462
+ guidance: "Can reveal walkthrough notes, field help, and AI explanations.",
463
+ density: "Can select compact, balanced, or dense multi-pane layouts.",
464
+ dataDepth: "Controls issue relation hydration, activity size, and inspector depth.",
465
+ control: "Shows advanced commands and raw config tabs only when allowed.",
466
+ media: "Supports text-first detail, visual graph, and hybrid assistant modes.",
467
+ pace: "Maps to motion tokens and confirmation behavior.",
468
+ narrative: "Can order issue summary before or after evidence/activity sections."
469
+ }
470
+ }
471
+ }
472
+ }
473
+ });
474
+ export {
475
+ PmWorkbenchBundle
476
+ };
@@ -0,0 +1,71 @@
1
+ // src/i18n/catalogs/en.ts
2
+ import { defineTranslation } from "@contractspec/lib.contracts-spec/translations";
3
+ var enMessages = defineTranslation({
4
+ meta: {
5
+ key: "surface-runtime.messages",
6
+ version: "1.0.0",
7
+ domain: "surface-runtime",
8
+ description: "User-facing strings for surface-runtime UI components",
9
+ owners: ["platform"],
10
+ stability: "experimental"
11
+ },
12
+ locale: "en",
13
+ fallback: "en",
14
+ messages: {
15
+ "overlay.conflicts.title": {
16
+ value: "Overlay conflicts",
17
+ description: "Title for overlay conflict resolution banner"
18
+ },
19
+ "overlay.conflicts.keepScope": {
20
+ value: "Keep {scope}",
21
+ description: "Button to keep overlay from scope A or B"
22
+ },
23
+ "patch.accept": {
24
+ value: "Accept",
25
+ description: "Accept patch proposal button"
26
+ },
27
+ "patch.reject": {
28
+ value: "Reject",
29
+ description: "Reject patch proposal button"
30
+ },
31
+ "patch.addWidget": {
32
+ value: "Add {title} to {slot}",
33
+ description: "Insert node proposal summary"
34
+ },
35
+ "patch.removeItem": {
36
+ value: "Remove item",
37
+ description: "Remove node proposal summary"
38
+ },
39
+ "patch.switchLayout": {
40
+ value: "Switch to {layoutId} layout",
41
+ description: "Set layout proposal summary"
42
+ },
43
+ "patch.showField": {
44
+ value: "Show field {fieldId}",
45
+ description: "Reveal field proposal summary"
46
+ },
47
+ "patch.hideField": {
48
+ value: "Hide field {fieldId}",
49
+ description: "Hide field proposal summary"
50
+ },
51
+ "patch.moveTo": {
52
+ value: "Move to {slot}",
53
+ description: "Move node proposal summary"
54
+ },
55
+ "patch.replaceItem": {
56
+ value: "Replace item",
57
+ description: "Replace node proposal summary"
58
+ },
59
+ "patch.promote": {
60
+ value: "Promote {actionId}",
61
+ description: "Promote action proposal summary"
62
+ },
63
+ "patch.changes": {
64
+ value: "{count} changes",
65
+ description: "Multiple patch ops summary"
66
+ }
67
+ }
68
+ });
69
+ export {
70
+ enMessages
71
+ };
@@ -0,0 +1,32 @@
1
+ // src/i18n/catalogs/es.ts
2
+ import { defineTranslation } from "@contractspec/lib.contracts-spec/translations";
3
+ var esMessages = defineTranslation({
4
+ meta: {
5
+ key: "surface-runtime.messages",
6
+ version: "1.0.0",
7
+ domain: "surface-runtime",
8
+ description: "User-facing strings for surface-runtime UI components",
9
+ owners: ["platform"],
10
+ stability: "experimental"
11
+ },
12
+ locale: "es",
13
+ fallback: "en",
14
+ messages: {
15
+ "overlay.conflicts.title": { value: "Conflictos de superposición" },
16
+ "overlay.conflicts.keepScope": { value: "Mantener {scope}" },
17
+ "patch.accept": { value: "Aceptar" },
18
+ "patch.reject": { value: "Rechazar" },
19
+ "patch.addWidget": { value: "Añadir {title} a {slot}" },
20
+ "patch.removeItem": { value: "Eliminar elemento" },
21
+ "patch.switchLayout": { value: "Cambiar a disposición {layoutId}" },
22
+ "patch.showField": { value: "Mostrar campo {fieldId}" },
23
+ "patch.hideField": { value: "Ocultar campo {fieldId}" },
24
+ "patch.moveTo": { value: "Mover a {slot}" },
25
+ "patch.replaceItem": { value: "Reemplazar elemento" },
26
+ "patch.promote": { value: "Promover {actionId}" },
27
+ "patch.changes": { value: "{count} cambios" }
28
+ }
29
+ });
30
+ export {
31
+ esMessages
32
+ };
@@ -0,0 +1,32 @@
1
+ // src/i18n/catalogs/fr.ts
2
+ import { defineTranslation } from "@contractspec/lib.contracts-spec/translations";
3
+ var frMessages = defineTranslation({
4
+ meta: {
5
+ key: "surface-runtime.messages",
6
+ version: "1.0.0",
7
+ domain: "surface-runtime",
8
+ description: "User-facing strings for surface-runtime UI components",
9
+ owners: ["platform"],
10
+ stability: "experimental"
11
+ },
12
+ locale: "fr",
13
+ fallback: "en",
14
+ messages: {
15
+ "overlay.conflicts.title": { value: "Conflits de superposition" },
16
+ "overlay.conflicts.keepScope": { value: "Conserver {scope}" },
17
+ "patch.accept": { value: "Accepter" },
18
+ "patch.reject": { value: "Rejeter" },
19
+ "patch.addWidget": { value: "Ajouter {title} à {slot}" },
20
+ "patch.removeItem": { value: "Supprimer l'élément" },
21
+ "patch.switchLayout": { value: "Passer à la disposition {layoutId}" },
22
+ "patch.showField": { value: "Afficher le champ {fieldId}" },
23
+ "patch.hideField": { value: "Masquer le champ {fieldId}" },
24
+ "patch.moveTo": { value: "Déplacer vers {slot}" },
25
+ "patch.replaceItem": { value: "Remplacer l'élément" },
26
+ "patch.promote": { value: "Promouvoir {actionId}" },
27
+ "patch.changes": { value: "{count} modifications" }
28
+ }
29
+ });
30
+ export {
31
+ frMessages
32
+ };