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,498 @@
1
+ import { normalizePath } from "../compiler/primitives/paths.ts";
2
+ import { hashStable } from "../compiler/primitives/hash.ts";
3
+ import { readDeltaGitSnapshot } from "../delta/git-observer.ts";
4
+ import { redactDeltaPayload } from "../delta/redaction.ts";
5
+ import { redactAgentPayload } from "./redaction.ts";
6
+ import {
7
+ AGENT_EVENT_SCHEMA,
8
+ type AgentEventEnvelope,
9
+ type AgentMemoryIntegrationKind,
10
+ type AgentMemorySourceName,
11
+ type AgentMemoryTrustLevel,
12
+ } from "./types.ts";
13
+
14
+ export interface NormalizeAgentEventInput {
15
+ workspaceRoot: string;
16
+ source: AgentMemorySourceName | string;
17
+ eventName?: string;
18
+ integration?: AgentMemoryIntegrationKind | string;
19
+ raw: Record<string, unknown>;
20
+ now?: string;
21
+ }
22
+
23
+ export function normalizeAgentEvent(input: NormalizeAgentEventInput): AgentEventEnvelope {
24
+ const source = normalizeSource(input.source);
25
+ const rawEventName = input.eventName ?? stringField(input.raw, "hook_event_name") ?? stringField(input.raw, "event") ?? "unknown";
26
+ const normalizedKind = normalizeEventKind(source, rawEventName, input.raw);
27
+ const redacted = redactAgentPayload(input.raw);
28
+ const derived = deriveSafeHookMetadata(input.raw, rawEventName);
29
+ const git = readDeltaGitSnapshot(input.workspaceRoot);
30
+ const actorName = source === "claude-code"
31
+ ? "Claude Code"
32
+ : source === "codex"
33
+ ? "Codex"
34
+ : source === "cursor"
35
+ ? "Cursor"
36
+ : "External Agent";
37
+ return {
38
+ schema: AGENT_EVENT_SCHEMA,
39
+ source: {
40
+ agent: source,
41
+ integration: input.integration ?? defaultIntegrationForSource(source),
42
+ version: stringField(input.raw, "version") ?? "unknown",
43
+ },
44
+ workspace: {
45
+ root: normalizePath(input.workspaceRoot),
46
+ gitBranch: git.branch,
47
+ gitHead: git.head,
48
+ },
49
+ session: {
50
+ externalSessionId: stringField(input.raw, "session_id") ?? stringField(input.raw, "sessionId") ?? stringField(input.raw, "conversation_id"),
51
+ forgeSessionId: stringField(input.raw, "forgeSessionId"),
52
+ turnId: stringField(input.raw, "turn_id") ?? stringField(input.raw, "turnId"),
53
+ },
54
+ event: {
55
+ kind: normalizedKind,
56
+ timestamp: stringField(input.raw, "timestamp") ?? input.now ?? new Date().toISOString(),
57
+ },
58
+ actor: {
59
+ kind: "agent",
60
+ name: stringField(input.raw, "actor") ?? actorName,
61
+ model: stringField(input.raw, "model"),
62
+ },
63
+ payload: {
64
+ ...redacted.value,
65
+ ...derived,
66
+ },
67
+ privacy: {
68
+ rawPromptStored: false,
69
+ rawCompletionStored: false,
70
+ rawToolArgsStored: false,
71
+ transcriptImported: false,
72
+ redacted: true,
73
+ sensitiveFieldsRemoved: redacted.sensitiveFieldsRemoved,
74
+ },
75
+ capture: {
76
+ trustLevel: trustLevelForIntegration(input.integration ?? defaultIntegrationForSource(source)),
77
+ confidence: confidenceForSource(source, input.integration ?? defaultIntegrationForSource(source)),
78
+ },
79
+ };
80
+ }
81
+
82
+ export function summarizeAgentEvent(envelope: AgentEventEnvelope): string {
83
+ const tool = stringField(envelope.payload, "toolName") ??
84
+ stringField(envelope.payload, "tool_name") ??
85
+ stringField(envelope.payload, "tool");
86
+ const command = stringField(envelope.payload, "command");
87
+ const commandSummary = stringField(envelope.payload, "commandSummary");
88
+ const resultStatus = stringField(envelope.payload, "resultStatus");
89
+ const promptSummary = stringField(envelope.payload, "promptSummary") ?? stringField(envelope.payload, "userPromptSummary");
90
+ if (envelope.event.kind === "agent.prompt.submitted" && promptSummary) {
91
+ return promptSummary;
92
+ }
93
+ if (envelope.event.kind === "approval.requested") {
94
+ return `${envelope.source.agent} requested approval${tool ? ` for ${tool}` : ""}${commandSummary ? `: ${commandSummary}` : ""}`;
95
+ }
96
+ if (tool) {
97
+ const status = resultStatus ?? statusFromKind(envelope.event.kind);
98
+ return `${envelope.source.agent} ${tool}${status ? ` ${status}` : ""}${commandSummary ? `: ${commandSummary}` : ""}`;
99
+ }
100
+ if (commandSummary ?? command) {
101
+ return `${envelope.source.agent} ran ${commandSummary ?? command}`;
102
+ }
103
+ return `${envelope.source.agent} ${envelope.event.kind}`;
104
+ }
105
+
106
+ export function extractAgentEventBindings(envelope: AgentEventEnvelope): {
107
+ toolName?: string;
108
+ command?: string;
109
+ exitCode?: number;
110
+ files: string[];
111
+ entries: string[];
112
+ proofs: string[];
113
+ status?: string;
114
+ } {
115
+ const payload = envelope.payload;
116
+ return {
117
+ toolName: stringField(payload, "toolName") ?? stringField(payload, "tool_name") ?? stringField(payload, "tool"),
118
+ command: stringField(payload, "command") ?? stringField(payload, "commandSummary"),
119
+ exitCode: numberField(payload, "exitCode") ?? numberField(payload, "exit_code"),
120
+ files: uniqueStrings([
121
+ ...arrayOfStrings(payload.files),
122
+ ...arrayOfStrings(payload.paths),
123
+ ...arrayOfStrings(payload.affectedFiles),
124
+ stringField(payload, "file_path"),
125
+ stringField(payload, "filePath"),
126
+ stringField(payload, "path"),
127
+ ].map((value) => value ? normalizePath(value) : value)),
128
+ entries: uniqueStrings([
129
+ ...arrayOfStrings(payload.entries),
130
+ stringField(payload, "entryName"),
131
+ stringField(payload, "entry_name"),
132
+ stringField(payload, "runtimeEntry"),
133
+ stringField(payload, "runtime_entry"),
134
+ ]),
135
+ proofs: uniqueStrings([
136
+ ...arrayOfStrings(payload.proofs),
137
+ stringField(payload, "proofKind"),
138
+ stringField(payload, "proof_kind"),
139
+ ]),
140
+ status: statusFromKind(envelope.event.kind),
141
+ };
142
+ }
143
+
144
+ function deriveSafeHookMetadata(raw: Record<string, unknown>, eventName: string): Record<string, unknown> {
145
+ const derived: Record<string, unknown> = {};
146
+ const toolName = stringField(raw, "tool_name") ?? stringField(raw, "toolName") ?? stringField(raw, "tool");
147
+ const toolUseId = stringField(raw, "tool_use_id") ?? stringField(raw, "toolUseId");
148
+ const permissionMode = stringField(raw, "permission_mode") ?? stringField(raw, "permissionMode");
149
+ const cwd = stringField(raw, "cwd");
150
+ const trigger = stringField(raw, "trigger");
151
+ const agentId = stringField(raw, "agent_id") ?? stringField(raw, "agentId");
152
+ const agentType = stringField(raw, "agent_type") ?? stringField(raw, "agentType");
153
+ if (toolName) {
154
+ derived.toolName = toolName;
155
+ }
156
+ if (toolUseId) {
157
+ derived.toolUseId = toolUseId;
158
+ }
159
+ if (permissionMode) {
160
+ derived.permissionMode = permissionMode;
161
+ }
162
+ if (cwd) {
163
+ derived.cwd = normalizePath(cwd);
164
+ }
165
+ if (trigger) {
166
+ derived.trigger = trigger;
167
+ }
168
+ if (agentId) {
169
+ derived.agentId = agentId;
170
+ }
171
+ if (agentType) {
172
+ derived.agentType = agentType;
173
+ }
174
+
175
+ const toolInput = objectField(raw, "tool_input") ?? objectField(raw, "toolInput");
176
+ const toolResponse = objectField(raw, "tool_response") ?? objectField(raw, "toolResponse");
177
+ const command = stringField(toolInput, "command") ?? stringField(raw, "command");
178
+ if (command) {
179
+ derived.commandHash = hashStable(command);
180
+ derived.commandStored = false;
181
+ derived.commandSummary = summarizeCommand(command);
182
+ derived.commandKind = classifyCommand(toolName, command);
183
+ }
184
+ const description = stringField(toolInput, "description");
185
+ if (description) {
186
+ derived.approvalDescriptionSummary = safeSummary(description, 180);
187
+ }
188
+ const exitCode = numberField(toolResponse, "exitCode") ?? numberField(toolResponse, "exit_code") ?? numberField(raw, "exitCode") ?? numberField(raw, "exit_code");
189
+ if (exitCode !== undefined) {
190
+ derived.exitCode = exitCode;
191
+ derived.resultStatus = exitCode === 0 ? "success" : "failed";
192
+ } else {
193
+ const status = stringField(toolResponse, "status") ?? stringField(raw, "status");
194
+ if (status) {
195
+ derived.resultStatus = status;
196
+ }
197
+ }
198
+ const responseSummary = summarizeToolResponse(toolResponse);
199
+ if (responseSummary) {
200
+ derived.responseSummary = responseSummary;
201
+ }
202
+ if (toolResponse) {
203
+ derived.responseHash = hashStable(JSON.stringify(toolResponse));
204
+ derived.responseStored = false;
205
+ }
206
+
207
+ const files = uniqueStrings([
208
+ ...extractPaths(toolInput),
209
+ ...extractPaths(toolResponse),
210
+ ...extractPaths(raw),
211
+ ...extractPathsFromCommand(command),
212
+ ].map((value) => normalizePath(value)));
213
+ if (files.length > 0) {
214
+ derived.files = files;
215
+ derived.affectedFiles = files;
216
+ }
217
+ const entries = uniqueStrings([
218
+ ...extractNamedValues(toolInput, ["entry", "entryName", "entry_name", "runtimeEntry", "runtime_entry"]),
219
+ ...extractNamedValues(raw, ["entry", "entryName", "entry_name", "runtimeEntry", "runtime_entry"]),
220
+ ...extractEntriesFromCommand(command),
221
+ ]);
222
+ if (entries.length > 0) {
223
+ derived.entries = entries;
224
+ }
225
+ if (eventName === "Stop" || eventName === "SubagentStop") {
226
+ const lastMessage = stringField(raw, "last_assistant_message") ?? stringField(raw, "lastAssistantMessage");
227
+ if (lastMessage) {
228
+ derived.lastAssistantMessageHash = hashStable(lastMessage);
229
+ derived.lastAssistantMessageStored = false;
230
+ derived.lastAssistantMessageSummary = safeSummary(lastMessage, 220);
231
+ }
232
+ }
233
+ return derived;
234
+ }
235
+
236
+ function normalizeSource(source: string): AgentMemorySourceName | string {
237
+ if (source === "claude" || source === "claude-code") {
238
+ return "claude-code";
239
+ }
240
+ if (source === "codex" || source === "cursor") {
241
+ return source;
242
+ }
243
+ return source || "generic";
244
+ }
245
+
246
+ function defaultIntegrationForSource(source: string): AgentMemoryIntegrationKind {
247
+ return source === "cursor" ? "mcp" : "native-hook";
248
+ }
249
+
250
+ function trustLevelForIntegration(integration: string): AgentMemoryTrustLevel {
251
+ if (integration === "native-hook") {
252
+ return "direct-hook";
253
+ }
254
+ if (integration === "mcp") {
255
+ return "mcp-tool";
256
+ }
257
+ if (integration === "cli-wrapper") {
258
+ return "forge-command";
259
+ }
260
+ if (integration === "file-watcher") {
261
+ return "file-watcher";
262
+ }
263
+ if (integration === "git-hook") {
264
+ return "git-observer";
265
+ }
266
+ return "manual-import";
267
+ }
268
+
269
+ function confidenceForSource(source: string, integration: string): number {
270
+ if (integration === "native-hook") {
271
+ return source === "generic" ? 0.78 : 0.94;
272
+ }
273
+ if (integration === "mcp") {
274
+ return 0.9;
275
+ }
276
+ if (integration === "cli-wrapper") {
277
+ return 0.84;
278
+ }
279
+ return 0.68;
280
+ }
281
+
282
+ function normalizeEventKind(source: string, eventName: string, raw: Record<string, unknown>): string {
283
+ const canonical = eventName.replace(/\s+/g, "");
284
+ switch (canonical) {
285
+ case "SessionStart":
286
+ return "agent.session.started";
287
+ case "SessionEnd":
288
+ return "agent.session.ended";
289
+ case "UserPromptSubmit":
290
+ return "agent.prompt.submitted";
291
+ case "PreToolUse":
292
+ return "agent.tool.requested";
293
+ case "PermissionRequest":
294
+ return "approval.requested";
295
+ case "PermissionDenied":
296
+ return "approval.denied";
297
+ case "PostToolUseFailure":
298
+ return "agent.tool.failed";
299
+ case "PostToolUse":
300
+ return "agent.tool.completed";
301
+ case "SubagentStart":
302
+ return "agent.subagent.started";
303
+ case "SubagentStop":
304
+ return "agent.subagent.ended";
305
+ case "PreCompact":
306
+ return "agent.memory.compaction.requested";
307
+ case "PostCompact":
308
+ return "agent.memory.compaction.completed";
309
+ case "FileChanged":
310
+ return "agent.file.changed";
311
+ case "Stop":
312
+ return source === "codex" ? "agent.turn.stopped" : "agent.turn.completed";
313
+ default: {
314
+ const tool = stringField(raw, "toolName") ?? stringField(raw, "tool_name");
315
+ if (tool && (canonical === "tool.call" || canonical === "tool_call" || canonical === "unknown")) {
316
+ return "agent.tool.called";
317
+ }
318
+ return canonical.includes(".") ? canonical : `agent.${canonical || "event"}`;
319
+ }
320
+ }
321
+ }
322
+
323
+ function statusFromKind(kind: string): string | undefined {
324
+ if (kind.endsWith(".completed") || kind === "agent.tool.called") {
325
+ return "completed";
326
+ }
327
+ if (kind.endsWith(".failed")) {
328
+ return "failed";
329
+ }
330
+ if (kind.endsWith(".requested")) {
331
+ return "requested";
332
+ }
333
+ if (kind.endsWith(".denied")) {
334
+ return "denied";
335
+ }
336
+ return undefined;
337
+ }
338
+
339
+ function objectField(value: Record<string, unknown>, key: string): Record<string, unknown> | undefined {
340
+ const field = value[key];
341
+ return field && typeof field === "object" && !Array.isArray(field) ? field as Record<string, unknown> : undefined;
342
+ }
343
+
344
+ function stringField(value: Record<string, unknown> | undefined, key: string): string | undefined {
345
+ const field = value?.[key];
346
+ return typeof field === "string" && field.length > 0 ? field : undefined;
347
+ }
348
+
349
+ function numberField(value: Record<string, unknown> | undefined, key: string): number | undefined {
350
+ const field = value?.[key];
351
+ return typeof field === "number" && Number.isFinite(field) ? field : undefined;
352
+ }
353
+
354
+ function arrayOfStrings(value: unknown): string[] {
355
+ return Array.isArray(value) ? value.filter((item): item is string => typeof item === "string" && item.length > 0) : [];
356
+ }
357
+
358
+ function uniqueStrings(values: Array<string | undefined>): string[] {
359
+ return [...new Set(values.filter((value): value is string => typeof value === "string" && value.length > 0))];
360
+ }
361
+
362
+ function summarizeCommand(command: string): string {
363
+ const normalized = command.replace(/\s+/g, " ").trim();
364
+ const singleLine = normalized.length > 220 ? `${normalized.slice(0, 217)}...` : normalized;
365
+ return safeSummary(singleLine, 220);
366
+ }
367
+
368
+ function safeSummary(value: string, maxLength: number): string {
369
+ const normalized = value.replace(/\s+/g, " ").trim();
370
+ const clipped = normalized.length > maxLength ? `${normalized.slice(0, Math.max(0, maxLength - 3))}...` : normalized;
371
+ const redacted = redactDeltaPayload({ summary: scrubSecretTokens(clipped) }).value.summary;
372
+ return typeof redacted === "string" ? redacted : clipped;
373
+ }
374
+
375
+ function scrubSecretTokens(value: string): string {
376
+ return value
377
+ .replace(/\bsk[-_][A-Za-z0-9_\-.]{8,}\b/g, "[REDACTED]")
378
+ .replace(/\bnpm_[A-Za-z0-9]{16,}\b/g, "[REDACTED]")
379
+ .replace(/\bgh[pousr]_[A-Za-z0-9_]{16,}\b/g, "[REDACTED]")
380
+ .replace(/\b(?:xox[baprs]-)[A-Za-z0-9-]{16,}\b/g, "[REDACTED]");
381
+ }
382
+
383
+ function classifyCommand(toolName: string | undefined, command: string): string {
384
+ if (toolName === "apply_patch") {
385
+ return "patch";
386
+ }
387
+ if (toolName?.startsWith("mcp__")) {
388
+ return "mcp";
389
+ }
390
+ if (/\b(git|npm|pnpm|bun|yarn|node|forge|gh|python|pytest|mvn|gradle)\b/.test(command)) {
391
+ return "shell";
392
+ }
393
+ return toolName ?? "command";
394
+ }
395
+
396
+ function summarizeToolResponse(value: Record<string, unknown> | undefined): string | undefined {
397
+ if (!value) {
398
+ return undefined;
399
+ }
400
+ const candidate = stringField(value, "summary") ??
401
+ stringField(value, "message") ??
402
+ stringField(value, "stderr") ??
403
+ stringField(value, "stdout") ??
404
+ stringField(value, "output") ??
405
+ stringField(value, "result");
406
+ return candidate ? safeSummary(candidate, 220) : undefined;
407
+ }
408
+
409
+ function extractPaths(value: unknown): string[] {
410
+ const paths: string[] = [];
411
+ visit(value, (key, child) => {
412
+ if (typeof child === "string" && isPathLikeKey(key) && looksLikePath(child)) {
413
+ paths.push(child);
414
+ }
415
+ if (Array.isArray(child) && isPathListKey(key)) {
416
+ for (const item of child) {
417
+ if (typeof item === "string" && looksLikePath(item)) {
418
+ paths.push(item);
419
+ }
420
+ }
421
+ }
422
+ });
423
+ return paths;
424
+ }
425
+
426
+ function extractNamedValues(value: unknown, keys: string[]): string[] {
427
+ const wanted = new Set(keys);
428
+ const values: string[] = [];
429
+ visit(value, (key, child) => {
430
+ if (wanted.has(key) && typeof child === "string" && child.length > 0) {
431
+ values.push(child);
432
+ }
433
+ });
434
+ return values;
435
+ }
436
+
437
+ function extractPathsFromCommand(command: string | undefined): string[] {
438
+ if (!command) {
439
+ return [];
440
+ }
441
+ const paths: string[] = [];
442
+ const patchPattern = /^\*\*\* (?:Add|Update|Delete) File: (.+)$/gm;
443
+ for (const match of command.matchAll(patchPattern)) {
444
+ if (match[1]) {
445
+ paths.push(match[1].trim());
446
+ }
447
+ }
448
+ const literalPathPattern = /(?:-LiteralPath|-Path)\s+["']?([^"'\s]+)["']?/g;
449
+ for (const match of command.matchAll(literalPathPattern)) {
450
+ if (match[1] && looksLikePath(match[1])) {
451
+ paths.push(match[1]);
452
+ }
453
+ }
454
+ return paths;
455
+ }
456
+
457
+ function extractEntriesFromCommand(command: string | undefined): string[] {
458
+ if (!command) {
459
+ return [];
460
+ }
461
+ const entries: string[] = [];
462
+ const forgeEntryPattern = /\b(?:run|query|explain|timeline)\s+([a-zA-Z0-9_.:-]+)/g;
463
+ for (const match of command.matchAll(forgeEntryPattern)) {
464
+ const value = match[1];
465
+ if (value && value.includes(".")) {
466
+ entries.push(value);
467
+ }
468
+ }
469
+ return entries;
470
+ }
471
+
472
+ function visit(value: unknown, callback: (key: string, value: unknown) => void, depth = 0): void {
473
+ if (!value || typeof value !== "object" || depth > 4) {
474
+ return;
475
+ }
476
+ if (Array.isArray(value)) {
477
+ for (const item of value.slice(0, 50)) {
478
+ visit(item, callback, depth + 1);
479
+ }
480
+ return;
481
+ }
482
+ for (const [key, child] of Object.entries(value as Record<string, unknown>)) {
483
+ callback(key, child);
484
+ visit(child, callback, depth + 1);
485
+ }
486
+ }
487
+
488
+ function isPathLikeKey(key: string): boolean {
489
+ return /^(file|filePath|file_path|path|uri|absolutePath|relativePath)$/i.test(key);
490
+ }
491
+
492
+ function isPathListKey(key: string): boolean {
493
+ return /^(files|paths|changedFiles|affectedFiles|artifact_paths|artifactPaths)$/i.test(key);
494
+ }
495
+
496
+ function looksLikePath(value: string): boolean {
497
+ return /[\\/]/.test(value) || /\.(ts|tsx|js|jsx|mjs|cjs|json|md|mdx|sql|css|html|yml|yaml|toml|lock|java|go|py|ps1)$/i.test(value);
498
+ }
@@ -0,0 +1,103 @@
1
+ import { hashStable } from "../compiler/primitives/hash.ts";
2
+ import { redactDeltaPayload } from "../delta/redaction.ts";
3
+
4
+ const RAW_TEXT_KEYS = new Set([
5
+ "prompt",
6
+ "userPrompt",
7
+ "last_assistant_message",
8
+ "lastAssistantMessage",
9
+ "completion",
10
+ "message",
11
+ "transcript",
12
+ "transcript_path",
13
+ "transcriptPath",
14
+ "output",
15
+ "stdout",
16
+ "stderr",
17
+ "result",
18
+ ]);
19
+
20
+ const RAW_ARGS_KEYS = new Set(["args", "arguments", "tool_input", "toolInput", "tool_response", "toolResponse", "input"]);
21
+
22
+ export interface AgentPayloadRedaction {
23
+ value: Record<string, unknown>;
24
+ sensitiveFieldsRemoved: string[];
25
+ }
26
+
27
+ export function redactAgentPayload(payload: Record<string, unknown>): AgentPayloadRedaction {
28
+ const removed: string[] = [];
29
+ const coarse = stripRawPayload(payload, [], removed) as Record<string, unknown>;
30
+ const redacted = redactDeltaPayload(coarse);
31
+ return {
32
+ value: redacted.value,
33
+ sensitiveFieldsRemoved: [...new Set([...removed, ...redacted.redaction.diagnostics])],
34
+ };
35
+ }
36
+
37
+ function stripRawPayload(value: unknown, path: string[], removed: string[]): unknown {
38
+ if (Array.isArray(value)) {
39
+ return value.slice(0, 50).map((item, index) => stripRawPayload(item, [...path, String(index)], removed));
40
+ }
41
+ if (!value || typeof value !== "object") {
42
+ return value;
43
+ }
44
+ const output: Record<string, unknown> = {};
45
+ for (const [key, child] of Object.entries(value as Record<string, unknown>)) {
46
+ const nextPath = [...path, key];
47
+ if (RAW_TEXT_KEYS.has(key)) {
48
+ removed.push(nextPath.join("."));
49
+ output[`${key}Hash`] = typeof child === "string" ? hashStable(child) : hashStable(JSON.stringify(child ?? null));
50
+ output[`${key}Stored`] = false;
51
+ if (typeof child === "string" && !isPromptLikeKey(key)) {
52
+ const summary = summarizeText(child);
53
+ if (summary) {
54
+ output[`${key}Summary`] = redactDeltaPayload({ summary }).value.summary;
55
+ }
56
+ }
57
+ continue;
58
+ }
59
+ if (RAW_ARGS_KEYS.has(key)) {
60
+ removed.push(nextPath.join("."));
61
+ output[`${key}Hash`] = hashStable(JSON.stringify(child ?? null));
62
+ output[`${key}Stored`] = false;
63
+ output[`${key}Shape`] = describeShape(child);
64
+ continue;
65
+ }
66
+ output[key] = stripRawPayload(child, nextPath, removed);
67
+ }
68
+ return output;
69
+ }
70
+
71
+ function isPromptLikeKey(key: string): boolean {
72
+ return key.toLowerCase().includes("prompt") || key.toLowerCase().includes("completion") || key.toLowerCase().includes("message");
73
+ }
74
+
75
+ function summarizeText(value: string): string | undefined {
76
+ const normalized = value.replace(/\s+/g, " ").trim();
77
+ if (!normalized) {
78
+ return undefined;
79
+ }
80
+ const clipped = normalized.length > 160 ? `${normalized.slice(0, 157)}...` : normalized;
81
+ return scrubSecretTokens(clipped);
82
+ }
83
+
84
+ function describeShape(value: unknown): unknown {
85
+ if (Array.isArray(value)) {
86
+ return { kind: "array", length: value.length };
87
+ }
88
+ if (value && typeof value === "object") {
89
+ return {
90
+ kind: "object",
91
+ keys: Object.keys(value as Record<string, unknown>).slice(0, 20).sort(),
92
+ };
93
+ }
94
+ return { kind: typeof value };
95
+ }
96
+
97
+ function scrubSecretTokens(value: string): string {
98
+ return value
99
+ .replace(/\bsk[-_][A-Za-z0-9_\-.]{8,}\b/g, "[REDACTED]")
100
+ .replace(/\bnpm_[A-Za-z0-9]{16,}\b/g, "[REDACTED]")
101
+ .replace(/\bgh[pousr]_[A-Za-z0-9_]{16,}\b/g, "[REDACTED]")
102
+ .replace(/\b(?:xox[baprs]-)[A-Za-z0-9-]{16,}\b/g, "[REDACTED]");
103
+ }
@@ -0,0 +1,51 @@
1
+ import type { AgentInstallResult } from "../types.ts";
2
+ import { privacyDefaults } from "./codex.ts";
3
+
4
+ const CLAUDE_EVENTS = [
5
+ "SessionStart",
6
+ "UserPromptSubmit",
7
+ "PreToolUse",
8
+ "PermissionRequest",
9
+ "PostToolUse",
10
+ "PostToolUseFailure",
11
+ "PermissionDenied",
12
+ "FileChanged",
13
+ "SubagentStart",
14
+ "SubagentStop",
15
+ "Stop",
16
+ "SessionEnd",
17
+ ];
18
+
19
+ export function claudeCodeInstallFiles(): Array<{ path: string; content: string }> {
20
+ const settings = {
21
+ hooks: Object.fromEntries(CLAUDE_EVENTS.map((event) => [
22
+ event,
23
+ [
24
+ {
25
+ matcher: "*",
26
+ hooks: [
27
+ {
28
+ type: "command",
29
+ command: `forge agent ingest claude-code --event ${event}`,
30
+ },
31
+ ],
32
+ },
33
+ ],
34
+ ])),
35
+ };
36
+ return [
37
+ { path: ".claude/settings.json", content: `${JSON.stringify(settings, null, 2)}\n` },
38
+ ];
39
+ }
40
+
41
+ export function claudeCodeInstallResult(filesWritten: string[], filesPlanned: string[]): AgentInstallResult {
42
+ return {
43
+ ok: true,
44
+ target: "claude-code",
45
+ filesWritten,
46
+ filesPlanned,
47
+ privacy: privacyDefaults(),
48
+ warnings: ["Claude transcript imports are opt-in only; hooks store redacted project memory."],
49
+ exitCode: 0,
50
+ };
51
+ }