forgeos 0.1.0-alpha.2 → 0.1.0-alpha.21

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 (406) hide show
  1. package/.npmignore +4 -0
  2. package/AGENTS.md +168 -81
  3. package/CHANGELOG.md +211 -0
  4. package/README.md +88 -14
  5. package/adapters/go/README.md +23 -0
  6. package/adapters/go/go.mod +3 -0
  7. package/adapters/go/http.go +149 -0
  8. package/adapters/go/registry.go +234 -0
  9. package/adapters/go/types.go +136 -0
  10. package/adapters/java/README.md +68 -0
  11. package/adapters/java/pom.xml +34 -0
  12. package/adapters/java/src/main/java/dev/forgeos/adapter/Auth.java +20 -0
  13. package/adapters/java/src/main/java/dev/forgeos/adapter/Diagnostic.java +16 -0
  14. package/adapters/java/src/main/java/dev/forgeos/adapter/Entry.java +38 -0
  15. package/adapters/java/src/main/java/dev/forgeos/adapter/EntryKind.java +16 -0
  16. package/adapters/java/src/main/java/dev/forgeos/adapter/ErrorInfo.java +4 -0
  17. package/adapters/java/src/main/java/dev/forgeos/adapter/Forge.java +94 -0
  18. package/adapters/java/src/main/java/dev/forgeos/adapter/ForgeCall.java +12 -0
  19. package/adapters/java/src/main/java/dev/forgeos/adapter/ForgeContext.java +11 -0
  20. package/adapters/java/src/main/java/dev/forgeos/adapter/ForgeHandler.java +8 -0
  21. package/adapters/java/src/main/java/dev/forgeos/adapter/ForgeHttpHandler.java +179 -0
  22. package/adapters/java/src/main/java/dev/forgeos/adapter/ForgeRegistry.java +121 -0
  23. package/adapters/java/src/main/java/dev/forgeos/adapter/Json.java +14 -0
  24. package/adapters/java/src/main/java/dev/forgeos/adapter/Manifest.java +14 -0
  25. package/adapters/java/src/main/java/dev/forgeos/adapter/RequestEnvelope.java +6 -0
  26. package/adapters/java/src/main/java/dev/forgeos/adapter/ResponseEnvelope.java +25 -0
  27. package/adapters/java/src/main/java/dev/forgeos/adapter/Risk.java +18 -0
  28. package/adapters/java/src/main/java/dev/forgeos/adapter/Schemas.java +36 -0
  29. package/adapters/java/src/main/java/dev/forgeos/adapter/Service.java +65 -0
  30. package/adapters/java/src/main/java/dev/forgeos/adapter/TransactionMode.java +18 -0
  31. package/adapters/java/src/main/java/dev/forgeos/adapter/TypedForgeHandler.java +6 -0
  32. package/adapters/java/target/classes/dev/forgeos/adapter/Auth.class +0 -0
  33. package/adapters/java/target/classes/dev/forgeos/adapter/Diagnostic.class +0 -0
  34. package/adapters/java/target/classes/dev/forgeos/adapter/Entry.class +0 -0
  35. package/adapters/java/target/classes/dev/forgeos/adapter/EntryKind.class +0 -0
  36. package/adapters/java/target/classes/dev/forgeos/adapter/ErrorInfo.class +0 -0
  37. package/adapters/java/target/classes/dev/forgeos/adapter/Forge.class +0 -0
  38. package/adapters/java/target/classes/dev/forgeos/adapter/ForgeCall.class +0 -0
  39. package/adapters/java/target/classes/dev/forgeos/adapter/ForgeContext.class +0 -0
  40. package/adapters/java/target/classes/dev/forgeos/adapter/ForgeHandler.class +0 -0
  41. package/adapters/java/target/classes/dev/forgeos/adapter/ForgeHttpHandler.class +0 -0
  42. package/adapters/java/target/classes/dev/forgeos/adapter/ForgeRegistry$EntryOption.class +0 -0
  43. package/adapters/java/target/classes/dev/forgeos/adapter/ForgeRegistry$RegisteredEntry.class +0 -0
  44. package/adapters/java/target/classes/dev/forgeos/adapter/ForgeRegistry$RegistryOption.class +0 -0
  45. package/adapters/java/target/classes/dev/forgeos/adapter/ForgeRegistry.class +0 -0
  46. package/adapters/java/target/classes/dev/forgeos/adapter/Json.class +0 -0
  47. package/adapters/java/target/classes/dev/forgeos/adapter/Manifest.class +0 -0
  48. package/adapters/java/target/classes/dev/forgeos/adapter/RequestEnvelope.class +0 -0
  49. package/adapters/java/target/classes/dev/forgeos/adapter/ResponseEnvelope.class +0 -0
  50. package/adapters/java/target/classes/dev/forgeos/adapter/Risk.class +0 -0
  51. package/adapters/java/target/classes/dev/forgeos/adapter/Schemas.class +0 -0
  52. package/adapters/java/target/classes/dev/forgeos/adapter/Service.class +0 -0
  53. package/adapters/java/target/classes/dev/forgeos/adapter/TransactionMode.class +0 -0
  54. package/adapters/java/target/classes/dev/forgeos/adapter/TypedForgeHandler.class +0 -0
  55. package/adapters/java/target/forge-java-adapter-0.1.0-alpha.11.jar +0 -0
  56. package/adapters/java/target/maven-archiver/pom.properties +3 -0
  57. package/adapters/java/target/maven-status/maven-compiler-plugin/compile/default-compile/createdFiles.lst +23 -0
  58. package/adapters/java/target/maven-status/maven-compiler-plugin/compile/default-compile/inputFiles.lst +20 -0
  59. package/adapters/java-spring-boot-starter/README.md +32 -0
  60. package/adapters/java-spring-boot-starter/pom.xml +36 -0
  61. package/adapters/java-spring-boot-starter/src/main/java/dev/forgeos/adapter/spring/ForgeCommand.java +22 -0
  62. package/adapters/java-spring-boot-starter/src/main/java/dev/forgeos/adapter/spring/ForgeExternalService.java +15 -0
  63. package/adapters/java-spring-boot-starter/src/main/java/dev/forgeos/adapter/spring/ForgeQuery.java +16 -0
  64. package/adapters/java-spring-boot-starter/src/main/java/dev/forgeos/adapter/spring/ForgeServiceBeanCondition.java +18 -0
  65. package/adapters/java-spring-boot-starter/src/main/java/dev/forgeos/adapter/spring/ForgeSpringAutoConfiguration.java +16 -0
  66. package/adapters/java-spring-boot-starter/src/main/java/dev/forgeos/adapter/spring/ForgeSpringRuntime.java +104 -0
  67. package/adapters/java-spring-boot-starter/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports +1 -0
  68. package/adapters/java-spring-boot-starter/target/classes/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports +1 -0
  69. package/adapters/java-spring-boot-starter/target/classes/dev/forgeos/adapter/spring/ForgeCommand.class +0 -0
  70. package/adapters/java-spring-boot-starter/target/classes/dev/forgeos/adapter/spring/ForgeExternalService.class +0 -0
  71. package/adapters/java-spring-boot-starter/target/classes/dev/forgeos/adapter/spring/ForgeQuery.class +0 -0
  72. package/adapters/java-spring-boot-starter/target/classes/dev/forgeos/adapter/spring/ForgeServiceBeanCondition.class +0 -0
  73. package/adapters/java-spring-boot-starter/target/classes/dev/forgeos/adapter/spring/ForgeSpringAutoConfiguration.class +0 -0
  74. package/adapters/java-spring-boot-starter/target/classes/dev/forgeos/adapter/spring/ForgeSpringRuntime.class +0 -0
  75. package/adapters/java-spring-boot-starter/target/forge-java-spring-boot-starter-0.1.0-alpha.11.jar +0 -0
  76. package/adapters/java-spring-boot-starter/target/maven-archiver/pom.properties +3 -0
  77. package/adapters/java-spring-boot-starter/target/maven-status/maven-compiler-plugin/compile/default-compile/createdFiles.lst +6 -0
  78. package/adapters/java-spring-boot-starter/target/maven-status/maven-compiler-plugin/compile/default-compile/inputFiles.lst +6 -0
  79. package/bin/forge.mjs +18 -0
  80. package/docs/changelog.md +242 -0
  81. package/docs/forge-protocol.md +189 -0
  82. package/examples/go-billing/go.mod +7 -0
  83. package/examples/go-billing/main.go +120 -0
  84. package/examples/java-billing/pom.xml +52 -0
  85. package/examples/java-billing/src/main/java/dev/forgeos/examples/billing/CreateInvoiceInput.java +4 -0
  86. package/examples/java-billing/src/main/java/dev/forgeos/examples/billing/Invoice.java +11 -0
  87. package/examples/java-billing/src/main/java/dev/forgeos/examples/billing/Main.java +127 -0
  88. package/examples/java-billing/target/classes/dev/forgeos/examples/billing/CreateInvoiceInput.class +0 -0
  89. package/examples/java-billing/target/classes/dev/forgeos/examples/billing/Invoice.class +0 -0
  90. package/examples/java-billing/target/classes/dev/forgeos/examples/billing/Main$EmptyInput.class +0 -0
  91. package/examples/java-billing/target/classes/dev/forgeos/examples/billing/Main$Options.class +0 -0
  92. package/examples/java-billing/target/classes/dev/forgeos/examples/billing/Main.class +0 -0
  93. package/examples/java-billing/target/java-billing-0.1.0-alpha.11-all.jar +0 -0
  94. package/examples/java-billing/target/java-billing-0.1.0-alpha.11.jar +0 -0
  95. package/examples/java-billing/target/maven-archiver/pom.properties +3 -0
  96. package/examples/java-billing/target/maven-status/maven-compiler-plugin/compile/default-compile/createdFiles.lst +5 -0
  97. package/examples/java-billing/target/maven-status/maven-compiler-plugin/compile/default-compile/inputFiles.lst +3 -0
  98. package/package.json +29 -7
  99. package/schemas/forge-manifest.schema.json +57 -0
  100. package/src/forge/_generated/releaseManifest.json +1 -2
  101. package/src/forge/_generated/releaseManifest.ts +3 -3
  102. package/src/forge/agent-adapters/index.ts +1511 -123
  103. package/src/forge/agent-adapters/types.ts +216 -1
  104. package/src/forge/agent-memory/bridge.ts +1245 -0
  105. package/src/forge/agent-memory/context-pack.ts +151 -0
  106. package/src/forge/agent-memory/hook-runner.ts +312 -0
  107. package/src/forge/agent-memory/mcp.ts +224 -0
  108. package/src/forge/agent-memory/normalize.ts +498 -0
  109. package/src/forge/agent-memory/redaction.ts +103 -0
  110. package/src/forge/agent-memory/sources/claude-code.ts +51 -0
  111. package/src/forge/agent-memory/sources/codex-hook-runner.mjs +273 -0
  112. package/src/forge/agent-memory/sources/codex.ts +119 -0
  113. package/src/forge/agent-memory/sources/cursor.ts +35 -0
  114. package/src/forge/agent-memory/types.ts +191 -0
  115. package/src/forge/bench.ts +248 -0
  116. package/src/forge/brownfield-import/index.ts +801 -0
  117. package/src/forge/brownfield-import/types.ts +127 -0
  118. package/src/forge/cair/action-journal.ts +61 -0
  119. package/src/forge/cair/action-parser.ts +314 -0
  120. package/src/forge/cair/action-validator.ts +40 -0
  121. package/src/forge/cair/actions.ts +1818 -0
  122. package/src/forge/cair/format.ts +77 -0
  123. package/src/forge/cair/index.ts +106 -0
  124. package/src/forge/cair/query.ts +478 -0
  125. package/src/forge/cair/snapshot.ts +315 -0
  126. package/src/forge/cair/types.ts +248 -0
  127. package/src/forge/cli/ai.ts +671 -3
  128. package/src/forge/cli/auth.ts +36 -1
  129. package/src/forge/cli/build.ts +20 -4
  130. package/src/forge/cli/changed.ts +300 -0
  131. package/src/forge/cli/codex-app-server.ts +877 -0
  132. package/src/forge/cli/commands.ts +1285 -7
  133. package/src/forge/cli/db.ts +121 -2
  134. package/src/forge/cli/deps.ts +79 -12
  135. package/src/forge/cli/dev.ts +502 -38
  136. package/src/forge/cli/docs.ts +265 -0
  137. package/src/forge/cli/handoff.ts +250 -0
  138. package/src/forge/cli/index.ts +1 -0
  139. package/src/forge/cli/main.ts +49 -3
  140. package/src/forge/cli/new.ts +3 -1
  141. package/src/forge/cli/next-actions.ts +23 -0
  142. package/src/forge/cli/output.ts +290 -1
  143. package/src/forge/cli/parse.ts +770 -36
  144. package/src/forge/cli/query.ts +32 -0
  145. package/src/forge/cli/release.ts +35 -11
  146. package/src/forge/cli/rls.ts +568 -17
  147. package/src/forge/cli/run.ts +41 -0
  148. package/src/forge/cli/secrets.ts +46 -1
  149. package/src/forge/cli/security.ts +381 -0
  150. package/src/forge/cli/self-host.ts +56 -14
  151. package/src/forge/cli/studio.ts +2163 -0
  152. package/src/forge/cli/verify.ts +1422 -32
  153. package/src/forge/compiler/agent-contract/build.ts +725 -41
  154. package/src/forge/compiler/agent-contract/types.ts +85 -0
  155. package/src/forge/compiler/ai-registry/build.ts +62 -1
  156. package/src/forge/compiler/ai-registry/constants.ts +1 -1
  157. package/src/forge/compiler/ai-registry/parse.ts +168 -5
  158. package/src/forge/compiler/api-surface/build.ts +47 -0
  159. package/src/forge/compiler/app-graph/build.ts +68 -8
  160. package/src/forge/compiler/app-graph/extract.ts +107 -0
  161. package/src/forge/compiler/app-graph/forge-apis.ts +1 -0
  162. package/src/forge/compiler/app-graph/module-graph.ts +73 -78
  163. package/src/forge/compiler/app-graph/parser.ts +24 -24
  164. package/src/forge/compiler/app-graph/profile.ts +26 -0
  165. package/src/forge/compiler/app-graph/versions.ts +1 -1
  166. package/src/forge/compiler/classifier/capabilities.ts +3 -2
  167. package/src/forge/compiler/classifier/classify.ts +32 -8
  168. package/src/forge/compiler/classifier/secrets.ts +3 -2
  169. package/src/forge/compiler/classifier/signals.ts +91 -1
  170. package/src/forge/compiler/client-sdk/build-manifest.ts +59 -0
  171. package/src/forge/compiler/client-sdk/render-client.ts +188 -13
  172. package/src/forge/compiler/data-graph/parse.ts +3 -3
  173. package/src/forge/compiler/data-graph/sql/ddl.ts +60 -2
  174. package/src/forge/compiler/data-graph/sql/serialize.ts +4 -0
  175. package/src/forge/compiler/data-graph/sql/types.ts +1 -0
  176. package/src/forge/compiler/dev-manifest/build.ts +3 -0
  177. package/src/forge/compiler/diagnostics/codes.ts +35 -0
  178. package/src/forge/compiler/diagnostics/create.ts +8 -3
  179. package/src/forge/compiler/diagnostics/index.ts +2 -0
  180. package/src/forge/compiler/emitter/barrel.ts +3 -0
  181. package/src/forge/compiler/emitter/render.ts +5 -0
  182. package/src/forge/compiler/external-manifest/registry.ts +205 -0
  183. package/src/forge/compiler/external-manifest/types.ts +91 -0
  184. package/src/forge/compiler/external-manifest/validate.ts +373 -0
  185. package/src/forge/compiler/frontend-graph/build.ts +85 -13
  186. package/src/forge/compiler/integration/add.ts +498 -22
  187. package/src/forge/compiler/integration/snapshot.ts +2 -0
  188. package/src/forge/compiler/make-registry/build.ts +19 -7
  189. package/src/forge/compiler/orchestrator/plan-profile.ts +23 -0
  190. package/src/forge/compiler/orchestrator/plan.ts +78 -7
  191. package/src/forge/compiler/orchestrator/profile.ts +65 -0
  192. package/src/forge/compiler/orchestrator/run.ts +97 -31
  193. package/src/forge/compiler/orchestrator/serialize.ts +101 -8
  194. package/src/forge/compiler/package-graph/compiler.ts +13 -3
  195. package/src/forge/compiler/package-manager/adapter.ts +4 -1
  196. package/src/forge/compiler/package-manager/commands.ts +4 -0
  197. package/src/forge/compiler/package-manager/executor.ts +30 -1
  198. package/src/forge/compiler/policy-registry/build.ts +44 -1
  199. package/src/forge/compiler/test-graph/build.ts +11 -3
  200. package/src/forge/compiler/types/ai-registry.ts +25 -1
  201. package/src/forge/compiler/types/app-graph.ts +9 -2
  202. package/src/forge/compiler/types/cli.ts +76 -1
  203. package/src/forge/compiler/types/dev-manifest.ts +3 -0
  204. package/src/forge/compiler/types/frontend-graph.ts +2 -2
  205. package/src/forge/delta/classifier.ts +52 -0
  206. package/src/forge/delta/explain.ts +126 -0
  207. package/src/forge/delta/git-observer.ts +43 -0
  208. package/src/forge/delta/ids.ts +44 -0
  209. package/src/forge/delta/index.ts +13 -0
  210. package/src/forge/delta/recorder.ts +402 -0
  211. package/src/forge/delta/redaction.ts +50 -0
  212. package/src/forge/delta/schema.ts +240 -0
  213. package/src/forge/delta/session.ts +142 -0
  214. package/src/forge/delta/status.ts +489 -0
  215. package/src/forge/delta/store.ts +2975 -0
  216. package/src/forge/delta/timeline.ts +104 -0
  217. package/src/forge/dev/server.ts +768 -15
  218. package/src/forge/dev/types.ts +15 -1
  219. package/src/forge/dev/watch.ts +17 -7
  220. package/src/forge/dev-console/cycle.ts +233 -21
  221. package/src/forge/dev-console/types.ts +46 -1
  222. package/src/forge/impact/index.ts +46 -8
  223. package/src/forge/impact/types.ts +6 -0
  224. package/src/forge/intent/index.ts +35 -16
  225. package/src/forge/make/index.ts +149 -6
  226. package/src/forge/make/templates.ts +343 -2
  227. package/src/forge/make/types.ts +3 -1
  228. package/src/forge/refactor/index.ts +1 -0
  229. package/src/forge/repair/rules/index.ts +2 -2
  230. package/src/forge/review/index.ts +158 -12
  231. package/src/forge/review/types.ts +15 -0
  232. package/src/forge/runtime/ai/context.ts +210 -5
  233. package/src/forge/runtime/ai/types.ts +70 -0
  234. package/src/forge/runtime/auth/claims.ts +32 -0
  235. package/src/forge/runtime/auth/errors.ts +2 -0
  236. package/src/forge/runtime/context/create-context.ts +30 -6
  237. package/src/forge/runtime/db/generated-client.ts +13 -2
  238. package/src/forge/runtime/db/memory-adapter.ts +2 -2
  239. package/src/forge/runtime/db/pglite-adapter.ts +77 -2
  240. package/src/forge/runtime/db/postgres-adapter.ts +6 -3
  241. package/src/forge/runtime/executor.ts +112 -2
  242. package/src/forge/runtime/external/bridge.ts +649 -0
  243. package/src/forge/runtime/runner/run-entry.ts +16 -7
  244. package/src/forge/runtime/telemetry/scrubber.ts +91 -10
  245. package/src/forge/runtime/webhooks/security.ts +184 -0
  246. package/src/forge/server.ts +100 -2
  247. package/src/forge/version.ts +1 -1
  248. package/src/forge/vue/index.ts +407 -0
  249. package/src/forge/workspace/change-summary.ts +209 -0
  250. package/src/forge/workspace/forge-cli.ts +14 -0
  251. package/src/forge/workspace/git-summary.ts +279 -0
  252. package/templates/agent-workroom/AGENTS.md +29 -0
  253. package/templates/agent-workroom/README.md +34 -0
  254. package/templates/agent-workroom/forge.config.ts +3 -0
  255. package/templates/agent-workroom/package.json +33 -0
  256. package/templates/agent-workroom/src/actions/indexAgentSignal.ts +10 -0
  257. package/templates/agent-workroom/src/commands/openWorkroom.ts +61 -0
  258. package/templates/agent-workroom/src/commands/recordAgentSignal.ts +119 -0
  259. package/templates/agent-workroom/src/commands/recordCheckRun.ts +52 -0
  260. package/templates/agent-workroom/src/forge/schema.ts +54 -0
  261. package/templates/agent-workroom/src/policies.ts +6 -0
  262. package/templates/agent-workroom/src/queries/listWorkrooms.ts +11 -0
  263. package/templates/agent-workroom/src/queries/liveWorkroom.ts +63 -0
  264. package/templates/agent-workroom/tsconfig.json +16 -0
  265. package/templates/agent-workroom/web/index.html +12 -0
  266. package/templates/agent-workroom/web/package.json +21 -0
  267. package/templates/agent-workroom/web/src/App.tsx +345 -0
  268. package/templates/agent-workroom/web/src/lib/forge.ts +13 -0
  269. package/templates/agent-workroom/web/src/main.tsx +13 -0
  270. package/templates/agent-workroom/web/src/styles.css +545 -0
  271. package/templates/agent-workroom/web/tsconfig.json +27 -0
  272. package/templates/b2b-support-web/package.json +2 -0
  273. package/templates/b2b-support-web/tsconfig.json +4 -1
  274. package/templates/b2b-support-web/web/package.json +1 -1
  275. package/templates/minimal-web/package.json +2 -1
  276. package/templates/minimal-web/tsconfig.json +3 -1
  277. package/templates/minimal-web/web/package.json +2 -2
  278. package/src/forge/_generated/actionSubscriptions.json +0 -2
  279. package/src/forge/_generated/actionSubscriptions.ts +0 -10
  280. package/src/forge/_generated/agentAdapterManifest.json +0 -2
  281. package/src/forge/_generated/agentAdapterManifest.ts +0 -73
  282. package/src/forge/_generated/agentContract.json +0 -2
  283. package/src/forge/_generated/agentContract.ts +0 -7696
  284. package/src/forge/_generated/agentQuickstart.md +0 -32
  285. package/src/forge/_generated/aiContext.ts +0 -59
  286. package/src/forge/_generated/aiModels.json +0 -2
  287. package/src/forge/_generated/aiModels.ts +0 -35
  288. package/src/forge/_generated/aiProviders.json +0 -2
  289. package/src/forge/_generated/aiProviders.ts +0 -23
  290. package/src/forge/_generated/aiRegistry.json +0 -2
  291. package/src/forge/_generated/aiRegistry.ts +0 -29
  292. package/src/forge/_generated/api.json +0 -2
  293. package/src/forge/_generated/api.ts +0 -8
  294. package/src/forge/_generated/appGraph.json +0 -2
  295. package/src/forge/_generated/appGraph.ts +0 -14667
  296. package/src/forge/_generated/appMap.md +0 -35
  297. package/src/forge/_generated/artifactManifest.json +0 -2
  298. package/src/forge/_generated/artifactManifest.ts +0 -7
  299. package/src/forge/_generated/authClaims.json +0 -2
  300. package/src/forge/_generated/authClaims.ts +0 -13
  301. package/src/forge/_generated/authConfig.json +0 -2
  302. package/src/forge/_generated/authConfig.ts +0 -17
  303. package/src/forge/_generated/authContext.ts +0 -23
  304. package/src/forge/_generated/authRegistry.json +0 -2
  305. package/src/forge/_generated/authRegistry.ts +0 -25
  306. package/src/forge/_generated/buildInfo.json +0 -2
  307. package/src/forge/_generated/buildInfo.ts +0 -9
  308. package/src/forge/_generated/capabilityMap.json +0 -2
  309. package/src/forge/_generated/capabilityMap.md +0 -15
  310. package/src/forge/_generated/capabilityMap.ts +0 -17
  311. package/src/forge/_generated/client.ts +0 -282
  312. package/src/forge/_generated/clientApi.ts +0 -9
  313. package/src/forge/_generated/clientManifest.json +0 -2
  314. package/src/forge/_generated/clientManifest.ts +0 -39
  315. package/src/forge/_generated/clientTypes.ts +0 -78
  316. package/src/forge/_generated/configRegistry.json +0 -2
  317. package/src/forge/_generated/configRegistry.ts +0 -4
  318. package/src/forge/_generated/dataGraph.json +0 -2
  319. package/src/forge/_generated/dataGraph.ts +0 -8
  320. package/src/forge/_generated/db.json +0 -2
  321. package/src/forge/_generated/db.ts +0 -2
  322. package/src/forge/_generated/dbSecurityManifest.json +0 -2
  323. package/src/forge/_generated/dbSecurityManifest.ts +0 -15
  324. package/src/forge/_generated/dbSessionContext.json +0 -2
  325. package/src/forge/_generated/dbSessionContext.ts +0 -39
  326. package/src/forge/_generated/deployManifest.json +0 -2
  327. package/src/forge/_generated/deployManifest.ts +0 -14
  328. package/src/forge/_generated/devManifest.json +0 -2
  329. package/src/forge/_generated/devManifest.ts +0 -47
  330. package/src/forge/_generated/envSchema.json +0 -2
  331. package/src/forge/_generated/envSchema.ts +0 -59
  332. package/src/forge/_generated/frontendGraph.json +0 -2
  333. package/src/forge/_generated/frontendGraph.ts +0 -27
  334. package/src/forge/_generated/importGuards.json +0 -2
  335. package/src/forge/_generated/importGuards.ts +0 -686
  336. package/src/forge/_generated/index.ts +0 -67
  337. package/src/forge/_generated/liveProductionManifest.json +0 -2
  338. package/src/forge/_generated/liveProductionManifest.ts +0 -23
  339. package/src/forge/_generated/liveProtocol.json +0 -2
  340. package/src/forge/_generated/liveProtocol.ts +0 -21
  341. package/src/forge/_generated/liveQueryRegistry.json +0 -2
  342. package/src/forge/_generated/liveQueryRegistry.ts +0 -9
  343. package/src/forge/_generated/liveTransportConfig.json +0 -2
  344. package/src/forge/_generated/liveTransportConfig.ts +0 -19
  345. package/src/forge/_generated/makeRegistry.json +0 -2
  346. package/src/forge/_generated/makeRegistry.ts +0 -163
  347. package/src/forge/_generated/makeTemplates.json +0 -2
  348. package/src/forge/_generated/makeTemplates.ts +0 -61
  349. package/src/forge/_generated/mockMap.json +0 -2
  350. package/src/forge/_generated/mockMap.ts +0 -7
  351. package/src/forge/_generated/operationPlaybooks.md +0 -147
  352. package/src/forge/_generated/packageGraph.json +0 -2
  353. package/src/forge/_generated/packageGraph.ts +0 -245249
  354. package/src/forge/_generated/packageUpgradeRegistry.json +0 -2
  355. package/src/forge/_generated/packageUpgradeRegistry.ts +0 -15
  356. package/src/forge/_generated/permissionMatrix.json +0 -2
  357. package/src/forge/_generated/permissionMatrix.ts +0 -7
  358. package/src/forge/_generated/policyRegistry.json +0 -2
  359. package/src/forge/_generated/policyRegistry.ts +0 -11
  360. package/src/forge/_generated/queryRegistry.json +0 -2
  361. package/src/forge/_generated/queryRegistry.ts +0 -9
  362. package/src/forge/_generated/react.d.ts +0 -22
  363. package/src/forge/_generated/react.ts +0 -29
  364. package/src/forge/_generated/reactManifest.json +0 -2
  365. package/src/forge/_generated/reactManifest.ts +0 -19
  366. package/src/forge/_generated/rlsPolicies.json +0 -2
  367. package/src/forge/_generated/rlsPolicies.sql +0 -34
  368. package/src/forge/_generated/rlsPolicies.ts +0 -6
  369. package/src/forge/_generated/runtimeGraph.json +0 -2
  370. package/src/forge/_generated/runtimeGraph.ts +0 -8
  371. package/src/forge/_generated/runtimeMatrix.json +0 -2
  372. package/src/forge/_generated/runtimeMatrix.ts +0 -327385
  373. package/src/forge/_generated/runtimeRegistry.ts +0 -2
  374. package/src/forge/_generated/runtimeRules.md +0 -79
  375. package/src/forge/_generated/secretRegistry.json +0 -2
  376. package/src/forge/_generated/secretRegistry.ts +0 -50
  377. package/src/forge/_generated/secretsContext.ts +0 -11
  378. package/src/forge/_generated/serverApi.ts +0 -10
  379. package/src/forge/_generated/sourceMapManifest.json +0 -2
  380. package/src/forge/_generated/sourceMapManifest.ts +0 -7
  381. package/src/forge/_generated/sqlPlan.json +0 -2
  382. package/src/forge/_generated/sqlPlan.ts +0 -88
  383. package/src/forge/_generated/subscriptionManifest.json +0 -2
  384. package/src/forge/_generated/subscriptionManifest.ts +0 -7
  385. package/src/forge/_generated/symbolicationManifest.json +0 -2
  386. package/src/forge/_generated/symbolicationManifest.ts +0 -17
  387. package/src/forge/_generated/telemetryRegistry.json +0 -2
  388. package/src/forge/_generated/telemetryRegistry.ts +0 -9
  389. package/src/forge/_generated/telemetrySinks.json +0 -2
  390. package/src/forge/_generated/telemetrySinks.ts +0 -11
  391. package/src/forge/_generated/tenantScope.json +0 -2
  392. package/src/forge/_generated/tenantScope.ts +0 -8
  393. package/src/forge/_generated/testGraph.json +0 -2
  394. package/src/forge/_generated/testGraph.ts +0 -3108
  395. package/src/forge/_generated/testPlanRegistry.json +0 -2
  396. package/src/forge/_generated/testPlanRegistry.ts +0 -33
  397. package/src/forge/_generated/uiRoutes.json +0 -2
  398. package/src/forge/_generated/uiRoutes.ts +0 -16
  399. package/src/forge/_generated/uiScenarios.json +0 -2
  400. package/src/forge/_generated/uiScenarios.ts +0 -30
  401. package/src/forge/_generated/uiTestManifest.json +0 -2
  402. package/src/forge/_generated/uiTestManifest.ts +0 -27
  403. package/src/forge/_generated/workflowRegistry.json +0 -2
  404. package/src/forge/_generated/workflowRegistry.ts +0 -9
  405. package/src/forge/_generated/workflowSubscriptions.json +0 -2
  406. package/src/forge/_generated/workflowSubscriptions.ts +0 -10
