@topogram/cli 0.3.63 → 0.3.65

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 (344) hide show
  1. package/package.json +1 -1
  2. package/src/adoption/plan/index.js +703 -0
  3. package/src/adoption/plan.d.ts +6 -0
  4. package/src/adoption/plan.js +12 -703
  5. package/src/adoption/reporting.d.ts +10 -0
  6. package/src/adoption/review-groups.d.ts +6 -0
  7. package/src/agent-brief.d.ts +3 -0
  8. package/src/agent-brief.js +495 -0
  9. package/src/agent-ops/query-builders/auth.js +375 -0
  10. package/src/agent-ops/query-builders/change-risk/change-plan.js +123 -0
  11. package/src/agent-ops/query-builders/change-risk/import-plan.js +49 -0
  12. package/src/agent-ops/query-builders/change-risk/maintained.js +286 -0
  13. package/src/agent-ops/query-builders/change-risk/review-packets.js +123 -0
  14. package/src/agent-ops/query-builders/change-risk/risk.js +189 -0
  15. package/src/agent-ops/query-builders/change-risk.js +25 -0
  16. package/src/agent-ops/query-builders/common.js +149 -0
  17. package/src/agent-ops/query-builders/maintained-risk.js +539 -0
  18. package/src/agent-ops/query-builders/maintained-shared.js +120 -0
  19. package/src/agent-ops/query-builders/multi-agent.js +547 -0
  20. package/src/agent-ops/query-builders/projection-impacts.js +514 -0
  21. package/src/agent-ops/query-builders/work-packets.js +417 -0
  22. package/src/agent-ops/query-builders/workflow-context-shared.js +300 -0
  23. package/src/agent-ops/query-builders/workflow-context.js +398 -0
  24. package/src/agent-ops/query-builders/workflow-presets-core.js +676 -0
  25. package/src/agent-ops/query-builders/workflow-presets.js +341 -0
  26. package/src/agent-ops/query-builders.d.ts +26 -0
  27. package/src/agent-ops/query-builders.js +42 -5021
  28. package/src/archive/archive.d.ts +2 -0
  29. package/src/archive/compact.d.ts +1 -0
  30. package/src/archive/unarchive.d.ts +1 -0
  31. package/src/catalog/constants.js +10 -0
  32. package/src/catalog/copy.js +60 -0
  33. package/src/catalog/diagnostics.js +15 -0
  34. package/src/catalog/entries.js +42 -0
  35. package/src/catalog/files.js +67 -0
  36. package/src/catalog/provenance.js +122 -0
  37. package/src/catalog/source.js +150 -0
  38. package/src/catalog/validation.js +252 -0
  39. package/src/catalog.d.ts +12 -0
  40. package/src/catalog.js +18 -750
  41. package/src/cli/catalog-alias.d.ts +1 -0
  42. package/src/cli/command-parser.js +38 -0
  43. package/src/cli/command-parsers/core.js +102 -0
  44. package/src/cli/command-parsers/generator.js +39 -0
  45. package/src/cli/command-parsers/import.js +44 -0
  46. package/src/cli/command-parsers/legacy-workflow.js +21 -0
  47. package/src/cli/command-parsers/project.js +47 -0
  48. package/src/cli/command-parsers/sdlc.js +47 -0
  49. package/src/cli/command-parsers/shared.js +51 -0
  50. package/src/cli/command-parsers/template.js +48 -0
  51. package/src/cli/commands/agent.js +47 -0
  52. package/src/cli/commands/catalog/check.js +31 -0
  53. package/src/cli/commands/catalog/copy.js +59 -0
  54. package/src/cli/commands/catalog/doctor.js +248 -0
  55. package/src/cli/commands/catalog/help.js +21 -0
  56. package/src/cli/commands/catalog/list.js +52 -0
  57. package/src/cli/commands/catalog/runner.js +92 -0
  58. package/src/cli/commands/catalog/shared.js +17 -0
  59. package/src/cli/commands/catalog/show.js +134 -0
  60. package/src/cli/commands/catalog.js +32 -0
  61. package/src/cli/commands/check.js +268 -0
  62. package/src/cli/commands/doctor.js +268 -0
  63. package/src/cli/commands/emit.js +149 -0
  64. package/src/cli/commands/generate.js +96 -0
  65. package/src/cli/commands/generator-policy/package-info.js +162 -0
  66. package/src/cli/commands/generator-policy/payloads.js +372 -0
  67. package/src/cli/commands/generator-policy/printers.js +159 -0
  68. package/src/cli/commands/generator-policy/runner.js +81 -0
  69. package/src/cli/commands/generator-policy/shared.js +39 -0
  70. package/src/cli/commands/generator-policy.js +17 -0
  71. package/src/cli/commands/generator.js +443 -0
  72. package/src/cli/commands/import/adopt.js +170 -0
  73. package/src/cli/commands/import/check.js +91 -0
  74. package/src/cli/commands/import/diff.js +84 -0
  75. package/src/cli/commands/import/help.js +47 -0
  76. package/src/cli/commands/import/paths.js +277 -0
  77. package/src/cli/commands/import/plan.js +284 -0
  78. package/src/cli/commands/import/refresh.js +470 -0
  79. package/src/cli/commands/import/status-history.js +196 -0
  80. package/src/cli/commands/import/workspace.js +230 -0
  81. package/src/cli/commands/import-runner.js +157 -0
  82. package/src/cli/commands/import.js +35 -0
  83. package/src/cli/commands/inspect.js +55 -0
  84. package/src/cli/commands/new.js +94 -0
  85. package/src/cli/commands/package/constants.js +17 -0
  86. package/src/cli/commands/package/doctor.js +240 -0
  87. package/src/cli/commands/package/help.js +27 -0
  88. package/src/cli/commands/package/lockfile.js +135 -0
  89. package/src/cli/commands/package/npm.js +97 -0
  90. package/src/cli/commands/package/reporting.js +35 -0
  91. package/src/cli/commands/package/runner.js +33 -0
  92. package/src/cli/commands/package/shared.js +9 -0
  93. package/src/cli/commands/package/update-cli.js +252 -0
  94. package/src/cli/commands/package/versions.js +35 -0
  95. package/src/cli/commands/package.js +31 -0
  96. package/src/cli/commands/query/change-plan.js +68 -0
  97. package/src/cli/commands/query/definitions.js +202 -0
  98. package/src/cli/commands/query/import-adopt.js +121 -0
  99. package/src/cli/commands/query/runner/artifacts.js +102 -0
  100. package/src/cli/commands/query/runner/boundaries.js +211 -0
  101. package/src/cli/commands/query/runner/change.js +182 -0
  102. package/src/cli/commands/query/runner/import-adopt.js +111 -0
  103. package/src/cli/commands/query/runner/index.js +31 -0
  104. package/src/cli/commands/query/runner/output.js +12 -0
  105. package/src/cli/commands/query/runner/workflow.js +241 -0
  106. package/src/cli/commands/query/runner.js +3 -0
  107. package/src/cli/commands/query/workflow-context.js +5 -0
  108. package/src/cli/commands/query/workspace.js +274 -0
  109. package/src/cli/commands/query.js +11 -0
  110. package/src/cli/commands/release-rollout.js +257 -0
  111. package/src/cli/commands/release-shared.js +528 -0
  112. package/src/cli/commands/release-status.js +429 -0
  113. package/src/cli/commands/release.js +107 -0
  114. package/src/cli/commands/sdlc.js +168 -0
  115. package/src/cli/commands/setup.js +76 -0
  116. package/src/cli/commands/source.js +291 -0
  117. package/src/cli/commands/template/baseline.js +100 -0
  118. package/src/cli/commands/template/check.js +466 -0
  119. package/src/cli/commands/template/constants.js +8 -0
  120. package/src/cli/commands/template/diagnostics.js +26 -0
  121. package/src/cli/commands/template/help.js +28 -0
  122. package/src/cli/commands/template/lifecycle.js +404 -0
  123. package/src/cli/commands/template/list-show.js +287 -0
  124. package/src/cli/commands/template/policy.js +422 -0
  125. package/src/cli/commands/template/shared.js +127 -0
  126. package/src/cli/commands/template/updates.js +352 -0
  127. package/src/cli/commands/template-runner.js +198 -0
  128. package/src/cli/commands/template.js +43 -0
  129. package/src/cli/commands/trust.js +219 -0
  130. package/src/cli/commands/version.js +40 -0
  131. package/src/cli/commands/widget.js +168 -0
  132. package/src/cli/commands/workflow.js +63 -0
  133. package/src/cli/dispatcher.js +392 -0
  134. package/src/cli/help-dispatch.js +188 -0
  135. package/src/cli/help.js +296 -0
  136. package/src/cli/migration-guidance.js +59 -0
  137. package/src/cli/options.js +96 -0
  138. package/src/cli/output-safety.js +107 -0
  139. package/src/cli/path-normalization.js +29 -0
  140. package/src/cli.js +47 -11711
  141. package/src/example-implementation.d.ts +2 -0
  142. package/src/format.d.ts +1 -0
  143. package/src/generator/api/contracts.js +497 -0
  144. package/src/generator/api/metadata.js +221 -0
  145. package/src/generator/api/openapi.js +559 -0
  146. package/src/generator/api/schema.js +124 -0
  147. package/src/generator/api/types.d.ts +98 -0
  148. package/src/generator/api.js +3 -1195
  149. package/src/generator/check.d.ts +1 -0
  150. package/src/generator/context/bundle.d.ts +1 -0
  151. package/src/generator/context/shared/domain-sdlc.js +282 -0
  152. package/src/generator/context/shared/maintained-boundary.js +665 -0
  153. package/src/generator/context/shared/metrics.js +85 -0
  154. package/src/generator/context/shared/primitives.js +64 -0
  155. package/src/generator/context/shared/relationships.js +453 -0
  156. package/src/generator/context/shared/summaries.js +263 -0
  157. package/src/generator/context/shared/types.d.ts +207 -0
  158. package/src/generator/context/shared.d.ts +44 -0
  159. package/src/generator/context/shared.js +80 -1390
  160. package/src/generator/context/slice/core.js +397 -0
  161. package/src/generator/context/slice/sdlc.js +417 -0
  162. package/src/generator/context/slice/ui-packets.js +183 -0
  163. package/src/generator/context/slice.js +2 -859
  164. package/src/generator/native/parity-bundle.js +2 -1
  165. package/src/generator/registry/index.js +507 -0
  166. package/src/generator/registry.js +18 -504
  167. package/src/generator/runtime/environment/index.js +666 -0
  168. package/src/generator/runtime/environment.js +4 -666
  169. package/src/generator/runtime/runtime-check/index.js +554 -0
  170. package/src/generator/runtime/runtime-check.js +4 -554
  171. package/src/generator/runtime/shared/index.js +572 -0
  172. package/src/generator/runtime/shared.js +19 -570
  173. package/src/generator/shared.d.ts +2 -0
  174. package/src/generator/surfaces/shared.d.ts +3 -0
  175. package/src/generator/surfaces/web/html-escape.js +22 -0
  176. package/src/generator/surfaces/web/react.js +10 -8
  177. package/src/generator/surfaces/web/sveltekit.js +7 -5
  178. package/src/generator/surfaces/web/vanilla.js +8 -4
  179. package/src/generator/widget-conformance/behavior-report.js +258 -0
  180. package/src/generator/widget-conformance/checks.js +371 -0
  181. package/src/generator/widget-conformance/projection-context.js +200 -0
  182. package/src/generator/widget-conformance/report.js +166 -0
  183. package/src/generator/widget-conformance/types.d.ts +121 -0
  184. package/src/generator/widget-conformance.js +3 -824
  185. package/src/generator.d.ts +2 -0
  186. package/src/github-client.js +520 -0
  187. package/src/import/core/context.d.ts +3 -0
  188. package/src/import/core/contracts.d.ts +1 -0
  189. package/src/import/core/registry.d.ts +4 -0
  190. package/src/import/core/runner/candidates.js +217 -0
  191. package/src/import/core/runner/options.js +22 -0
  192. package/src/import/core/runner/reports.js +50 -0
  193. package/src/import/core/runner/run.js +79 -0
  194. package/src/import/core/runner/tracks.js +150 -0
  195. package/src/import/core/runner/ui-drafts.js +337 -0
  196. package/src/import/core/runner.js +3 -698
  197. package/src/import/core/shared/api-routes.js +221 -0
  198. package/src/import/core/shared/candidates.js +97 -0
  199. package/src/import/core/shared/files.js +177 -0
  200. package/src/import/core/shared/next-app.js +389 -0
  201. package/src/import/core/shared/types.d.ts +51 -0
  202. package/src/import/core/shared/ui-routes.js +230 -0
  203. package/src/import/core/shared.js +67 -910
  204. package/src/import/extractors/api/flutter-dio.js +4 -8
  205. package/src/import/extractors/api/react-native-repository.js +4 -8
  206. package/src/import/index.d.ts +4 -0
  207. package/src/import/provenance.d.ts +4 -0
  208. package/src/new-project/constants.js +128 -0
  209. package/src/new-project/create.js +83 -0
  210. package/src/new-project/json.js +28 -0
  211. package/src/new-project/metadata.js +96 -0
  212. package/src/new-project/package-spec.js +161 -0
  213. package/src/new-project/project-files.js +348 -0
  214. package/src/new-project/template-policy.js +269 -0
  215. package/src/new-project/template-resolution.js +368 -0
  216. package/src/new-project/template-snapshots.js +430 -0
  217. package/src/new-project/template-updates.js +512 -0
  218. package/src/new-project/types.d.ts +83 -0
  219. package/src/new-project.js +6 -2188
  220. package/src/npm-safety.js +79 -0
  221. package/src/parser.d.ts +87 -0
  222. package/src/parser.js +118 -0
  223. package/src/path-helpers.d.ts +1 -0
  224. package/src/path-helpers.js +20 -0
  225. package/src/policy/review-boundaries.d.ts +15 -0
  226. package/src/project-config/index.js +564 -0
  227. package/src/project-config.js +19 -560
  228. package/src/reconcile/docs.d.ts +8 -0
  229. package/src/reconcile/journeys.d.ts +1 -0
  230. package/src/resolver/enrich/acceptance-criterion.js +2 -0
  231. package/src/resolver/enrich/bug.js +2 -0
  232. package/src/resolver/enrich/pitch.js +2 -0
  233. package/src/resolver/enrich/requirement.js +2 -0
  234. package/src/resolver/enrich/task.js +2 -0
  235. package/src/resolver/index.js +19 -2089
  236. package/src/resolver/normalize.js +384 -1
  237. package/src/resolver/plans.js +168 -0
  238. package/src/resolver/projections-api.js +494 -0
  239. package/src/resolver/projections-db.js +133 -0
  240. package/src/resolver/projections-ui.js +317 -0
  241. package/src/resolver/shapes.js +251 -0
  242. package/src/resolver/shared.js +278 -0
  243. package/src/resolver/widgets.js +132 -0
  244. package/src/resolver.d.ts +1 -0
  245. package/src/runtime-support.js +29 -0
  246. package/src/sdlc/adopt.d.ts +1 -0
  247. package/src/sdlc/check.d.ts +1 -0
  248. package/src/sdlc/explain.d.ts +1 -0
  249. package/src/sdlc/release.d.ts +1 -0
  250. package/src/sdlc/scaffold.d.ts +1 -0
  251. package/src/sdlc/transition.d.ts +1 -0
  252. package/src/template-trust/constants.js +62 -0
  253. package/src/template-trust/content.js +258 -0
  254. package/src/template-trust/diff.js +92 -0
  255. package/src/template-trust/policy.js +61 -0
  256. package/src/template-trust/record.js +90 -0
  257. package/src/template-trust/status.js +182 -0
  258. package/src/template-trust.js +24 -687
  259. package/src/text-helpers.d.ts +7 -0
  260. package/src/text-helpers.js +245 -0
  261. package/src/topogram-config.js +306 -0
  262. package/src/topogram-types.d.ts +69 -0
  263. package/src/validator/common.js +488 -0
  264. package/src/validator/data-model.js +237 -0
  265. package/src/validator/docs.js +167 -0
  266. package/src/validator/expressions.js +146 -1
  267. package/src/validator/index.d.ts +23 -0
  268. package/src/validator/index.js +32 -3585
  269. package/src/validator/kinds.d.ts +41 -0
  270. package/src/validator/kinds.js +2 -0
  271. package/src/validator/model-helpers.js +46 -0
  272. package/src/validator/per-kind/acceptance-criterion.js +5 -0
  273. package/src/validator/per-kind/bug.js +6 -0
  274. package/src/validator/per-kind/domain.js +15 -2
  275. package/src/validator/per-kind/pitch.js +7 -0
  276. package/src/validator/per-kind/requirement.js +5 -0
  277. package/src/validator/per-kind/task.js +7 -0
  278. package/src/validator/per-kind/widget.js +14 -0
  279. package/src/validator/projections/api-http-async.js +410 -0
  280. package/src/validator/projections/api-http-authz.js +88 -0
  281. package/src/validator/projections/api-http-core.js +205 -0
  282. package/src/validator/projections/api-http-policies.js +339 -0
  283. package/src/validator/projections/api-http-responses.js +233 -0
  284. package/src/validator/projections/api-http.js +44 -0
  285. package/src/validator/projections/db.js +353 -0
  286. package/src/validator/projections/generator-defaults.js +45 -0
  287. package/src/validator/projections/helpers.js +87 -0
  288. package/src/validator/projections/ui-helpers.js +214 -0
  289. package/src/validator/projections/ui-navigation.js +344 -0
  290. package/src/validator/projections/ui-structure.js +364 -0
  291. package/src/validator/projections/ui-widgets.js +493 -0
  292. package/src/validator/projections/ui.js +46 -0
  293. package/src/validator/registry.js +48 -1
  294. package/src/validator/utils.d.ts +20 -0
  295. package/src/validator/utils.js +115 -12
  296. package/src/validator.d.ts +2 -0
  297. package/src/widget-behavior.d.ts +1 -0
  298. package/src/workflows/adoption/index.js +26 -0
  299. package/src/workflows/docs-generate.js +262 -0
  300. package/src/workflows/docs-scan.js +703 -0
  301. package/src/workflows/docs.js +15 -0
  302. package/src/workflows/import-app/api/collect.js +221 -0
  303. package/src/workflows/import-app/api/openapi.js +257 -0
  304. package/src/workflows/import-app/api/routes.js +327 -0
  305. package/src/workflows/import-app/api/sources.js +22 -0
  306. package/src/workflows/import-app/api.js +4 -0
  307. package/src/workflows/import-app/db.js +538 -0
  308. package/src/workflows/import-app/index.js +30 -0
  309. package/src/workflows/import-app/shared.js +218 -0
  310. package/src/workflows/import-app/ui.js +443 -0
  311. package/src/workflows/import-app/workflow.js +159 -0
  312. package/src/workflows/reconcile/adoption-plan/build.js +208 -0
  313. package/src/workflows/reconcile/adoption-plan/dependencies.js +75 -0
  314. package/src/workflows/reconcile/adoption-plan/outputs.js +143 -0
  315. package/src/workflows/reconcile/adoption-plan/paths.js +58 -0
  316. package/src/workflows/reconcile/adoption-plan/projection-patches.js +177 -0
  317. package/src/workflows/reconcile/adoption-plan/reasons.js +107 -0
  318. package/src/workflows/reconcile/adoption-plan.js +32 -0
  319. package/src/workflows/reconcile/auth/closures.js +115 -0
  320. package/src/workflows/reconcile/auth/formatters.js +142 -0
  321. package/src/workflows/reconcile/auth/inference.js +330 -0
  322. package/src/workflows/reconcile/auth/roles.js +122 -0
  323. package/src/workflows/reconcile/auth.js +37 -0
  324. package/src/workflows/reconcile/bundle-core/index.js +600 -0
  325. package/src/workflows/reconcile/bundle-core.js +14 -0
  326. package/src/workflows/reconcile/bundle-shared.js +75 -0
  327. package/src/workflows/reconcile/candidate-model.js +477 -0
  328. package/src/workflows/reconcile/canonical-surface.js +264 -0
  329. package/src/workflows/reconcile/gap-report.js +333 -0
  330. package/src/workflows/reconcile/ids.js +6 -0
  331. package/src/workflows/reconcile/impacts/adoption-plan.js +192 -0
  332. package/src/workflows/reconcile/impacts/indexes.js +101 -0
  333. package/src/workflows/reconcile/impacts/patches.js +252 -0
  334. package/src/workflows/reconcile/impacts/reports.js +80 -0
  335. package/src/workflows/reconcile/impacts.js +16 -0
  336. package/src/workflows/reconcile/index.js +7 -0
  337. package/src/workflows/reconcile/renderers.js +461 -0
  338. package/src/workflows/reconcile/summary.js +90 -0
  339. package/src/workflows/reconcile/workflow.js +309 -0
  340. package/src/workflows/shared.js +189 -0
  341. package/src/workflows/types.d.ts +93 -0
  342. package/src/workflows.d.ts +1 -0
  343. package/src/workflows.js +10 -7652
  344. package/src/workspace-docs.d.ts +29 -0
