forgeos 0.1.0-alpha.2 → 0.1.0-alpha.20

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 +199 -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 +212 -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 +1192 -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 +84 -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 +736 -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,649 @@
1
+ import { spawn } from "node:child_process";
2
+ import { join } from "node:path";
3
+ import { createDiagnostic } from "../../compiler/diagnostics/create.ts";
4
+ import {
5
+ FORGE_EXTERNAL_RUNTIME_BAD_RESPONSE,
6
+ FORGE_EXTERNAL_RUNTIME_FAILED,
7
+ FORGE_EXTERNAL_RUNTIME_NOT_FOUND,
8
+ FORGE_EXTERNAL_RUNTIME_UNSUPPORTED,
9
+ } from "../../compiler/diagnostics/codes.ts";
10
+ import { GENERATED_DIR } from "../../compiler/emitter/constants.ts";
11
+ import { nodeFileSystem } from "../../compiler/fs/index.ts";
12
+ import { stripDeterministicHeader } from "../../compiler/primitives/header.ts";
13
+ import type {
14
+ ForgeExternalService,
15
+ ForgeExternalServiceEntry,
16
+ ForgeExternalServiceGraph,
17
+ } from "../../compiler/external-manifest/types.ts";
18
+ import type { Diagnostic } from "../../compiler/types/diagnostic.ts";
19
+ import type { QueryDefinition } from "../../compiler/types/query-registry.ts";
20
+ import type { RuntimeEntry } from "../../compiler/types/runtime-graph.ts";
21
+ import type { AuthContext } from "../auth/types.ts";
22
+ import { snapshotAuth } from "../auth/types.ts";
23
+ import type { DbAdapter } from "../db/adapter.ts";
24
+ import { checkCommandPolicy, checkQueryPolicy } from "../policy/check.ts";
25
+ import { createTelemetryContext } from "../telemetry/context.ts";
26
+ import { generateRequestId, generateTraceId } from "../telemetry/correlation.ts";
27
+
28
+ export interface ExternalRuntimeCall {
29
+ kind: "command" | "query";
30
+ serviceName: string;
31
+ entryName: string;
32
+ args?: unknown;
33
+ auth?: AuthContext;
34
+ requestHeaders?: Headers;
35
+ requestId?: string;
36
+ }
37
+
38
+ export interface ExternalRuntimeResult {
39
+ ok: boolean;
40
+ result?: unknown;
41
+ diagnostics: Diagnostic[];
42
+ traceId: string;
43
+ exitCode: 0 | 1;
44
+ service?: ForgeExternalService;
45
+ entry?: ForgeExternalServiceEntry;
46
+ }
47
+
48
+ interface ExternalEnvelope {
49
+ ok?: boolean;
50
+ result?: unknown;
51
+ diagnostics?: Diagnostic[];
52
+ error?: { code?: string; message?: string; details?: unknown };
53
+ traceId?: string;
54
+ }
55
+
56
+ function readGeneratedJson<T>(workspaceRoot: string, relative: string): T | null {
57
+ const absolute = join(workspaceRoot, relative);
58
+ if (!nodeFileSystem.exists(absolute)) {
59
+ return null;
60
+ }
61
+ const raw = stripDeterministicHeader(nodeFileSystem.readText(absolute) ?? "");
62
+ return JSON.parse(raw) as T;
63
+ }
64
+
65
+ export function loadExternalServiceGraph(
66
+ workspaceRoot: string,
67
+ ): { graph: ForgeExternalServiceGraph | null; diagnostics: Diagnostic[] } {
68
+ const graph = readGeneratedJson<ForgeExternalServiceGraph>(
69
+ workspaceRoot,
70
+ `${GENERATED_DIR}/externalServices.json`,
71
+ );
72
+ if (!graph) {
73
+ return {
74
+ graph: null,
75
+ diagnostics: [
76
+ createDiagnostic({
77
+ severity: "error",
78
+ code: FORGE_EXTERNAL_RUNTIME_NOT_FOUND,
79
+ message: `missing ${GENERATED_DIR}/externalServices.json; run forge generate first`,
80
+ file: `${GENERATED_DIR}/externalServices.json`,
81
+ docs: ["docs/forge-protocol.md"],
82
+ }),
83
+ ],
84
+ };
85
+ }
86
+ return { graph, diagnostics: graph.diagnostics ?? [] };
87
+ }
88
+
89
+ function findExternalEntry(
90
+ graph: ForgeExternalServiceGraph,
91
+ serviceName: string,
92
+ entryName: string,
93
+ kind: "command" | "query",
94
+ ): { service: ForgeExternalService; entry: ForgeExternalServiceEntry } | null {
95
+ const service = graph.services.find((candidate) => candidate.name === serviceName);
96
+ const entry = service?.entries.find(
97
+ (candidate) => candidate.name === entryName && candidate.kind === kind,
98
+ );
99
+ return service && entry ? { service, entry } : null;
100
+ }
101
+
102
+ export function parseExternalQualifiedName(name: string): {
103
+ serviceName: string;
104
+ entryName: string;
105
+ } | null {
106
+ const [serviceName, ...entryParts] = name.split(".");
107
+ const entryName = entryParts.join(".");
108
+ return serviceName && entryName ? { serviceName, entryName } : null;
109
+ }
110
+
111
+ export function resolveExternalQualifiedName(
112
+ workspaceRoot: string,
113
+ qualifiedName: string,
114
+ kind: "command" | "query",
115
+ ): { service: ForgeExternalService; entry: ForgeExternalServiceEntry; serviceName: string; entryName: string } | null {
116
+ const parsed = parseExternalQualifiedName(qualifiedName);
117
+ if (!parsed) {
118
+ return null;
119
+ }
120
+ const loaded = loadExternalServiceGraph(workspaceRoot);
121
+ if (!loaded.graph) {
122
+ return null;
123
+ }
124
+ const found = findExternalEntry(loaded.graph, parsed.serviceName, parsed.entryName, kind);
125
+ return found ? { ...found, ...parsed } : null;
126
+ }
127
+
128
+ function runtimeEntryFor(entry: ForgeExternalServiceEntry): RuntimeEntry {
129
+ return {
130
+ id: `external:${entry.service}:${entry.kind}:${entry.name}`,
131
+ kind: "command",
132
+ name: `${entry.service}.${entry.name}`,
133
+ qualifiedName: `${entry.service}.${entry.name}`,
134
+ file: `external:${entry.service}`,
135
+ moduleId: `external:${entry.service}`,
136
+ runtimeContext: "command",
137
+ dependencies: [],
138
+ };
139
+ }
140
+
141
+ function queryDefinitionFor(entry: ForgeExternalServiceEntry): QueryDefinition {
142
+ return {
143
+ name: `${entry.service}.${entry.name}`,
144
+ qualifiedName: `${entry.service}.${entry.name}`,
145
+ file: `external:${entry.service}`,
146
+ symbolId: `external:${entry.service}:query:${entry.name}`,
147
+ moduleId: `external:${entry.service}`,
148
+ };
149
+ }
150
+
151
+ function authHeaders(auth: AuthContext | undefined, traceId: string): Record<string, string> {
152
+ const headers: Record<string, string> = {
153
+ "x-forge-trace-id": traceId,
154
+ "x-forge-auth-kind": auth?.kind ?? "anonymous",
155
+ };
156
+ if (auth?.kind === "user") {
157
+ headers["x-forge-user-id"] = auth.userId;
158
+ if (auth.tenantId) headers["x-forge-tenant-id"] = auth.tenantId;
159
+ if (auth.role) headers["x-forge-role"] = auth.role;
160
+ if (auth.roles) headers["x-forge-roles"] = JSON.stringify(auth.roles);
161
+ if (auth.permissions) headers["x-forge-permissions"] = JSON.stringify(auth.permissions);
162
+ }
163
+ if (auth?.kind === "system" && auth.tenantId) {
164
+ headers["x-forge-tenant-id"] = auth.tenantId;
165
+ }
166
+ return headers;
167
+ }
168
+
169
+ function inboundAuthHeader(headers: Headers | undefined): string | null {
170
+ return headers?.get("authorization") ?? headers?.get("Authorization") ?? null;
171
+ }
172
+
173
+ function normalizeExternalPayload(
174
+ body: unknown,
175
+ responseOk: boolean,
176
+ status: number,
177
+ fallbackTraceId: string,
178
+ ): ExternalRuntimeResult {
179
+ const envelope = body && typeof body === "object" ? (body as ExternalEnvelope) : {};
180
+ const diagnostics = Array.isArray(envelope.diagnostics) ? envelope.diagnostics : [];
181
+ const traceId = envelope.traceId ?? fallbackTraceId;
182
+
183
+ if (envelope.ok === false || !responseOk) {
184
+ const message =
185
+ envelope.error?.message ??
186
+ diagnostics.find((diagnostic) => diagnostic.message)?.message ??
187
+ `external runtime returned HTTP ${status}`;
188
+ return {
189
+ ok: false,
190
+ diagnostics: [
191
+ ...diagnostics,
192
+ createDiagnostic({
193
+ severity: "error",
194
+ code: envelope.error?.code ?? FORGE_EXTERNAL_RUNTIME_FAILED,
195
+ message,
196
+ docs: ["docs/forge-protocol.md"],
197
+ }),
198
+ ],
199
+ traceId,
200
+ exitCode: 1,
201
+ };
202
+ }
203
+
204
+ return {
205
+ ok: true,
206
+ result: "result" in envelope ? envelope.result : body,
207
+ diagnostics,
208
+ traceId,
209
+ exitCode: 0,
210
+ };
211
+ }
212
+
213
+ function externalRequestBody(call: ExternalRuntimeCall, traceId: string): unknown {
214
+ return {
215
+ args: call.args ?? {},
216
+ auth: snapshotAuth(call.auth ?? { kind: "anonymous" }),
217
+ forge: {
218
+ service: call.serviceName,
219
+ entry: call.entryName,
220
+ kind: call.kind,
221
+ traceId,
222
+ },
223
+ };
224
+ }
225
+
226
+ function withArgsSearchParams(url: URL, args: unknown): URL {
227
+ url.searchParams.set("args", JSON.stringify(args ?? {}));
228
+ return url;
229
+ }
230
+
231
+ async function runHttpExternalEntry(
232
+ service: ForgeExternalService,
233
+ entry: ForgeExternalServiceEntry,
234
+ call: ExternalRuntimeCall,
235
+ traceId: string,
236
+ ): Promise<ExternalRuntimeResult> {
237
+ if (!service.baseUrl) {
238
+ return {
239
+ ok: false,
240
+ diagnostics: [
241
+ createDiagnostic({
242
+ severity: "error",
243
+ code: FORGE_EXTERNAL_RUNTIME_UNSUPPORTED,
244
+ message: `external service '${service.name}' uses http transport but has no baseUrl`,
245
+ file: `external:${service.name}`,
246
+ docs: ["docs/forge-protocol.md"],
247
+ }),
248
+ ],
249
+ traceId,
250
+ exitCode: 1,
251
+ service,
252
+ entry,
253
+ };
254
+ }
255
+
256
+ const method = entry.method ?? "POST";
257
+ const url = new URL(entry.path ?? `/${entry.kind}s/${entry.name}`, service.baseUrl);
258
+ if (method === "GET") {
259
+ withArgsSearchParams(url, call.args);
260
+ }
261
+
262
+ const authorization = inboundAuthHeader(call.requestHeaders);
263
+ const response = await fetch(url, {
264
+ method,
265
+ headers: {
266
+ "content-type": "application/json",
267
+ "x-forge-external-service": service.name,
268
+ "x-forge-external-entry": entry.name,
269
+ "x-forge-external-kind": entry.kind,
270
+ ...authHeaders(call.auth, traceId),
271
+ ...(authorization ? { authorization } : {}),
272
+ },
273
+ body: method === "GET" ? undefined : JSON.stringify(externalRequestBody(call, traceId)),
274
+ });
275
+
276
+ let body: unknown;
277
+ try {
278
+ body = await response.json();
279
+ } catch {
280
+ return {
281
+ ok: false,
282
+ diagnostics: [
283
+ createDiagnostic({
284
+ severity: "error",
285
+ code: FORGE_EXTERNAL_RUNTIME_BAD_RESPONSE,
286
+ message: `external ${entry.kind} '${service.name}.${entry.name}' returned non-JSON response`,
287
+ file: `external:${service.name}`,
288
+ docs: ["docs/forge-protocol.md"],
289
+ }),
290
+ ],
291
+ traceId,
292
+ exitCode: 1,
293
+ service,
294
+ entry,
295
+ };
296
+ }
297
+
298
+ return {
299
+ ...normalizeExternalPayload(body, response.ok, response.status, traceId),
300
+ service,
301
+ entry,
302
+ };
303
+ }
304
+
305
+ export function parseExternalCommandLine(command: string): string[] {
306
+ return parseExternalCommandLineDetailed(command).args;
307
+ }
308
+
309
+ export interface ExternalCommandLineParseResult {
310
+ args: string[];
311
+ diagnostics: string[];
312
+ }
313
+
314
+ export function parseExternalCommandLineDetailed(command: string): ExternalCommandLineParseResult {
315
+ const parts: string[] = [];
316
+ let current = "";
317
+ let quote: "'" | "\"" | null = null;
318
+ let escaping = false;
319
+ let tokenStarted = false;
320
+ const diagnostics: string[] = [];
321
+
322
+ const push = () => {
323
+ if (tokenStarted || current.length > 0) {
324
+ parts.push(current);
325
+ current = "";
326
+ tokenStarted = false;
327
+ }
328
+ };
329
+
330
+ for (const char of command) {
331
+ if (escaping) {
332
+ current += char;
333
+ tokenStarted = true;
334
+ escaping = false;
335
+ continue;
336
+ }
337
+ if (quote === "'") {
338
+ if (char === "'") {
339
+ quote = null;
340
+ } else {
341
+ current += char;
342
+ tokenStarted = true;
343
+ }
344
+ continue;
345
+ }
346
+ if (quote === "\"") {
347
+ if (char === "\"") {
348
+ quote = null;
349
+ } else if (char === "\\") {
350
+ escaping = true;
351
+ } else {
352
+ current += char;
353
+ tokenStarted = true;
354
+ }
355
+ continue;
356
+ }
357
+ if (char === "\\") {
358
+ escaping = true;
359
+ continue;
360
+ }
361
+ if (char === "'" || char === "\"") {
362
+ quote = char;
363
+ tokenStarted = true;
364
+ continue;
365
+ }
366
+ if (/\s/.test(char)) {
367
+ push();
368
+ continue;
369
+ }
370
+ current += char;
371
+ tokenStarted = true;
372
+ }
373
+ if (escaping) {
374
+ current += "\\";
375
+ tokenStarted = true;
376
+ diagnostics.push("stdio command ends with a trailing escape");
377
+ }
378
+ if (quote) {
379
+ diagnostics.push(`stdio command has an unterminated ${quote === "'" ? "single" : "double"} quote`);
380
+ }
381
+ push();
382
+ return { args: parts, diagnostics };
383
+ }
384
+
385
+ async function runStdioExternalEntry(
386
+ service: ForgeExternalService,
387
+ entry: ForgeExternalServiceEntry,
388
+ call: ExternalRuntimeCall,
389
+ traceId: string,
390
+ ): Promise<ExternalRuntimeResult> {
391
+ if (!service.command && !service.commandArgs?.length) {
392
+ return {
393
+ ok: false,
394
+ diagnostics: [
395
+ createDiagnostic({
396
+ severity: "error",
397
+ code: FORGE_EXTERNAL_RUNTIME_UNSUPPORTED,
398
+ message: `external service '${service.name}' uses stdio transport but has no command`,
399
+ file: `external:${service.name}`,
400
+ docs: ["docs/forge-protocol.md"],
401
+ }),
402
+ ],
403
+ traceId,
404
+ exitCode: 1,
405
+ service,
406
+ entry,
407
+ };
408
+ }
409
+
410
+ const parsedCommand = service.commandArgs?.length
411
+ ? { args: service.commandArgs, diagnostics: [] }
412
+ : parseExternalCommandLineDetailed(service.command ?? "");
413
+ if (parsedCommand.diagnostics.length > 0) {
414
+ return {
415
+ ok: false,
416
+ diagnostics: [
417
+ createDiagnostic({
418
+ severity: "error",
419
+ code: FORGE_EXTERNAL_RUNTIME_UNSUPPORTED,
420
+ message: `external service '${service.name}' has an invalid stdio command: ${parsedCommand.diagnostics.join("; ")}`,
421
+ file: `external:${service.name}`,
422
+ docs: ["docs/forge-protocol.md"],
423
+ }),
424
+ ],
425
+ traceId,
426
+ exitCode: 1,
427
+ service,
428
+ entry,
429
+ };
430
+ }
431
+ const [executable, ...args] = parsedCommand.args;
432
+ if (!executable) {
433
+ return {
434
+ ok: false,
435
+ diagnostics: [
436
+ createDiagnostic({
437
+ severity: "error",
438
+ code: FORGE_EXTERNAL_RUNTIME_UNSUPPORTED,
439
+ message: `external service '${service.name}' has an empty stdio command`,
440
+ file: `external:${service.name}`,
441
+ }),
442
+ ],
443
+ traceId,
444
+ exitCode: 1,
445
+ service,
446
+ entry,
447
+ };
448
+ }
449
+
450
+ return new Promise((resolve) => {
451
+ const child = spawn(executable, args, {
452
+ stdio: ["pipe", "pipe", "pipe"],
453
+ windowsHide: true,
454
+ });
455
+ let stdout = "";
456
+ let stderr = "";
457
+ child.stdout.setEncoding("utf8");
458
+ child.stderr.setEncoding("utf8");
459
+ child.stdout.on("data", (chunk) => { stdout += chunk; });
460
+ child.stderr.on("data", (chunk) => { stderr += chunk; });
461
+ child.on("error", (error) => {
462
+ resolve({
463
+ ok: false,
464
+ diagnostics: [
465
+ createDiagnostic({
466
+ severity: "error",
467
+ code: FORGE_EXTERNAL_RUNTIME_FAILED,
468
+ message: `external stdio command failed to start: ${error.message}`,
469
+ file: `external:${service.name}`,
470
+ }),
471
+ ],
472
+ traceId,
473
+ exitCode: 1,
474
+ service,
475
+ entry,
476
+ });
477
+ });
478
+ child.on("close", (code) => {
479
+ try {
480
+ const parsed = stdout.trim().length > 0 ? JSON.parse(stdout) : {};
481
+ resolve({
482
+ ...normalizeExternalPayload(parsed, code === 0, code ?? 1, traceId),
483
+ service,
484
+ entry,
485
+ });
486
+ } catch {
487
+ resolve({
488
+ ok: false,
489
+ diagnostics: [
490
+ createDiagnostic({
491
+ severity: "error",
492
+ code: FORGE_EXTERNAL_RUNTIME_BAD_RESPONSE,
493
+ message: `external stdio command returned invalid JSON${stderr ? `: ${stderr}` : ""}`,
494
+ file: `external:${service.name}`,
495
+ }),
496
+ ],
497
+ traceId,
498
+ exitCode: 1,
499
+ service,
500
+ entry,
501
+ });
502
+ }
503
+ });
504
+ child.stdin.end(JSON.stringify(externalRequestBody(call, traceId)));
505
+ });
506
+ }
507
+
508
+ export async function runExternalEntry(
509
+ workspaceRoot: string,
510
+ call: ExternalRuntimeCall,
511
+ runtime?: { adapter?: DbAdapter | null },
512
+ ): Promise<ExternalRuntimeResult> {
513
+ const traceId = generateTraceId();
514
+ const loaded = loadExternalServiceGraph(workspaceRoot);
515
+ if (!loaded.graph) {
516
+ return {
517
+ ok: false,
518
+ diagnostics: loaded.diagnostics,
519
+ traceId,
520
+ exitCode: 1,
521
+ };
522
+ }
523
+
524
+ const found = findExternalEntry(
525
+ loaded.graph,
526
+ call.serviceName,
527
+ call.entryName,
528
+ call.kind,
529
+ );
530
+ if (!found) {
531
+ return {
532
+ ok: false,
533
+ diagnostics: [
534
+ ...loaded.diagnostics,
535
+ createDiagnostic({
536
+ severity: "error",
537
+ code: FORGE_EXTERNAL_RUNTIME_NOT_FOUND,
538
+ message: `external ${call.kind} '${call.serviceName}.${call.entryName}' not found`,
539
+ file: `${GENERATED_DIR}/externalServices.json`,
540
+ }),
541
+ ],
542
+ traceId,
543
+ exitCode: 1,
544
+ };
545
+ }
546
+
547
+ const { service, entry } = found;
548
+ const telemetry = runtime?.adapter
549
+ ? createTelemetryContext({
550
+ adapter: runtime.adapter,
551
+ traceId,
552
+ requestId: call.requestId ?? generateRequestId(),
553
+ runtime: { kind: call.kind, name: `${service.name}.${entry.name}` },
554
+ bufferInTransaction: false,
555
+ workspaceRoot,
556
+ })
557
+ : undefined;
558
+
559
+ const policyCheck = call.kind === "command"
560
+ ? await checkCommandPolicy({
561
+ workspaceRoot,
562
+ entry: runtimeEntryFor(entry),
563
+ auth: call.auth ?? { kind: "anonymous" },
564
+ telemetry,
565
+ })
566
+ : await checkQueryPolicy({
567
+ workspaceRoot,
568
+ query: queryDefinitionFor(entry),
569
+ auth: call.auth ?? { kind: "anonymous" },
570
+ telemetry,
571
+ });
572
+ if (!policyCheck.allowed) {
573
+ return {
574
+ ok: false,
575
+ diagnostics: [...loaded.diagnostics, ...policyCheck.diagnostics],
576
+ traceId,
577
+ exitCode: 1,
578
+ service,
579
+ entry,
580
+ };
581
+ }
582
+
583
+ await telemetry?.capture("forge.external.started", {
584
+ service: service.name,
585
+ entry: entry.name,
586
+ kind: entry.kind,
587
+ transport: service.transport,
588
+ });
589
+
590
+ try {
591
+ const result = service.transport === "http"
592
+ ? await runHttpExternalEntry(service, entry, call, traceId)
593
+ : service.transport === "stdio"
594
+ ? await runStdioExternalEntry(service, entry, call, traceId)
595
+ : {
596
+ ok: false,
597
+ diagnostics: [
598
+ createDiagnostic({
599
+ severity: "error",
600
+ code: FORGE_EXTERNAL_RUNTIME_UNSUPPORTED,
601
+ message: `external transport '${service.transport}' is not executable by this Forge runtime`,
602
+ file: `external:${service.name}`,
603
+ docs: ["docs/forge-protocol.md"],
604
+ }),
605
+ ],
606
+ traceId,
607
+ exitCode: 1 as const,
608
+ service,
609
+ entry,
610
+ };
611
+
612
+ await telemetry?.capture(result.ok ? "forge.external.completed" : "forge.external.failed", {
613
+ service: service.name,
614
+ entry: entry.name,
615
+ kind: entry.kind,
616
+ transport: service.transport,
617
+ });
618
+
619
+ return {
620
+ ...result,
621
+ diagnostics: [...loaded.diagnostics, ...result.diagnostics],
622
+ };
623
+ } catch (error) {
624
+ const message = error instanceof Error ? error.message : "external runtime failed";
625
+ await telemetry?.capture("forge.external.failed", {
626
+ service: service.name,
627
+ entry: entry.name,
628
+ kind: entry.kind,
629
+ error: message,
630
+ });
631
+ return {
632
+ ok: false,
633
+ diagnostics: [
634
+ ...loaded.diagnostics,
635
+ createDiagnostic({
636
+ severity: "error",
637
+ code: FORGE_EXTERNAL_RUNTIME_FAILED,
638
+ message,
639
+ file: `external:${service.name}`,
640
+ docs: ["docs/forge-protocol.md"],
641
+ }),
642
+ ],
643
+ traceId,
644
+ exitCode: 1,
645
+ service,
646
+ entry,
647
+ };
648
+ }
649
+ }
@@ -1,5 +1,3 @@
1
- import { join } from "node:path";
2
- import { pathToFileURL } from "node:url";
3
1
  import { createDiagnostic } from "../../compiler/diagnostics/create.ts";
