forgeos 0.1.0-alpha.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 (540) hide show
  1. package/.npmignore +1 -0
  2. package/AGENTS.md +277 -0
  3. package/CHANGELOG.md +8 -0
  4. package/CONTRIBUTING.md +58 -0
  5. package/README.md +377 -0
  6. package/bin/forge-bun.mjs +110 -0
  7. package/bin/forge.mjs +19 -0
  8. package/package.json +96 -0
  9. package/packages/eslint-plugin-forge/index.ts +15 -0
  10. package/packages/eslint-plugin-forge/package.json +10 -0
  11. package/packages/eslint-plugin-forge/src/check-source.ts +95 -0
  12. package/packages/eslint-plugin-forge/src/load-artifacts.ts +24 -0
  13. package/packages/eslint-plugin-forge/src/rule-no-forge-guard-violation.ts +93 -0
  14. package/src/forge/_generated/actionSubscriptions.json +2 -0
  15. package/src/forge/_generated/actionSubscriptions.ts +10 -0
  16. package/src/forge/_generated/agentAdapterManifest.json +2 -0
  17. package/src/forge/_generated/agentAdapterManifest.ts +73 -0
  18. package/src/forge/_generated/agentContract.json +2 -0
  19. package/src/forge/_generated/agentContract.ts +912 -0
  20. package/src/forge/_generated/agentQuickstart.md +32 -0
  21. package/src/forge/_generated/aiContext.ts +59 -0
  22. package/src/forge/_generated/aiModels.json +2 -0
  23. package/src/forge/_generated/aiModels.ts +35 -0
  24. package/src/forge/_generated/aiProviders.json +2 -0
  25. package/src/forge/_generated/aiProviders.ts +23 -0
  26. package/src/forge/_generated/aiRegistry.json +2 -0
  27. package/src/forge/_generated/aiRegistry.ts +29 -0
  28. package/src/forge/_generated/api.json +2 -0
  29. package/src/forge/_generated/api.ts +8 -0
  30. package/src/forge/_generated/appGraph.json +2 -0
  31. package/src/forge/_generated/appGraph.ts +14511 -0
  32. package/src/forge/_generated/appMap.md +35 -0
  33. package/src/forge/_generated/artifactManifest.json +2 -0
  34. package/src/forge/_generated/artifactManifest.ts +7 -0
  35. package/src/forge/_generated/authClaims.json +2 -0
  36. package/src/forge/_generated/authClaims.ts +13 -0
  37. package/src/forge/_generated/authConfig.json +2 -0
  38. package/src/forge/_generated/authConfig.ts +17 -0
  39. package/src/forge/_generated/authContext.ts +23 -0
  40. package/src/forge/_generated/authRegistry.json +2 -0
  41. package/src/forge/_generated/authRegistry.ts +25 -0
  42. package/src/forge/_generated/buildInfo.json +2 -0
  43. package/src/forge/_generated/buildInfo.ts +9 -0
  44. package/src/forge/_generated/capabilityMap.json +2 -0
  45. package/src/forge/_generated/capabilityMap.md +15 -0
  46. package/src/forge/_generated/capabilityMap.ts +17 -0
  47. package/src/forge/_generated/client.ts +282 -0
  48. package/src/forge/_generated/clientApi.ts +9 -0
  49. package/src/forge/_generated/clientManifest.json +2 -0
  50. package/src/forge/_generated/clientManifest.ts +39 -0
  51. package/src/forge/_generated/clientTypes.ts +78 -0
  52. package/src/forge/_generated/configRegistry.json +2 -0
  53. package/src/forge/_generated/configRegistry.ts +4 -0
  54. package/src/forge/_generated/dataGraph.json +2 -0
  55. package/src/forge/_generated/dataGraph.ts +8 -0
  56. package/src/forge/_generated/db.json +2 -0
  57. package/src/forge/_generated/db.ts +2 -0
  58. package/src/forge/_generated/dbSecurityManifest.json +2 -0
  59. package/src/forge/_generated/dbSecurityManifest.ts +15 -0
  60. package/src/forge/_generated/dbSessionContext.json +2 -0
  61. package/src/forge/_generated/dbSessionContext.ts +39 -0
  62. package/src/forge/_generated/deployManifest.json +2 -0
  63. package/src/forge/_generated/deployManifest.ts +14 -0
  64. package/src/forge/_generated/devManifest.json +2 -0
  65. package/src/forge/_generated/devManifest.ts +47 -0
  66. package/src/forge/_generated/envSchema.json +2 -0
  67. package/src/forge/_generated/envSchema.ts +59 -0
  68. package/src/forge/_generated/frontendGraph.json +2 -0
  69. package/src/forge/_generated/frontendGraph.ts +27 -0
  70. package/src/forge/_generated/importGuards.json +2 -0
  71. package/src/forge/_generated/importGuards.ts +652 -0
  72. package/src/forge/_generated/index.ts +67 -0
  73. package/src/forge/_generated/liveProductionManifest.json +2 -0
  74. package/src/forge/_generated/liveProductionManifest.ts +23 -0
  75. package/src/forge/_generated/liveProtocol.json +2 -0
  76. package/src/forge/_generated/liveProtocol.ts +21 -0
  77. package/src/forge/_generated/liveQueryRegistry.json +2 -0
  78. package/src/forge/_generated/liveQueryRegistry.ts +9 -0
  79. package/src/forge/_generated/liveTransportConfig.json +2 -0
  80. package/src/forge/_generated/liveTransportConfig.ts +19 -0
  81. package/src/forge/_generated/makeRegistry.json +2 -0
  82. package/src/forge/_generated/makeRegistry.ts +163 -0
  83. package/src/forge/_generated/makeTemplates.json +2 -0
  84. package/src/forge/_generated/makeTemplates.ts +61 -0
  85. package/src/forge/_generated/mockMap.json +2 -0
  86. package/src/forge/_generated/mockMap.ts +7 -0
  87. package/src/forge/_generated/operationPlaybooks.md +145 -0
  88. package/src/forge/_generated/packageGraph.json +2 -0
  89. package/src/forge/_generated/packageGraph.ts +168569 -0
  90. package/src/forge/_generated/packageUpgradeRegistry.json +2 -0
  91. package/src/forge/_generated/packageUpgradeRegistry.ts +15 -0
  92. package/src/forge/_generated/permissionMatrix.json +2 -0
  93. package/src/forge/_generated/permissionMatrix.ts +7 -0
  94. package/src/forge/_generated/policyRegistry.json +2 -0
  95. package/src/forge/_generated/policyRegistry.ts +11 -0
  96. package/src/forge/_generated/queryRegistry.json +2 -0
  97. package/src/forge/_generated/queryRegistry.ts +9 -0
  98. package/src/forge/_generated/react.d.ts +22 -0
  99. package/src/forge/_generated/react.ts +29 -0
  100. package/src/forge/_generated/reactManifest.json +2 -0
  101. package/src/forge/_generated/reactManifest.ts +19 -0
  102. package/src/forge/_generated/releaseManifest.json +2 -0
  103. package/src/forge/_generated/releaseManifest.ts +25 -0
  104. package/src/forge/_generated/rlsPolicies.json +2 -0
  105. package/src/forge/_generated/rlsPolicies.sql +34 -0
  106. package/src/forge/_generated/rlsPolicies.ts +6 -0
  107. package/src/forge/_generated/runtimeGraph.json +2 -0
  108. package/src/forge/_generated/runtimeGraph.ts +8 -0
  109. package/src/forge/_generated/runtimeMatrix.json +2 -0
  110. package/src/forge/_generated/runtimeMatrix.ts +229125 -0
  111. package/src/forge/_generated/runtimeRegistry.ts +2 -0
  112. package/src/forge/_generated/runtimeRules.md +79 -0
  113. package/src/forge/_generated/secretRegistry.json +2 -0
  114. package/src/forge/_generated/secretRegistry.ts +50 -0
  115. package/src/forge/_generated/secretsContext.ts +11 -0
  116. package/src/forge/_generated/serverApi.ts +10 -0
  117. package/src/forge/_generated/sourceMapManifest.json +2 -0
  118. package/src/forge/_generated/sourceMapManifest.ts +7 -0
  119. package/src/forge/_generated/sqlPlan.json +2 -0
  120. package/src/forge/_generated/sqlPlan.ts +88 -0
  121. package/src/forge/_generated/subscriptionManifest.json +2 -0
  122. package/src/forge/_generated/subscriptionManifest.ts +7 -0
  123. package/src/forge/_generated/symbolicationManifest.json +2 -0
  124. package/src/forge/_generated/symbolicationManifest.ts +17 -0
  125. package/src/forge/_generated/telemetryRegistry.json +2 -0
  126. package/src/forge/_generated/telemetryRegistry.ts +9 -0
  127. package/src/forge/_generated/telemetrySinks.json +2 -0
  128. package/src/forge/_generated/telemetrySinks.ts +11 -0
  129. package/src/forge/_generated/tenantScope.json +2 -0
  130. package/src/forge/_generated/tenantScope.ts +8 -0
  131. package/src/forge/_generated/testGraph.json +2 -0
  132. package/src/forge/_generated/testGraph.ts +3054 -0
  133. package/src/forge/_generated/testPlanRegistry.json +2 -0
  134. package/src/forge/_generated/testPlanRegistry.ts +33 -0
  135. package/src/forge/_generated/uiRoutes.json +2 -0
  136. package/src/forge/_generated/uiRoutes.ts +16 -0
  137. package/src/forge/_generated/uiScenarios.json +2 -0
  138. package/src/forge/_generated/uiScenarios.ts +30 -0
  139. package/src/forge/_generated/uiTestManifest.json +2 -0
  140. package/src/forge/_generated/uiTestManifest.ts +27 -0
  141. package/src/forge/_generated/workflowRegistry.json +2 -0
  142. package/src/forge/_generated/workflowRegistry.ts +9 -0
  143. package/src/forge/_generated/workflowSubscriptions.json +2 -0
  144. package/src/forge/_generated/workflowSubscriptions.ts +10 -0
  145. package/src/forge/agent-adapters/index.ts +1002 -0
  146. package/src/forge/agent-adapters/types.ts +135 -0
  147. package/src/forge/cli/agent-contract.ts +50 -0
  148. package/src/forge/cli/ai.ts +148 -0
  149. package/src/forge/cli/auth.ts +198 -0
  150. package/src/forge/cli/build.ts +105 -0
  151. package/src/forge/cli/bun-exec.ts +4 -0
  152. package/src/forge/cli/commands.ts +1130 -0
  153. package/src/forge/cli/db.ts +316 -0
  154. package/src/forge/cli/deps.ts +277 -0
  155. package/src/forge/cli/dev.ts +529 -0
  156. package/src/forge/cli/doctor.ts +209 -0
  157. package/src/forge/cli/feature.ts +485 -0
  158. package/src/forge/cli/index.ts +25 -0
  159. package/src/forge/cli/lint-forge.ts +119 -0
  160. package/src/forge/cli/live.ts +179 -0
  161. package/src/forge/cli/main.ts +92 -0
  162. package/src/forge/cli/make.ts +133 -0
  163. package/src/forge/cli/new.ts +505 -0
  164. package/src/forge/cli/outbox.ts +297 -0
  165. package/src/forge/cli/output.ts +114 -0
  166. package/src/forge/cli/parse.ts +2211 -0
  167. package/src/forge/cli/policy.ts +204 -0
  168. package/src/forge/cli/query.ts +91 -0
  169. package/src/forge/cli/refactor.ts +221 -0
  170. package/src/forge/cli/release.ts +285 -0
  171. package/src/forge/cli/rls.ts +322 -0
  172. package/src/forge/cli/run.ts +76 -0
  173. package/src/forge/cli/secrets.ts +274 -0
  174. package/src/forge/cli/self-host.ts +468 -0
  175. package/src/forge/cli/serve.ts +93 -0
  176. package/src/forge/cli/telemetry.ts +219 -0
  177. package/src/forge/cli/verify.ts +587 -0
  178. package/src/forge/cli/version.ts +1 -0
  179. package/src/forge/cli/windows.ts +413 -0
  180. package/src/forge/cli/worker.ts +87 -0
  181. package/src/forge/cli/workflow.ts +424 -0
  182. package/src/forge/compiler/action-subscriptions/build.ts +116 -0
  183. package/src/forge/compiler/action-subscriptions/constants.ts +2 -0
  184. package/src/forge/compiler/action-subscriptions/index.ts +6 -0
  185. package/src/forge/compiler/action-subscriptions/parse.ts +6 -0
  186. package/src/forge/compiler/agent-contract/build.ts +1651 -0
  187. package/src/forge/compiler/agent-contract/types.ts +326 -0
  188. package/src/forge/compiler/ai-registry/build.ts +165 -0
  189. package/src/forge/compiler/ai-registry/constants.ts +2 -0
  190. package/src/forge/compiler/ai-registry/parse.ts +56 -0
  191. package/src/forge/compiler/api-surface/build.ts +107 -0
  192. package/src/forge/compiler/app-graph/build.ts +121 -0
  193. package/src/forge/compiler/app-graph/classify.ts +10 -0
  194. package/src/forge/compiler/app-graph/dup-symbol.ts +29 -0
  195. package/src/forge/compiler/app-graph/extract.ts +124 -0
  196. package/src/forge/compiler/app-graph/forge-apis.ts +29 -0
  197. package/src/forge/compiler/app-graph/index.ts +15 -0
  198. package/src/forge/compiler/app-graph/module-graph.ts +320 -0
  199. package/src/forge/compiler/app-graph/parser.ts +119 -0
  200. package/src/forge/compiler/app-graph/symbols.ts +48 -0
  201. package/src/forge/compiler/app-graph/tsconfig-hash.ts +62 -0
  202. package/src/forge/compiler/app-graph/types.ts +43 -0
  203. package/src/forge/compiler/app-graph/versions.ts +14 -0
  204. package/src/forge/compiler/cache/index.ts +17 -0
  205. package/src/forge/compiler/cache/key.ts +46 -0
  206. package/src/forge/compiler/cache/scheduler.ts +72 -0
  207. package/src/forge/compiler/cache/store.ts +78 -0
  208. package/src/forge/compiler/classifier/capabilities.ts +78 -0
  209. package/src/forge/compiler/classifier/classify.ts +113 -0
  210. package/src/forge/compiler/classifier/contexts.ts +188 -0
  211. package/src/forge/compiler/classifier/index.ts +18 -0
  212. package/src/forge/compiler/classifier/runtime-matrix.ts +45 -0
  213. package/src/forge/compiler/classifier/secrets.ts +41 -0
  214. package/src/forge/compiler/classifier/signals.ts +129 -0
  215. package/src/forge/compiler/client-sdk/build-manifest.ts +151 -0
  216. package/src/forge/compiler/client-sdk/render-client.ts +432 -0
  217. package/src/forge/compiler/data-graph/build.ts +131 -0
  218. package/src/forge/compiler/data-graph/constants.ts +5 -0
  219. package/src/forge/compiler/data-graph/index.ts +6 -0
  220. package/src/forge/compiler/data-graph/parse.ts +176 -0
  221. package/src/forge/compiler/data-graph/rls/build.ts +222 -0
  222. package/src/forge/compiler/data-graph/rls/types.ts +62 -0
  223. package/src/forge/compiler/data-graph/sql/ddl.ts +390 -0
  224. package/src/forge/compiler/data-graph/sql/naming.ts +10 -0
  225. package/src/forge/compiler/data-graph/sql/serialize.ts +85 -0
  226. package/src/forge/compiler/data-graph/sql/types.ts +37 -0
  227. package/src/forge/compiler/dev-manifest/build.ts +170 -0
  228. package/src/forge/compiler/dev-manifest/constants.ts +5 -0
  229. package/src/forge/compiler/diagnostics/codes.ts +611 -0
  230. package/src/forge/compiler/diagnostics/create.ts +245 -0
  231. package/src/forge/compiler/diagnostics/index.ts +55 -0
  232. package/src/forge/compiler/emitter/artifact-kind.ts +14 -0
  233. package/src/forge/compiler/emitter/barrel.ts +44 -0
  234. package/src/forge/compiler/emitter/constants.ts +7 -0
  235. package/src/forge/compiler/emitter/emit.ts +237 -0
  236. package/src/forge/compiler/emitter/index.ts +24 -0
  237. package/src/forge/compiler/emitter/lock.ts +62 -0
  238. package/src/forge/compiler/emitter/render.ts +73 -0
  239. package/src/forge/compiler/emitter/write.ts +35 -0
  240. package/src/forge/compiler/frontend-graph/build.ts +495 -0
  241. package/src/forge/compiler/fs/index.ts +23 -0
  242. package/src/forge/compiler/fs/memory.ts +233 -0
  243. package/src/forge/compiler/fs/node.ts +139 -0
  244. package/src/forge/compiler/fs/profile.ts +108 -0
  245. package/src/forge/compiler/fs/types.ts +52 -0
  246. package/src/forge/compiler/guards/artifacts.ts +96 -0
  247. package/src/forge/compiler/guards/check-ai-usage.ts +98 -0
  248. package/src/forge/compiler/guards/check-import-guards.ts +106 -0
  249. package/src/forge/compiler/guards/check-process-env.ts +98 -0
  250. package/src/forge/compiler/guards/check-query-usage.ts +76 -0
  251. package/src/forge/compiler/guards/index.ts +11 -0
  252. package/src/forge/compiler/guards/propagate-contexts.ts +57 -0
  253. package/src/forge/compiler/index.ts +17 -0
  254. package/src/forge/compiler/integration/add.ts +496 -0
  255. package/src/forge/compiler/integration/index.ts +17 -0
  256. package/src/forge/compiler/integration/plan.ts +283 -0
  257. package/src/forge/compiler/integration/render.ts +189 -0
  258. package/src/forge/compiler/integration/snapshot.ts +52 -0
  259. package/src/forge/compiler/integration/templates/ai.ts +131 -0
  260. package/src/forge/compiler/integration/templates/index.ts +8 -0
  261. package/src/forge/compiler/integration/templates/posthog.ts +145 -0
  262. package/src/forge/compiler/integration/templates/render.ts +113 -0
  263. package/src/forge/compiler/integration/templates/sentry.ts +151 -0
  264. package/src/forge/compiler/integration/templates/stripe.ts +109 -0
  265. package/src/forge/compiler/integration/templates/types.ts +14 -0
  266. package/src/forge/compiler/integration/templates/zod.ts +55 -0
  267. package/src/forge/compiler/live-production/types.ts +122 -0
  268. package/src/forge/compiler/live-query-registry/build.ts +150 -0
  269. package/src/forge/compiler/live-query-registry/constants.ts +2 -0
  270. package/src/forge/compiler/make-registry/build.ts +179 -0
  271. package/src/forge/compiler/orchestrator/discover.ts +214 -0
  272. package/src/forge/compiler/orchestrator/fast-check.ts +117 -0
  273. package/src/forge/compiler/orchestrator/generate-lock.ts +138 -0
  274. package/src/forge/compiler/orchestrator/guards.ts +5 -0
  275. package/src/forge/compiler/orchestrator/index.ts +27 -0
  276. package/src/forge/compiler/orchestrator/manifest-hashes.ts +21 -0
  277. package/src/forge/compiler/orchestrator/manifest.ts +92 -0
  278. package/src/forge/compiler/orchestrator/orphans.ts +51 -0
  279. package/src/forge/compiler/orchestrator/plan.ts +876 -0
  280. package/src/forge/compiler/orchestrator/profile.ts +36 -0
  281. package/src/forge/compiler/orchestrator/run.ts +277 -0
  282. package/src/forge/compiler/orchestrator/serialize.ts +886 -0
  283. package/src/forge/compiler/orchestrator/session.ts +96 -0
  284. package/src/forge/compiler/orchestrator/types.ts +31 -0
  285. package/src/forge/compiler/orchestrator/verify.ts +38 -0
  286. package/src/forge/compiler/orchestrator/workspace-index.ts +154 -0
  287. package/src/forge/compiler/package-graph/capabilities-stub.ts +33 -0
  288. package/src/forge/compiler/package-graph/checksum.ts +97 -0
  289. package/src/forge/compiler/package-graph/compiler.ts +392 -0
  290. package/src/forge/compiler/package-graph/constants.ts +4 -0
  291. package/src/forge/compiler/package-graph/dts-extractor.ts +142 -0
  292. package/src/forge/compiler/package-graph/exports-discovery.ts +84 -0
  293. package/src/forge/compiler/package-graph/extract-dts.ts +32 -0
  294. package/src/forge/compiler/package-graph/index.ts +33 -0
  295. package/src/forge/compiler/package-graph/jsdoc.ts +62 -0
  296. package/src/forge/compiler/package-graph/read-file.ts +21 -0
  297. package/src/forge/compiler/package-graph/resolve.ts +127 -0
  298. package/src/forge/compiler/package-manager/adapter.ts +237 -0
  299. package/src/forge/compiler/package-manager/bun-executable.ts +92 -0
  300. package/src/forge/compiler/package-manager/commands.ts +47 -0
  301. package/src/forge/compiler/package-manager/detect.ts +79 -0
  302. package/src/forge/compiler/package-manager/executor.ts +117 -0
  303. package/src/forge/compiler/package-manager/index.ts +22 -0
  304. package/src/forge/compiler/package-manager/parse-spec.ts +16 -0
  305. package/src/forge/compiler/package-manager/version.ts +27 -0
  306. package/src/forge/compiler/package-upgrades/apply.ts +195 -0
  307. package/src/forge/compiler/package-upgrades/comparator.ts +181 -0
  308. package/src/forge/compiler/package-upgrades/impact.ts +139 -0
  309. package/src/forge/compiler/package-upgrades/markdown.ts +97 -0
  310. package/src/forge/compiler/package-upgrades/planner.ts +532 -0
  311. package/src/forge/compiler/package-upgrades/risk.ts +208 -0
  312. package/src/forge/compiler/package-upgrades/types.ts +174 -0
  313. package/src/forge/compiler/policy-registry/build.ts +266 -0
  314. package/src/forge/compiler/policy-registry/constants.ts +2 -0
  315. package/src/forge/compiler/policy-registry/parse.ts +81 -0
  316. package/src/forge/compiler/primitives/compare.ts +26 -0
  317. package/src/forge/compiler/primitives/hash.ts +40 -0
  318. package/src/forge/compiler/primitives/header.ts +45 -0
  319. package/src/forge/compiler/primitives/index.ts +45 -0
  320. package/src/forge/compiler/primitives/paths.ts +24 -0
  321. package/src/forge/compiler/primitives/result.ts +164 -0
  322. package/src/forge/compiler/primitives/serialize.ts +66 -0
  323. package/src/forge/compiler/primitives/sort.ts +87 -0
  324. package/src/forge/compiler/query-registry/build.ts +114 -0
  325. package/src/forge/compiler/query-registry/constants.ts +2 -0
  326. package/src/forge/compiler/recipes/definitions.ts +289 -0
  327. package/src/forge/compiler/recipes/helpers.ts +37 -0
  328. package/src/forge/compiler/recipes/index.ts +21 -0
  329. package/src/forge/compiler/recipes/registry.ts +102 -0
  330. package/src/forge/compiler/release/build.ts +100 -0
  331. package/src/forge/compiler/release/types.ts +119 -0
  332. package/src/forge/compiler/runtime-graph/build.ts +137 -0
  333. package/src/forge/compiler/runtime-graph/constants.ts +5 -0
  334. package/src/forge/compiler/runtime-graph/index.ts +5 -0
  335. package/src/forge/compiler/sandbox/artifact-sanitize.ts +26 -0
  336. package/src/forge/compiler/sandbox/backends/child.ts +123 -0
  337. package/src/forge/compiler/sandbox/backends/docker.ts +173 -0
  338. package/src/forge/compiler/sandbox/index.ts +51 -0
  339. package/src/forge/compiler/sandbox/inspect.ts +143 -0
  340. package/src/forge/compiler/sandbox/inspector-entry.ts +115 -0
  341. package/src/forge/compiler/sandbox/limits.ts +31 -0
  342. package/src/forge/compiler/sandbox/scrub-env.ts +60 -0
  343. package/src/forge/compiler/sandbox/secret-scan.ts +54 -0
  344. package/src/forge/compiler/sandbox/serialize.ts +106 -0
  345. package/src/forge/compiler/sandbox/types.ts +7 -0
  346. package/src/forge/compiler/secret-registry/build.ts +123 -0
  347. package/src/forge/compiler/telemetry-registry/build.ts +89 -0
  348. package/src/forge/compiler/telemetry-registry/constants.ts +2 -0
  349. package/src/forge/compiler/telemetry-registry/parse.ts +13 -0
  350. package/src/forge/compiler/test-graph/build.ts +277 -0
  351. package/src/forge/compiler/types/action-subscriptions.ts +19 -0
  352. package/src/forge/compiler/types/ai-registry.ts +33 -0
  353. package/src/forge/compiler/types/app-graph.ts +80 -0
  354. package/src/forge/compiler/types/capability.ts +29 -0
  355. package/src/forge/compiler/types/classification.ts +9 -0
  356. package/src/forge/compiler/types/cli.ts +159 -0
  357. package/src/forge/compiler/types/data-graph.ts +24 -0
  358. package/src/forge/compiler/types/dev-manifest.ts +41 -0
  359. package/src/forge/compiler/types/diagnostic.ts +12 -0
  360. package/src/forge/compiler/types/emit.ts +25 -0
  361. package/src/forge/compiler/types/frontend-graph.ts +81 -0
  362. package/src/forge/compiler/types/import-guards.ts +19 -0
  363. package/src/forge/compiler/types/index.ts +98 -0
  364. package/src/forge/compiler/types/integration.ts +25 -0
  365. package/src/forge/compiler/types/json.ts +3 -0
  366. package/src/forge/compiler/types/live-query-registry.ts +32 -0
  367. package/src/forge/compiler/types/lock.ts +37 -0
  368. package/src/forge/compiler/types/package-graph.ts +84 -0
  369. package/src/forge/compiler/types/policy-registry.ts +69 -0
  370. package/src/forge/compiler/types/query-registry.ts +18 -0
  371. package/src/forge/compiler/types/runtime-graph.ts +30 -0
  372. package/src/forge/compiler/types/runtime-matrix.ts +16 -0
  373. package/src/forge/compiler/types/runtime.ts +30 -0
  374. package/src/forge/compiler/types/sandbox.ts +24 -0
  375. package/src/forge/compiler/types/secret-registry.ts +38 -0
  376. package/src/forge/compiler/types/telemetry-registry.ts +26 -0
  377. package/src/forge/compiler/types/test-graph.ts +45 -0
  378. package/src/forge/compiler/types/workflow-registry.ts +42 -0
  379. package/src/forge/compiler/workflow-registry/build.ts +180 -0
  380. package/src/forge/compiler/workflow-registry/constants.ts +2 -0
  381. package/src/forge/compiler/workflow-registry/index.ts +5 -0
  382. package/src/forge/compiler/workflow-registry/parse.ts +19 -0
  383. package/src/forge/dev/server.ts +1379 -0
  384. package/src/forge/dev/types.ts +49 -0
  385. package/src/forge/dev/watch.ts +109 -0
  386. package/src/forge/dev-console/cycle.ts +652 -0
  387. package/src/forge/dev-console/types.ts +99 -0
  388. package/src/forge/feature/compiler.ts +656 -0
  389. package/src/forge/feature/examples.ts +125 -0
  390. package/src/forge/feature/types.ts +177 -0
  391. package/src/forge/impact/index.ts +1160 -0
  392. package/src/forge/impact/types.ts +151 -0
  393. package/src/forge/intent/index.ts +490 -0
  394. package/src/forge/intent/types.ts +73 -0
  395. package/src/forge/make/fields.ts +146 -0
  396. package/src/forge/make/index.ts +1101 -0
  397. package/src/forge/make/naming.ts +42 -0
  398. package/src/forge/make/templates.ts +525 -0
  399. package/src/forge/make/types.ts +151 -0
  400. package/src/forge/platform/module.ts +20 -0
  401. package/src/forge/policy.ts +1 -0
  402. package/src/forge/react/index.ts +418 -0
  403. package/src/forge/refactor/index.ts +1936 -0
  404. package/src/forge/refactor/text-utils.ts +34 -0
  405. package/src/forge/refactor/types.ts +191 -0
  406. package/src/forge/refactor/workspace-fs.ts +171 -0
  407. package/src/forge/repair/index.ts +656 -0
  408. package/src/forge/repair/rules/index.ts +476 -0
  409. package/src/forge/repair/types.ts +175 -0
  410. package/src/forge/review/index.ts +992 -0
  411. package/src/forge/review/types.ts +196 -0
  412. package/src/forge/runtime/ai/check.ts +86 -0
  413. package/src/forge/runtime/ai/context.ts +394 -0
  414. package/src/forge/runtime/ai/cost-estimator.ts +41 -0
  415. package/src/forge/runtime/ai/mock.ts +49 -0
  416. package/src/forge/runtime/ai/providers.ts +78 -0
  417. package/src/forge/runtime/ai/state.ts +17 -0
  418. package/src/forge/runtime/ai/types.ts +67 -0
  419. package/src/forge/runtime/auth/authenticate.ts +58 -0
  420. package/src/forge/runtime/auth/claims.ts +119 -0
  421. package/src/forge/runtime/auth/config.ts +148 -0
  422. package/src/forge/runtime/auth/errors.ts +45 -0
  423. package/src/forge/runtime/auth/evaluate.ts +126 -0
  424. package/src/forge/runtime/auth/resolve.ts +74 -0
  425. package/src/forge/runtime/auth/types.ts +87 -0
  426. package/src/forge/runtime/auth/verifier.ts +138 -0
  427. package/src/forge/runtime/context/create-context.ts +204 -0
  428. package/src/forge/runtime/context/create-query-context.ts +34 -0
  429. package/src/forge/runtime/db/adapter.ts +31 -0
  430. package/src/forge/runtime/db/factory.ts +83 -0
  431. package/src/forge/runtime/db/generated-client.ts +294 -0
  432. package/src/forge/runtime/db/memory-adapter.ts +706 -0
  433. package/src/forge/runtime/db/migrate.ts +132 -0
  434. package/src/forge/runtime/db/outbox.ts +54 -0
  435. package/src/forge/runtime/db/pglite-adapter.ts +51 -0
  436. package/src/forge/runtime/db/postgres-adapter.ts +112 -0
  437. package/src/forge/runtime/db/read-only-client.ts +97 -0
  438. package/src/forge/runtime/db/session-context.ts +62 -0
  439. package/src/forge/runtime/executor.ts +446 -0
  440. package/src/forge/runtime/live/dependency-tracker.ts +57 -0
  441. package/src/forge/runtime/live/invalidation-log.ts +189 -0
  442. package/src/forge/runtime/live/live-query-runner.ts +267 -0
  443. package/src/forge/runtime/live/registry.ts +28 -0
  444. package/src/forge/runtime/live/sse.ts +75 -0
  445. package/src/forge/runtime/live/subscription-manager.ts +443 -0
  446. package/src/forge/runtime/live/types.ts +143 -0
  447. package/src/forge/runtime/outbox/claim.ts +153 -0
  448. package/src/forge/runtime/outbox/process.ts +298 -0
  449. package/src/forge/runtime/outbox/retry.ts +8 -0
  450. package/src/forge/runtime/outbox/subscriptions.ts +33 -0
  451. package/src/forge/runtime/outbox/types.ts +69 -0
  452. package/src/forge/runtime/policy/check.ts +157 -0
  453. package/src/forge/runtime/policy/load.ts +55 -0
  454. package/src/forge/runtime/query/registry.ts +19 -0
  455. package/src/forge/runtime/query/run-query.ts +347 -0
  456. package/src/forge/runtime/release/runtime.ts +322 -0
  457. package/src/forge/runtime/release/symbolicate.ts +175 -0
  458. package/src/forge/runtime/runner/command-transaction.ts +193 -0
  459. package/src/forge/runtime/runner/run-entry.ts +226 -0
  460. package/src/forge/runtime/secrets/check.ts +78 -0
  461. package/src/forge/runtime/secrets/create-context.ts +138 -0
  462. package/src/forge/runtime/secrets/env-loader.ts +94 -0
  463. package/src/forge/runtime/secrets/runtime-bundle.ts +47 -0
  464. package/src/forge/runtime/secrets/types.ts +31 -0
  465. package/src/forge/runtime/telemetry/buffer.ts +87 -0
  466. package/src/forge/runtime/telemetry/context.ts +192 -0
  467. package/src/forge/runtime/telemetry/correlation.ts +13 -0
  468. package/src/forge/runtime/telemetry/flush.ts +190 -0
  469. package/src/forge/runtime/telemetry/process.ts +20 -0
  470. package/src/forge/runtime/telemetry/scrubber.ts +115 -0
  471. package/src/forge/runtime/telemetry/sinks/local-jsonl.ts +39 -0
  472. package/src/forge/runtime/telemetry/sinks/posthog.ts +64 -0
  473. package/src/forge/runtime/telemetry/sinks/sentry.ts +60 -0
  474. package/src/forge/runtime/telemetry/spans.ts +58 -0
  475. package/src/forge/runtime/telemetry/types.ts +64 -0
  476. package/src/forge/runtime/workflows/cancel.ts +26 -0
  477. package/src/forge/runtime/workflows/create-run.ts +98 -0
  478. package/src/forge/runtime/workflows/process-run.ts +182 -0
  479. package/src/forge/runtime/workflows/process-step.ts +190 -0
  480. package/src/forge/runtime/workflows/process.ts +260 -0
  481. package/src/forge/runtime/workflows/registry.ts +51 -0
  482. package/src/forge/runtime/workflows/resolve-step.ts +46 -0
  483. package/src/forge/runtime/workflows/retry-run.ts +44 -0
  484. package/src/forge/runtime/workflows/retry.ts +8 -0
  485. package/src/forge/runtime/workflows/sanitize.ts +19 -0
  486. package/src/forge/runtime/workflows/start-from-outbox.ts +71 -0
  487. package/src/forge/runtime/workflows/types.ts +77 -0
  488. package/src/forge/server.ts +96 -0
  489. package/src/forge/ui/index.ts +770 -0
  490. package/src/forge/ui/types.ts +191 -0
  491. package/templates/b2b-support-web/.env.example +22 -0
  492. package/templates/b2b-support-web/.vscode/settings.json +14 -0
  493. package/templates/b2b-support-web/AGENTS.md +108 -0
  494. package/templates/b2b-support-web/README.md +48 -0
  495. package/templates/b2b-support-web/forge.config.ts +3 -0
  496. package/templates/b2b-support-web/package.json +34 -0
  497. package/templates/b2b-support-web/src/actions/captureTicketCreated.ts +14 -0
  498. package/templates/b2b-support-web/src/commands/closeTicket.ts +20 -0
  499. package/templates/b2b-support-web/src/commands/createTicket.ts +47 -0
  500. package/templates/b2b-support-web/src/commands/manageBilling.ts +9 -0
  501. package/templates/b2b-support-web/src/forge/schema.ts +35 -0
  502. package/templates/b2b-support-web/src/policies.ts +9 -0
  503. package/templates/b2b-support-web/src/queries/getTicket.ts +6 -0
  504. package/templates/b2b-support-web/src/queries/listTickets.ts +6 -0
  505. package/templates/b2b-support-web/src/queries/liveTickets.ts +9 -0
  506. package/templates/b2b-support-web/src/workflows/triageTicketWorkflow.ts +64 -0
  507. package/templates/b2b-support-web/tsconfig.json +14 -0
  508. package/templates/b2b-support-web/web/app/globals.css +77 -0
  509. package/templates/b2b-support-web/web/app/layout.tsx +13 -0
  510. package/templates/b2b-support-web/web/app/page.tsx +13 -0
  511. package/templates/b2b-support-web/web/app/providers.tsx +21 -0
  512. package/templates/b2b-support-web/web/app/tickets/page.tsx +21 -0
  513. package/templates/b2b-support-web/web/components/CreateTicketForm.tsx +43 -0
  514. package/templates/b2b-support-web/web/components/PolicyDeniedDemo.tsx +31 -0
  515. package/templates/b2b-support-web/web/components/TicketList.tsx +52 -0
  516. package/templates/b2b-support-web/web/components/TraceDetails.tsx +18 -0
  517. package/templates/b2b-support-web/web/components/TriageStatus.tsx +13 -0
  518. package/templates/b2b-support-web/web/lib/forge.ts +13 -0
  519. package/templates/b2b-support-web/web/next-env.d.ts +5 -0
  520. package/templates/b2b-support-web/web/next.config.ts +8 -0
  521. package/templates/b2b-support-web/web/package.json +21 -0
  522. package/templates/b2b-support-web/web/tsconfig.json +30 -0
  523. package/templates/minimal-web/.vscode/settings.json +14 -0
  524. package/templates/minimal-web/README.md +21 -0
  525. package/templates/minimal-web/forge.config.ts +3 -0
  526. package/templates/minimal-web/package.json +32 -0
  527. package/templates/minimal-web/src/actions/logNoteCreated.ts +11 -0
  528. package/templates/minimal-web/src/commands/createNote.ts +26 -0
  529. package/templates/minimal-web/src/forge/schema.ts +12 -0
  530. package/templates/minimal-web/src/policies.ts +6 -0
  531. package/templates/minimal-web/src/queries/listNotes.ts +8 -0
  532. package/templates/minimal-web/src/queries/liveNotes.ts +8 -0
  533. package/templates/minimal-web/tsconfig.json +15 -0
  534. package/templates/minimal-web/web/index.html +12 -0
  535. package/templates/minimal-web/web/package.json +21 -0
  536. package/templates/minimal-web/web/src/App.tsx +89 -0
  537. package/templates/minimal-web/web/src/lib/forge.ts +13 -0
  538. package/templates/minimal-web/web/src/main.tsx +13 -0
  539. package/templates/minimal-web/web/src/styles.css +156 -0
  540. package/templates/minimal-web/web/tsconfig.json +18 -0
