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,279 @@
1
+ import { spawnSync } from "node:child_process";
2
+ import { readFileSync } from "node:fs";
3
+ import { join } from "node:path";
4
+ import {
5
+ categorizeFiles,
6
+ classifyChangeType,
7
+ compactFiles,
8
+ filterVolatileForgeState,
9
+ type CategorizedFileSummary,
10
+ type ChangeClassifier,
11
+ type FileListSummary,
12
+ } from "./change-summary.ts";
13
+
14
+ export interface WorkspaceGitSummary {
15
+ available: boolean;
16
+ branch?: string;
17
+ commit?: string;
18
+ changed: FileListSummary;
19
+ staged: FileListSummary;
20
+ unstaged: FileListSummary;
21
+ untracked: FileListSummary;
22
+ changeSummary: {
23
+ changed: CategorizedFileSummary;
24
+ staged: CategorizedFileSummary;
25
+ unstaged: CategorizedFileSummary;
26
+ untracked: CategorizedFileSummary;
27
+ };
28
+ error?: string;
29
+ }
30
+
31
+ function emptySummary(error?: string): WorkspaceGitSummary {
32
+ return {
33
+ available: false,
34
+ changed: compactFiles([]),
35
+ staged: compactFiles([]),
36
+ unstaged: compactFiles([]),
37
+ untracked: compactFiles([]),
38
+ changeSummary: {
39
+ changed: categorizeFiles([]),
40
+ staged: categorizeFiles([]),
41
+ unstaged: categorizeFiles([]),
42
+ untracked: categorizeFiles([]),
43
+ },
44
+ ...(error ? { error } : {}),
45
+ };
46
+ }
47
+
48
+ function runGit(
49
+ args: string[],
50
+ workspaceRoot: string,
51
+ options: { trim?: boolean } = {},
52
+ ): { ok: boolean; stdout: string; error?: string } {
53
+ const result = spawnSync("git", args, {
54
+ cwd: workspaceRoot,
55
+ encoding: "utf8",
56
+ windowsHide: true,
57
+ });
58
+ if (result.status !== 0) {
59
+ return {
60
+ ok: false,
61
+ stdout: "",
62
+ error: (result.stderr || result.stdout || "git command failed").trim(),
63
+ };
64
+ }
65
+ return { ok: true, stdout: options.trim === false ? result.stdout : result.stdout.trim() };
66
+ }
67
+
68
+ function pathspecLiteral(file: string): string {
69
+ return `:(literal)${file}`;
70
+ }
71
+
72
+ function diffTouchesOnlyGeneratedMetadata(diff: string): boolean {
73
+ let changedLines = 0;
74
+ for (const line of diff.split(/\r?\n/)) {
75
+ if (
76
+ line.startsWith("diff --git ") ||
77
+ line.startsWith("index ") ||
78
+ line.startsWith("@@ ") ||
79
+ line.startsWith("--- ") ||
80
+ line.startsWith("+++ ")
81
+ ) {
82
+ continue;
83
+ }
84
+ if (!line.startsWith("+") && !line.startsWith("-")) {
85
+ continue;
86
+ }
87
+ changedLines += 1;
88
+ if (!line.slice(1).trimStart().startsWith("// @forge-generated")) {
89
+ return false;
90
+ }
91
+ }
92
+ return changedLines > 0;
93
+ }
94
+
95
+ function generatedLineSet(text: string): Set<number> {
96
+ const generated = new Set<number>();
97
+ const lines = text.split(/\r?\n/);
98
+ if (lines[0]?.trimStart().startsWith("// @forge-generated")) {
99
+ generated.add(1);
100
+ }
101
+ let inGeneratedBlock = false;
102
+ lines.forEach((line, index) => {
103
+ const lineNumber = index + 1;
104
+ if (line.includes("<!-- forge-generated:start -->")) {
105
+ inGeneratedBlock = true;
106
+ generated.add(lineNumber);
107
+ return;
108
+ }
109
+ if (inGeneratedBlock) {
110
+ generated.add(lineNumber);
111
+ }
112
+ if (line.includes("<!-- forge-generated:end -->")) {
113
+ inGeneratedBlock = false;
114
+ }
115
+ });
116
+ return generated;
117
+ }
118
+
119
+ function headFileText(workspaceRoot: string, file: string): string | null {
120
+ const result = runGit(["show", `HEAD:${file}`], workspaceRoot, { trim: false });
121
+ return result.ok ? result.stdout : null;
122
+ }
123
+
124
+ function worktreeFileText(workspaceRoot: string, file: string): string | null {
125
+ try {
126
+ return readFileSync(join(workspaceRoot, file), "utf8");
127
+ } catch {
128
+ return null;
129
+ }
130
+ }
131
+
132
+ function parseHunkStart(header: string, marker: "-" | "+"): number {
133
+ const pattern = marker === "-"
134
+ ? /^@@ -(\d+)(?:,\d+)? \+\d+(?:,\d+)? @@/
135
+ : /^@@ -\d+(?:,\d+)? \+(\d+)(?:,\d+)? @@/;
136
+ const match = header.match(pattern);
137
+ return match ? Number(match[1]) : 1;
138
+ }
139
+
140
+ function diffTouchesOnlyGeneratedRegions(diff: string, oldText: string | null, newText: string | null): boolean {
141
+ if (!oldText || !newText) {
142
+ return diffTouchesOnlyGeneratedMetadata(diff);
143
+ }
144
+ const oldGenerated = generatedLineSet(oldText);
145
+ const newGenerated = generatedLineSet(newText);
146
+ if (oldGenerated.size === 0 && newGenerated.size === 0) {
147
+ return diffTouchesOnlyGeneratedMetadata(diff);
148
+ }
149
+
150
+ let changedLines = 0;
151
+ let oldLine = 1;
152
+ let newLine = 1;
153
+ for (const line of diff.split(/\r?\n/)) {
154
+ if (line.startsWith("@@ ")) {
155
+ oldLine = parseHunkStart(line, "-");
156
+ newLine = parseHunkStart(line, "+");
157
+ continue;
158
+ }
159
+ if (
160
+ line.startsWith("diff --git ") ||
161
+ line.startsWith("index ") ||
162
+ line.startsWith("--- ") ||
163
+ line.startsWith("+++ ")
164
+ ) {
165
+ continue;
166
+ }
167
+ if (line.startsWith("-")) {
168
+ changedLines += 1;
169
+ if (!oldGenerated.has(oldLine) && !line.slice(1).trimStart().startsWith("// @forge-generated")) {
170
+ return false;
171
+ }
172
+ oldLine += 1;
173
+ continue;
174
+ }
175
+ if (line.startsWith("+")) {
176
+ changedLines += 1;
177
+ if (!newGenerated.has(newLine) && !line.slice(1).trimStart().startsWith("// @forge-generated")) {
178
+ return false;
179
+ }
180
+ newLine += 1;
181
+ continue;
182
+ }
183
+ if (line.startsWith(" ")) {
184
+ oldLine += 1;
185
+ newLine += 1;
186
+ }
187
+ }
188
+ return changedLines > 0;
189
+ }
190
+
191
+ function generatedMetadataOnlyChange(workspaceRoot: string, file: string): boolean {
192
+ const pathspec = pathspecLiteral(file);
193
+ const diffs = [
194
+ runGit(["diff", "--unified=0", "--", pathspec], workspaceRoot, { trim: false }),
195
+ runGit(["diff", "--cached", "--unified=0", "--", pathspec], workspaceRoot, { trim: false }),
196
+ ]
197
+ .filter((result) => result.ok)
198
+ .map((result) => result.stdout)
199
+ .filter((stdout) => stdout.trim().length > 0);
200
+ const oldText = headFileText(workspaceRoot, file);
201
+ const newText = worktreeFileText(workspaceRoot, file);
202
+ return diffs.length > 0 && diffs.every((diff) => diffTouchesOnlyGeneratedRegions(diff, oldText, newText));
203
+ }
204
+
205
+ function workspaceChangeClassifier(workspaceRoot: string): ChangeClassifier {
206
+ const cache = new Map<string, ReturnType<ChangeClassifier>>();
207
+ return (file) => {
208
+ const cached = cache.get(file);
209
+ if (cached) {
210
+ return cached;
211
+ }
212
+ const baseType = classifyChangeType(file);
213
+ const type = baseType === "docs" && generatedMetadataOnlyChange(workspaceRoot, file)
214
+ ? "generated"
215
+ : baseType;
216
+ cache.set(file, type);
217
+ return type;
218
+ };
219
+ }
220
+
221
+ function parseStatusPath(line: string): string {
222
+ const raw = line.slice(2).trimStart();
223
+ const renamed = raw.split(" -> ");
224
+ return (renamed[renamed.length - 1] ?? raw).replace(/\\/g, "/");
225
+ }
226
+
227
+ export function buildWorkspaceGitSummary(workspaceRoot: string): WorkspaceGitSummary {
228
+ const root = runGit(["rev-parse", "--show-toplevel"], workspaceRoot);
229
+ if (!root.ok) {
230
+ return emptySummary(root.error);
231
+ }
232
+
233
+ const status = runGit(["status", "--porcelain=v1", "-uall"], workspaceRoot, { trim: false });
234
+ const branch = runGit(["rev-parse", "--abbrev-ref", "HEAD"], workspaceRoot);
235
+ const commit = runGit(["rev-parse", "--short", "HEAD"], workspaceRoot);
236
+ const lines = status.ok ? status.stdout.split(/\r?\n/).filter(Boolean) : [];
237
+ const staged: string[] = [];
238
+ const unstaged: string[] = [];
239
+ const untracked: string[] = [];
240
+
241
+ for (const line of lines) {
242
+ const x = line[0] ?? " ";
243
+ const y = line[1] ?? " ";
244
+ const file = parseStatusPath(line);
245
+ if (x === "?" && y === "?") {
246
+ untracked.push(file);
247
+ continue;
248
+ }
249
+ if (x !== " ") {
250
+ staged.push(file);
251
+ }
252
+ if (y !== " ") {
253
+ unstaged.push(file);
254
+ }
255
+ }
256
+
257
+ const stagedFiles = filterVolatileForgeState([...new Set(staged)].sort());
258
+ const unstagedFiles = filterVolatileForgeState([...new Set(unstaged)].sort());
259
+ const untrackedFiles = filterVolatileForgeState([...new Set(untracked)].sort());
260
+ const changedFiles = [...new Set([...stagedFiles, ...unstagedFiles, ...untrackedFiles])].sort();
261
+ const classify = workspaceChangeClassifier(workspaceRoot);
262
+
263
+ return {
264
+ available: true,
265
+ ...(branch.ok ? { branch: branch.stdout } : {}),
266
+ ...(commit.ok ? { commit: commit.stdout } : {}),
267
+ changed: compactFiles(changedFiles),
268
+ staged: compactFiles(stagedFiles),
269
+ unstaged: compactFiles(unstagedFiles),
270
+ untracked: compactFiles(untrackedFiles),
271
+ changeSummary: {
272
+ changed: categorizeFiles(changedFiles, 8, classify),
273
+ staged: categorizeFiles(stagedFiles, 8, classify),
274
+ unstaged: categorizeFiles(unstagedFiles, 8, classify),
275
+ untracked: categorizeFiles(untrackedFiles, 8, classify),
276
+ },
277
+ ...(status.ok ? {} : { error: status.error }),
278
+ };
279
+ }
@@ -0,0 +1,29 @@
1
+ # AGENTS.md
2
+
3
+ This ForgeOS app is an agent workroom. External code agents edit the app in this
4
+ directory while ForgeOS records context, hooks, checks, generated artifacts, and
5
+ handoff state.
6
+
7
+ Before editing:
8
+
9
+ ```bash
10
+ forge dev --once --json
11
+ forge agent print-context --json
12
+ forge check --json
13
+ ```
14
+
15
+ After editing:
16
+
17
+ ```bash
18
+ forge generate
19
+ forge check
20
+ forge verify agent
21
+ ```
22
+
23
+ Do not edit:
24
+
25
+ - `src/forge/_generated/**`
26
+ - `forge.lock`
27
+
28
+ Use `forge studio attach . --target codex --preview-port 5174` to connect this
29
+ workspace to Forge Studio.
@@ -0,0 +1,34 @@
1
+ # __FORGE_APP_TITLE__
2
+
3
+ An agent-native ForgeOS workroom. It is not a chat app. Codex, Claude Code, or
4
+ Cursor work in this directory, while the UI shows the app preview, terminal-like
5
+ agent signals, generated freshness, diff focus, checks, and handoff evidence
6
+ collected by ForgeOS.
7
+
8
+ ## Run
9
+
10
+ ```bash
11
+ __PACKAGE_MANAGER__ install
12
+ __PACKAGE_MANAGER__ run generate
13
+ __PACKAGE_MANAGER__ run dev
14
+ ```
15
+
16
+ The workroom runs on the Forge web port. The app being built should normally run
17
+ on the next port, for example `http://127.0.0.1:5174`.
18
+
19
+ ## Attach to Studio
20
+
21
+ ```bash
22
+ forge studio attach . --target codex --preview-port 5174
23
+ forge agent doctor --target codex
24
+ ```
25
+
26
+ Agent hooks can feed the room through ForgeOS commands or `forge agent ingest`.
27
+
28
+ The browser does not chat with an AI model. It observes:
29
+
30
+ - the app preview, usually on `http://127.0.0.1:5174`;
31
+ - hook or ingest events from the external code agent;
32
+ - authored-vs-generated diff focus;
33
+ - generated artifact state (`fresh`, `regenerated`, or `stale-risk`);
34
+ - verification evidence from `forge check` / `forge verify agent`.
@@ -0,0 +1,3 @@
1
+ export default {
2
+ sourceRoots: ["src"],
3
+ };
@@ -0,0 +1,33 @@
1
+ {
2
+ "name": "__FORGE_APP_NAME__",
3
+ "private": true,
4
+ "type": "module",
5
+ "workspaces": [
6
+ "web"
7
+ ],
8
+ "packageManager": "__PACKAGE_MANAGER_SPEC__",
9
+ "scripts": {
10
+ "forge": "forge",
11
+ "generate": "forge generate",
12
+ "check": "forge check",
13
+ "verify": "forge verify agent",
14
+ "dev": "forge dev",
15
+ "dev:api": "forge dev --api-only",
16
+ "dev:web": "cd web && __PACKAGE_MANAGER__ run dev",
17
+ "typecheck": "tsc --noEmit && cd web && __PACKAGE_MANAGER__ run typecheck"
18
+ },
19
+ "dependencies": {
20
+ "@electric-sql/pglite": "^0.2.17",
21
+ "forge": "__FORGE_PACKAGE_SPEC__"
22
+ },
23
+ "devDependencies": {
24
+ "@types/node": "^24.0.0",
25
+ "typescript": "^5.7.3"
26
+ },
27
+ "forge": {
28
+ "template": "agent-workroom",
29
+ "sourceRoots": [
30
+ "src"
31
+ ]
32
+ }
33
+ }
@@ -0,0 +1,10 @@
1
+ import { action } from "forge/server";
2
+
3
+ export const indexAgentSignal = action({
4
+ event: "agent.signal.recorded",
5
+ handler: async (_ctx, event) => ({
6
+ ok: true,
7
+ indexed: true,
8
+ event,
9
+ }),
10
+ });
@@ -0,0 +1,61 @@
1
+ import { can, command } from "forge/server";
2
+
3
+ function text(value: unknown, fallback: string): string {
4
+ return typeof value === "string" && value.trim().length > 0 ? value.trim() : fallback;
5
+ }
6
+
7
+ function count(value: unknown): number {
8
+ return typeof value === "number" && Number.isFinite(value) && value >= 0 ? value : 0;
9
+ }
10
+
11
+ export const openWorkroom = command({
12
+ auth: can("workroom.write"),
13
+ handler: async (ctx, args) => {
14
+ const input = args as {
15
+ appName?: unknown;
16
+ appPath?: unknown;
17
+ previewUrl?: unknown;
18
+ previewStatus?: unknown;
19
+ previewStatusReason?: unknown;
20
+ agent?: unknown;
21
+ objective?: unknown;
22
+ generatedState?: unknown;
23
+ generatedChangedFiles?: unknown;
24
+ authoredFiles?: unknown;
25
+ generatedFiles?: unknown;
26
+ authoredDiffCommand?: unknown;
27
+ generatedDiffCommand?: unknown;
28
+ terminalCommand?: unknown;
29
+ terminalCwd?: unknown;
30
+ };
31
+ const now = new Date().toISOString();
32
+ const session = await ctx.db.agentSessions.insert({
33
+ appName: text(input.appName, "__FORGE_APP_TITLE__"),
34
+ appPath: text(input.appPath, "."),
35
+ previewUrl: text(input.previewUrl, "http://127.0.0.1:5174"),
36
+ previewStatus: text(input.previewStatus, "not-checked"),
37
+ previewStatusReason: text(input.previewStatusReason, "Preview has not been probed yet."),
38
+ agent: text(input.agent, "codex"),
39
+ status: "observing",
40
+ objective: text(input.objective, "External agent development session"),
41
+ generatedState: text(input.generatedState, "fresh"),
42
+ generatedChangedFiles: count(input.generatedChangedFiles),
43
+ authoredFiles: count(input.authoredFiles),
44
+ generatedFiles: count(input.generatedFiles),
45
+ authoredDiffCommand: text(input.authoredDiffCommand, 'git diff -- . ":(exclude)src/forge/_generated/**" ":(exclude)forge.lock"'),
46
+ generatedDiffCommand: text(input.generatedDiffCommand, "git diff -- src/forge/_generated forge.lock"),
47
+ terminalCommand: text(input.terminalCommand, "codex"),
48
+ terminalCwd: text(input.terminalCwd, "."),
49
+ createdAt: now,
50
+ updatedAt: now,
51
+ });
52
+
53
+ await ctx.emit("workroom.opened", {
54
+ sessionId: session.id,
55
+ appName: session.appName,
56
+ agent: session.agent,
57
+ });
58
+
59
+ return session;
60
+ },
61
+ });
@@ -0,0 +1,119 @@
1
+ import { can, command } from "forge/server";
2
+
3
+ function text(value: unknown, fallback: string): string {
4
+ return typeof value === "string" && value.trim().length > 0 ? value.trim() : fallback;
5
+ }
6
+
7
+ function optionalText(value: unknown): string | undefined {
8
+ return typeof value === "string" && value.trim().length > 0 ? value.trim() : undefined;
9
+ }
10
+
11
+ function optionalCount(value: unknown): number | undefined {
12
+ return typeof value === "number" && Number.isFinite(value) && value >= 0 ? value : undefined;
13
+ }
14
+
15
+ export const recordAgentSignal = command({
16
+ auth: can("workroom.write"),
17
+ handler: async (ctx, args) => {
18
+ const input = args as {
19
+ sessionId?: unknown;
20
+ source?: unknown;
21
+ kind?: unknown;
22
+ title?: unknown;
23
+ detail?: unknown;
24
+ filesChanged?: unknown;
25
+ status?: unknown;
26
+ previewStatus?: unknown;
27
+ previewStatusReason?: unknown;
28
+ generatedState?: unknown;
29
+ generatedChangedFiles?: unknown;
30
+ authoredFiles?: unknown;
31
+ generatedFiles?: unknown;
32
+ authoredDiffCommand?: unknown;
33
+ generatedDiffCommand?: unknown;
34
+ terminalCommand?: unknown;
35
+ terminalCwd?: unknown;
36
+ };
37
+ let session =
38
+ typeof input.sessionId === "string" && input.sessionId.trim().length > 0
39
+ ? await ctx.db.agentSessions.get(input.sessionId)
40
+ : null;
41
+ if (!session) {
42
+ const sessions = await ctx.db.agentSessions.all();
43
+ session = sessions
44
+ .slice()
45
+ .sort((a, b) => String(b.updatedAt).localeCompare(String(a.updatedAt)))[0] ?? null;
46
+ }
47
+ if (!session) {
48
+ const createdAt = new Date().toISOString();
49
+ session = await ctx.db.agentSessions.insert({
50
+ appName: "__FORGE_APP_TITLE__",
51
+ appPath: ".",
52
+ previewUrl: "http://127.0.0.1:5174",
53
+ previewStatus: "not-checked",
54
+ previewStatusReason: "Preview has not been probed yet.",
55
+ agent: "codex",
56
+ status: "observing",
57
+ objective: "External agent development session",
58
+ generatedState: "fresh",
59
+ generatedChangedFiles: 0,
60
+ authoredFiles: 0,
61
+ generatedFiles: 0,
62
+ authoredDiffCommand: 'git diff -- . ":(exclude)src/forge/_generated/**" ":(exclude)forge.lock"',
63
+ generatedDiffCommand: "git diff -- src/forge/_generated forge.lock",
64
+ terminalCommand: "codex",
65
+ terminalCwd: ".",
66
+ createdAt,
67
+ updatedAt: createdAt,
68
+ });
69
+ }
70
+ if (!session) {
71
+ throw new Error("session not found");
72
+ }
73
+
74
+ const now = new Date().toISOString();
75
+ const signal = await ctx.db.agentSignals.insert({
76
+ sessionId: session.id,
77
+ source: text(input.source, session.agent),
78
+ kind: text(input.kind, "hook"),
79
+ title: text(input.title, "Agent activity recorded"),
80
+ detail: text(input.detail, "A code agent changed the workspace."),
81
+ filesChanged: Array.isArray(input.filesChanged)
82
+ ? input.filesChanged.filter((file) => typeof file === "string").join(", ")
83
+ : text(input.filesChanged, ""),
84
+ status: text(input.status, "info"),
85
+ createdAt: now,
86
+ });
87
+
88
+ const update: Record<string, unknown> = {
89
+ status: signal.status === "error" ? "needs-attention" : "observing",
90
+ updatedAt: now,
91
+ };
92
+ for (const [key, value] of Object.entries({
93
+ previewStatus: optionalText(input.previewStatus),
94
+ previewStatusReason: optionalText(input.previewStatusReason),
95
+ generatedState: optionalText(input.generatedState),
96
+ generatedChangedFiles: optionalCount(input.generatedChangedFiles),
97
+ authoredFiles: optionalCount(input.authoredFiles),
98
+ generatedFiles: optionalCount(input.generatedFiles),
99
+ authoredDiffCommand: optionalText(input.authoredDiffCommand),
100
+ generatedDiffCommand: optionalText(input.generatedDiffCommand),
101
+ terminalCommand: optionalText(input.terminalCommand),
102
+ terminalCwd: optionalText(input.terminalCwd),
103
+ })) {
104
+ if (value !== undefined) {
105
+ update[key] = value;
106
+ }
107
+ }
108
+
109
+ await ctx.db.agentSessions.update(session.id, update);
110
+ await ctx.emit("agent.signal.recorded", {
111
+ sessionId: session.id,
112
+ signalId: signal.id,
113
+ kind: signal.kind,
114
+ status: signal.status,
115
+ });
116
+
117
+ return signal;
118
+ },
119
+ });
@@ -0,0 +1,52 @@
1
+ import { can, command } from "forge/server";
2
+
3
+ function text(value: unknown, fallback: string): string {
4
+ return typeof value === "string" && value.trim().length > 0 ? value.trim() : fallback;
5
+ }
6
+
7
+ function duration(value: unknown): number {
8
+ return typeof value === "number" && Number.isFinite(value) && value >= 0 ? value : 0;
9
+ }
10
+
11
+ export const recordCheckRun = command({
12
+ auth: can("workroom.write"),
13
+ handler: async (ctx, args) => {
14
+ const input = args as {
15
+ sessionId?: unknown;
16
+ command?: unknown;
17
+ status?: unknown;
18
+ output?: unknown;
19
+ durationMs?: unknown;
20
+ };
21
+ const sessionId = text(input.sessionId, "");
22
+ if (!sessionId) {
23
+ throw new Error("sessionId is required");
24
+ }
25
+ const session = await ctx.db.agentSessions.get(sessionId);
26
+ if (!session) {
27
+ throw new Error("session not found");
28
+ }
29
+
30
+ const now = new Date().toISOString();
31
+ const run = await ctx.db.checkRuns.insert({
32
+ sessionId,
33
+ command: text(input.command, "forge check"),
34
+ status: text(input.status, "passed"),
35
+ output: text(input.output, ""),
36
+ durationMs: duration(input.durationMs),
37
+ createdAt: now,
38
+ });
39
+
40
+ await ctx.db.agentSessions.update(sessionId, {
41
+ status: run.status === "failed" ? "needs-attention" : "verified",
42
+ updatedAt: now,
43
+ });
44
+ await ctx.emit("check.run.recorded", {
45
+ sessionId,
46
+ runId: run.id,
47
+ status: run.status,
48
+ });
49
+
50
+ return run;
51
+ },
52
+ });
@@ -0,0 +1,54 @@
1
+ import { defineTable } from "forge/server";
2
+
3
+ export const agentSessions = defineTable({
4
+ name: "agentSessions",
5
+ fields: {
6
+ id: "uuid",
7
+ appName: "text",
8
+ appPath: "text",
9
+ previewUrl: "text",
10
+ previewStatus: "text",
11
+ previewStatusReason: "text",
12
+ agent: "text",
13
+ status: "text",
14
+ objective: "text",
15
+ generatedState: "text",
16
+ generatedChangedFiles: "number",
17
+ authoredFiles: "number",
18
+ generatedFiles: "number",
19
+ authoredDiffCommand: "text",
20
+ generatedDiffCommand: "text",
21
+ terminalCommand: "text",
22
+ terminalCwd: "text",
23
+ createdAt: "timestamp",
24
+ updatedAt: "timestamp",
25
+ },
26
+ });
27
+
28
+ export const agentSignals = defineTable({
29
+ name: "agentSignals",
30
+ fields: {
31
+ id: "uuid",
32
+ sessionId: "uuid",
33
+ source: "text",
34
+ kind: "text",
35
+ title: "text",
36
+ detail: "text",
37
+ filesChanged: "text",
38
+ status: "text",
39
+ createdAt: "timestamp",
40
+ },
41
+ });
42
+
43
+ export const checkRuns = defineTable({
44
+ name: "checkRuns",
45
+ fields: {
46
+ id: "uuid",
47
+ sessionId: "uuid",
48
+ command: "text",
49
+ status: "text",
50
+ output: "text",
51
+ durationMs: "number",
52
+ createdAt: "timestamp",
53
+ },
54
+ });
@@ -0,0 +1,6 @@
1
+ import { canRole, definePolicies } from "forge/policy";
2
+
3
+ export const policies = definePolicies({
4
+ "workroom.read": canRole("owner", "admin", "member"),
5
+ "workroom.write": canRole("owner", "admin", "member"),
6
+ });