@@ -0,0 +1,801 @@
1
+ import { existsSync, mkdirSync, readFileSync, readdirSync, statSync, writeFileSync } from "node:fs";
2
+ import { basename, dirname, extname, join, relative, sep } from "node:path";
3
+ import type {
4
+ BrownfieldImportArtifacts,
5
+ BrownfieldImportCommandOptions,
6
+ BrownfieldImportResult,
7
+ ImportedCandidateEntry,
8
+ ImportedDependencyInventory,
9
+ ImportedFrontendCall,
10
+ ImportedInventory,
11
+ ImportedEntryKind,
12
+ ImportedRiskFinding,
13
+ ImportedRiskReport,
14
+ ImportedRoute,
15
+ ImportedRouteSource,
16
+ } from "./types.ts";
17
+
18
+ const IMPORT_DIR = ".forge/import";
19
+ const SOURCE_EXTENSIONS = new Set([".ts", ".tsx", ".js", ".jsx", ".mjs", ".cjs"]);
20
+ const IGNORED_DIRS = new Set([
21
+ ".git",
22
+ ".forge",
23
+ ".next",
24
+ ".nuxt",
25
+ ".output",
26
+ "__tests__",
27
+ "_generated",
28
+ "coverage",
29
+ "dist",
30
+ "build",
31
+ "node_modules",
32
+ "out",
33
+ "target",
34
+ "test",
35
+ "tests",
36
+ ]);
37
+ const PREFERRED_SOURCE_ROOTS = ["src", "app", "pages", "server", "web", "apps", "packages"];
38
+
39
+ export const BROWNFIELD_IMPORT_ARTIFACTS: BrownfieldImportArtifacts = {
40
+ inventory: `${IMPORT_DIR}/inventory.json`,
41
+ routes: `${IMPORT_DIR}/routes.json`,
42
+ frontendCalls: `${IMPORT_DIR}/frontendCalls.json`,
43
+ candidateEntries: `${IMPORT_DIR}/candidateEntries.json`,
44
+ riskReport: `${IMPORT_DIR}/riskReport.json`,
45
+ migrationPlan: `${IMPORT_DIR}/migrationPlan.md`,
46
+ importedAgentContract: `${IMPORT_DIR}/importedAgentContract.json`,
47
+ };
48
+
49
+ interface SourceFile {
50
+ relativePath: string;
51
+ absolutePath: string;
52
+ text: string;
53
+ }
54
+
55
+ function normalizePath(path: string): string {
56
+ return path.split(sep).join("/");
57
+ }
58
+
59
+ function artifactPath(workspaceRoot: string, relativePath: string): string {
60
+ return join(workspaceRoot, ...relativePath.split("/"));
61
+ }
62
+
63
+ function readJson<T>(path: string): T | null {
64
+ if (!existsSync(path)) {
65
+ return null;
66
+ }
67
+ try {
68
+ return JSON.parse(readFileSync(path, "utf8")) as T;
69
+ } catch {
70
+ return null;
71
+ }
72
+ }
73
+
74
+ function readPackageJson(workspaceRoot: string): Record<string, unknown> {
75
+ return readJson<Record<string, unknown>>(join(workspaceRoot, "package.json")) ?? {};
76
+ }
77
+
78
+ function objectKeys(value: unknown): string[] {
79
+ if (!value || typeof value !== "object" || Array.isArray(value)) {
80
+ return [];
81
+ }
82
+ return Object.keys(value as Record<string, unknown>).sort();
83
+ }
84
+
85
+ function collectSourceFiles(workspaceRoot: string): SourceFile[] {
86
+ const files: SourceFile[] = [];
87
+ const visit = (absoluteDir: string): void => {
88
+ for (const entry of readdirSync(absoluteDir, { withFileTypes: true })) {
89
+ const absolutePath = join(absoluteDir, entry.name);
90
+ const relativePath = normalizePath(relative(workspaceRoot, absolutePath));
91
+ if (entry.isDirectory()) {
92
+ if (
93
+ !entry.name.startsWith(".") &&
94
+ !IGNORED_DIRS.has(entry.name) &&
95
+ !relativePath.includes("/src/forge/_generated")
96
+ ) {
97
+ visit(absolutePath);
98
+ }
99
+ continue;
100
+ }
101
+ if (!entry.isFile() || !SOURCE_EXTENSIONS.has(extname(entry.name))) {
102
+ continue;
103
+ }
104
+ if (statSync(absolutePath).size > 1_000_000) {
105
+ continue;
106
+ }
107
+ files.push({
108
+ absolutePath,
109
+ relativePath,
110
+ text: readFileSync(absolutePath, "utf8"),
111
+ });
112
+ }
113
+ };
114
+ const preferredRoots = PREFERRED_SOURCE_ROOTS
115
+ .map((name) => join(workspaceRoot, name))
116
+ .filter((absolutePath) => existsSync(absolutePath) && statSync(absolutePath).isDirectory());
117
+ const roots = preferredRoots.length > 0 ? preferredRoots : [workspaceRoot];
118
+ for (const root of roots) {
119
+ visit(root);
120
+ }
121
+ return files.sort((left, right) => left.relativePath.localeCompare(right.relativePath));
122
+ }
123
+
124
+ function hasAny(names: string[], candidates: string[]): boolean {
125
+ return candidates.some((candidate) => names.includes(candidate));
126
+ }
127
+
128
+ function buildDependencyInventory(workspaceRoot: string): {
129
+ packageName?: string;
130
+ dependencies: ImportedDependencyInventory;
131
+ } {
132
+ const pkg = readPackageJson(workspaceRoot);
133
+ const dependencies = objectKeys(pkg.dependencies);
134
+ const devDependencies = objectKeys(pkg.devDependencies);
135
+ const all = [...dependencies, ...devDependencies];
136
+ const scripts = objectKeys(pkg.scripts);
137
+ const frameworks = [
138
+ hasAny(all, ["next"]) ? "next" : null,
139
+ hasAny(all, ["react"]) ? "react" : null,
140
+ hasAny(all, ["vue", "nuxt"]) ? "vue" : null,
141
+ hasAny(all, ["nuxt"]) ? "nuxt" : null,
142
+ hasAny(all, ["express"]) ? "express" : null,
143
+ hasAny(all, ["@nestjs/core"]) ? "nest" : null,
144
+ hasAny(all, ["fastify"]) ? "fastify" : null,
145
+ hasAny(all, ["hono"]) ? "hono" : null,
146
+ ].filter((value): value is string => value !== null);
147
+ const dataPackages = all.filter((name) =>
148
+ ["@prisma/client", "prisma", "drizzle-orm", "typeorm", "mongoose", "sequelize", "knex"].includes(name),
149
+ );
150
+ const externalPackages = all.filter((name) =>
151
+ [
152
+ "stripe",
153
+ "resend",
154
+ "nodemailer",
155
+ "openai",
156
+ "@ai-sdk/openai",
157
+ "@ai-sdk/anthropic",
158
+ "aws-sdk",
159
+ "@aws-sdk/client-s3",
160
+ "sendgrid",
161
+ "@sendgrid/mail",
162
+ "twilio",
163
+ ].includes(name),
164
+ );
165
+ return {
166
+ packageName: typeof pkg.name === "string" ? pkg.name : undefined,
167
+ dependencies: {
168
+ dependencies,
169
+ devDependencies,
170
+ scripts,
171
+ frameworks: Array.from(new Set(frameworks)).sort(),
172
+ dataPackages,
173
+ externalPackages,
174
+ },
175
+ };
176
+ }
177
+
178
+ function stableId(prefix: string, parts: string[]): string {
179
+ let hash = 2166136261;
180
+ for (const char of parts.join("|")) {
181
+ hash ^= char.charCodeAt(0);
182
+ hash = Math.imul(hash, 16777619);
183
+ }
184
+ return `${prefix}_${(hash >>> 0).toString(36)}`;
185
+ }
186
+
187
+ function stripExtension(segment: string): string {
188
+ return segment.replace(/\.(ts|tsx|js|jsx|mjs|cjs)$/u, "");
189
+ }
190
+
191
+ function normalizeRouteSegment(segment: string): string | null {
192
+ const clean = stripExtension(segment);
193
+ if (clean === "index" || clean === "route" || clean === "page") {
194
+ return null;
195
+ }
196
+ if (clean.startsWith("(") && clean.endsWith(")")) {
197
+ return null;
198
+ }
199
+ const catchAll = clean.match(/^\[\.\.\.(.+)\]$/u);
200
+ if (catchAll) {
201
+ return `:${catchAll[1]}*`;
202
+ }
203
+ const dynamic = clean.match(/^\[(.+)\]$/u);
204
+ if (dynamic) {
205
+ return `:${dynamic[1]}`;
206
+ }
207
+ return clean;
208
+ }
209
+
210
+ function routePathFromFile(relativePath: string, marker: string): string {
211
+ const normalized = relativePath.startsWith("/") ? relativePath : `/${relativePath}`;
212
+ const markerIndex = normalized.indexOf(marker);
213
+ const afterMarker = markerIndex >= 0 ? normalized.slice(markerIndex + marker.length) : normalized;
214
+ const segments = afterMarker
215
+ .split("/")
216
+ .map(normalizeRouteSegment)
217
+ .filter((segment): segment is string => Boolean(segment));
218
+ return `/${segments.join("/")}`.replace(/\/+/gu, "/");
219
+ }
220
+
221
+ function joinRoutePath(base: string, child: string): string {
222
+ return `/${[base, child].map((part) => part.replace(/^\/|\/$/gu, "")).filter(Boolean).join("/")}`.replace(/\/+/gu, "/");
223
+ }
224
+
225
+ function pathIncludesRouteMarker(relativePath: string, marker: string): boolean {
226
+ const normalized = relativePath.startsWith("/") ? relativePath : `/${relativePath}`;
227
+ return normalized.includes(marker);
228
+ }
229
+
230
+ function addRoute(
231
+ routes: ImportedRoute[],
232
+ method: string,
233
+ path: string,
234
+ file: string,
235
+ source: ImportedRouteSource,
236
+ confidence: number,
237
+ handler?: string,
238
+ ): void {
239
+ routes.push({
240
+ id: stableId("route", [method.toUpperCase(), path, file, source, handler ?? ""]),
241
+ method: method.toUpperCase(),
242
+ path,
243
+ file,
244
+ source,
245
+ handler,
246
+ confidence,
247
+ });
248
+ }
249
+
250
+ function detectRoutes(files: SourceFile[]): ImportedRoute[] {
251
+ const routes: ImportedRoute[] = [];
252
+ const methods = ["GET", "POST", "PUT", "PATCH", "DELETE", "HEAD", "OPTIONS"];
253
+ for (const file of files) {
254
+ if (pathIncludesRouteMarker(file.relativePath, "/app/api/") && basename(file.relativePath).startsWith("route.")) {
255
+ const path = routePathFromFile(file.relativePath, "/app/");
256
+ for (const method of methods) {
257
+ if (new RegExp(`export\\s+(?:async\\s+)?function\\s+${method}\\b`, "u").test(file.text)) {
258
+ addRoute(routes, method, path, file.relativePath, "next-app-router", 0.92, method);
259
+ }
260
+ }
261
+ }
262
+
263
+ if (pathIncludesRouteMarker(file.relativePath, "/pages/api/")) {
264
+ const path = routePathFromFile(file.relativePath, "/pages/");
265
+ addRoute(routes, "ANY", path, file.relativePath, "next-pages-api", 0.78, "default");
266
+ }
267
+
268
+ const expressRoute = /\b(?:app|router)\s*\.\s*(get|post|put|patch|delete|all)\s*\(\s*["'`]([^"'`]+)["'`]/giu;
269
+ for (const match of file.text.matchAll(expressRoute)) {
270
+ addRoute(routes, match[1] ?? "all", match[2] ?? "/", file.relativePath, "express", 0.84);
271
+ }
272
+
273
+ const controller = file.text.match(/@Controller\s*\(\s*["'`]([^"'`]*)["'`]\s*\)/u);
274
+ if (controller) {
275
+ const nestRoute = /@(Get|Post|Put|Patch|Delete|All)\s*\(\s*(?:["'`]([^"'`]*)["'`])?\s*\)/giu;
276
+ for (const match of file.text.matchAll(nestRoute)) {
277
+ addRoute(
278
+ routes,
279
+ match[1] ?? "All",
280
+ joinRoutePath(controller[1] ?? "", match[2] ?? ""),
281
+ file.relativePath,
282
+ "nest",
283
+ 0.78,
284
+ );
285
+ }
286
+ }
287
+ }
288
+ const seen = new Set<string>();
289
+ return routes
290
+ .filter((route) => {
291
+ const key = `${route.method}:${route.path}:${route.file}:${route.source}`;
292
+ if (seen.has(key)) {
293
+ return false;
294
+ }
295
+ seen.add(key);
296
+ return true;
297
+ })
298
+ .sort((left, right) => `${left.path}:${left.method}`.localeCompare(`${right.path}:${right.method}`));
299
+ }
300
+
301
+ function detectFrontendCalls(files: SourceFile[], routes: ImportedRoute[]): ImportedFrontendCall[] {
302
+ const calls: ImportedFrontendCall[] = [];
303
+ const addCall = (file: string, client: "fetch" | "axios", method: string, url: string): void => {
304
+ const route = routes.find((candidate) => candidate.path === url || url.startsWith(`${candidate.path}/`));
305
+ calls.push({
306
+ id: stableId("call", [file, client, method, url]),
307
+ file,
308
+ client,
309
+ method: method.toUpperCase(),
310
+ url,
311
+ routeId: route?.id,
312
+ confidence: route ? 0.78 : 0.55,
313
+ });
314
+ };
315
+ for (const file of files) {
316
+ if (
317
+ pathIncludesRouteMarker(file.relativePath, "/app/api/") ||
318
+ pathIncludesRouteMarker(file.relativePath, "/pages/api/")
319
+ ) {
320
+ continue;
321
+ }
322
+ const fetchCall = /\bfetch\s*\(\s*["'`]([^"'`]+)["'`]\s*(?:,\s*\{(?<options>[\s\S]{0,300}?)\})?/giu;
323
+ for (const match of file.text.matchAll(fetchCall)) {
324
+ const url = match[1] ?? "";
325
+ if (!url.startsWith("/api/") && !url.startsWith("http")) {
326
+ continue;
327
+ }
328
+ const options = match.groups?.options ?? "";
329
+ const method = options.match(/method\s*:\s*["'`]([A-Z]+)["'`]/iu)?.[1] ?? "GET";
330
+ addCall(file.relativePath, "fetch", method, url);
331
+ }
332
+ const axiosMethodCall = /\baxios\.(get|post|put|patch|delete)\s*\(\s*["'`]([^"'`]+)["'`]/giu;
333
+ for (const match of file.text.matchAll(axiosMethodCall)) {
334
+ const url = match[2] ?? "";
335
+ if (url.startsWith("/api/") || url.startsWith("http")) {
336
+ addCall(file.relativePath, "axios", match[1] ?? "GET", url);
337
+ }
338
+ }
339
+ }
340
+ return calls.sort((left, right) => `${left.file}:${left.url}`.localeCompare(`${right.file}:${right.url}`));
341
+ }
342
+
343
+ function collectEnv(workspaceRoot: string, files: SourceFile[]): ImportedInventory["env"] {
344
+ const names = new Set<string>();
345
+ for (const file of files) {
346
+ for (const match of file.text.matchAll(/\bprocess\.env\.([A-Z0-9_]+)/gu)) {
347
+ names.add(match[1] ?? "");
348
+ }
349
+ }
350
+ const envFiles = [".env", ".env.local", ".env.example", ".env.sample"]
351
+ .filter((name) => existsSync(join(workspaceRoot, name)))
352
+ .sort();
353
+ for (const envFile of envFiles) {
354
+ const raw = readFileSync(join(workspaceRoot, envFile), "utf8");
355
+ for (const line of raw.split(/\r?\n/u)) {
356
+ const match = line.match(/^\s*([A-Z0-9_]+)\s*=/u);
357
+ if (match) {
358
+ names.add(match[1] ?? "");
359
+ }
360
+ }
361
+ }
362
+ return {
363
+ processEnv: Array.from(names).filter(Boolean).sort(),
364
+ envFiles,
365
+ };
366
+ }
367
+
368
+ function sourceTextForRoute(route: ImportedRoute, files: SourceFile[]): string {
369
+ const text = files.find((file) => file.relativePath === route.file)?.text ?? "";
370
+ if (!text) {
371
+ return "";
372
+ }
373
+ return scopedSourceTextForRoute(route, text) ?? text;
374
+ }
375
+
376
+ function scopedSourceTextForRoute(route: ImportedRoute, text: string): string | null {
377
+ if (route.source === "next-app-router" && route.handler) {
378
+ return sliceUntilNextMatch(
379
+ text,
380
+ new RegExp(`export\\s+(?:async\\s+)?function\\s+${escapeRegExp(route.handler)}\\b`, "u"),
381
+ /export\s+(?:async\s+)?function\s+(?:GET|POST|PUT|PATCH|DELETE|HEAD|OPTIONS)\b/gu,
382
+ );
383
+ }
384
+ if (route.source === "express") {
385
+ const method = route.method.toLowerCase();
386
+ const matcher = new RegExp(`\\b(?:app|router)\\s*\\.\\s*${escapeRegExp(method)}\\s*\\(\\s*["'\`]${escapeRegExp(route.path)}["'\`]`, "u");
387
+ return sliceUntilNextMatch(
388
+ text,
389
+ matcher,
390
+ /\b(?:app|router)\s*\.\s*(?:get|post|put|patch|delete|all)\s*\(\s*["'`][^"'`]+["'`]/giu,
391
+ );
392
+ }
393
+ if (route.source === "nest") {
394
+ const method = route.method.charAt(0).toUpperCase() + route.method.slice(1).toLowerCase();
395
+ return sliceUntilNextMatch(
396
+ text,
397
+ new RegExp(`@${escapeRegExp(method)}\\s*\\(`, "u"),
398
+ /@(Get|Post|Put|Patch|Delete|All)\s*\(/gu,
399
+ );
400
+ }
401
+ return null;
402
+ }
403
+
404
+ function sliceUntilNextMatch(text: string, startPattern: RegExp, nextPattern: RegExp): string | null {
405
+ const start = text.search(startPattern);
406
+ if (start < 0) {
407
+ return null;
408
+ }
409
+ nextPattern.lastIndex = start + 1;
410
+ const next = nextPattern.exec(text);
411
+ return text.slice(start, next?.index ?? text.length);
412
+ }
413
+
414
+ function escapeRegExp(value: string): string {
415
+ return value.replace(/[.*+?^${}()|[\]\\]/gu, "\\$&");
416
+ }
417
+
418
+ function classifyCandidate(route: ImportedRoute, text: string): Pick<ImportedCandidateEntry, "kind" | "confidence" | "risks" | "evidence" | "needsApproval"> {
419
+ const method = route.method.toUpperCase();
420
+ const lowerPath = route.path.toLowerCase();
421
+ const lowerText = text.toLowerCase();
422
+ const risks = new Set<string>();
423
+ const evidence: string[] = [`${route.source} ${method} ${route.path}`];
424
+ const isQuery = method === "GET" || method === "HEAD";
425
+ const isDestructive = method === "DELETE" || /(delete|remove|destroy|cancel|refund|void|purge)/u.test(lowerPath);
426
+ const writes = /(\.create\(|\.update\(|\.delete\(|\.upsert\(|\.insert\(|\.save\(|\.destroy\(|\.remove\()/u.test(lowerText);
427
+ const external = /(stripe|resend|sendgrid|twilio|nodemailer|openai|anthropic|https?:\/\/|\.send\()/u.test(lowerText);
428
+ const auth = /(auth|session|currentuser|getserversession|clerk|nextauth|requireuser|requireauth)/u.test(lowerText);
429
+ const tenant = /(tenantid|tenant_id|organizationid|orgid|accountid)/u.test(lowerText);
430
+ const methodUnknown = method === "ANY" || method === "ALL";
431
+ const ambiguousPostQuery = method === "POST" &&
432
+ /(?:^|\/)(search|query|filter|lookup|graphql)(?:$|\/)/u.test(lowerPath) &&
433
+ !writes &&
434
+ !isDestructive &&
435
+ !external;
436
+ if ((!isQuery && !ambiguousPostQuery) || writes) {
437
+ risks.add("writes-state");
438
+ }
439
+ if (ambiguousPostQuery) {
440
+ risks.add("ambiguous-post-query");
441
+ }
442
+ if (isDestructive) {
443
+ risks.add("destructive");
444
+ }
445
+ if (external) {
446
+ risks.add("external-side-effect");
447
+ }
448
+ if (!auth) {
449
+ risks.add("auth-unknown");
450
+ }
451
+ if (tenant) {
452
+ risks.add("tenant-sensitive");
453
+ }
454
+ if (/\bprocess\.env\./u.test(text)) {
455
+ risks.add("secret-sensitive");
456
+ }
457
+ if (methodUnknown) {
458
+ risks.add("method-unknown");
459
+ }
460
+ const commandLike = !isQuery || writes || isDestructive || external;
461
+ if (ambiguousPostQuery) {
462
+ return {
463
+ kind: "command-candidate",
464
+ confidence: 0.55,
465
+ risks: Array.from(risks).sort(),
466
+ evidence,
467
+ needsApproval: true,
468
+ };
469
+ }
470
+ return {
471
+ kind: commandLike ? "command" : "query",
472
+ confidence: commandLike ? (isDestructive ? 0.9 : 0.78) : 0.86,
473
+ risks: Array.from(risks).sort(),
474
+ evidence,
475
+ needsApproval: commandLike || external || isDestructive || methodUnknown,
476
+ };
477
+ }
478
+
479
+ function nameForCandidate(route: ImportedRoute, kind: ImportedEntryKind): string {
480
+ const nouns = route.path
481
+ .replace(/^\/api\//u, "")
482
+ .replace(/:\w+\*?/gu, "byId")
483
+ .split("/")
484
+ .filter(Boolean)
485
+ .map((segment) => segment.replace(/[^a-zA-Z0-9]/gu, ""))
486
+ .filter(Boolean);
487
+ const base = nouns.length > 0 ? nouns.join(".") : "imported.route";
488
+ const method = route.method.toUpperCase();
489
+ const action =
490
+ kind === "query" ? "read" :
491
+ kind === "command-candidate" ? "candidate" :
492
+ method === "POST" ? "create" :
493
+ method === "PUT" || method === "PATCH" ? "update" :
494
+ method === "DELETE" ? "delete" :
495
+ "call";
496
+ return `${base}.${action}`;
497
+ }
498
+
499
+ function buildCandidates(routes: ImportedRoute[], files: SourceFile[]): ImportedCandidateEntry[] {
500
+ const usedNames = new Map<string, number>();
501
+ return routes.map((route) => {
502
+ const text = sourceTextForRoute(route, files);
503
+ const classification = classifyCandidate(route, text);
504
+ const baseName = nameForCandidate(route, classification.kind);
505
+ const count = usedNames.get(baseName) ?? 0;
506
+ usedNames.set(baseName, count + 1);
507
+ const name = count === 0 ? baseName : `${baseName}${count + 1}`;
508
+ return {
509
+ id: stableId("entry", [route.id, name]),
510
+ name,
511
+ kind: classification.kind,
512
+ method: route.method,
513
+ path: route.path,
514
+ routeId: route.id,
515
+ file: route.file,
516
+ origin: "imported",
517
+ assurance: "static-scan",
518
+ reviewStatus: "needs-review",
519
+ visibleToAgent: false,
520
+ needsApproval: classification.needsApproval,
521
+ confidence: classification.confidence,
522
+ risks: classification.risks,
523
+ evidence: classification.evidence,
524
+ };
525
+ });
526
+ }
527
+
528
+ function buildRiskReport(
529
+ routes: ImportedRoute[],
530
+ frontendCalls: ImportedFrontendCall[],
531
+ candidates: ImportedCandidateEntry[],
532
+ files: SourceFile[],
533
+ ): ImportedRiskReport {
534
+ const findings: ImportedRiskFinding[] = [];
535
+ for (const candidate of candidates) {
536
+ if (candidate.visibleToAgent) {
537
+ findings.push({
538
+ code: "FORGE_IMPORT_VISIBLE",
539
+ severity: "error",
540
+ file: candidate.file,
541
+ routeId: candidate.routeId,
542
+ message: "Imported entries must stay hidden from agents until a human approves them.",
543
+ });
544
+ }
545
+ if (candidate.risks.includes("auth-unknown") && candidate.kind === "command") {
546
+ findings.push({
547
+ code: "FORGE_IMPORT_AUTH_UNKNOWN",
548
+ severity: "warning",
549
+ file: candidate.file,
550
+ routeId: candidate.routeId,
551
+ message: `${candidate.name} looks command-like but static scan did not find an obvious auth guard.`,
552
+ });
553
+ }
554
+ if (candidate.risks.includes("destructive")) {
555
+ findings.push({
556
+ code: "FORGE_IMPORT_DESTRUCTIVE",
557
+ severity: "warning",
558
+ file: candidate.file,
559
+ routeId: candidate.routeId,
560
+ message: `${candidate.name} is destructive and must keep needsApproval=true.`,
561
+ });
562
+ }
563
+ const text = sourceTextForRoute({ ...candidate, id: candidate.routeId, source: "unknown", confidence: 0 }, files);
564
+ if (/\b(req\.body|body|input)\.tenantId\b/u.test(text) || /\b(req\.body|body|input)\.tenant_id\b/u.test(text)) {
565
+ findings.push({
566
+ code: "FORGE_IMPORT_TENANT_SPOOFABLE",
567
+ severity: "warning",
568
+ file: candidate.file,
569
+ routeId: candidate.routeId,
570
+ message: `${candidate.name} appears to accept tenant identity from input; review tenant isolation before migration.`,
571
+ });
572
+ }
573
+ }
574
+ return {
575
+ schemaVersion: "0.1.0",
576
+ summary: {
577
+ routeCount: routes.length,
578
+ frontendCallCount: frontendCalls.length,
579
+ candidateCount: candidates.length,
580
+ commandCount: candidates.filter((candidate) => candidate.kind === "command").length,
581
+ queryCount: candidates.filter((candidate) => candidate.kind === "query").length,
582
+ hiddenFromAgents: candidates.filter((candidate) => !candidate.visibleToAgent).length,
583
+ needsApproval: candidates.filter((candidate) => candidate.needsApproval).length,
584
+ },
585
+ findings,
586
+ };
587
+ }
588
+
589
+ function buildInventory(workspaceRoot: string, files: SourceFile[]): ImportedInventory {
590
+ const dependencyInventory = buildDependencyInventory(workspaceRoot);
591
+ return {
592
+ schemaVersion: "0.1.0",
593
+ origin: "imported",
594
+ assurance: "static-scan",
595
+ workspaceRoot,
596
+ generatedAt: new Date().toISOString(),
597
+ packageName: dependencyInventory.packageName,
598
+ dependencies: dependencyInventory.dependencies,
599
+ filesScanned: files.length,
600
+ sourceFiles: files.map((file) => file.relativePath),
601
+ env: collectEnv(workspaceRoot, files),
602
+ };
603
+ }
604
+
605
+ function buildMigrationPlan(
606
+ inventory: ImportedInventory,
607
+ routes: ImportedRoute[],
608
+ frontendCalls: ImportedFrontendCall[],
609
+ candidates: ImportedCandidateEntry[],
610
+ riskReport: ImportedRiskReport,
611
+ ): string {
612
+ const lines = [
613
+ "# Brownfield Import Migration Plan",
614
+ "",
615
+ "This plan was produced by `forge import analyze` from static evidence only. Imported entries are hidden from agents until reviewed.",
616
+ "",
617
+ "## Summary",
618
+ "",
619
+ `- Package: ${inventory.packageName ?? "unknown"}`,
620
+ `- Files scanned: ${inventory.filesScanned}`,
621
+ `- Routes detected: ${routes.length}`,
622
+ `- Frontend calls detected: ${frontendCalls.length}`,
623
+ `- Candidate entries: ${candidates.length}`,
624
+ `- Hidden from agents: ${riskReport.summary.hiddenFromAgents}`,
625
+ `- Entries requiring approval: ${riskReport.summary.needsApproval}`,
626
+ "",
627
+ "## Review Order",
628
+ "",
629
+ "1. Review destructive and external-side-effect candidates first.",
630
+ "2. Confirm auth and tenant boundaries before exposing any imported entry.",
631
+ "3. Convert read-only GET candidates into Forge queries only after validating schema ownership.",
632
+ "4. Convert mutating candidates into Forge commands/actions with `ctx.emit` or durable workflows for side effects.",
633
+ "5. Replace frontend raw API calls with generated Forge client bindings after each reviewed migration.",
634
+ "",
635
+ "## Candidate Entries",
636
+ "",
637
+ ...candidates.map((candidate) =>
638
+ `- \`${candidate.name}\` (${candidate.kind}, ${candidate.method} ${candidate.path}) - confidence ${candidate.confidence.toFixed(2)}, risks: ${candidate.risks.join(", ") || "none"}`,
639
+ ),
640
+ "",
641
+ "## Findings",
642
+ "",
643
+ ...(riskReport.findings.length === 0
644
+ ? ["- No high-signal risk findings beyond conservative imported defaults."]
645
+ : riskReport.findings.map((finding) => `- ${finding.severity.toUpperCase()} ${finding.code}: ${finding.message}`)),
646
+ "",
647
+ ];
648
+ return `${lines.join("\n")}\n`;
649
+ }
650
+
651
+ function buildImportedAgentContract(
652
+ inventory: ImportedInventory,
653
+ routes: ImportedRoute[],
654
+ frontendCalls: ImportedFrontendCall[],
655
+ candidates: ImportedCandidateEntry[],
656
+ riskReport: ImportedRiskReport,
657
+ ): Record<string, unknown> {
658
+ return {
659
+ schemaVersion: "0.1.0",
660
+ origin: "imported",
661
+ assurance: "static-scan",
662
+ reviewStatus: "needs-review",
663
+ visibleToAgent: false,
664
+ generatedAt: inventory.generatedAt,
665
+ summary: riskReport.summary,
666
+ frameworks: inventory.dependencies.frameworks,
667
+ routes,
668
+ frontendCalls,
669
+ entries: candidates,
670
+ findings: riskReport.findings,
671
+ };
672
+ }
673
+
674
+ function writeArtifact(workspaceRoot: string, relativePath: string, value: unknown): void {
675
+ const absolute = artifactPath(workspaceRoot, relativePath);
676
+ mkdirSync(dirname(absolute), { recursive: true });
677
+ const content = typeof value === "string" ? value : `${JSON.stringify(value, null, 2)}\n`;
678
+ writeFileSync(absolute, content, "utf8");
679
+ }
680
+
681
+ function analyze(workspaceRoot: string, dryRun: boolean): BrownfieldImportResult {
682
+ const files = collectSourceFiles(workspaceRoot);
683
+ const inventory = buildInventory(workspaceRoot, files);
684
+ const routes = detectRoutes(files);
685
+ const frontendCalls = detectFrontendCalls(files, routes);
686
+ const candidateEntries = buildCandidates(routes, files);
687
+ const riskReport = buildRiskReport(routes, frontendCalls, candidateEntries, files);
688
+ const migrationPlan = buildMigrationPlan(inventory, routes, frontendCalls, candidateEntries, riskReport);
689
+ const importedAgentContract = buildImportedAgentContract(inventory, routes, frontendCalls, candidateEntries, riskReport);
690
+ if (!dryRun) {
691
+ writeArtifact(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.inventory, inventory);
692
+ writeArtifact(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.routes, routes);
693
+ writeArtifact(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.frontendCalls, frontendCalls);
694
+ writeArtifact(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.candidateEntries, candidateEntries);
695
+ writeArtifact(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.riskReport, riskReport);
696
+ writeArtifact(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.migrationPlan, migrationPlan);
697
+ writeArtifact(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.importedAgentContract, importedAgentContract);
698
+ }
699
+ return {
700
+ schemaVersion: "0.1.0",
701
+ feature: "H49",
702
+ subcommand: "analyze",
703
+ workspaceRoot,
704
+ wroteArtifacts: !dryRun,
705
+ artifacts: BROWNFIELD_IMPORT_ARTIFACTS,
706
+ inventory,
707
+ routes,
708
+ frontendCalls,
709
+ candidateEntries,
710
+ riskReport,
711
+ migrationPlan,
712
+ exitCode: 0,
713
+ };
714
+ }
715
+
716
+ export function inspectBrownfieldImport(workspaceRoot: string): BrownfieldImportResult {
717
+ const inventory = readJson<ImportedInventory>(artifactPath(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.inventory));
718
+ const routes = readJson<ImportedRoute[]>(artifactPath(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.routes)) ?? [];
719
+ const frontendCalls =
720
+ readJson<ImportedFrontendCall[]>(artifactPath(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.frontendCalls)) ?? [];
721
+ const candidateEntries =
722
+ readJson<ImportedCandidateEntry[]>(artifactPath(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.candidateEntries)) ?? [];
723
+ const riskReport = readJson<ImportedRiskReport>(artifactPath(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.riskReport));
724
+ const migrationPlanPath = artifactPath(workspaceRoot, BROWNFIELD_IMPORT_ARTIFACTS.migrationPlan);
725
+ const migrationPlan = existsSync(migrationPlanPath) ? readFileSync(migrationPlanPath, "utf8") : null;
726
+ const missing = !inventory || !riskReport;
727
+ return {
728
+ schemaVersion: "0.1.0",
729
+ feature: "H49",
730
+ subcommand: "inspect",
731
+ workspaceRoot,
732
+ wroteArtifacts: false,
733
+ artifacts: BROWNFIELD_IMPORT_ARTIFACTS,
734
+ inventory,
735
+ routes,
736
+ frontendCalls,
737
+ candidateEntries,
738
+ riskReport,
739
+ migrationPlan,
740
+ exitCode: missing ? 1 : 0,
741
+ failureKind: missing ? "missing_import_artifacts" : undefined,
742
+ };
743
+ }
744
+
745
+ export function runBrownfieldImportCommand(options: BrownfieldImportCommandOptions): BrownfieldImportResult {
746
+ if (options.subcommand === "analyze") {
747
+ return analyze(options.workspaceRoot, options.dryRun);
748
+ }
749
+ const result = inspectBrownfieldImport(options.workspaceRoot);
750
+ if (options.entry) {
751
+ result.candidateEntries = result.candidateEntries.filter((entry) => entry.id === options.entry || entry.name === options.entry);
752
+ }
753
+ if (options.target === "routes") {
754
+ result.frontendCalls = [];
755
+ result.candidateEntries = [];
756
+ } else if (options.target === "frontend-calls") {
757
+ result.routes = [];
758
+ result.candidateEntries = [];
759
+ } else if (options.target === "candidate-entries") {
760
+ result.routes = [];
761
+ result.frontendCalls = [];
762
+ }
763
+ return result;
764
+ }
765
+
766
+ export function formatBrownfieldImportJson(result: BrownfieldImportResult): Record<string, unknown> {
767
+ return {
768
+ schemaVersion: result.schemaVersion,
769
+ feature: result.feature,
770
+ subcommand: result.subcommand,
771
+ workspaceRoot: result.workspaceRoot,
772
+ wroteArtifacts: result.wroteArtifacts,
773
+ artifacts: result.artifacts,
774
+ inventory: result.inventory,
775
+ routes: result.routes,
776
+ frontendCalls: result.frontendCalls,
777
+ candidateEntries: result.candidateEntries,
778
+ riskReport: result.riskReport,
779
+ migrationPlan: result.migrationPlan,
780
+ exitCode: result.exitCode,
781
+ failureKind: result.failureKind ?? null,
782
+ };
783
+ }
784
+
785
+ export function formatBrownfieldImportHuman(result: BrownfieldImportResult): string {
786
+ if (result.exitCode !== 0 && !result.inventory) {
787
+ return "No brownfield import artifacts found. Run `forge import analyze` first.\n";
788
+ }
789
+ const summary = result.riskReport?.summary;
790
+ return [
791
+ `forge import ${result.subcommand}`,
792
+ `artifacts: ${result.wroteArtifacts ? "written" : "read"} at ${IMPORT_DIR}`,
793
+ `files scanned: ${result.inventory?.filesScanned ?? 0}`,
794
+ `routes: ${summary?.routeCount ?? result.routes.length}`,
795
+ `frontend calls: ${summary?.frontendCallCount ?? result.frontendCalls.length}`,
796
+ `candidate entries: ${summary?.candidateCount ?? result.candidateEntries.length}`,
797
+ `hidden from agents: ${summary?.hiddenFromAgents ?? result.candidateEntries.filter((entry) => !entry.visibleToAgent).length}`,
798
+ `needs approval: ${summary?.needsApproval ?? result.candidateEntries.filter((entry) => entry.needsApproval).length}`,
799
+ "",
800
+ ].join("\n");
801
+ }