@@ -0,0 +1,75 @@
1
+ // @ts-check
2
+
3
+ /** @param {string} value @returns {any} */
4
+ export function provenanceList(value) {
5
+ if (!value) {
6
+ return [];
7
+ }
8
+ return Array.isArray(value) ? value.filter(Boolean) : [value].filter(Boolean);
9
+ }
10
+
11
+ /** @param {CandidateBundle} bundle @returns {any} */
12
+ export function collectBundleProvenance(bundle) {
13
+ const values = new Set();
14
+ for (const entry of bundle.docs || []) {
15
+ for (const provenance of provenanceList(entry.provenance)) values.add(provenance);
16
+ }
17
+ for (const entry of bundle.workflows || []) {
18
+ for (const provenance of provenanceList(entry.provenance)) values.add(provenance);
19
+ }
20
+ for (const entry of bundle.capabilities || []) {
21
+ for (const provenance of provenanceList(entry.provenance)) values.add(provenance);
22
+ }
23
+ for (const entry of bundle.screens || []) {
24
+ for (const provenance of provenanceList(entry.provenance)) values.add(provenance);
25
+ }
26
+ for (const entry of bundle.entities || []) {
27
+ for (const provenance of provenanceList(entry.provenance)) values.add(provenance);
28
+ }
29
+ return values;
30
+ }
31
+
32
+ /** @param {CandidateBundle} bundle @returns {any} */
33
+ export function contextualEvidenceScore(bundle) {
34
+ return (
35
+ (bundle.docs || []).length * 4 +
36
+ (bundle.workflows || []).length * 4 +
37
+ (bundle.capabilities || []).length * 3 +
38
+ (bundle.screens || []).length * 2 +
39
+ (bundle.entities || []).length * 2
40
+ );
41
+ }
42
+
43
+ /** @param {CandidateBundle} bundle @returns {any} */
44
+ export function collectBundleDocIds(bundle) {
45
+ return new Set((bundle.docs || []).map((/** @type {any} */ entry) => entry.id));
46
+ }
47
+
48
+ /** @param {CandidateBundle} bundle @returns {any} */
49
+ export function collectBundleCapabilityIds(bundle) {
50
+ return new Set([
51
+ ...(bundle.capabilities || []).map((/** @type {any} */ entry) => entry.id_hint),
52
+ ...(bundle.verifications || []).flatMap((/** @type {any} */ entry) => entry.related_capabilities || [])
53
+ ]);
54
+ }
55
+
56
+ /** @param {CandidateBundle} bundle @returns {any} */
57
+ export function collectBundleRuleIds(bundle) {
58
+ return [...new Set([
59
+ ...(bundle.docs || []).flatMap((/** @type {any} */ entry) => entry.related_rules || []),
60
+ ...(bundle.docLinkSuggestions || []).flatMap((/** @type {any} */ entry) => entry.add_related_rules || []),
61
+ ...(bundle.docMetadataPatches || []).flatMap((/** @type {any} */ entry) => entry.related_rules || [])
62
+ ])].sort();
63
+ }
64
+
65
+
66
+ /** @param {CandidateBundle} bundle @returns {any} */
67
+ export function primaryEntityIdForBundle(bundle) {
68
+ const entityIds = [
69
+ ...(bundle.entities || []).map((/** @type {any} */ entry) => entry.id_hint),
70
+ ...(bundle.capabilities || []).map((/** @type {any} */ entry) => entry.entity_id).filter(Boolean),
71
+ ...(bundle.screens || []).map((/** @type {any} */ entry) => entry.entity_id).filter(Boolean),
72
+ ...(bundle.workflows || []).map((/** @type {any} */ entry) => entry.entity_id).filter(Boolean)
73
+ ];
74
+ return entityIds.find(Boolean) || null;
75
+ }
@@ -0,0 +1,477 @@
1
+ // @ts-check
2
+ import { buildBundleDocDriftSummaries as buildBundleDocDriftSummariesReconcile, buildBundleDocMetadataPatches as buildBundleDocMetadataPatchesReconcile } from "../../reconcile/docs.js";
3
+ import { canonicalCandidateTerm, idHintify, titleCase } from "../../text-helpers.js";
4
+ import { confidenceRank, docDirForKind, renderCandidateActor, renderCandidateRole } from "../docs.js";
5
+ import { inferCapabilityEntityId } from "../import-app/index.js";
6
+ import { readTextIfExists, renderMarkdownDoc } from "../shared.js";
7
+ import {
8
+ annotateDocLinkSuggestionsWithAuthRoleGuidance,
9
+ classifyBundleAuthRoleGuidance,
10
+ inferBundleAuthClaimHints,
11
+ inferBundleAuthOwnershipHints,
12
+ inferBundleAuthPermissionHints,
13
+ inferBundleAuthRoleGuidance
14
+ } from "./auth.js";
15
+ import { collectBundleCapabilityIds, collectBundleDocIds, collectBundleProvenance, contextualEvidenceScore, provenanceList } from "./bundle-shared.js";
16
+ import { addBundleJourneyDrafts, buildBundleMergeHints, bundleLabelFromConceptId, getOrCreateCandidateBundle, renderCandidateBundleReadme } from "./bundle-core.js";
17
+ import { buildTopogramApiCapabilityIndex, buildBundleDocLinkSuggestions, collectCanonicalUiSurface, collectCanonicalWorkflowSurface, matchImportedApiCapability } from "./canonical-surface.js";
18
+ import { buildBundleAdoptionPlan, buildCanonicalShapeIndex, buildProjectionEntityIndex, buildProjectionImpacts, buildProjectionPatchCandidates, buildUiImpacts, buildWorkflowImpacts } from "./impacts.js";
19
+ import {
20
+ renderCandidateCapability,
21
+ renderCandidateEntity,
22
+ renderCandidateEnum,
23
+ renderCandidateShape,
24
+ renderCandidateUiReportDoc,
25
+ renderCandidateVerification,
26
+ renderCandidateWidget,
27
+ renderCandidateWorkflowDecision,
28
+ renderCandidateWorkflowDoc,
29
+ renderDocLinkPatchDoc,
30
+ renderDocMetadataPatchDoc,
31
+ renderProjectionPatchDoc,
32
+ shapeIdForCapability
33
+ } from "./renderers.js";
34
+
35
+ /** @param {CandidateBundle} bundle @returns {any} */
36
+ export function bundleNoiseSuppressionReason(bundle) {
37
+ if ((bundle.actors || []).length > 0 || (bundle.roles || []).length > 0 || (bundle.capabilities || []).length > 0 || (bundle.workflows || []).length > 0 || (bundle.docs || []).length > 0 || (bundle.screens || []).length > 0) {
38
+ return null;
39
+ }
40
+ const noiseEntities = (bundle.entities || []).filter((/** @type {any} */ entry) => entry.noise_candidate);
41
+ if (noiseEntities.length === 0) {
42
+ return null;
43
+ }
44
+ if (noiseEntities.length === (bundle.entities || []).length) {
45
+ return noiseEntities[0].noise_reason || "Rails implementation-noise bundle.";
46
+ }
47
+ return null;
48
+ }
49
+
50
+ /** @param {Map<string, CandidateBundle>} bundles @param {WorkflowRecord} candidate @returns {any} */
51
+ export function bestContextBundleForCandidate(bundles, candidate) {
52
+ const candidateProvenance = new Set(provenanceList(candidate.provenance));
53
+ const relatedDocs = new Set(candidate.related_docs || []);
54
+ const relatedCapabilities = new Set(candidate.related_capabilities || []);
55
+ let best = null;
56
+ for (const bundle of bundles.values()) {
57
+ const evidenceScore = contextualEvidenceScore(bundle);
58
+ if (evidenceScore === 0) {
59
+ continue;
60
+ }
61
+ const provenanceOverlap = candidateProvenance.size > 0
62
+ ? [...candidateProvenance].filter((/** @type {any} */ item) => collectBundleProvenance(bundle).has(item)).length
63
+ : 0;
64
+ const docLinkOverlap = relatedDocs.size > 0
65
+ ? [...relatedDocs].filter((/** @type {any} */ item) => collectBundleDocIds(bundle).has(item)).length
66
+ : 0;
67
+ const capabilityLinkOverlap = relatedCapabilities.size > 0
68
+ ? [...relatedCapabilities].filter((/** @type {any} */ item) => collectBundleCapabilityIds(bundle).has(item)).length
69
+ : 0;
70
+ if (provenanceOverlap === 0 && docLinkOverlap === 0 && capabilityLinkOverlap === 0) {
71
+ continue;
72
+ }
73
+ const score = docLinkOverlap * 1000 + capabilityLinkOverlap * 750 + provenanceOverlap * 100 + evidenceScore;
74
+ if (!best || score > best.score || (score === best.score && bundle.slug.localeCompare(best.bundle.slug) < 0)) {
75
+ best = { bundle, score };
76
+ }
77
+ }
78
+ return best?.bundle || null;
79
+ }
80
+
81
+ /** @param {ResolvedGraph} graph @param {ImportArtifacts} appImport @param {any} topogramRoot @returns {any} */
82
+ export function buildCandidateModelBundles(graph, appImport, topogramRoot) {
83
+ const dbCandidates = appImport.candidates.db || { entities: [], enums: [] };
84
+ const apiCandidates = appImport.candidates.api || { capabilities: [] };
85
+ const uiCandidates = appImport.candidates.ui || { screens: [], routes: [], actions: [], widgets: [] };
86
+ const uiWidgetCandidates = uiCandidates.widgets || uiCandidates.components || [];
87
+ const workflowCandidates = appImport.candidates.workflows || { workflows: [], workflow_states: [], workflow_transitions: [] };
88
+ const verificationCandidates = appImport.candidates.verification || { verifications: [], scenarios: [], frameworks: [], scripts: [] };
89
+ const docCandidates = appImport.candidates.docs || [];
90
+ const actorCandidates = appImport.candidates.actors || [];
91
+ const roleCandidates = appImport.candidates.roles || [];
92
+ const knownEnums = new Set((dbCandidates.enums || []).map((/** @type {any} */ entry) => entry.id_hint));
93
+ const canonicalActorIds = new Set((graph?.byKind.actor || []).map((/** @type {any} */ entry) => entry.id));
94
+ const canonicalRoleIds = new Set((graph?.byKind.role || []).map((/** @type {any} */ entry) => entry.id));
95
+ const canonicalEntityIds = new Set((graph?.byKind.entity || []).map((/** @type {any} */ entry) => entry.id));
96
+ const canonicalEnumIds = new Set((graph?.byKind.enum || []).map((/** @type {any} */ entry) => entry.id));
97
+ const canonicalWidgetIds = new Set((graph?.byKind.widget || []).map((/** @type {any} */ entry) => entry.id));
98
+ const canonicalUi = collectCanonicalUiSurface(graph || { byKind: { projection: [] } });
99
+ const canonicalWorkflow = collectCanonicalWorkflowSurface(graph || { byKind: { decision: [] }, docs: [] });
100
+ const canonicalDocsByKind = new Map();
101
+ for (const doc of graph?.docs || []) {
102
+ if (!canonicalDocsByKind.has(doc.kind)) {
103
+ canonicalDocsByKind.set(doc.kind, new Set());
104
+ }
105
+ canonicalDocsByKind.get(doc.kind).add(doc.id);
106
+ }
107
+ const topogramApiCapabilities = graph ? buildTopogramApiCapabilityIndex(graph) : [];
108
+ const canonicalShapeIndex = buildCanonicalShapeIndex(graph);
109
+ const canonicalVerificationIds = new Set((graph?.byKind.verification || []).map((/** @type {any} */ entry) => entry.id));
110
+ const projectionIndex = buildProjectionEntityIndex(graph);
111
+ const bundles = new Map();
112
+ const enumCandidatesById = new Map((dbCandidates.enums || []).map((/** @type {any} */ entry) => [entry.id_hint, entry]));
113
+ const verificationScenariosByVerificationId = new Map();
114
+ for (const scenario of verificationCandidates.scenarios || []) {
115
+ const bucket = verificationScenariosByVerificationId.get(scenario.verification_id) || [];
116
+ bucket.push(scenario);
117
+ verificationScenariosByVerificationId.set(scenario.verification_id, bucket);
118
+ }
119
+
120
+ for (const entry of dbCandidates.enums || []) {
121
+ if (canonicalEnumIds.has(entry.id_hint)) continue;
122
+ getOrCreateCandidateBundle(bundles, `enum_${entry.id_hint}`, titleCase(entry.id_hint)).enums.push(entry);
123
+ }
124
+ for (const entry of dbCandidates.entities || []) {
125
+ const bundle = getOrCreateCandidateBundle(bundles, entry.id_hint, entry.label);
126
+ bundle.importedFieldEvidence = [
127
+ ...(bundle.importedFieldEvidence || []),
128
+ ...((entry.fields || []).map((/** @type {any} */ field) => ({
129
+ entity_id: entry.id_hint,
130
+ name: field.name,
131
+ field_type: field.field_type,
132
+ required: field.required
133
+ })))
134
+ ];
135
+ if (!canonicalEntityIds.has(entry.id_hint)) {
136
+ bundle.entities.push(entry);
137
+ }
138
+ for (const field of entry.fields || []) {
139
+ const enumId = idHintify(field.field_type);
140
+ const enumEntry = enumCandidatesById.get(enumId);
141
+ if (!enumEntry || canonicalEnumIds.has(enumId)) {
142
+ continue;
143
+ }
144
+ if (!bundle.enums.some((/** @type {any} */ candidate) => candidate.id_hint === enumId)) {
145
+ bundle.enums.push(enumEntry);
146
+ }
147
+ bundles.delete(`enum_${enumId}`);
148
+ }
149
+ }
150
+ for (const entry of apiCandidates.capabilities || []) {
151
+ const matchedCapability = graph ? matchImportedApiCapability(entry, topogramApiCapabilities) : null;
152
+ if (matchedCapability) {
153
+ continue;
154
+ }
155
+ const conceptId = inferCapabilityEntityId(entry);
156
+ const bundle = getOrCreateCandidateBundle(bundles, conceptId, titleCase(conceptId.replace(/^entity_/, "")));
157
+ bundle.capabilities.push(entry);
158
+ const inputFields = entry.input_fields || [];
159
+ const outputFields = entry.output_fields || [];
160
+ if (inputFields.length > 0) {
161
+ bundle.shapes.push({
162
+ id: shapeIdForCapability(entry, "input"),
163
+ label: `${titleCase(entry.id_hint.replace(/^cap_/, ""))} Input`,
164
+ fields: inputFields
165
+ });
166
+ }
167
+ if (outputFields.length > 0) {
168
+ bundle.shapes.push({
169
+ id: shapeIdForCapability(entry, "output"),
170
+ label: `${titleCase(entry.id_hint.replace(/^cap_/, ""))} Output`,
171
+ fields: outputFields
172
+ });
173
+ }
174
+ }
175
+ for (const entry of uiCandidates.screens || []) {
176
+ if (canonicalUi.screens.includes(entry.id_hint)) {
177
+ continue;
178
+ }
179
+ const conceptId = entry.entity_id || entry.concept_id || `entity_${canonicalCandidateTerm(entry.id_hint)}`;
180
+ const bundle = getOrCreateCandidateBundle(bundles, conceptId, bundleLabelFromConceptId(conceptId || entry.id_hint));
181
+ bundle.screens.push(entry);
182
+ }
183
+ for (const entry of uiCandidates.routes || []) {
184
+ if (canonicalUi.routes.includes(entry.path)) {
185
+ continue;
186
+ }
187
+ const conceptId = entry.entity_id || entry.concept_id || `entity_${canonicalCandidateTerm(entry.screen_id || entry.id_hint)}`;
188
+ const bundle = getOrCreateCandidateBundle(bundles, conceptId, bundleLabelFromConceptId(conceptId || entry.screen_id || entry.id_hint));
189
+ bundle.uiRoutes.push(entry);
190
+ }
191
+ for (const entry of uiCandidates.actions || []) {
192
+ const conceptId = entry.entity_id || entry.concept_id || `entity_${canonicalCandidateTerm(entry.screen_id || entry.id_hint)}`;
193
+ const bundle = getOrCreateCandidateBundle(bundles, conceptId, bundleLabelFromConceptId(conceptId || entry.screen_id || entry.id_hint));
194
+ bundle.uiActions.push(entry);
195
+ }
196
+ /** @param {WorkflowRecord} entry @returns {any} */
197
+ function widgetConceptId(entry) {
198
+ if (entry.entity_id || entry.concept_id) {
199
+ return entry.entity_id || entry.concept_id;
200
+ }
201
+ const screenStem = String(entry.screen_id || entry.id_hint || "")
202
+ .replace(/_(list|index|table|grid|results)$/, "")
203
+ .replace(/^list_/, "");
204
+ return `entity_${canonicalCandidateTerm(screenStem || entry.id_hint)}`;
205
+ }
206
+
207
+ for (const entry of uiWidgetCandidates) {
208
+ if (canonicalWidgetIds.has(entry.id_hint)) {
209
+ continue;
210
+ }
211
+ const conceptId = widgetConceptId(entry);
212
+ const bundle = getOrCreateCandidateBundle(bundles, conceptId, bundleLabelFromConceptId(conceptId || entry.screen_id || entry.id_hint));
213
+ bundle.widgets.push(entry);
214
+ }
215
+ for (const entry of workflowCandidates.workflows || []) {
216
+ if (canonicalWorkflow.workflow_docs.includes(entry.id_hint)) {
217
+ continue;
218
+ }
219
+ const bundle = getOrCreateCandidateBundle(bundles, entry.entity_id || `entity_${canonicalCandidateTerm(entry.id_hint)}`, titleCase((entry.entity_id || entry.id_hint).replace(/^entity_/, "")));
220
+ bundle.workflows.push(entry);
221
+ }
222
+ for (const entry of workflowCandidates.workflow_states || []) {
223
+ if (canonicalWorkflow.workflow_docs.includes(entry.workflow_id)) {
224
+ continue;
225
+ }
226
+ const bundle = getOrCreateCandidateBundle(bundles, entry.entity_id || `entity_${canonicalCandidateTerm(entry.workflow_id || entry.id_hint)}`, titleCase((entry.entity_id || entry.workflow_id || entry.id_hint).replace(/^entity_/, "")));
227
+ bundle.workflowStates.push(entry);
228
+ }
229
+ for (const entry of workflowCandidates.workflow_transitions || []) {
230
+ if (canonicalWorkflow.workflow_docs.includes(entry.workflow_id)) {
231
+ continue;
232
+ }
233
+ const bundle = getOrCreateCandidateBundle(bundles, entry.entity_id || `entity_${canonicalCandidateTerm(entry.workflow_id || entry.id_hint)}`, titleCase((entry.entity_id || entry.workflow_id || entry.id_hint).replace(/^entity_/, "")));
234
+ bundle.workflowTransitions.push(entry);
235
+ }
236
+ for (const entry of verificationCandidates.verifications || []) {
237
+ if (canonicalVerificationIds.has(entry.id_hint)) {
238
+ continue;
239
+ }
240
+ const relatedCapabilityId =
241
+ entry.related_capabilities?.[0] ||
242
+ verificationScenariosByVerificationId.get(entry.id_hint)?.flatMap((/** @type {any} */ scenario) => scenario.related_capabilities || [])[0] ||
243
+ null;
244
+ const conceptId = relatedCapabilityId
245
+ ? inferCapabilityEntityId({
246
+ id_hint: relatedCapabilityId,
247
+ endpoint: { path: `/${relatedCapabilityId.replace(/^cap_(create|get|update|delete|list|complete|close|approve|reject|request_revision|export|download)_/, "").replace(/_/g, "-")}` }
248
+ })
249
+ : `surface_${canonicalCandidateTerm(entry.id_hint)}`;
250
+ const bundle = getOrCreateCandidateBundle(bundles, conceptId, bundleLabelFromConceptId(conceptId || entry.id_hint));
251
+ bundle.verifications.push({
252
+ ...entry,
253
+ scenarios: verificationScenariosByVerificationId.get(entry.id_hint) || []
254
+ });
255
+ }
256
+ for (const entry of docCandidates || []) {
257
+ const hasCanonicalDoc = (canonicalDocsByKind.get(entry.kind) || new Set()).has(entry.id);
258
+ const canonicalEntityHint = `entity_${canonicalCandidateTerm(entry.id)}`;
259
+ const semanticallyAnchored =
260
+ (entry.related_entities || []).length > 0 ||
261
+ (entry.related_capabilities || []).length > 0 ||
262
+ canonicalEntityIds.has(canonicalEntityHint);
263
+ if (!semanticallyAnchored && entry.kind !== "workflow") {
264
+ continue;
265
+ }
266
+ const conceptId = semanticallyAnchored
267
+ ? (
268
+ entry.related_entities?.[0] ||
269
+ (entry.related_capabilities?.[0] ? inferCapabilityEntityId({ id_hint: entry.related_capabilities[0], endpoint: { path: `/${entry.related_capabilities[0].replace(/^cap_(create|get|update|delete|list)_/, "").replace(/_/g, "-")}` } }) : `entity_${canonicalCandidateTerm(entry.id)}`)
270
+ )
271
+ : `flow_${canonicalCandidateTerm(entry.id)}`;
272
+ const bundle = getOrCreateCandidateBundle(bundles, conceptId, titleCase(conceptId.replace(/^(entity|flow)_/, "")));
273
+ bundle.docs.push({
274
+ ...entry,
275
+ existing_canonical: hasCanonicalDoc
276
+ });
277
+ }
278
+ for (const entry of actorCandidates || []) {
279
+ if (canonicalActorIds.has(entry.id_hint)) continue;
280
+ const contextualBundle = bestContextBundleForCandidate(bundles, entry);
281
+ (contextualBundle || getOrCreateCandidateBundle(bundles, entry.id_hint, entry.label)).actors.push(entry);
282
+ }
283
+ for (const entry of roleCandidates || []) {
284
+ if (canonicalRoleIds.has(entry.id_hint)) continue;
285
+ const contextualBundle = bestContextBundleForCandidate(bundles, entry);
286
+ (contextualBundle || getOrCreateCandidateBundle(bundles, entry.id_hint, entry.label)).roles.push(entry);
287
+ }
288
+
289
+ addBundleJourneyDrafts(bundles, graph);
290
+
291
+ /** @type {any[]} */
292
+
293
+ const suppressedNoiseBundles = [];
294
+ const finalizedBundles = [...bundles.values()]
295
+ .filter((/** @type {any} */ bundle) =>
296
+ bundle.actors.length > 0 ||
297
+ bundle.roles.length > 0 ||
298
+ bundle.entities.length > 0 ||
299
+ bundle.enums.length > 0 ||
300
+ bundle.capabilities.length > 0 ||
301
+ bundle.shapes.length > 0 ||
302
+ bundle.widgets.length > 0 ||
303
+ bundle.screens.length > 0 ||
304
+ bundle.uiRoutes.length > 0 ||
305
+ bundle.uiActions.length > 0 ||
306
+ bundle.workflows.length > 0 ||
307
+ bundle.verifications.length > 0 ||
308
+ bundle.workflowStates.length > 0 ||
309
+ bundle.workflowTransitions.length > 0
310
+ )
311
+ .map((/** @type {any} */ bundle) => {
312
+ const sortedBundle = {
313
+ ...bundle,
314
+ actors: bundle.actors.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
315
+ roles: bundle.roles.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
316
+ entities: bundle.entities.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
317
+ enums: bundle.enums.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
318
+ capabilities: bundle.capabilities.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
319
+ shapes: bundle.shapes.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id.localeCompare(b.id)),
320
+ widgets: bundle.widgets.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
321
+ screens: bundle.screens.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
322
+ uiRoutes: bundle.uiRoutes.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
323
+ uiActions: bundle.uiActions.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
324
+ workflows: bundle.workflows.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
325
+ verifications: bundle.verifications.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
326
+ workflowStates: bundle.workflowStates.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
327
+ workflowTransitions: bundle.workflowTransitions.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id_hint.localeCompare(b.id_hint)),
328
+ docs: bundle.docs.sort((/** @type {any} */ a, /** @type {any} */ b) => a.id.localeCompare(b.id))
329
+ };
330
+ const mergeHints = buildBundleMergeHints(sortedBundle, canonicalEntityIds);
331
+ const projectionImpacts = buildProjectionImpacts({ ...sortedBundle, mergeHints }, projectionIndex);
332
+ const uiImpacts = buildUiImpacts(sortedBundle, graph);
333
+ const workflowImpacts = buildWorkflowImpacts(sortedBundle, graph);
334
+ const authPermissionHints = inferBundleAuthPermissionHints(sortedBundle);
335
+ const authClaimHints = inferBundleAuthClaimHints(sortedBundle);
336
+ const authOwnershipHints = inferBundleAuthOwnershipHints(sortedBundle);
337
+ const authRoleGuidance = inferBundleAuthRoleGuidance({
338
+ ...sortedBundle,
339
+ authPermissionHints,
340
+ authClaimHints,
341
+ authOwnershipHints
342
+ });
343
+ const docLinkSuggestions = buildBundleDocLinkSuggestions(sortedBundle, graph);
344
+ const enrichedBundle = {
345
+ ...sortedBundle,
346
+ mergeHints,
347
+ projectionImpacts,
348
+ uiImpacts,
349
+ workflowImpacts,
350
+ authPermissionHints,
351
+ authClaimHints,
352
+ authOwnershipHints,
353
+ authRoleGuidance,
354
+ docLinkSuggestions,
355
+ docDriftSummaries: buildBundleDocDriftSummariesReconcile(sortedBundle, graph, topogramRoot, confidenceRank, readTextIfExists),
356
+ docMetadataPatches: []
357
+ };
358
+ const projectionPatches = buildProjectionPatchCandidates(enrichedBundle);
359
+ const adoptionPlan = buildBundleAdoptionPlan({ ...enrichedBundle, projectionPatches }, canonicalShapeIndex);
360
+ const classifiedAuthRoleGuidance = classifyBundleAuthRoleGuidance({ ...enrichedBundle, projectionPatches, adoptionPlan });
361
+ return {
362
+ ...enrichedBundle,
363
+ authRoleGuidance: classifiedAuthRoleGuidance,
364
+ docLinkSuggestions: annotateDocLinkSuggestionsWithAuthRoleGuidance(docLinkSuggestions, classifiedAuthRoleGuidance),
365
+ projectionPatches,
366
+ adoptionPlan
367
+ };
368
+ })
369
+ .map((/** @type {any} */ bundle) => ({
370
+ ...bundle,
371
+ docMetadataPatches: buildBundleDocMetadataPatchesReconcile(bundle, confidenceRank)
372
+ }))
373
+ .filter((/** @type {any} */ bundle) => {
374
+ const reason = bundleNoiseSuppressionReason(bundle);
375
+ if (!reason) {
376
+ return true;
377
+ }
378
+ suppressedNoiseBundles.push({
379
+ slug: bundle.slug,
380
+ id: bundle.id,
381
+ reason
382
+ });
383
+ return false;
384
+ })
385
+ .sort((/** @type {any} */ a, /** @type {any} */ b) => a.slug.localeCompare(b.slug));
386
+ return { bundles: finalizedBundles, suppressedNoiseBundles };
387
+ }
388
+
389
+ /** @param {ResolvedGraph} graph @param {ImportArtifacts} appImport @param {any} topogramRoot @returns {any} */
390
+ export function buildCandidateModelFiles(graph, appImport, topogramRoot) {
391
+ /** @type {WorkflowFiles} */
392
+ /** @type {WorkflowFiles} */
393
+ const files = {};
394
+ const { bundles, suppressedNoiseBundles } = buildCandidateModelBundles(graph, appImport, topogramRoot);
395
+ const knownEnums = new Set(
396
+ bundles.flatMap((/** @type {any} */ bundle) => bundle.enums.map((/** @type {any} */ entry) => entry.id_hint))
397
+ );
398
+
399
+ for (const bundle of bundles) {
400
+ const bundleRoot = `candidates/reconcile/model/bundles/${bundle.slug}`;
401
+ files[`${bundleRoot}/README.md`] = renderCandidateBundleReadme(bundle);
402
+ for (const entry of bundle.actors) {
403
+ files[`${bundleRoot}/actors/${entry.id_hint}.tg`] = renderCandidateActor(entry);
404
+ }
405
+ for (const entry of bundle.roles) {
406
+ files[`${bundleRoot}/roles/${entry.id_hint}.tg`] = renderCandidateRole(entry);
407
+ }
408
+ for (const entry of bundle.enums) {
409
+ files[`${bundleRoot}/enums/${entry.id_hint}.tg`] = renderCandidateEnum(entry);
410
+ }
411
+ for (const entry of bundle.entities) {
412
+ files[`${bundleRoot}/entities/${entry.id_hint}.tg`] = renderCandidateEntity(entry, knownEnums);
413
+ }
414
+ for (const shape of bundle.shapes) {
415
+ files[`${bundleRoot}/shapes/${shape.id}.tg`] = renderCandidateShape(shape.id, shape.label, shape.fields);
416
+ }
417
+ for (const entry of bundle.capabilities) {
418
+ const inputShapeId = bundle.shapes.find((/** @type {any} */ shape) => shape.id === shapeIdForCapability(entry, "input")) ? shapeIdForCapability(entry, "input") : null;
419
+ const outputShapeId = bundle.shapes.find((/** @type {any} */ shape) => shape.id === shapeIdForCapability(entry, "output")) ? shapeIdForCapability(entry, "output") : null;
420
+ files[`${bundleRoot}/capabilities/${entry.id_hint}.tg`] = renderCandidateCapability(entry, inputShapeId, outputShapeId);
421
+ }
422
+ for (const entry of bundle.verifications || []) {
423
+ files[`${bundleRoot}/verifications/${entry.id_hint}.tg`] = renderCandidateVerification(entry, entry.scenarios || []);
424
+ }
425
+ for (const entry of bundle.widgets || []) {
426
+ files[`${bundleRoot}/widgets/${entry.id_hint}.tg`] = renderCandidateWidget(entry);
427
+ }
428
+ for (const entry of bundle.docs) {
429
+ if (entry.existing_canonical) {
430
+ continue;
431
+ }
432
+ const docDir = docDirForKind(entry.kind);
433
+ files[`${bundleRoot}/docs/${docDir}/${entry.id}.md`] = renderMarkdownDoc(
434
+ entry.metadata || {
435
+ id: entry.id,
436
+ kind: entry.kind,
437
+ title: entry.title,
438
+ status: "inferred",
439
+ source_of_truth: entry.source_of_truth || "imported",
440
+ confidence: entry.confidence || "low",
441
+ review_required: true,
442
+ related_entities: entry.related_entities || [],
443
+ related_capabilities: entry.related_capabilities || [],
444
+ related_actors: entry.related_actors || [],
445
+ related_roles: entry.related_roles || [],
446
+ related_rules: entry.related_rules || [],
447
+ related_workflows: entry.related_workflows || [],
448
+ provenance: entry.provenance || [],
449
+ tags: entry.tags || ["import", entry.kind]
450
+ },
451
+ entry.body || "Candidate imported doc."
452
+ );
453
+ }
454
+ for (const entry of bundle.workflows) {
455
+ const states = bundle.workflowStates.filter((/** @type {any} */ state) => state.workflow_id === entry.id_hint);
456
+ const transitions = bundle.workflowTransitions.filter((/** @type {any} */ transition) => transition.workflow_id === entry.id_hint);
457
+ files[`${bundleRoot}/docs/workflows/${entry.id_hint}.md`] = renderCandidateWorkflowDoc(entry, states, transitions);
458
+ files[`${bundleRoot}/decisions/dec_${entry.id_hint.replace(/^workflow_/, "")}.tg`] = renderCandidateWorkflowDecision(entry, states, transitions);
459
+ }
460
+ for (const screen of bundle.screens) {
461
+ const routes = bundle.uiRoutes.filter((/** @type {any} */ route) => route.screen_id === screen.id_hint);
462
+ const actions = bundle.uiActions.filter((/** @type {any} */ action) => action.screen_id === screen.id_hint);
463
+ files[`${bundleRoot}/docs/reports/ui-${screen.id_hint}.md`] = renderCandidateUiReportDoc(screen, routes, actions);
464
+ }
465
+ for (const patch of bundle.projectionPatches || []) {
466
+ files[`${bundleRoot}/${patch.patch_rel_path}`] = renderProjectionPatchDoc(patch);
467
+ }
468
+ for (const patch of bundle.docLinkSuggestions || []) {
469
+ files[`${bundleRoot}/${patch.patch_rel_path}`] = renderDocLinkPatchDoc(patch);
470
+ }
471
+ for (const patch of bundle.docMetadataPatches || []) {
472
+ files[`${bundleRoot}/${patch.patch_rel_path}`] = renderDocMetadataPatchDoc(patch);
473
+ }
474
+ }
475
+
476
+ return { files, bundles, suppressedNoiseBundles };
477
+ }