@@ -0,0 +1,770 @@
1
+ import { nodeFileSystem } from "../compiler/fs/index.ts";
2
+ import { dirname, join } from "node:path";
3
+ import { spawnSync } from "node:child_process";
4
+ import { createDiagnostic } from "../compiler/diagnostics/create.ts";
5
+ import { GENERATOR_VERSION } from "../compiler/emitter/constants.ts";
6
+ import { hashStable } from "../compiler/primitives/hash.ts";
7
+ import { serializeCanonical } from "../compiler/primitives/serialize.ts";
8
+ import { stripDeterministicHeader } from "../compiler/primitives/header.ts";
9
+ import type { ApiSurface } from "../compiler/api-surface/build.ts";
10
+ import type { AppGraph, SourceFile } from "../compiler/types/app-graph.ts";
11
+ import type { Diagnostic } from "../compiler/types/diagnostic.ts";
12
+ import type {
13
+ UiCommandOptions,
14
+ UiCommandResult,
15
+ UiGeneratedArtifacts,
16
+ UiRoute,
17
+ UiRunReport,
18
+ UiScenario,
19
+ UiScenarioResult,
20
+ UiScenarioStep,
21
+ UiScenariosArtifact,
22
+ UiTestManifest,
23
+ UiRoutesArtifact,
24
+ } from "./types.ts";
25
+
26
+ const UI_RUN_VERSION = "ui-run-0.1.0";
27
+ const UI_RUN_DIR = ".forge/ui-runs";
28
+ const GENERATED = "src/forge/_generated";
29
+
30
+ function normalize(path: string): string {
31
+ return path.replace(/\\/g, "/").replace(/^\.\//, "");
32
+ }
33
+
34
+ function diagnostic(severity: Diagnostic["severity"], code: string, message: string, file?: string): Diagnostic {
35
+ return createDiagnostic({ severity, code, message, ...(file ? { file } : {}) });
36
+ }
37
+
38
+ function readText(workspaceRoot: string, relative: string): string {
39
+ const absolute = join(workspaceRoot, normalize(relative));
40
+ if (!nodeFileSystem.exists(absolute)) return "";
41
+ return (nodeFileSystem.readText(absolute) ?? "");
42
+ }
43
+
44
+ function readJson<T>(workspaceRoot: string, relative: string, fallback: T): T {
45
+ const text = readText(workspaceRoot, relative);
46
+ if (!text) return fallback;
47
+ return JSON.parse(stripDeterministicHeader(text)) as T;
48
+ }
49
+
50
+ function writeText(workspaceRoot: string, relative: string, content: string): void {
51
+ const absolute = join(workspaceRoot, normalize(relative));
52
+ nodeFileSystem.mkdirp(dirname(absolute));
53
+ nodeFileSystem.writeText(absolute, content);
54
+ }
55
+
56
+ function uniqueSorted(values: Array<string | undefined>): string[] {
57
+ return [...new Set(values.filter((value): value is string => Boolean(value)))].sort();
58
+ }
59
+
60
+ function routeName(path: string): string {
61
+ if (path === "/") return "home";
62
+ return path.replace(/^\//, "").replace(/[^a-zA-Z0-9]+/g, "-") || "route";
63
+ }
64
+
65
+ function detectRoutesFromSources(sources: SourceFile[]): UiRoute[] {
66
+ const routes: UiRoute[] = [];
67
+ for (const source of sources) {
68
+ const path = normalize(source.path);
69
+ if (!path.startsWith("web/app/") || !path.endsWith("/page.tsx")) continue;
70
+ const route = path
71
+ .replace(/^web\/app/, "")
72
+ .replace(/\/page\.tsx$/, "") || "/";
73
+ routes.push({
74
+ path: route,
75
+ name: routeName(route),
76
+ uses: {
77
+ commands: [],
78
+ queries: [],
79
+ liveQueries: [],
80
+ components: [],
81
+ },
82
+ });
83
+ }
84
+ return routes.sort((a, b) => a.path.localeCompare(b.path));
85
+ }
86
+
87
+ function defaultRoutes(api: ApiSurface, sources: SourceFile[]): UiRoute[] {
88
+ const routes = detectRoutesFromSources(sources);
89
+ if (!routes.some((route) => route.path === "/")) {
90
+ routes.unshift({
91
+ path: "/",
92
+ name: "home",
93
+ uses: { commands: [], queries: [], liveQueries: [], components: [] },
94
+ });
95
+ }
96
+ const commandNames = Object.keys(api.commands).sort();
97
+ const liveQueryNames = Object.keys(api.liveQueries).sort();
98
+ if (
99
+ (commandNames.some((name) => /ticket/i.test(name)) ||
100
+ liveQueryNames.some((name) => /ticket/i.test(name))) &&
101
+ !routes.some((route) => route.path === "/tickets")
102
+ ) {
103
+ routes.push({
104
+ path: "/tickets",
105
+ name: "tickets",
106
+ uses: {
107
+ commands: commandNames.filter((name) => /ticket|billing/i.test(name)),
108
+ queries: Object.keys(api.queries).filter((name) => /ticket/i.test(name)).sort(),
109
+ liveQueries: liveQueryNames.filter((name) => /ticket/i.test(name)),
110
+ components: ["TicketList", "CreateTicketForm"],
111
+ },
112
+ });
113
+ }
114
+ return routes.sort((a, b) => a.path.localeCompare(b.path));
115
+ }
116
+
117
+ function emptyRequires(): UiScenario["requires"] {
118
+ return {
119
+ commands: [],
120
+ queries: [],
121
+ liveQueries: [],
122
+ policies: [],
123
+ components: [],
124
+ workflows: [],
125
+ };
126
+ }
127
+
128
+ function buildDefaultScenarios(api: ApiSurface, appGraph: AppGraph, routes: UiRoute[]): UiScenario[] {
129
+ const commands = Object.keys(api.commands).sort();
130
+ const liveQueries = Object.keys(api.liveQueries).sort();
131
+ const workflows = Object.keys(api.workflows).sort();
132
+ const policies = appGraph.symbols.filter((symbol) => symbol.kind === "policy").map((symbol) => symbol.name).sort();
133
+ const scenarios: UiScenario[] = [
134
+ {
135
+ name: "home-loads",
136
+ description: "Load the home route and verify the app renders.",
137
+ route: "/",
138
+ cost: "browser",
139
+ steps: [
140
+ { kind: "goto", path: "/" },
141
+ { kind: "expectVisible", selector: "[data-forge-testid='app-root'], body" },
142
+ ],
143
+ requires: emptyRequires(),
144
+ },
145
+ ];
146
+
147
+ if (routes.some((route) => route.path === "/tickets")) {
148
+ scenarios.push({
149
+ name: "tickets-page-loads",
150
+ description: "Load the tickets page and verify the generated form/list selectors.",
151
+ route: "/tickets",
152
+ cost: "browser",
153
+ steps: [
154
+ { kind: "goto", path: "/tickets" },
155
+ { kind: "expectVisible", selector: "[data-forge-testid='ticket-title-input']" },
156
+ { kind: "expectVisible", selector: "[data-forge-testid='ticket-list']" },
157
+ ],
158
+ requires: {
159
+ ...emptyRequires(),
160
+ components: ["CreateTicketForm", "TicketList"],
161
+ },
162
+ });
163
+ scenarios.push({
164
+ name: "tickets-live-update",
165
+ description: "Create a ticket and verify liveQuery updates the ticket list.",
166
+ route: "/tickets",
167
+ cost: "browser",
168
+ steps: [
169
+ { kind: "goto", path: "/tickets" },
170
+ { kind: "fill", selector: "[data-forge-testid='ticket-title-input']", value: "Ticket from UI smoke" },
171
+ { kind: "click", selector: "[data-forge-testid='create-ticket-button']" },
172
+ { kind: "expectText", selector: "[data-forge-testid='ticket-list']", text: "Ticket from UI smoke", timeoutMs: 5000 },
173
+ { kind: "waitForLiveRevision", minRevision: 1, timeoutMs: 5000 },
174
+ ],
175
+ requires: {
176
+ ...emptyRequires(),
177
+ commands: commands.filter((name) => /create.*ticket|ticket.*create/i.test(name)),
178
+ liveQueries: liveQueries.filter((name) => /ticket/i.test(name)),
179
+ policies: policies.filter((name) => /tickets\.(create|read)/i.test(name)),
180
+ components: ["CreateTicketForm", "TicketList"],
181
+ },
182
+ });
183
+ scenarios.push({
184
+ name: "policy-denied-visible",
185
+ description: "Verify policy denied errors surface with a traceId.",
186
+ route: "/tickets",
187
+ cost: "browser",
188
+ steps: [
189
+ { kind: "goto", path: "/tickets" },
190
+ { kind: "click", selector: "[data-forge-testid='billing-manage-demo']" },
191
+ { kind: "expectText", selector: "[data-forge-testid='policy-denied-error']", text: "FORGE_POLICY_DENIED", timeoutMs: 5000 },
192
+ { kind: "expectText", selector: "[data-forge-testid='policy-denied-error']", text: "trace", timeoutMs: 5000 },
193
+ ],
194
+ requires: {
195
+ ...emptyRequires(),
196
+ commands: commands.filter((name) => /billing/i.test(name)),
197
+ policies: policies.filter((name) => /billing\.manage/i.test(name)),
198
+ components: ["TicketList"],
199
+ },
200
+ });
201
+ }
202
+
203
+ if (workflows.some((name) => /triage|ai/i.test(name))) {
204
+ scenarios.push({
205
+ name: "ai-triage-mock-visible",
206
+ description: "Create a ticket and verify AI mock workflow output appears.",
207
+ route: "/tickets",
208
+ cost: "slow",
209
+ steps: [
210
+ { kind: "goto", path: "/tickets" },
211
+ { kind: "fill", selector: "[data-forge-testid='ticket-title-input']", value: "AI triage smoke" },
212
+ { kind: "click", selector: "[data-forge-testid='create-ticket-button']" },
213
+ { kind: "expectVisible", selector: "[data-forge-testid='triage-summary']", timeoutMs: 10000 },
214
+ ],
215
+ requires: {
216
+ ...emptyRequires(),
217
+ commands: commands.filter((name) => /ticket/i.test(name)),
218
+ liveQueries: liveQueries.filter((name) => /ticket/i.test(name)),
219
+ workflows: workflows.filter((name) => /triage|ai/i.test(name)),
220
+ components: ["TicketList"],
221
+ },
222
+ });
223
+ }
224
+
225
+ return scenarios.sort((a, b) => a.name.localeCompare(b.name));
226
+ }
227
+
228
+ export function buildUiGeneratedArtifacts(input: {
229
+ appGraph: AppGraph;
230
+ apiSurface: ApiSurface;
231
+ sources: SourceFile[];
232
+ }): UiGeneratedArtifacts {
233
+ const routes = defaultRoutes(input.apiSurface, input.sources);
234
+ const scenarios = buildDefaultScenarios(input.apiSurface, input.appGraph, routes);
235
+ const manifest: UiTestManifest = {
236
+ schemaVersion: "0.1.0",
237
+ generatorVersion: GENERATOR_VERSION,
238
+ framework: input.sources.some((source) => source.path.startsWith("web/app/")) ? "next" : "unknown",
239
+ webRoot: input.sources.some((source) => source.path.startsWith("web/")) ? "web" : "",
240
+ defaultBaseUrl: "http://127.0.0.1:3000",
241
+ runtimeUrl: "http://127.0.0.1:3765",
242
+ routes,
243
+ scenarios: scenarios.map((scenario) => scenario.name),
244
+ selectors: uniqueSorted(
245
+ scenarios.flatMap((scenario) =>
246
+ scenario.steps.flatMap((step) =>
247
+ "selector" in step ? [step.selector] : [],
248
+ ),
249
+ ),
250
+ ),
251
+ };
252
+ return {
253
+ manifest,
254
+ scenarios: { schemaVersion: "0.1.0", scenarios },
255
+ routes: { schemaVersion: "0.1.0", routes },
256
+ };
257
+ }
258
+
259
+ export function serializeUiTestManifestJson(manifest: UiTestManifest): string {
260
+ return serializeCanonical(manifest);
261
+ }
262
+
263
+ export function serializeUiTestManifestTs(manifest: UiTestManifest): string {
264
+ return `export const uiTestManifest = ${JSON.stringify(JSON.parse(serializeUiTestManifestJson(manifest)), null, 2)} as const;\n`;
265
+ }
266
+
267
+ export function serializeUiScenariosJson(scenarios: UiScenariosArtifact): string {
268
+ return serializeCanonical(scenarios);
269
+ }
270
+
271
+ export function serializeUiScenariosTs(scenarios: UiScenariosArtifact): string {
272
+ return `export const uiScenarios = ${JSON.stringify(JSON.parse(serializeUiScenariosJson(scenarios)), null, 2)} as const;\n`;
273
+ }
274
+
275
+ export function serializeUiRoutesJson(routes: UiRoutesArtifact): string {
276
+ return serializeCanonical(routes);
277
+ }
278
+
279
+ export function serializeUiRoutesTs(routes: UiRoutesArtifact): string {
280
+ return `export const uiRoutes = ${JSON.stringify(JSON.parse(serializeUiRoutesJson(routes)), null, 2)} as const;\n`;
281
+ }
282
+
283
+ export function validateUiScenario(scenario: UiScenario): Diagnostic[] {
284
+ const diagnostics: Diagnostic[] = [];
285
+ if (!scenario.name) {
286
+ diagnostics.push(diagnostic("error", "FORGE_UI_SCENARIO_INVALID", "scenario name is required"));
287
+ }
288
+ if (!scenario.route.startsWith("/")) {
289
+ diagnostics.push(diagnostic("error", "FORGE_UI_SCENARIO_INVALID", `scenario ${scenario.name} route must start with /`));
290
+ }
291
+ if (scenario.steps.length === 0) {
292
+ diagnostics.push(diagnostic("error", "FORGE_UI_SCENARIO_INVALID", `scenario ${scenario.name} has no steps`));
293
+ }
294
+ for (const [index, step] of scenario.steps.entries()) {
295
+ if ("selector" in step && !step.selector) {
296
+ diagnostics.push(diagnostic("error", "FORGE_UI_SELECTOR_NOT_FOUND", `scenario ${scenario.name} step ${index + 1} selector is empty`));
297
+ }
298
+ if (step.kind === "goto" && !step.path.startsWith("/")) {
299
+ diagnostics.push(diagnostic("error", "FORGE_UI_ROUTE_FAILED", `scenario ${scenario.name} goto path must start with /`));
300
+ }
301
+ }
302
+ return diagnostics;
303
+ }
304
+
305
+ function loadUiManifest(workspaceRoot: string): UiTestManifest {
306
+ return readJson<UiTestManifest>(workspaceRoot, `${GENERATED}/uiTestManifest.json`, {
307
+ schemaVersion: "0.1.0",
308
+ generatorVersion: GENERATOR_VERSION,
309
+ framework: "unknown",
310
+ webRoot: "",
311
+ defaultBaseUrl: "http://127.0.0.1:3000",
312
+ runtimeUrl: "http://127.0.0.1:3765",
313
+ routes: [{ path: "/", name: "home", uses: { commands: [], queries: [], liveQueries: [], components: [] } }],
314
+ scenarios: ["home-loads"],
315
+ selectors: ["body"],
316
+ });
317
+ }
318
+
319
+ function loadUiScenarios(workspaceRoot: string): UiScenario[] {
320
+ return readJson<UiScenariosArtifact>(workspaceRoot, `${GENERATED}/uiScenarios.json`, {
321
+ schemaVersion: "0.1.0",
322
+ scenarios: [{
323
+ name: "home-loads",
324
+ description: "Load the home route.",
325
+ route: "/",
326
+ cost: "browser",
327
+ steps: [{ kind: "goto", path: "/" }, { kind: "expectVisible", selector: "body" }],
328
+ requires: { commands: [], queries: [], liveQueries: [], policies: [], components: [], workflows: [] },
329
+ }],
330
+ }).scenarios;
331
+ }
332
+
333
+ function scenarioFailure(name: string, route: string, message: string): UiScenarioResult {
334
+ return {
335
+ name,
336
+ ok: false,
337
+ route,
338
+ durationMs: 0,
339
+ steps: [],
340
+ failure: {
341
+ kind: "playwright-missing",
342
+ message,
343
+ suggestedCommands: [
344
+ "bun add -d @playwright/test",
345
+ "bunx playwright install",
346
+ "forge ui doctor --json",
347
+ ],
348
+ },
349
+ };
350
+ }
351
+
352
+ function suggestedCommands(results: UiScenarioResult[]): string[] {
353
+ const commands = new Set<string>([
354
+ "forge ui doctor --json",
355
+ "forge review --changed",
356
+ ]);
357
+ for (const result of results) {
358
+ if (result.failure?.kind === "live-query-no-update") {
359
+ commands.add("forge live status --json");
360
+ commands.add("forge live invalidations --json");
361
+ commands.add("forge repair diagnose --from-last-ui-run --json");
362
+ }
363
+ if (result.traceId) {
364
+ commands.add(`forge telemetry inspect ${result.traceId} --json`);
365
+ commands.add(`forge repair diagnose --trace ${result.traceId} --json`);
366
+ }
367
+ for (const command of result.failure?.suggestedCommands ?? []) {
368
+ commands.add(command);
369
+ }
370
+ }
371
+ return [...commands].sort();
372
+ }
373
+
374
+ function makeRunId(input: unknown): string {
375
+ return `ui_${hashStable(JSON.stringify(input)).slice(0, 12)}`;
376
+ }
377
+
378
+ function emptyReport(options: UiCommandOptions, scenarios: UiScenario[], diagnostics: Diagnostic[], started: number): UiRunReport {
379
+ const results = scenarios.map((scenario) =>
380
+ scenarioFailure(scenario.name, scenario.route, "Playwright is not installed; run forge ui doctor for setup details."),
381
+ );
382
+ const failed = results.length;
383
+ const report: UiRunReport = {
384
+ schemaVersion: "0.1.0",
385
+ uiRunVersion: UI_RUN_VERSION,
386
+ id: makeRunId({ scenarios: scenarios.map((scenario) => scenario.name), diagnostics: diagnostics.map((item) => item.code) }),
387
+ config: {
388
+ baseUrl: options.baseUrl,
389
+ runtimeUrl: options.runtimeUrl,
390
+ browser: options.browser,
391
+ headed: options.headed,
392
+ trace: options.trace,
393
+ screenshot: options.screenshot,
394
+ video: options.video,
395
+ },
396
+ scenarios: results,
397
+ summary: {
398
+ ok: false,
399
+ passed: 0,
400
+ failed,
401
+ skipped: 0,
402
+ durationMs: Date.now() - started,
403
+ },
404
+ artifacts: {
405
+ screenshots: [],
406
+ traces: [],
407
+ videos: [],
408
+ logs: [],
409
+ console: `${UI_RUN_DIR}/last/console.json`,
410
+ network: `${UI_RUN_DIR}/last/network.json`,
411
+ },
412
+ suggestedCommands: [],
413
+ diagnostics,
414
+ };
415
+ report.suggestedCommands = suggestedCommands(results);
416
+ return report;
417
+ }
418
+
419
+ async function importPlaywright(): Promise<unknown | null> {
420
+ try {
421
+ const dynamicImport = new Function("specifier", "return import(specifier)") as (specifier: string) => Promise<unknown>;
422
+ return await dynamicImport("playwright");
423
+ } catch {
424
+ return null;
425
+ }
426
+ }
427
+
428
+ async function runWithPlaywright(options: UiCommandOptions, scenarios: UiScenario[], started: number): Promise<UiRunReport> {
429
+ const playwright = await importPlaywright() as Record<string, { launch: (options: { headless: boolean }) => Promise<unknown> }> | null;
430
+ if (!playwright || !playwright[options.browser]) {
431
+ const diag = diagnostic("error", "FORGE_UI_PLAYWRIGHT_MISSING", "Playwright is not installed. Add @playwright/test/playwright and install browsers.");
432
+ return emptyReport(options, scenarios, [diag], started);
433
+ }
434
+
435
+ // The real adapter is intentionally small: declarative scenarios are executed
436
+ // through Playwright when the package is available. Unit tests keep this path
437
+ // behind optional dependency detection.
438
+ const browser = await playwright[options.browser].launch({ headless: !options.headed }) as {
439
+ newPage: () => Promise<{
440
+ goto: (url: string, options?: { timeout?: number }) => Promise<unknown>;
441
+ click: (selector: string, options?: { timeout?: number }) => Promise<unknown>;
442
+ fill: (selector: string, value: string, options?: { timeout?: number }) => Promise<unknown>;
443
+ waitForSelector: (selector: string, options?: { timeout?: number; state?: string }) => Promise<unknown>;
444
+ textContent: (selector: string, options?: { timeout?: number }) => Promise<string | null>;
445
+ screenshot: (options: { path: string; fullPage?: boolean }) => Promise<unknown>;
446
+ close: () => Promise<void>;
447
+ }>;
448
+ close: () => Promise<void>;
449
+ };
450
+ const results: UiScenarioResult[] = [];
451
+ const screenshots: string[] = [];
452
+ for (const scenario of scenarios) {
453
+ const scenarioStarted = Date.now();
454
+ const steps: UiScenarioResult["steps"] = [];
455
+ const page = await browser.newPage();
456
+ let failed: UiScenarioResult["failure"];
457
+ try {
458
+ for (const step of scenario.steps) {
459
+ const stepStarted = Date.now();
460
+ await executeStep(page, options, step);
461
+ steps.push({ kind: step.kind, ok: true, durationMs: Date.now() - stepStarted });
462
+ }
463
+ } catch (error) {
464
+ const screenshot = `${UI_RUN_DIR}/${makeRunId(scenario.name)}/screenshots/failure-${scenario.name}.png`;
465
+ const absolute = join(options.workspaceRoot, screenshot);
466
+ nodeFileSystem.mkdirp(dirname(absolute));
467
+ try {
468
+ await page.screenshot({ path: absolute, fullPage: true });
469
+ screenshots.push(screenshot);
470
+ } catch {
471
+ // Screenshot failures are surfaced by the main failure message.
472
+ }
473
+ failed = {
474
+ kind: "expected-text-missing",
475
+ message: error instanceof Error ? error.message : "UI scenario failed",
476
+ screenshot,
477
+ suggestedCommands: ["forge ui report last", "forge repair diagnose --from-last-ui-run --json"],
478
+ };
479
+ } finally {
480
+ await page.close();
481
+ }
482
+ results.push({
483
+ name: scenario.name,
484
+ ok: !failed,
485
+ route: scenario.route,
486
+ durationMs: Date.now() - scenarioStarted,
487
+ steps,
488
+ failure: failed,
489
+ });
490
+ }
491
+ await browser.close();
492
+ return buildReportFromResults(options, results, [], screenshots, started);
493
+ }
494
+
495
+ async function executeStep(page: {
496
+ goto: (url: string, options?: { timeout?: number }) => Promise<unknown>;
497
+ click: (selector: string, options?: { timeout?: number }) => Promise<unknown>;
498
+ fill: (selector: string, value: string, options?: { timeout?: number }) => Promise<unknown>;
499
+ waitForSelector: (selector: string, options?: { timeout?: number; state?: string }) => Promise<unknown>;
500
+ textContent: (selector: string, options?: { timeout?: number }) => Promise<string | null>;
501
+ screenshot: (options: { path: string; fullPage?: boolean }) => Promise<unknown>;
502
+ }, options: UiCommandOptions, step: UiScenarioStep): Promise<void> {
503
+ if (step.kind === "goto") {
504
+ await page.goto(new URL(step.path, options.baseUrl).toString(), { timeout: options.timeoutMs });
505
+ return;
506
+ }
507
+ if (step.kind === "click") {
508
+ await page.click(step.selector, { timeout: options.timeoutMs });
509
+ return;
510
+ }
511
+ if (step.kind === "fill") {
512
+ await page.fill(step.selector, step.value, { timeout: options.timeoutMs });
513
+ return;
514
+ }
515
+ if (step.kind === "expectVisible") {
516
+ await page.waitForSelector(step.selector, { timeout: step.timeoutMs ?? options.timeoutMs, state: "visible" });
517
+ return;
518
+ }
519
+ if (step.kind === "expectNotVisible") {
520
+ await page.waitForSelector(step.selector, { timeout: step.timeoutMs ?? options.timeoutMs, state: "hidden" });
521
+ return;
522
+ }
523
+ if (step.kind === "expectText") {
524
+ const text = await page.textContent(step.selector, { timeout: step.timeoutMs ?? options.timeoutMs });
525
+ if (!text?.includes(step.text)) {
526
+ throw new Error(`Expected ${step.selector} to contain '${step.text}'`);
527
+ }
528
+ return;
529
+ }
530
+ if (step.kind === "captureScreenshot") {
531
+ const path = `${UI_RUN_DIR}/snapshots/${step.name}.png`;
532
+ const absolute = join(options.workspaceRoot, path);
533
+ nodeFileSystem.mkdirp(dirname(absolute));
534
+ await page.screenshot({ path: absolute, fullPage: true });
535
+ }
536
+ }
537
+
538
+ function buildReportFromResults(
539
+ options: UiCommandOptions,
540
+ results: UiScenarioResult[],
541
+ diagnostics: Diagnostic[],
542
+ screenshots: string[],
543
+ started: number,
544
+ ): UiRunReport {
545
+ const failed = results.filter((result) => !result.ok).length;
546
+ const passed = results.filter((result) => result.ok).length;
547
+ const report: UiRunReport = {
548
+ schemaVersion: "0.1.0",
549
+ uiRunVersion: UI_RUN_VERSION,
550
+ id: makeRunId({ scenarios: results.map((result) => result.name), failed, passed }),
551
+ config: {
552
+ baseUrl: options.baseUrl,
553
+ runtimeUrl: options.runtimeUrl,
554
+ browser: options.browser,
555
+ headed: options.headed,
556
+ trace: options.trace,
557
+ screenshot: options.screenshot,
558
+ video: options.video,
559
+ },
560
+ scenarios: results,
561
+ summary: {
562
+ ok: failed === 0 && diagnostics.every((item) => item.severity !== "error"),
563
+ passed,
564
+ failed,
565
+ skipped: 0,
566
+ durationMs: Date.now() - started,
567
+ },
568
+ artifacts: {
569
+ screenshots,
570
+ traces: [],
571
+ videos: [],
572
+ logs: [],
573
+ console: `${UI_RUN_DIR}/last/console.json`,
574
+ network: `${UI_RUN_DIR}/last/network.json`,
575
+ },
576
+ suggestedCommands: suggestedCommands(results),
577
+ diagnostics,
578
+ };
579
+ return report;
580
+ }
581
+
582
+ function renderReportMarkdown(report: UiRunReport): string {
583
+ return `# Forge UI Run
584
+
585
+ Run: ${report.id}
586
+ OK: ${report.summary.ok ? "yes" : "no"}
587
+ Passed: ${report.summary.passed}
588
+ Failed: ${report.summary.failed}
589
+
590
+ ## Scenarios
591
+
592
+ ${report.scenarios.map((scenario) => `- ${scenario.ok ? "OK" : "FAIL"} ${scenario.name}${scenario.failure ? `: ${scenario.failure.message}` : ""}`).join("\n") || "- none"}
593
+
594
+ ## Suggested Commands
595
+
596
+ \`\`\`bash
597
+ ${report.suggestedCommands.join("\n")}
598
+ \`\`\`
599
+ `;
600
+ }
601
+
602
+ export function writeUiReport(workspaceRoot: string, report: UiRunReport): void {
603
+ const dir = `${UI_RUN_DIR}/${report.id}`;
604
+ writeText(workspaceRoot, `${dir}/report.json`, serializeCanonical(report));
605
+ writeText(workspaceRoot, `${dir}/report.md`, renderReportMarkdown(report));
606
+ writeText(workspaceRoot, `${dir}/console.json`, "[]\n");
607
+ writeText(workspaceRoot, `${dir}/network.json`, "[]\n");
608
+ writeText(workspaceRoot, `${UI_RUN_DIR}/last.json`, serializeCanonical(report));
609
+ }
610
+
611
+ function selectScenarios(options: UiCommandOptions, scenarios: UiScenario[]): UiScenario[] {
612
+ if (options.subcommand === "route") {
613
+ const path = options.routePath ?? "/";
614
+ return [{
615
+ name: `route-${routeName(path)}`,
616
+ description: `Load route ${path}`,
617
+ route: path,
618
+ cost: "browser",
619
+ steps: [{ kind: "goto", path }, { kind: "expectVisible", selector: "body" }],
620
+ requires: { commands: [], queries: [], liveQueries: [], policies: [], components: [], workflows: [] },
621
+ }];
622
+ }
623
+ if (options.subcommand === "snapshot") {
624
+ const path = options.routePath ?? "/";
625
+ return [{
626
+ name: options.snapshotName ?? `snapshot-${routeName(path)}`,
627
+ description: `Capture snapshot for ${path}`,
628
+ route: path,
629
+ cost: "browser",
630
+ steps: [{ kind: "goto", path }, { kind: "captureScreenshot", name: options.snapshotName ?? routeName(path) }],
631
+ requires: { commands: [], queries: [], liveQueries: [], policies: [], components: [], workflows: [] },
632
+ }];
633
+ }
634
+ if (options.scenarioName) {
635
+ return scenarios.filter((scenario) => scenario.name === options.scenarioName);
636
+ }
637
+ if (options.subcommand === "smoke") {
638
+ return scenarios.filter((scenario) => scenario.cost === "browser").slice(0, 4);
639
+ }
640
+ return options.all ? scenarios : scenarios.slice(0, 1);
641
+ }
642
+
643
+ export async function runUiCommand(options: UiCommandOptions): Promise<UiCommandResult> {
644
+ if (options.subcommand === "doctor") {
645
+ return runUiDoctor(options);
646
+ }
647
+ if (options.subcommand === "list") {
648
+ const scenarios = loadUiScenarios(options.workspaceRoot);
649
+ return { ok: true, manifest: loadUiManifest(options.workspaceRoot), scenarios, diagnostics: [], exitCode: 0 };
650
+ }
651
+ if (options.subcommand === "report") {
652
+ return readUiReport(options.workspaceRoot, options.reportId ?? "last");
653
+ }
654
+
655
+ const started = Date.now();
656
+ const allScenarios = loadUiScenarios(options.workspaceRoot);
657
+ const selected = selectScenarios(options, allScenarios);
658
+ const validation = selected.flatMap(validateUiScenario);
659
+ if (validation.some((item) => item.severity === "error")) {
660
+ const report = emptyReport(options, selected, validation, started);
661
+ writeUiReport(options.workspaceRoot, report);
662
+ return { ok: false, report, diagnostics: validation, exitCode: 1 };
663
+ }
664
+ const report = await runWithPlaywright(options, selected, started);
665
+ writeUiReport(options.workspaceRoot, report);
666
+ return {
667
+ ok: report.summary.ok,
668
+ report,
669
+ diagnostics: report.diagnostics,
670
+ exitCode: report.summary.ok ? 0 : 1,
671
+ };
672
+ }
673
+
674
+ function runUiDoctor(options: UiCommandOptions): UiCommandResult {
675
+ const manifest = loadUiManifest(options.workspaceRoot);
676
+ const diagnostics: Diagnostic[] = [];
677
+ const checks = [
678
+ nodeFileSystem.exists(join(options.workspaceRoot, "node_modules/playwright")) ||
679
+ nodeFileSystem.exists(join(options.workspaceRoot, "node_modules/@playwright/test")),
680
+ manifest.routes.length > 0,
681
+ manifest.scenarios.length > 0,
682
+ ];
683
+ if (!checks[0]) {
684
+ diagnostics.push(diagnostic("error", "FORGE_UI_PLAYWRIGHT_MISSING", "Playwright is not installed; run bun add -d @playwright/test && bunx playwright install."));
685
+ }
686
+ if (!checks[1]) {
687
+ diagnostics.push(diagnostic("warning", "FORGE_UI_ROUTE_FAILED", "No UI routes are present in uiTestManifest."));
688
+ }
689
+ if (!checks[2]) {
690
+ diagnostics.push(diagnostic("warning", "FORGE_UI_TESTID_MISSING", "No UI scenarios are present in uiScenarios."));
691
+ }
692
+ return {
693
+ ok: diagnostics.every((item) => item.severity !== "error"),
694
+ manifest,
695
+ diagnostics,
696
+ exitCode: diagnostics.some((item) => item.severity === "error") ? 1 : 0,
697
+ };
698
+ }
699
+
700
+ function readUiReport(workspaceRoot: string, id: string): UiCommandResult {
701
+ const path =
702
+ id === "last"
703
+ ? `${UI_RUN_DIR}/last.json`
704
+ : `${UI_RUN_DIR}/${id}/report.json`;
705
+ const absolute = join(workspaceRoot, path);
706
+ if (!nodeFileSystem.exists(absolute)) {
707
+ const diag = diagnostic("error", "FORGE_UI_REPORT_NOT_FOUND", `UI report not found: ${id}`, path);
708
+ return { ok: false, diagnostics: [diag], exitCode: 1 };
709
+ }
710
+ const report = JSON.parse((nodeFileSystem.readText(absolute) ?? "")) as UiRunReport;
711
+ return { ok: report.summary.ok, report, diagnostics: report.diagnostics, exitCode: report.summary.ok ? 0 : 1 };
712
+ }
713
+
714
+ export function listUiRuns(workspaceRoot: string): Array<{ id: string; path: string }> {
715
+ const dir = join(workspaceRoot, UI_RUN_DIR);
716
+ if (!nodeFileSystem.exists(dir)) return [];
717
+ return nodeFileSystem
718
+ .readDir(dir)
719
+ .filter((entry) => entry.isDirectory)
720
+ .map((entry) => ({ id: entry.name, path: `${UI_RUN_DIR}/${entry.name}/report.json` }))
721
+ .sort((a, b) => a.id.localeCompare(b.id));
722
+ }
723
+
724
+ export function formatUiJson(result: UiCommandResult): string {
725
+ if (result.report) {
726
+ return `${JSON.stringify(result.report, null, 2)}\n`;
727
+ }
728
+ return `${JSON.stringify(result, null, 2)}\n`;
729
+ }
730
+
731
+ export function formatUiHuman(result: UiCommandResult): string {
732
+ if (result.report) {
733
+ return renderReportMarkdown(result.report);
734
+ }
735
+ if (result.scenarios) {
736
+ return `Forge UI Scenarios
737
+
738
+ ${result.scenarios.map((scenario) => `- ${scenario.name}: ${scenario.route}`).join("\n")}
739
+ `;
740
+ }
741
+ if (result.manifest) {
742
+ return `Forge UI Doctor
743
+
744
+ ${result.diagnostics.map((diag) => `${diag.severity} ${diag.code}: ${diag.message}`).join("\n") || "OK"}
745
+ `;
746
+ }
747
+ return `${result.diagnostics.map((diag) => `${diag.severity} ${diag.code}: ${diag.message}`).join("\n")}\n`;
748
+ }
749
+
750
+ export function runUiListCommand(workspaceRoot: string): UiCommandResult {
751
+ return {
752
+ ok: true,
753
+ reports: listUiRuns(workspaceRoot),
754
+ diagnostics: [],
755
+ exitCode: 0,
756
+ };
757
+ }
758
+
759
+ export function runForgeCommandForUi(workspaceRoot: string, command: string): { ok: boolean; output: string } {
760
+ const parts = command.split(/\s+/).filter(Boolean);
761
+ const result = spawnSync(parts[0], parts.slice(1), {
762
+ cwd: workspaceRoot,
763
+ encoding: "utf8",
764
+ windowsHide: true,
765
+ });
766
+ return {
767
+ ok: result.status === 0,
768
+ output: `${result.stdout ?? ""}${result.stderr ?? ""}`,
769
+ };
770
+ }