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,73 @@
1
+ import type { EmitFile } from "../types/emit.ts";
2
+ import {
3
+ hashStable,
4
+ prependDeterministicHeader,
5
+ normalizeNewlines,
6
+ serializeCanonical,
7
+ } from "../primitives/index.ts";
8
+ import { detectArtifactKind } from "./artifact-kind.ts";
9
+
10
+ export interface RenderContext {
11
+ generatorVersion: string;
12
+ inputHash: string;
13
+ }
14
+
15
+ function renderJsonBody(content: string): string {
16
+ try {
17
+ const parsed: unknown = JSON.parse(content);
18
+ return serializeCanonical(parsed);
19
+ } catch {
20
+ return normalizeNewlines(content);
21
+ }
22
+ }
23
+
24
+ function renderTypeScriptBody(content: string): string {
25
+ return normalizeNewlines(content);
26
+ }
27
+
28
+ function renderMarkdownBody(content: string): string {
29
+ return normalizeNewlines(content);
30
+ }
31
+
32
+ function renderTextBody(content: string): string {
33
+ return normalizeNewlines(content);
34
+ }
35
+
36
+ /**
37
+ * Normalize file body bytes by artifact kind (no deterministic header).
38
+ */
39
+ export function renderBody(file: EmitFile): string {
40
+ const kind = detectArtifactKind(file.path);
41
+
42
+ switch (kind) {
43
+ case "json":
44
+ if (file.canonical) {
45
+ return normalizeNewlines(file.content);
46
+ }
47
+ return renderJsonBody(file.content);
48
+ case "typescript":
49
+ return renderTypeScriptBody(file.content);
50
+ case "markdown":
51
+ return renderMarkdownBody(file.content);
52
+ default:
53
+ return renderTextBody(file.content);
54
+ }
55
+ }
56
+
57
+ /**
58
+ * Pure render: same EmitFile + context → same bytes (header included).
59
+ */
60
+ export function render(file: EmitFile, context: RenderContext): string {
61
+ const body = renderBody(file);
62
+
63
+ if (file.contentHash !== hashStable(body)) {
64
+ throw new Error(
65
+ `EmitFile contentHash mismatch for ${file.path}: expected ${hashStable(body)}, got ${file.contentHash}`,
66
+ );
67
+ }
68
+
69
+ return prependDeterministicHeader(body, {
70
+ generatorVersion: context.generatorVersion,
71
+ inputHash: context.inputHash,
72
+ });
73
+ }
@@ -0,0 +1,35 @@
1
+ import { nodeFileSystem } from "../fs/index.ts";
2
+ import type { FileSystem } from "../fs/index.ts";
3
+
4
+ /**
5
+ * Thin async I/O helpers for the emitter, delegating to an injectable
6
+ * {@link FileSystem}. The async signatures are preserved so callers in
7
+ * `emit.ts` are unchanged; tests can pass an `InMemoryFileSystem` to run the
8
+ * emitter without touching disk. Atomic write semantics live in
9
+ * {@link NodeFileSystem.writeText}.
10
+ */
11
+ export async function readTextFileIfExists(
12
+ absolutePath: string,
13
+ fs: FileSystem = nodeFileSystem,
14
+ ): Promise<string | null> {
15
+ return fs.readText(absolutePath);
16
+ }
17
+
18
+ export async function writeFileAtomic(
19
+ absolutePath: string,
20
+ content: string,
21
+ fs: FileSystem = nodeFileSystem,
22
+ ): Promise<void> {
23
+ fs.writeText(absolutePath, content);
24
+ }
25
+
26
+ export async function removeFileIfExists(
27
+ absolutePath: string,
28
+ fs: FileSystem = nodeFileSystem,
29
+ ): Promise<boolean> {
30
+ const existed = fs.exists(absolutePath);
31
+ if (existed) {
32
+ fs.remove(absolutePath);
33
+ }
34
+ return existed;
35
+ }
@@ -0,0 +1,495 @@
1
+ import { basename, dirname, join, relative } from "node:path";
2
+ import { nodeFileSystem } from "../fs/index.ts";
3
+ import type { ClientManifest } from "../client-sdk/build-manifest.ts";
4
+ import type {
5
+ FrontendClientBindingInfo,
6
+ FrontendComponentInfo,
7
+ FrontendGraph,
8
+ FrontendProviderInfo,
9
+ FrontendRouteInfo,
10
+ } from "../types/frontend-graph.ts";
11
+ import { serializeCanonical } from "../primitives/serialize.ts";
12
+ import { createDiagnostic } from "../diagnostics/create.ts";
13
+
14
+ const WEB_ROOT = "web";
15
+ const API_URL_ENV = "NEXT_PUBLIC_FORGE_URL";
16
+ const VITE_API_URL_ENV = "VITE_FORGE_URL";
17
+
18
+ function toPosix(path: string): string {
19
+ return path.replace(/\\/g, "/");
20
+ }
21
+
22
+ function uniqueSorted(items: string[]): string[] {
23
+ return [...new Set(items.filter(Boolean))].sort();
24
+ }
25
+
26
+ function readJson<T>(path: string): T | null {
27
+ if (!nodeFileSystem.exists(path)) {
28
+ return null;
29
+ }
30
+ try {
31
+ return JSON.parse(nodeFileSystem.readText(path) ?? "") as T;
32
+ } catch {
33
+ return null;
34
+ }
35
+ }
36
+
37
+ function walkFiles(root: string): string[] {
38
+ if (!nodeFileSystem.exists(root)) {
39
+ return [];
40
+ }
41
+ const files: string[] = [];
42
+ function walk(dir: string): void {
43
+ for (const entry of nodeFileSystem.readDir(dir)) {
44
+ const absolute = join(dir, entry.name);
45
+ if (entry.isDirectory) {
46
+ if (entry.name === "node_modules" || entry.name === ".next" || entry.name === "dist") {
47
+ continue;
48
+ }
49
+ walk(absolute);
50
+ } else if (entry.isFile) {
51
+ files.push(absolute);
52
+ }
53
+ }
54
+ }
55
+ walk(root);
56
+ return files.sort((a, b) => toPosix(a).localeCompare(toPosix(b)));
57
+ }
58
+
59
+ type WebPackageJson = {
60
+ packageManager?: string;
61
+ scripts?: Record<string, string>;
62
+ dependencies?: Record<string, string>;
63
+ devDependencies?: Record<string, string>;
64
+ };
65
+
66
+ function detectFramework(webRoot: string): FrontendGraph["framework"] {
67
+ const pkg = readJson<WebPackageJson>(
68
+ join(webRoot, "package.json"),
69
+ );
70
+ const deps = { ...(pkg?.dependencies ?? {}), ...(pkg?.devDependencies ?? {}) };
71
+ if (deps.next || nodeFileSystem.exists(join(webRoot, "next.config.ts"))) {
72
+ return "next";
73
+ }
74
+ if (deps.vite || nodeFileSystem.exists(join(webRoot, "vite.config.ts"))) {
75
+ return "vite";
76
+ }
77
+ if (nodeFileSystem.exists(join(webRoot, "index.html"))) {
78
+ return "static";
79
+ }
80
+ return "unknown";
81
+ }
82
+
83
+ function detectPackageManager(pkg: WebPackageJson | null): FrontendGraph["webManifest"]["packageManager"] {
84
+ const raw = pkg?.packageManager;
85
+ if (raw?.startsWith("bun@")) return "bun";
86
+ if (raw?.startsWith("pnpm@")) return "pnpm";
87
+ if (raw?.startsWith("yarn@")) return "yarn";
88
+ if (raw?.startsWith("npm@")) return "npm";
89
+ return raw ? "unknown" : undefined;
90
+ }
91
+
92
+ function routePathForFile(webRoot: string, file: string): string | null {
93
+ const rel = toPosix(relative(webRoot, file));
94
+ if (rel === "index.html") {
95
+ return "/";
96
+ }
97
+ if (
98
+ rel === "src/App.tsx" ||
99
+ rel === "src/App.ts" ||
100
+ rel === "src/App.jsx" ||
101
+ rel === "src/App.js"
102
+ ) {
103
+ return "/";
104
+ }
105
+ if (rel === "app/page.tsx" || rel === "app/page.ts" || rel === "app/page.jsx" || rel === "app/page.js") {
106
+ return "/";
107
+ }
108
+ const match = rel.match(/^app\/(.+)\/page\.(tsx|ts|jsx|js)$/);
109
+ if (match) {
110
+ const route = match[1]
111
+ .split("/")
112
+ .filter((segment) => !segment.startsWith("(") && !segment.endsWith(")"))
113
+ .join("/");
114
+ return `/${route}`;
115
+ }
116
+ return null;
117
+ }
118
+
119
+ function componentNameForFile(file: string): string {
120
+ return basename(file).replace(/\.(tsx|ts|jsx|js)$/, "");
121
+ }
122
+
123
+ function componentNameForText(file: string, text: string): string {
124
+ const exportMatch = text.match(/export\s+function\s+([A-Z][A-Za-z0-9_]*)\s*\(/);
125
+ if (exportMatch?.[1]) {
126
+ return exportMatch[1];
127
+ }
128
+ const fnMatch = text.match(/function\s+([A-Z][A-Za-z0-9_]*)\s*\(/);
129
+ if (fnMatch?.[1]) {
130
+ return fnMatch[1];
131
+ }
132
+ return componentNameForFile(file);
133
+ }
134
+
135
+ function isComponentFile(webRoot: string, file: string, text: string): boolean {
136
+ const rel = toPosix(relative(webRoot, file));
137
+ if (!/\.(tsx|jsx)$/.test(file)) {
138
+ return false;
139
+ }
140
+ if (rel === "src/main.tsx" || rel === "src/main.jsx") {
141
+ return false;
142
+ }
143
+ if (rel.startsWith("components/") || rel.startsWith("src/components/")) {
144
+ return true;
145
+ }
146
+ return /export\s+(default\s+)?function\s+[A-Z]/.test(text) || /<[A-Z][A-Za-z0-9_.]*[\s/>]/.test(text);
147
+ }
148
+
149
+ function detectUses(text: string, clientManifest: ClientManifest) {
150
+ const rawForgeFetches = [
151
+ ...[...text.matchAll(/fetch\((["'`])([^"'`]*(?:\/commands\/|\/queries\/|\/live\/)[^"'`]*)\1/g)]
152
+ .map((match) => match[2] ?? ""),
153
+ ...[...text.matchAll(/(["'`])(\/(?:commands|queries|live)\/[^"'`]+)\1/g)]
154
+ .map((match) => match[2] ?? ""),
155
+ ];
156
+ return {
157
+ usesCommands: uniqueSorted(clientManifest.commands.filter((name) => text.includes(name))),
158
+ usesQueries: uniqueSorted(clientManifest.queries.filter((name) => text.includes(name))),
159
+ usesLiveQueries: uniqueSorted(clientManifest.liveQueries.filter((name) => text.includes(name))),
160
+ rawForgeFetches: uniqueSorted(rawForgeFetches),
161
+ };
162
+ }
163
+
164
+ function mergeUses(
165
+ left: ReturnType<typeof detectUses>,
166
+ right: ReturnType<typeof detectUses>,
167
+ ): ReturnType<typeof detectUses> {
168
+ return {
169
+ usesCommands: uniqueSorted([...left.usesCommands, ...right.usesCommands]),
170
+ usesQueries: uniqueSorted([...left.usesQueries, ...right.usesQueries]),
171
+ usesLiveQueries: uniqueSorted([...left.usesLiveQueries, ...right.usesLiveQueries]),
172
+ rawForgeFetches: uniqueSorted([...left.rawForgeFetches, ...right.rawForgeFetches]),
173
+ };
174
+ }
175
+
176
+ function detectRouteUses(file: string, text: string, clientManifest: ClientManifest) {
177
+ let uses = detectUses(text, clientManifest);
178
+ if (!file.endsWith(".html")) {
179
+ return uses;
180
+ }
181
+ for (const match of text.matchAll(/<script[^>]+src=["']([^"']+)["']/g)) {
182
+ const scriptPath = match[1];
183
+ if (!scriptPath || /^https?:\/\//.test(scriptPath)) {
184
+ continue;
185
+ }
186
+ const absolute = join(dirname(file), scriptPath.replace(/^\//, ""));
187
+ if (nodeFileSystem.exists(absolute)) {
188
+ uses = mergeUses(uses, detectUses(nodeFileSystem.readText(absolute) ?? "", clientManifest));
189
+ }
190
+ }
191
+ return uses;
192
+ }
193
+
194
+ function devCommandFor(webRoot: string, framework: FrontendGraph["framework"]): string {
195
+ const pkg = readJson<{ scripts?: Record<string, string> }>(join(webRoot, "package.json"));
196
+ if (pkg?.scripts?.dev) {
197
+ return "cd web && bun run dev";
198
+ }
199
+ if (nodeFileSystem.exists(join(webRoot, "server.ts"))) {
200
+ return "bun web/server.ts";
201
+ }
202
+ if (framework === "static") {
203
+ return "forge dev --web";
204
+ }
205
+ return "none";
206
+ }
207
+
208
+ function apiEnvFor(framework: FrontendGraph["framework"]): string {
209
+ return framework === "vite" || framework === "static" ? VITE_API_URL_ENV : API_URL_ENV;
210
+ }
211
+
212
+ function stringProp(text: string, prop: "userId" | "tenantId" | "role"): string | undefined {
213
+ const literal = text.match(new RegExp(`${prop}\\s*:\\s*["'\`]([^"'\`]+)["'\`]`));
214
+ if (literal?.[1]) {
215
+ return literal[1];
216
+ }
217
+ const identifier = text.match(new RegExp(`${prop}\\s*:\\s*([A-Za-z_$][A-Za-z0-9_$]*)`));
218
+ const name = identifier?.[1];
219
+ if (!name) {
220
+ return undefined;
221
+ }
222
+ const constant = text.match(new RegExp(`(?:const|let|var)\\s+${name}\\s*=\\s*["'\`]([^"'\`]+)["'\`]`));
223
+ return constant?.[1];
224
+ }
225
+
226
+ function defaultWebPort(framework: FrontendGraph["framework"]): number {
227
+ return framework === "next" ? 3000 : 5173;
228
+ }
229
+
230
+ function componentNamesInText(text: string, components: FrontendComponentInfo[]): string[] {
231
+ return uniqueSorted(
232
+ components
233
+ .filter((component) => new RegExp(`<${component.name}(\\s|>|/)`).test(text))
234
+ .map((component) => component.name),
235
+ );
236
+ }
237
+
238
+ function bindingEntries(input: {
239
+ file: string;
240
+ route?: string;
241
+ component?: string;
242
+ usesCommands: string[];
243
+ usesQueries: string[];
244
+ usesLiveQueries: string[];
245
+ rawForgeFetches: string[];
246
+ }): FrontendClientBindingInfo[] {
247
+ return [
248
+ ...input.usesCommands.map((name) => ({ kind: "command" as const, name })),
249
+ ...input.usesQueries.map((name) => ({ kind: "query" as const, name })),
250
+ ...input.usesLiveQueries.map((name) => ({ kind: "liveQuery" as const, name })),
251
+ ...input.rawForgeFetches.map((name) => ({ kind: "rawFetch" as const, name })),
252
+ ].map((binding) => ({
253
+ ...binding,
254
+ file: input.file,
255
+ ...(input.route ? { route: input.route } : {}),
256
+ ...(input.component ? { component: input.component } : {}),
257
+ }));
258
+ }
259
+
260
+ export function buildFrontendGraph(input: {
261
+ workspaceRoot: string;
262
+ clientManifest: ClientManifest;
263
+ apiPort?: number;
264
+ webPort?: number;
265
+ }): FrontendGraph {
266
+ const webRoot = join(input.workspaceRoot, WEB_ROOT);
267
+ if (!nodeFileSystem.exists(webRoot)) {
268
+ return {
269
+ schemaVersion: "0.1.0",
270
+ present: false,
271
+ framework: "none",
272
+ routes: [],
273
+ components: [],
274
+ providers: [],
275
+ bridgeFiles: [],
276
+ webManifest: {
277
+ present: false,
278
+ framework: "none",
279
+ scripts: {},
280
+ urls: {
281
+ api: `http://127.0.0.1:${input.apiPort ?? 3765}`,
282
+ },
283
+ env: {
284
+ apiUrl: API_URL_ENV,
285
+ },
286
+ bridge: {
287
+ files: [],
288
+ valid: false,
289
+ },
290
+ },
291
+ clientBindings: [],
292
+ diagnostics: [],
293
+ };
294
+ }
295
+
296
+ const pkg = readJson<WebPackageJson>(join(webRoot, "package.json"));
297
+ const framework = detectFramework(webRoot);
298
+ const apiUrlEnv = apiEnvFor(framework);
299
+ const defaultApiUrl = `http://127.0.0.1:${input.apiPort ?? 3765}`;
300
+ const devUrl = `http://127.0.0.1:${input.webPort ?? defaultWebPort(framework)}`;
301
+ const files = walkFiles(webRoot);
302
+ const sourceFiles = files.filter((file) => /\.(tsx|ts|jsx|js|html)$/.test(file));
303
+ const routes: FrontendRouteInfo[] = [];
304
+ const components: FrontendComponentInfo[] = [];
305
+ const providers: FrontendProviderInfo[] = [];
306
+ const bridgeFiles: string[] = [];
307
+ const diagnostics: FrontendGraph["diagnostics"] = [];
308
+ const textByRel = new Map<string, string>();
309
+
310
+ for (const file of sourceFiles) {
311
+ const rel = toPosix(relative(input.workspaceRoot, file));
312
+ const text = nodeFileSystem.readText(file) ?? "";
313
+ const isBridgeFile =
314
+ rel === "web/lib/forge.ts" ||
315
+ rel === "web/lib/forge.tsx" ||
316
+ rel === "web/src/lib/forge.ts" ||
317
+ rel === "web/src/lib/forge.tsx";
318
+ textByRel.set(rel, text);
319
+ const uses = detectUses(text, input.clientManifest);
320
+ if (isComponentFile(webRoot, file, text)) {
321
+ components.push({ name: componentNameForText(file, text), file: rel, ...uses });
322
+ }
323
+ if (!isBridgeFile && text.includes("ForgeProvider")) {
324
+ const devAuth =
325
+ text.includes("devAuth") ||
326
+ (text.includes("userId") && text.includes("tenantId") && text.includes("role"));
327
+ providers.push({
328
+ name: "ForgeProvider",
329
+ file: rel,
330
+ ...(text.includes(API_URL_ENV) ? { apiUrlEnv: API_URL_ENV } : {}),
331
+ ...(text.includes(VITE_API_URL_ENV) ? { apiUrlEnv: VITE_API_URL_ENV } : {}),
332
+ devAuth,
333
+ ...(devAuth && stringProp(text, "userId") ? { devAuthUserId: stringProp(text, "userId") } : {}),
334
+ ...(devAuth && stringProp(text, "tenantId") ? { devAuthTenantId: stringProp(text, "tenantId") } : {}),
335
+ ...(devAuth && stringProp(text, "role") ? { devAuthRole: stringProp(text, "role") } : {}),
336
+ });
337
+ }
338
+ if (
339
+ rel === "web/lib/forge.ts" ||
340
+ rel === "web/lib/forge.tsx" ||
341
+ rel === "web/src/lib/forge.ts" ||
342
+ rel === "web/src/lib/forge.tsx"
343
+ ) {
344
+ bridgeFiles.push(rel);
345
+ }
346
+ if (!isBridgeFile && (/from\s+["']\.\.\/\.\.\/src\/forge\/_generated/.test(text) || /from\s+["'][^"']*\/src\/forge\/_generated/.test(text))) {
347
+ diagnostics.push(createDiagnostic({
348
+ severity: "warning",
349
+ code: "FORGE_FRONTEND_SERVER_IMPORT",
350
+ message: "frontend imports generated files directly; prefer the local web/lib/forge bridge",
351
+ file: rel,
352
+ fixHint: "Import generated client APIs through web/lib/forge.ts so agents and humans have one stable frontend bridge.",
353
+ suggestedCommands: ["forge inspect frontend --json", "forge make ui --framework vite --dry-run --json"],
354
+ docs: ["src/forge/_generated/frontendGraph.json", "src/forge/_generated/appMap.md"],
355
+ }));
356
+ }
357
+ if (uses.rawForgeFetches.length > 0) {
358
+ diagnostics.push(createDiagnostic({
359
+ severity: "warning",
360
+ code: "FORGE_FRONTEND_DIRECT_RUNTIME_FETCH",
361
+ message: "frontend calls Forge runtime endpoints directly; prefer generated hooks from the web bridge",
362
+ file: rel,
363
+ fixHint: "Replace raw /commands, /queries, or /live fetches with useCommand, useQuery, or useLiveQuery from the local Forge bridge.",
364
+ suggestedCommands: ["forge inspect frontend --json", "forge repair diagnose --diagnostic FORGE_FRONTEND_DIRECT_RUNTIME_FETCH --json"],
365
+ docs: ["src/forge/_generated/frontendGraph.json", "AGENTS.md"],
366
+ }));
367
+ }
368
+ }
369
+
370
+ for (const file of sourceFiles) {
371
+ const rel = toPosix(relative(input.workspaceRoot, file));
372
+ const text = textByRel.get(rel) ?? "";
373
+ const routePath = routePathForFile(webRoot, file);
374
+ if (routePath) {
375
+ if (framework === "vite" && rel === "web/index.html") {
376
+ continue;
377
+ }
378
+ const componentNames = componentNamesInText(text, components);
379
+ const componentUses = components
380
+ .filter((component) => componentNames.includes(component.name))
381
+ .reduce(
382
+ (acc, component) =>
383
+ mergeUses(acc, {
384
+ usesCommands: component.usesCommands,
385
+ usesQueries: component.usesQueries,
386
+ usesLiveQueries: component.usesLiveQueries,
387
+ rawForgeFetches: component.rawForgeFetches,
388
+ }),
389
+ detectRouteUses(file, text, input.clientManifest),
390
+ );
391
+ routes.push({
392
+ path: routePath,
393
+ file: rel,
394
+ components: componentNames,
395
+ ...componentUses,
396
+ });
397
+ }
398
+ }
399
+
400
+ if (providers.length === 0 && framework !== "static") {
401
+ diagnostics.push(createDiagnostic({
402
+ severity: "warning",
403
+ code: "FORGE_FRONTEND_PROVIDER_MISSING",
404
+ message: "web app does not expose a ForgeProvider; generated hooks may not be wired",
405
+ fixHint: "Mount ForgeProvider once in the web app root/provider layer and pass devAuth for local development.",
406
+ suggestedCommands: ["forge inspect frontend --json", "forge make ui --framework vite --dry-run --json"],
407
+ docs: ["src/forge/_generated/frontendGraph.json", "AGENTS.md"],
408
+ }));
409
+ }
410
+ if (bridgeFiles.length === 0 && framework !== "static") {
411
+ diagnostics.push(createDiagnostic({
412
+ severity: "warning",
413
+ code: "FORGE_FRONTEND_BRIDGE_MISSING",
414
+ message: "web/**/lib/forge.ts bridge is missing; agents may use fragile generated import paths",
415
+ fixHint: "Create web/lib/forge.ts or web/src/lib/forge.ts and re-export api, client, ForgeProvider, and generated hooks from src/forge/_generated.",
416
+ suggestedCommands: ["forge make ui --framework vite --dry-run --json", "forge generate"],
417
+ docs: ["src/forge/_generated/frontendGraph.json", "AGENTS.md"],
418
+ }));
419
+ }
420
+
421
+ const clientBindings = uniqueSorted([
422
+ ...routes.flatMap((route) =>
423
+ bindingEntries({
424
+ file: route.file,
425
+ route: route.path,
426
+ usesCommands: route.usesCommands,
427
+ usesQueries: route.usesQueries,
428
+ usesLiveQueries: route.usesLiveQueries,
429
+ rawForgeFetches: route.rawForgeFetches,
430
+ }).map((binding) => JSON.stringify(binding)),
431
+ ),
432
+ ...components.flatMap((component) =>
433
+ bindingEntries({
434
+ file: component.file,
435
+ component: component.name,
436
+ usesCommands: component.usesCommands,
437
+ usesQueries: component.usesQueries,
438
+ usesLiveQueries: component.usesLiveQueries,
439
+ rawForgeFetches: component.rawForgeFetches,
440
+ }).map((binding) => JSON.stringify(binding)),
441
+ ),
442
+ ]).map((binding) => JSON.parse(binding) as FrontendClientBindingInfo);
443
+
444
+ return {
445
+ schemaVersion: "0.1.0",
446
+ present: true,
447
+ framework,
448
+ root: WEB_ROOT,
449
+ dev: {
450
+ command: devCommandFor(webRoot, framework),
451
+ url: devUrl,
452
+ apiUrlEnv,
453
+ defaultApiUrl,
454
+ },
455
+ routes: routes.sort((a, b) => a.path.localeCompare(b.path)),
456
+ components: components.sort((a, b) => a.file.localeCompare(b.file)),
457
+ providers: providers.sort((a, b) => a.file.localeCompare(b.file)),
458
+ bridgeFiles: uniqueSorted(bridgeFiles),
459
+ webManifest: {
460
+ present: true,
461
+ framework,
462
+ root: WEB_ROOT,
463
+ ...(detectPackageManager(pkg) ? { packageManager: detectPackageManager(pkg) } : {}),
464
+ scripts: {
465
+ ...(pkg?.scripts?.dev ? { dev: pkg.scripts.dev } : {}),
466
+ ...(pkg?.scripts?.build ? { build: pkg.scripts.build } : {}),
467
+ ...(pkg?.scripts?.typecheck ? { typecheck: pkg.scripts.typecheck } : {}),
468
+ },
469
+ urls: {
470
+ dev: devUrl,
471
+ api: defaultApiUrl,
472
+ },
473
+ env: {
474
+ apiUrl: apiUrlEnv,
475
+ },
476
+ bridge: {
477
+ files: uniqueSorted(bridgeFiles),
478
+ valid: bridgeFiles.length > 0,
479
+ },
480
+ },
481
+ clientBindings: clientBindings.sort((a, b) =>
482
+ `${a.file}:${a.kind}:${a.name}`.localeCompare(`${b.file}:${b.kind}:${b.name}`),
483
+ ),
484
+ diagnostics,
485
+ };
486
+ }
487
+
488
+ export function serializeFrontendGraphJson(graph: FrontendGraph): string {
489
+ return serializeCanonical(graph);
490
+ }
491
+
492
+ export function serializeFrontendGraphTs(graph: FrontendGraph): string {
493
+ const parsed = JSON.parse(serializeFrontendGraphJson(graph)) as unknown;
494
+ return `export const frontendGraph = ${JSON.stringify(parsed, null, 2)} as const;\n`;
495
+ }
@@ -0,0 +1,23 @@
1
+ export type { FileSystem, DirEntry, FileStat } from "./types.ts";
2
+ export { NodeFileSystem } from "./node.ts";
3
+ export { InMemoryFileSystem } from "./memory.ts";
4
+ export {
5
+ createProfiledFileSystem,
6
+ formatFileSystemProfile,
7
+ getFileSystemProfile,
8
+ isForgeProfileEnabled,
9
+ resetFileSystemProfile,
10
+ } from "./profile.ts";
11
+
12
+ import { NodeFileSystem } from "./node.ts";
13
+ import {
14
+ createProfiledFileSystem,
15
+ isForgeProfileEnabled,
16
+ resetFileSystemProfile,
17
+ } from "./profile.ts";
18
+
19
+ resetFileSystemProfile();
20
+ const baseFileSystem = new NodeFileSystem();
21
+ export const nodeFileSystem = isForgeProfileEnabled()
22
+ ? createProfiledFileSystem(baseFileSystem)
23
+ : baseFileSystem;