4
2
  import {
5
3
  FORGE_DB_ADAPTER_UNAVAILABLE,
@@ -43,6 +41,7 @@ export interface RunEntryRuntime {
43
41
  export interface ResolvedHandler {
44
42
  invoke: (args: unknown) => Promise<unknown>;
45
43
  usesContext: boolean;
44
+ handler?: (ctx: unknown, args: unknown) => unknown | Promise<unknown>;
46
45
  }
47
46
 
48
47
  function needsDatabaseHint(
@@ -83,6 +82,7 @@ export function resolveHandlerFromModule(
83
82
 
84
83
  return {
85
84
  usesContext: true,
85
+ handler,
86
86
  invoke: async (args) => {
87
87
  if (runtime?.adapter && runtime.tableMap) {
88
88
  const tx = adapterAsTransaction(runtime.adapter);
@@ -162,11 +162,20 @@ export async function executeResolvedEntry(
162
162
  runtime?.adapter &&
163
163
  runtime.tableMap
164
164
  ) {
165
- const absolutePath = join(workspaceRoot, entry.file);
166
- const mod = (await import(pathToFileURL(absolutePath).href)) as Record<string, unknown>;
167
- const exported = mod[entry.name];
168
- const handler = (exported as { handler: (ctx: unknown, args: unknown) => unknown })
169
- .handler;
165
+ const handler = resolved.handler;
166
+ if (!handler) {
167
+ return {
168
+ ok: false,
169
+ diagnostics: [
170
+ createDiagnostic({
171
+ severity: "error",
172
+ code: FORGE_RUNTIME_NOT_FOUND,
173
+ message: `runtime entry '${entry.name}' is missing a command handler`,
174
+ file: entry.file,
175
+ }),
176
+ ],
177
+ };
178
+ }
170
179
 
171
180
  return runCommandWithTransaction(
172
181
  entry,