gsd-pi 2.74.0-dev.2b524c3 → 2.74.0-dev.658744a

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 (506) hide show
  1. package/dist/cli.js +85 -0
  2. package/dist/headless-query.js +4 -1
  3. package/dist/help-text.js +23 -0
  4. package/dist/resources/extensions/gsd/activity-log.js +16 -0
  5. package/dist/resources/extensions/gsd/auto/detect-stuck.js +11 -4
  6. package/dist/resources/extensions/gsd/auto/loop.js +147 -10
  7. package/dist/resources/extensions/gsd/auto/phases.js +209 -10
  8. package/dist/resources/extensions/gsd/auto/session.js +10 -0
  9. package/dist/resources/extensions/gsd/auto-dispatch.js +11 -1
  10. package/dist/resources/extensions/gsd/auto-model-selection.js +54 -8
  11. package/dist/resources/extensions/gsd/auto-post-unit.js +220 -17
  12. package/dist/resources/extensions/gsd/auto-prompts.js +12 -0
  13. package/dist/resources/extensions/gsd/auto-recovery.js +24 -10
  14. package/dist/resources/extensions/gsd/auto-unit-closeout.js +18 -0
  15. package/dist/resources/extensions/gsd/auto-verification.js +100 -2
  16. package/dist/resources/extensions/gsd/auto-worktree.js +2 -0
  17. package/dist/resources/extensions/gsd/auto.js +36 -4
  18. package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +5 -3
  19. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +30 -8
  20. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +61 -9
  21. package/dist/resources/extensions/gsd/cache.js +16 -5
  22. package/dist/resources/extensions/gsd/commands/catalog.js +31 -1
  23. package/dist/resources/extensions/gsd/commands/handlers/core.js +5 -1
  24. package/dist/resources/extensions/gsd/commands/handlers/ops.js +25 -0
  25. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +68 -9
  26. package/dist/resources/extensions/gsd/commands-add-tests.js +111 -0
  27. package/dist/resources/extensions/gsd/commands-backlog.js +140 -0
  28. package/dist/resources/extensions/gsd/commands-do.js +79 -0
  29. package/dist/resources/extensions/gsd/commands-extract-learnings.js +225 -0
  30. package/dist/resources/extensions/gsd/commands-maintenance.js +6 -6
  31. package/dist/resources/extensions/gsd/commands-pr-branch.js +180 -0
  32. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +51 -4
  33. package/dist/resources/extensions/gsd/commands-session-report.js +82 -0
  34. package/dist/resources/extensions/gsd/commands-ship.js +187 -0
  35. package/dist/resources/extensions/gsd/db-writer.js +3 -5
  36. package/dist/resources/extensions/gsd/docs/preferences-reference.md +16 -1
  37. package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +144 -0
  38. package/dist/resources/extensions/gsd/ecosystem/loader.js +145 -0
  39. package/dist/resources/extensions/gsd/git-service.js +49 -1
  40. package/dist/resources/extensions/gsd/graph-context.js +157 -0
  41. package/dist/resources/extensions/gsd/gsd-db.js +581 -2
  42. package/dist/resources/extensions/gsd/guided-flow.js +31 -6
  43. package/dist/resources/extensions/gsd/index.js +15 -2
  44. package/dist/resources/extensions/gsd/init-wizard.js +1 -0
  45. package/dist/resources/extensions/gsd/journal.js +27 -0
  46. package/dist/resources/extensions/gsd/md-importer.js +3 -4
  47. package/dist/resources/extensions/gsd/memory-store.js +19 -51
  48. package/dist/resources/extensions/gsd/metrics.js +19 -0
  49. package/dist/resources/extensions/gsd/milestone-validation-gates.js +13 -12
  50. package/dist/resources/extensions/gsd/native-git-bridge.js +7 -4
  51. package/dist/resources/extensions/gsd/parallel-orchestrator.js +33 -1
  52. package/dist/resources/extensions/gsd/preferences-models.js +20 -3
  53. package/dist/resources/extensions/gsd/preferences-types.js +2 -0
  54. package/dist/resources/extensions/gsd/preferences-validation.js +118 -2
  55. package/dist/resources/extensions/gsd/preferences.js +31 -0
  56. package/dist/resources/extensions/gsd/prompts/add-tests.md +35 -0
  57. package/dist/resources/extensions/gsd/safety/evidence-collector.js +15 -30
  58. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +12 -2
  59. package/dist/resources/extensions/gsd/state.js +5 -1
  60. package/dist/resources/extensions/gsd/templates/PREFERENCES.md +19 -0
  61. package/dist/resources/extensions/gsd/tools/complete-slice.js +20 -0
  62. package/dist/resources/extensions/gsd/tools/validate-milestone.js +39 -4
  63. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +3 -14
  64. package/dist/resources/extensions/gsd/triage-resolution.js +2 -5
  65. package/dist/resources/extensions/gsd/unit-ownership.js +1 -1
  66. package/dist/resources/extensions/gsd/uok/audit-toggle.js +7 -0
  67. package/dist/resources/extensions/gsd/uok/audit.js +40 -0
  68. package/dist/resources/extensions/gsd/uok/contracts.js +1 -0
  69. package/dist/resources/extensions/gsd/uok/execution-graph.js +179 -0
  70. package/dist/resources/extensions/gsd/uok/flags.js +29 -0
  71. package/dist/resources/extensions/gsd/uok/gate-runner.js +109 -0
  72. package/dist/resources/extensions/gsd/uok/gitops.js +53 -0
  73. package/dist/resources/extensions/gsd/uok/kernel.js +80 -0
  74. package/dist/resources/extensions/gsd/uok/loop-adapter.js +133 -0
  75. package/dist/resources/extensions/gsd/uok/model-policy.js +66 -0
  76. package/dist/resources/extensions/gsd/uok/plan-v2.js +132 -0
  77. package/dist/resources/extensions/gsd/workflow-logger.js +22 -0
  78. package/dist/resources/extensions/gsd/workflow-manifest.js +8 -69
  79. package/dist/resources/extensions/gsd/workflow-migration.js +21 -22
  80. package/dist/resources/extensions/gsd/workflow-projections.js +4 -1
  81. package/dist/resources/extensions/gsd/workflow-reconcile.js +14 -11
  82. package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
  83. package/dist/tsconfig.extensions.tsbuildinfo +1 -0
  84. package/dist/web/standalone/.next/BUILD_ID +1 -1
  85. package/dist/web/standalone/.next/app-path-routes-manifest.json +9 -9
  86. package/dist/web/standalone/.next/build-manifest.json +2 -2
  87. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  88. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  89. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  90. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  91. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  92. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  93. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  94. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  95. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  96. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  97. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  98. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  99. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  100. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  102. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  103. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  104. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  105. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +1 -1
  106. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +1 -1
  107. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
  108. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
  109. package/dist/web/standalone/.next/server/app/index.html +1 -1
  110. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  112. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  113. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  114. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  115. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  116. package/dist/web/standalone/.next/server/app-paths-manifest.json +9 -9
  117. package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
  118. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  119. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  120. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  121. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  122. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  123. package/package.json +3 -2
  124. package/packages/daemon/package.json +2 -2
  125. package/packages/mcp-server/dist/index.d.ts +3 -0
  126. package/packages/mcp-server/dist/index.d.ts.map +1 -1
  127. package/packages/mcp-server/dist/index.js +3 -0
  128. package/packages/mcp-server/dist/index.js.map +1 -1
  129. package/packages/mcp-server/dist/readers/graph.d.ts +87 -0
  130. package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -0
  131. package/packages/mcp-server/dist/readers/graph.js +655 -0
  132. package/packages/mcp-server/dist/readers/graph.js.map +1 -0
  133. package/packages/mcp-server/dist/readers/index.d.ts +2 -0
  134. package/packages/mcp-server/dist/readers/index.d.ts.map +1 -1
  135. package/packages/mcp-server/dist/readers/index.js +1 -0
  136. package/packages/mcp-server/dist/readers/index.js.map +1 -1
  137. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  138. package/packages/mcp-server/dist/server.js +65 -0
  139. package/packages/mcp-server/dist/server.js.map +1 -1
  140. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  141. package/packages/mcp-server/dist/workflow-tools.js +88 -6
  142. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  143. package/packages/mcp-server/package.json +2 -2
  144. package/packages/mcp-server/src/index.ts +15 -0
  145. package/packages/mcp-server/src/readers/graph.test.ts +604 -0
  146. package/packages/mcp-server/src/readers/graph.ts +855 -0
  147. package/packages/mcp-server/src/readers/index.ts +12 -0
  148. package/packages/mcp-server/src/server.ts +83 -0
  149. package/packages/mcp-server/src/workflow-tools.ts +95 -10
  150. package/packages/mcp-server/tsconfig.json +1 -0
  151. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -0
  152. package/packages/native/package.json +2 -2
  153. package/packages/native/tsconfig.tsbuildinfo +1 -0
  154. package/packages/pi-agent-core/package.json +1 -1
  155. package/packages/pi-agent-core/tsconfig.json +1 -0
  156. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -0
  157. package/packages/pi-ai/dist/index.d.ts +1 -9
  158. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  159. package/packages/pi-ai/dist/index.js +1 -9
  160. package/packages/pi-ai/dist/index.js.map +1 -1
  161. package/packages/pi-ai/dist/models/capability-patches.d.ts +19 -0
  162. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -0
  163. package/packages/pi-ai/dist/models/capability-patches.js +36 -0
  164. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -0
  165. package/packages/pi-ai/dist/{models.custom.d.ts → models/custom.d.ts} +1 -1
  166. package/packages/pi-ai/dist/models/custom.d.ts.map +1 -0
  167. package/packages/pi-ai/dist/{models.custom.js → models/custom.js} +4 -4
  168. package/packages/pi-ai/dist/models/custom.js.map +1 -0
  169. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts +1482 -0
  170. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts.map +1 -0
  171. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js +1484 -0
  172. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js.map +1 -0
  173. package/packages/pi-ai/dist/models/generated/anthropic.d.ts +377 -0
  174. package/packages/pi-ai/dist/models/generated/anthropic.d.ts.map +1 -0
  175. package/packages/pi-ai/dist/models/generated/anthropic.js +379 -0
  176. package/packages/pi-ai/dist/models/generated/anthropic.js.map +1 -0
  177. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts +700 -0
  178. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts.map +1 -0
  179. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js +702 -0
  180. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js.map +1 -0
  181. package/packages/pi-ai/dist/models/generated/cerebras.d.ts +71 -0
  182. package/packages/pi-ai/dist/models/generated/cerebras.d.ts.map +1 -0
  183. package/packages/pi-ai/dist/models/generated/cerebras.js +73 -0
  184. package/packages/pi-ai/dist/models/generated/cerebras.js.map +1 -0
  185. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts +590 -0
  186. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts.map +1 -0
  187. package/packages/pi-ai/dist/models/generated/github-copilot.js +444 -0
  188. package/packages/pi-ai/dist/models/generated/github-copilot.js.map +1 -0
  189. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts +156 -0
  190. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts.map +1 -0
  191. package/packages/pi-ai/dist/models/generated/google-antigravity.js +158 -0
  192. package/packages/pi-ai/dist/models/generated/google-antigravity.js.map +1 -0
  193. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts +105 -0
  194. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts.map +1 -0
  195. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js +107 -0
  196. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js.map +1 -0
  197. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts +207 -0
  198. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts.map +1 -0
  199. package/packages/pi-ai/dist/models/generated/google-vertex.js +209 -0
  200. package/packages/pi-ai/dist/models/generated/google-vertex.js.map +1 -0
  201. package/packages/pi-ai/dist/models/generated/google.d.ts +462 -0
  202. package/packages/pi-ai/dist/models/generated/google.d.ts.map +1 -0
  203. package/packages/pi-ai/dist/models/generated/google.js +464 -0
  204. package/packages/pi-ai/dist/models/generated/google.js.map +1 -0
  205. package/packages/pi-ai/dist/models/generated/groq.d.ts +309 -0
  206. package/packages/pi-ai/dist/models/generated/groq.d.ts.map +1 -0
  207. package/packages/pi-ai/dist/models/generated/groq.js +311 -0
  208. package/packages/pi-ai/dist/models/generated/groq.js.map +1 -0
  209. package/packages/pi-ai/dist/models/generated/huggingface.d.ts +383 -0
  210. package/packages/pi-ai/dist/models/generated/huggingface.d.ts.map +1 -0
  211. package/packages/pi-ai/dist/models/generated/huggingface.js +347 -0
  212. package/packages/pi-ai/dist/models/generated/huggingface.js.map +1 -0
  213. package/packages/pi-ai/dist/{models.generated.d.ts → models/generated/index.d.ts} +1 -1
  214. package/packages/pi-ai/dist/{models.generated.d.ts.map → models/generated/index.d.ts.map} +1 -1
  215. package/packages/pi-ai/dist/models/generated/index.js +51 -0
  216. package/packages/pi-ai/dist/models/generated/index.js.map +1 -0
  217. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts +37 -0
  218. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts.map +1 -0
  219. package/packages/pi-ai/dist/models/generated/kimi-coding.js +39 -0
  220. package/packages/pi-ai/dist/models/generated/kimi-coding.js.map +1 -0
  221. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts +105 -0
  222. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts.map +1 -0
  223. package/packages/pi-ai/dist/models/generated/minimax-cn.js +107 -0
  224. package/packages/pi-ai/dist/models/generated/minimax-cn.js.map +1 -0
  225. package/packages/pi-ai/dist/models/generated/minimax.d.ts +105 -0
  226. package/packages/pi-ai/dist/models/generated/minimax.d.ts.map +1 -0
  227. package/packages/pi-ai/dist/models/generated/minimax.js +107 -0
  228. package/packages/pi-ai/dist/models/generated/minimax.js.map +1 -0
  229. package/packages/pi-ai/dist/models/generated/mistral.d.ts +445 -0
  230. package/packages/pi-ai/dist/models/generated/mistral.d.ts.map +1 -0
  231. package/packages/pi-ai/dist/models/generated/mistral.js +447 -0
  232. package/packages/pi-ai/dist/models/generated/mistral.js.map +1 -0
  233. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +139 -0
  234. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -0
  235. package/packages/pi-ai/dist/models/generated/openai-codex.js +141 -0
  236. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -0
  237. package/packages/pi-ai/dist/models/generated/openai.d.ts +700 -0
  238. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -0
  239. package/packages/pi-ai/dist/models/generated/openai.js +702 -0
  240. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -0
  241. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts +122 -0
  242. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts.map +1 -0
  243. package/packages/pi-ai/dist/models/generated/opencode-go.js +124 -0
  244. package/packages/pi-ai/dist/models/generated/opencode-go.js.map +1 -0
  245. package/packages/pi-ai/dist/models/generated/opencode.d.ts +530 -0
  246. package/packages/pi-ai/dist/models/generated/opencode.d.ts.map +1 -0
  247. package/packages/pi-ai/dist/models/generated/opencode.js +532 -0
  248. package/packages/pi-ai/dist/models/generated/opencode.js.map +1 -0
  249. package/packages/pi-ai/dist/models/generated/openrouter.d.ts +4270 -0
  250. package/packages/pi-ai/dist/models/generated/openrouter.d.ts.map +1 -0
  251. package/packages/pi-ai/dist/models/generated/openrouter.js +4272 -0
  252. package/packages/pi-ai/dist/models/generated/openrouter.js.map +1 -0
  253. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts +2604 -0
  254. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts.map +1 -0
  255. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js +2606 -0
  256. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js.map +1 -0
  257. package/packages/pi-ai/dist/models/generated/xai.d.ts +411 -0
  258. package/packages/pi-ai/dist/models/generated/xai.d.ts.map +1 -0
  259. package/packages/pi-ai/dist/models/generated/xai.js +413 -0
  260. package/packages/pi-ai/dist/models/generated/xai.js.map +1 -0
  261. package/packages/pi-ai/dist/models/generated/zai.d.ts +276 -0
  262. package/packages/pi-ai/dist/models/generated/zai.d.ts.map +1 -0
  263. package/packages/pi-ai/dist/models/generated/zai.js +239 -0
  264. package/packages/pi-ai/dist/models/generated/zai.js.map +1 -0
  265. package/packages/pi-ai/dist/models/index.d.ts +27 -0
  266. package/packages/pi-ai/dist/models/index.d.ts.map +1 -0
  267. package/packages/pi-ai/dist/models/index.js +80 -0
  268. package/packages/pi-ai/dist/models/index.js.map +1 -0
  269. package/packages/pi-ai/dist/models.d.ts +1 -36
  270. package/packages/pi-ai/dist/models.d.ts.map +1 -1
  271. package/packages/pi-ai/dist/models.generated.test.js +1 -2
  272. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  273. package/packages/pi-ai/dist/models.js +3 -112
  274. package/packages/pi-ai/dist/models.js.map +1 -1
  275. package/packages/pi-ai/dist/models.test.js +6 -5
  276. package/packages/pi-ai/dist/models.test.js.map +1 -1
  277. package/packages/pi-ai/package.json +1 -1
  278. package/packages/pi-ai/scripts/generate-models.ts +74 -40
  279. package/packages/pi-ai/src/index.ts +1 -9
  280. package/packages/pi-ai/src/models/capability-patches.ts +40 -0
  281. package/packages/pi-ai/src/{models.custom.ts → models/custom.ts} +4 -4
  282. package/packages/pi-ai/src/models/generated/amazon-bedrock.ts +1486 -0
  283. package/packages/pi-ai/src/models/generated/anthropic.ts +381 -0
  284. package/packages/pi-ai/src/models/generated/azure-openai-responses.ts +704 -0
  285. package/packages/pi-ai/src/models/generated/cerebras.ts +75 -0
  286. package/packages/pi-ai/src/models/generated/github-copilot.ts +446 -0
  287. package/packages/pi-ai/src/models/generated/google-antigravity.ts +160 -0
  288. package/packages/pi-ai/src/models/generated/google-gemini-cli.ts +109 -0
  289. package/packages/pi-ai/src/models/generated/google-vertex.ts +211 -0
  290. package/packages/pi-ai/src/models/generated/google.ts +466 -0
  291. package/packages/pi-ai/src/models/generated/groq.ts +313 -0
  292. package/packages/pi-ai/src/models/generated/huggingface.ts +349 -0
  293. package/packages/pi-ai/src/models/generated/index.ts +52 -0
  294. package/packages/pi-ai/src/models/generated/kimi-coding.ts +41 -0
  295. package/packages/pi-ai/src/models/generated/minimax-cn.ts +109 -0
  296. package/packages/pi-ai/src/models/generated/minimax.ts +109 -0
  297. package/packages/pi-ai/src/models/generated/mistral.ts +449 -0
  298. package/packages/pi-ai/src/models/generated/openai-codex.ts +143 -0
  299. package/packages/pi-ai/src/models/generated/openai.ts +704 -0
  300. package/packages/pi-ai/src/models/generated/opencode-go.ts +126 -0
  301. package/packages/pi-ai/src/models/generated/opencode.ts +534 -0
  302. package/packages/pi-ai/src/models/generated/openrouter.ts +4274 -0
  303. package/packages/pi-ai/src/models/generated/vercel-ai-gateway.ts +2608 -0
  304. package/packages/pi-ai/src/models/generated/xai.ts +415 -0
  305. package/packages/pi-ai/src/models/generated/zai.ts +241 -0
  306. package/packages/pi-ai/src/models/index.ts +106 -0
  307. package/packages/pi-ai/src/models.generated.test.ts +1 -2
  308. package/packages/pi-ai/src/models.test.ts +6 -5
  309. package/packages/pi-ai/src/models.ts +3 -153
  310. package/packages/pi-ai/tsconfig.json +1 -0
  311. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -0
  312. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  313. package/packages/pi-coding-agent/dist/core/agent-session.js +8 -2
  314. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  315. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +472 -0
  316. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  317. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts +2 -0
  318. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts.map +1 -0
  319. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js +52 -0
  320. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js.map +1 -0
  321. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  322. package/packages/pi-coding-agent/dist/core/model-registry.js +2 -2
  323. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  324. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +11 -0
  325. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  326. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +1 -0
  327. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  328. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +23 -9
  329. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
  330. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts +11 -0
  331. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -0
  332. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +47 -0
  333. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -0
  334. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +8 -0
  335. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  336. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +68 -8
  337. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  338. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.d.ts.map +1 -1
  339. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js +22 -22
  340. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js.map +1 -1
  341. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  342. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +232 -18
  343. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  344. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts +2 -0
  345. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts.map +1 -0
  346. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +38 -0
  347. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -0
  348. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +14 -0
  349. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  350. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +70 -6
  351. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  352. package/packages/pi-coding-agent/src/core/agent-session.ts +12 -6
  353. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +612 -0
  354. package/packages/pi-coding-agent/src/core/model-registry-env-fallback.test.ts +59 -0
  355. package/packages/pi-coding-agent/src/core/model-registry.ts +2 -1
  356. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +19 -0
  357. package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +25 -10
  358. package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +67 -0
  359. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +83 -7
  360. package/packages/pi-coding-agent/src/modes/interactive/components/user-message.ts +23 -26
  361. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +298 -41
  362. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +44 -0
  363. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +92 -6
  364. package/packages/pi-coding-agent/src/types/ambient-modules.d.ts +69 -0
  365. package/packages/pi-coding-agent/tsconfig.json +3 -2
  366. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -0
  367. package/packages/pi-tui/package.json +1 -1
  368. package/packages/pi-tui/tsconfig.json +1 -0
  369. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -0
  370. package/packages/rpc-client/package.json +1 -1
  371. package/packages/rpc-client/tsconfig.json +1 -0
  372. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -0
  373. package/src/resources/extensions/gsd/activity-log.ts +21 -0
  374. package/src/resources/extensions/gsd/auto/detect-stuck.ts +12 -4
  375. package/src/resources/extensions/gsd/auto/loop-deps.ts +10 -0
  376. package/src/resources/extensions/gsd/auto/loop.ts +159 -10
  377. package/src/resources/extensions/gsd/auto/phases.ts +261 -10
  378. package/src/resources/extensions/gsd/auto/session.ts +10 -0
  379. package/src/resources/extensions/gsd/auto-dispatch.ts +16 -6
  380. package/src/resources/extensions/gsd/auto-model-selection.ts +69 -8
  381. package/src/resources/extensions/gsd/auto-post-unit.ts +238 -18
  382. package/src/resources/extensions/gsd/auto-prompts.ts +13 -0
  383. package/src/resources/extensions/gsd/auto-recovery.ts +29 -9
  384. package/src/resources/extensions/gsd/auto-unit-closeout.ts +25 -1
  385. package/src/resources/extensions/gsd/auto-verification.ts +129 -2
  386. package/src/resources/extensions/gsd/auto-worktree.ts +1 -0
  387. package/src/resources/extensions/gsd/auto.ts +41 -2
  388. package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +5 -3
  389. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +38 -8
  390. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +72 -8
  391. package/src/resources/extensions/gsd/cache.ts +16 -5
  392. package/src/resources/extensions/gsd/commands/catalog.ts +31 -1
  393. package/src/resources/extensions/gsd/commands/handlers/core.ts +5 -1
  394. package/src/resources/extensions/gsd/commands/handlers/ops.ts +25 -0
  395. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +74 -9
  396. package/src/resources/extensions/gsd/commands-add-tests.ts +137 -0
  397. package/src/resources/extensions/gsd/commands-backlog.ts +182 -0
  398. package/src/resources/extensions/gsd/commands-do.ts +109 -0
  399. package/src/resources/extensions/gsd/commands-extract-learnings.ts +304 -0
  400. package/src/resources/extensions/gsd/commands-maintenance.ts +6 -6
  401. package/src/resources/extensions/gsd/commands-pr-branch.ts +234 -0
  402. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +58 -4
  403. package/src/resources/extensions/gsd/commands-session-report.ts +101 -0
  404. package/src/resources/extensions/gsd/commands-ship.ts +219 -0
  405. package/src/resources/extensions/gsd/db-writer.ts +3 -5
  406. package/src/resources/extensions/gsd/docs/preferences-reference.md +16 -1
  407. package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +228 -0
  408. package/src/resources/extensions/gsd/ecosystem/loader.ts +201 -0
  409. package/src/resources/extensions/gsd/git-service.ts +68 -0
  410. package/src/resources/extensions/gsd/graph-context.ts +212 -0
  411. package/src/resources/extensions/gsd/gsd-db.ts +788 -3
  412. package/src/resources/extensions/gsd/guided-flow.ts +36 -2
  413. package/src/resources/extensions/gsd/index.ts +18 -2
  414. package/src/resources/extensions/gsd/init-wizard.ts +3 -2
  415. package/src/resources/extensions/gsd/journal.ts +30 -0
  416. package/src/resources/extensions/gsd/md-importer.ts +3 -5
  417. package/src/resources/extensions/gsd/memory-store.ts +31 -62
  418. package/src/resources/extensions/gsd/metrics.ts +26 -0
  419. package/src/resources/extensions/gsd/milestone-validation-gates.ts +13 -14
  420. package/src/resources/extensions/gsd/native-git-bridge.ts +11 -12
  421. package/src/resources/extensions/gsd/parallel-orchestrator.ts +40 -1
  422. package/src/resources/extensions/gsd/preferences-models.ts +20 -3
  423. package/src/resources/extensions/gsd/preferences-types.ts +38 -0
  424. package/src/resources/extensions/gsd/preferences-validation.ts +117 -2
  425. package/src/resources/extensions/gsd/preferences.ts +34 -0
  426. package/src/resources/extensions/gsd/prompts/add-tests.md +35 -0
  427. package/src/resources/extensions/gsd/safety/evidence-collector.ts +15 -31
  428. package/src/resources/extensions/gsd/session-lock.ts +14 -2
  429. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +20 -1
  430. package/src/resources/extensions/gsd/state.ts +9 -2
  431. package/src/resources/extensions/gsd/templates/PREFERENCES.md +19 -0
  432. package/src/resources/extensions/gsd/tests/artifacts-table-preserved-on-cache-invalidate.test.ts +177 -0
  433. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +7 -3
  434. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +20 -0
  435. package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +7 -3
  436. package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +272 -0
  437. package/src/resources/extensions/gsd/tests/auto-warning-noise-regression.test.ts +117 -0
  438. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +6 -2
  439. package/src/resources/extensions/gsd/tests/commands-backlog.test.ts +158 -0
  440. package/src/resources/extensions/gsd/tests/commands-do.test.ts +127 -0
  441. package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +340 -0
  442. package/src/resources/extensions/gsd/tests/commands-pr-branch.test.ts +68 -0
  443. package/src/resources/extensions/gsd/tests/commands-session-report.test.ts +82 -0
  444. package/src/resources/extensions/gsd/tests/commands-ship.test.ts +71 -0
  445. package/src/resources/extensions/gsd/tests/commands-workflow-custom.test.ts +14 -0
  446. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  447. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  448. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +154 -0
  449. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +10 -7
  450. package/src/resources/extensions/gsd/tests/graph-context.test.ts +337 -0
  451. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
  452. package/src/resources/extensions/gsd/tests/health-widget.test.ts +1 -1
  453. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +71 -4
  454. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -2
  455. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -3
  456. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +140 -0
  457. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  458. package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +2 -1
  459. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +40 -1
  460. package/src/resources/extensions/gsd/tests/preferences.test.ts +145 -0
  461. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +57 -2
  462. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +1 -1
  463. package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +180 -0
  464. package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -5
  465. package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +101 -0
  466. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +85 -0
  467. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +69 -0
  468. package/src/resources/extensions/gsd/tests/uok-flags.test.ts +39 -0
  469. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +70 -0
  470. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +85 -0
  471. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +35 -0
  472. package/src/resources/extensions/gsd/tests/uok-model-policy.test.ts +89 -0
  473. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +167 -0
  474. package/src/resources/extensions/gsd/tests/uok-preferences.test.ts +42 -0
  475. package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +39 -0
  476. package/src/resources/extensions/gsd/tests/workflow-logger-wiring.test.ts +223 -0
  477. package/src/resources/extensions/gsd/tools/complete-slice.ts +26 -0
  478. package/src/resources/extensions/gsd/tools/validate-milestone.ts +48 -3
  479. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +3 -11
  480. package/src/resources/extensions/gsd/triage-resolution.ts +2 -7
  481. package/src/resources/extensions/gsd/types.ts +14 -1
  482. package/src/resources/extensions/gsd/unit-ownership.ts +2 -2
  483. package/src/resources/extensions/gsd/uok/audit-toggle.ts +9 -0
  484. package/src/resources/extensions/gsd/uok/audit.ts +51 -0
  485. package/src/resources/extensions/gsd/uok/contracts.ts +135 -0
  486. package/src/resources/extensions/gsd/uok/execution-graph.ts +241 -0
  487. package/src/resources/extensions/gsd/uok/flags.ts +45 -0
  488. package/src/resources/extensions/gsd/uok/gate-runner.ts +146 -0
  489. package/src/resources/extensions/gsd/uok/gitops.ts +75 -0
  490. package/src/resources/extensions/gsd/uok/kernel.ts +105 -0
  491. package/src/resources/extensions/gsd/uok/loop-adapter.ts +162 -0
  492. package/src/resources/extensions/gsd/uok/model-policy.ts +112 -0
  493. package/src/resources/extensions/gsd/uok/plan-v2.ts +156 -0
  494. package/src/resources/extensions/gsd/workflow-logger.ts +27 -1
  495. package/src/resources/extensions/gsd/workflow-manifest.ts +9 -104
  496. package/src/resources/extensions/gsd/workflow-migration.ts +21 -29
  497. package/src/resources/extensions/gsd/workflow-projections.ts +8 -1
  498. package/src/resources/extensions/gsd/workflow-reconcile.ts +15 -15
  499. package/src/resources/extensions/ttsr/ttsr-manager.ts +10 -5
  500. package/packages/pi-ai/dist/models.custom.d.ts.map +0 -1
  501. package/packages/pi-ai/dist/models.custom.js.map +0 -1
  502. package/packages/pi-ai/dist/models.generated.js +0 -14343
  503. package/packages/pi-ai/dist/models.generated.js.map +0 -1
  504. package/packages/pi-ai/src/models.generated.ts +0 -14345
  505. /package/dist/web/standalone/.next/static/{YzIEI9sxJy4t5xgClF08g → Es_JWCfFZjIvYZShmjyye}/_buildManifest.js +0 -0
  506. /package/dist/web/standalone/.next/static/{YzIEI9sxJy4t5xgClF08g → Es_JWCfFZjIvYZShmjyye}/_ssgManifest.js +0 -0
@@ -36,7 +36,7 @@ import { captureAvailableSkills, resetSkillTelemetry, } from "./skill-telemetry.
36
36
  import { getRtkSessionSavings } from "../shared/rtk-session-stats.js";
37
37
  import { deactivateGSD } from "../shared/gsd-phase-state.js";
38
38
  import { initMetrics, resetMetrics, getLedger, getProjectTotals, formatCost, formatTokenCount, } from "./metrics.js";
39
- import { logWarning } from "./workflow-logger.js";
39
+ import { setLogBasePath, logWarning } from "./workflow-logger.js";
40
40
  import { homedir } from "node:os";
41
41
  import { join } from "node:path";
42
42
  import { pathToFileURL } from "node:url";
@@ -61,10 +61,12 @@ import { clearCmuxSidebar, logCmuxEvent, syncCmuxSidebar } from "../cmux/index.j
61
61
  // ── Extracted modules ──────────────────────────────────────────────────────
62
62
  import { startUnitSupervision } from "./auto-timers.js";
63
63
  import { runPostUnitVerification } from "./auto-verification.js";
64
- import { postUnitPreVerification, postUnitPostVerification, } from "./auto-post-unit.js";
64
+ import { autoCommitUnit, postUnitPreVerification, postUnitPostVerification, } from "./auto-post-unit.js";
65
65
  import { bootstrapAutoSession, openProjectDbIfPresent } from "./auto-start.js";
66
66
  import { initHealthWidget } from "./health-widget.js";
67
67
  import { autoLoop, resolveAgentEnd, resolveAgentEndCancelled, _resetPendingResolve, isSessionSwitchInFlight } from "./auto-loop.js";
68
+ import { runAutoLoopWithUok } from "./uok/kernel.js";
69
+ import { resolveUokFlags } from "./uok/flags.js";
68
70
  import { WorktreeResolver, } from "./worktree-resolver.js";
69
71
  import { reorderForCaching } from "./prompt-ordering.js";
70
72
  // ─── Session State ─────────────────────────────────────────────────────────
@@ -360,10 +362,22 @@ function clearUnitTimeout() {
360
362
  }
361
363
  /** Build snapshot metric opts. */
362
364
  function buildSnapshotOpts(_unitType, _unitId) {
365
+ const prefs = loadEffectiveGSDPreferences()?.preferences;
366
+ const uokFlags = resolveUokFlags(prefs);
363
367
  return {
364
368
  ...(s.autoStartTime > 0 ? { autoSessionKey: String(s.autoStartTime) } : {}),
365
369
  promptCharCount: s.lastPromptCharCount,
366
370
  baselineCharCount: s.lastBaselineCharCount,
371
+ traceId: s.currentTraceId ?? undefined,
372
+ turnId: s.currentTurnId ?? undefined,
373
+ ...(uokFlags.gitops
374
+ ? {
375
+ gitAction: uokFlags.gitopsTurnAction,
376
+ gitPush: uokFlags.gitopsTurnPush,
377
+ gitStatus: s.lastGitActionStatus ?? undefined,
378
+ gitError: s.lastGitActionFailure ?? undefined,
379
+ }
380
+ : {}),
367
381
  ...(s.currentUnitRouting ?? {}),
368
382
  };
369
383
  }
@@ -884,6 +898,7 @@ function buildLoopDeps() {
884
898
  getMainBranch,
885
899
  // Unit closeout + runtime records
886
900
  closeoutUnit,
901
+ autoCommitUnit,
887
902
  recordOutcome,
888
903
  writeLock,
889
904
  captureAvailableSkills,
@@ -1081,6 +1096,11 @@ export async function startAuto(ctx, pi, base, verboseMode, options) {
1081
1096
  s.stepMode = requestedStepMode;
1082
1097
  s.cmdCtx = ctx;
1083
1098
  s.basePath = base;
1099
+ // Ensure the workflow-logger audit log is pinned to the project root
1100
+ // even when auto-mode is entered via a path that bypasses the
1101
+ // bootstrap/dynamic-tools ensureDbOpen() → setLogBasePath() chain
1102
+ // (e.g. /clear resume, hot-reload).
1103
+ setLogBasePath(base);
1084
1104
  s.unitDispatchCount.clear();
1085
1105
  s.unitLifetimeDispatches.clear();
1086
1106
  if (!getLedger())
@@ -1159,7 +1179,13 @@ export async function startAuto(ctx, pi, base, verboseMode, options) {
1159
1179
  writeLock(lockBase(), "resuming", s.currentMilestoneId ?? "unknown");
1160
1180
  logCmuxEvent(loadEffectiveGSDPreferences()?.preferences, s.stepMode ? "Step-mode resumed." : "Auto-mode resumed.", "progress");
1161
1181
  captureProjectRootEnv(s.originalBasePath || s.basePath);
1162
- await autoLoop(ctx, pi, s, buildLoopDeps());
1182
+ await runAutoLoopWithUok({
1183
+ ctx,
1184
+ pi,
1185
+ s,
1186
+ deps: buildLoopDeps(),
1187
+ runLegacyLoop: autoLoop,
1188
+ });
1163
1189
  cleanupAfterLoopExit(ctx);
1164
1190
  return;
1165
1191
  }
@@ -1183,7 +1209,13 @@ export async function startAuto(ctx, pi, base, verboseMode, options) {
1183
1209
  }
1184
1210
  logCmuxEvent(loadEffectiveGSDPreferences()?.preferences, requestedStepMode ? "Step-mode started." : "Auto-mode started.", "progress");
1185
1211
  // Dispatch the first unit
1186
- await autoLoop(ctx, pi, s, buildLoopDeps());
1212
+ await runAutoLoopWithUok({
1213
+ ctx,
1214
+ pi,
1215
+ s,
1216
+ deps: buildLoopDeps(),
1217
+ runLegacyLoop: autoLoop,
1218
+ });
1187
1219
  cleanupAfterLoopExit(ctx);
1188
1220
  }
1189
1221
  // ─── Agent End Handler ────────────────────────────────────────────────────────
@@ -1,5 +1,6 @@
1
1
  import { getAutoDashboardData, startAuto } from "../auto.js";
2
2
  import { resetTransientRetryState } from "./agent-end-recovery.js";
3
+ import { resetSessionTimeoutState } from "../auto/phases.js";
3
4
  const defaultDeps = {
4
5
  getSnapshot: () => getAutoDashboardData(),
5
6
  startAuto,
@@ -14,10 +15,11 @@ export async function resumeAutoAfterProviderDelay(pi, ctx, deps = defaultDeps)
14
15
  ctx.ui.notify("Provider error recovery delay elapsed, but no paused auto-mode base path was available. Leaving auto-mode paused.", "warning");
15
16
  return "missing-base";
16
17
  }
17
- // Reset the transient retry counter before restarting — without this,
18
- // consecutiveTransientCount accumulates across pause/resume cycles and
19
- // permanently locks out auto-resume after MAX_TRANSIENT_AUTO_RESUMES errors.
18
+ // Reset retry counters before restarting — without this, counters
19
+ // accumulate across pause/resume cycles and permanently lock out
20
+ // auto-resume after their respective MAX thresholds.
20
21
  resetTransientRetryState();
22
+ resetSessionTimeoutState();
21
23
  await deps.startAuto(ctx, pi, snapshot.basePath, false, { step: snapshot.stepMode });
22
24
  return "resumed";
23
25
  }
@@ -1,7 +1,7 @@
1
1
  // GSD2 — Extension registration: wires all GSD tools, commands, and hooks into pi
2
- import { registerGSDCommand } from "../commands.js";
3
2
  import { registerExitCommand } from "../exit-command.js";
4
3
  import { registerWorktreeCommand } from "../worktree-command.js";
4
+ import { loadEcosystemExtensions } from "../ecosystem/loader.js";
5
5
  import { registerDbTools } from "./db-tools.js";
6
6
  import { registerDynamicTools } from "./dynamic-tools.js";
7
7
  import { registerJournalTools } from "./journal-tools.js";
@@ -9,6 +9,7 @@ import { registerQueryTools } from "./query-tools.js";
9
9
  import { registerHooks } from "./register-hooks.js";
10
10
  import { registerShortcuts } from "./register-shortcuts.js";
11
11
  import { writeCrashLog } from "./crash-log.js";
12
+ import { logWarning } from "../workflow-logger.js";
12
13
  export { writeCrashLog } from "./crash-log.js";
13
14
  export function handleRecoverableExtensionProcessError(err) {
14
15
  if (err.code === "EPIPE") {
@@ -52,20 +53,41 @@ function installEpipeGuard() {
52
53
  }
53
54
  }
54
55
  export function registerGsdExtension(pi) {
55
- registerGSDCommand(pi);
56
+ // Note: registerGSDCommand is called by index.ts before this function,
57
+ // so we intentionally skip it here to avoid double-registration.
56
58
  registerWorktreeCommand(pi);
57
59
  registerExitCommand(pi);
58
60
  installEpipeGuard();
61
+ // Ecosystem handlers captured by the GSDExtensionAPI wrapper for the
62
+ // GSD-owned `before_agent_start` dispatch step (#3338).
63
+ const ecosystemHandlers = [];
59
64
  pi.registerCommand("kill", {
60
65
  description: "Exit GSD immediately (no cleanup)",
61
66
  handler: async (_args, _ctx) => {
62
67
  process.exit(0);
63
68
  },
64
69
  });
65
- registerDynamicTools(pi);
66
- registerDbTools(pi);
67
- registerJournalTools(pi);
68
- registerQueryTools(pi);
69
- registerShortcuts(pi);
70
- registerHooks(pi);
70
+ // Wrap non-critical registrations individually so one failure
71
+ // doesn't prevent the others from loading.
72
+ const nonCriticalRegistrations = [
73
+ ["dynamic-tools", () => registerDynamicTools(pi)],
74
+ ["db-tools", () => registerDbTools(pi)],
75
+ ["journal-tools", () => registerJournalTools(pi)],
76
+ ["query-tools", () => registerQueryTools(pi)],
77
+ ["shortcuts", () => registerShortcuts(pi)],
78
+ ["hooks", () => registerHooks(pi, ecosystemHandlers)],
79
+ ["ecosystem", () => {
80
+ void loadEcosystemExtensions(pi, ecosystemHandlers).catch((err) => {
81
+ logWarning("ecosystem", `loader failed: ${err instanceof Error ? err.message : String(err)}`);
82
+ });
83
+ }],
84
+ ];
85
+ for (const [name, register] of nonCriticalRegistrations) {
86
+ try {
87
+ register();
88
+ }
89
+ catch (err) {
90
+ logWarning("bootstrap", `Failed to register ${name}: ${err instanceof Error ? err.message : String(err)}`);
91
+ }
92
+ }
71
93
  }
@@ -1,5 +1,7 @@
1
1
  import { join } from "node:path";
2
2
  import { isToolCallEventType } from "@gsd/pi-coding-agent";
3
+ import { updateSnapshot } from "../ecosystem/gsd-extension-api.js";
4
+ import { getEcosystemReadyPromise } from "../ecosystem/loader.js";
3
5
  import { buildMilestoneFileName, resolveMilestonePath, resolveSliceFile, resolveSlicePath } from "../paths.js";
4
6
  import { buildBeforeAgentStartResult } from "./system-context.js";
5
7
  import { handleAgentEnd } from "./agent-end-recovery.js";
@@ -29,7 +31,7 @@ async function syncServiceTierStatus(ctx) {
29
31
  const { getEffectiveServiceTier, formatServiceTierFooterStatus } = await import("../service-tier.js");
30
32
  ctx.ui.setStatus("gsd-fast", formatServiceTierFooterStatus(getEffectiveServiceTier(), ctx.model?.id));
31
33
  }
32
- export function registerHooks(pi) {
34
+ export function registerHooks(pi, ecosystemHandlers) {
33
35
  pi.on("session_start", async (_event, ctx) => {
34
36
  initNotificationStore(process.cwd());
35
37
  installNotifyInterceptor(ctx);
@@ -39,8 +41,12 @@ export function registerHooks(pi) {
39
41
  resetToolCallLoopGuard();
40
42
  resetAskUserQuestionsCache();
41
43
  await syncServiceTierStatus(ctx);
42
- const { prepareWorkflowMcpForProject } = await import("../workflow-mcp-auto-prep.js");
43
- prepareWorkflowMcpForProject(ctx, process.cwd());
44
+ // Skip MCP auto-prep when running inside an auto-worktree (see session_switch below).
45
+ const { isInAutoWorktree } = await import("../auto-worktree.js");
46
+ if (!isInAutoWorktree(process.cwd())) {
47
+ const { prepareWorkflowMcpForProject } = await import("../workflow-mcp-auto-prep.js");
48
+ prepareWorkflowMcpForProject(ctx, process.cwd());
49
+ }
44
50
  // Apply show_token_cost preference (#1515)
45
51
  try {
46
52
  const { loadEffectiveGSDPreferences } = await import("../preferences.js");
@@ -80,12 +86,56 @@ export function registerHooks(pi) {
80
86
  resetAskUserQuestionsCache();
81
87
  clearDiscussionFlowState();
82
88
  await syncServiceTierStatus(ctx);
83
- const { prepareWorkflowMcpForProject } = await import("../workflow-mcp-auto-prep.js");
84
- prepareWorkflowMcpForProject(ctx, process.cwd());
89
+ // Skip MCP auto-prep when running inside an auto-worktree. The worktree
90
+ // already has .mcp.json from createAutoWorktree, and re-running the writer
91
+ // post-chdir rewrites the file mid-run (non-idempotent due to cwd-relative
92
+ // CLI path resolution), dirtying the tree and breaking the milestone merge.
93
+ const { isInAutoWorktree } = await import("../auto-worktree.js");
94
+ if (!isInAutoWorktree(process.cwd())) {
95
+ const { prepareWorkflowMcpForProject } = await import("../workflow-mcp-auto-prep.js");
96
+ prepareWorkflowMcpForProject(ctx, process.cwd());
97
+ }
85
98
  loadToolApiKeys();
86
99
  });
87
100
  pi.on("before_agent_start", async (event, ctx) => {
88
- return buildBeforeAgentStartResult(event, ctx);
101
+ // Wait for ecosystem loader to finish (no-op after first turn).
102
+ await getEcosystemReadyPromise();
103
+ // GSD's own context injection (existing behavior — unchanged).
104
+ const gsdResult = await buildBeforeAgentStartResult(event, ctx);
105
+ // Refresh the snapshot used by ecosystem getPhase()/getActiveUnit().
106
+ // deriveState has its own ~100ms cache so this is cheap on repeat calls.
107
+ try {
108
+ const state = await deriveState(process.cwd());
109
+ updateSnapshot(state);
110
+ }
111
+ catch {
112
+ updateSnapshot(null);
113
+ }
114
+ // Chain ecosystem handlers using pi's runner.ts chaining protocol:
115
+ // each handler sees the systemPrompt mutated by prior handlers.
116
+ let currentSystemPrompt = gsdResult?.systemPrompt ?? event.systemPrompt;
117
+ // `any` because pi's BeforeAgentStartEventResult.message uses an internal
118
+ // CustomMessage type that's not re-exported (see ecosystem/gsd-extension-api.ts).
119
+ let lastMessage = gsdResult?.message;
120
+ for (const handler of ecosystemHandlers) {
121
+ try {
122
+ const r = await handler({ ...event, systemPrompt: currentSystemPrompt }, ctx);
123
+ if (r?.systemPrompt !== undefined)
124
+ currentSystemPrompt = r.systemPrompt;
125
+ if (r?.message)
126
+ lastMessage = r.message;
127
+ }
128
+ catch (err) {
129
+ safetyLogWarning("ecosystem", `before_agent_start handler failed: ${err instanceof Error ? err.message : String(err)}`);
130
+ }
131
+ }
132
+ // Compose result. Return undefined if nothing changed (preserves runner contract).
133
+ if (currentSystemPrompt === event.systemPrompt && !lastMessage)
134
+ return undefined;
135
+ return {
136
+ systemPrompt: currentSystemPrompt !== event.systemPrompt ? currentSystemPrompt : undefined,
137
+ message: lastMessage,
138
+ };
89
139
  });
90
140
  pi.on("agent_end", async (event, ctx) => {
91
141
  resetToolCallLoopGuard();
@@ -116,8 +166,10 @@ export function registerHooks(pi) {
116
166
  const state = await deriveState(basePath);
117
167
  if (!state.activeMilestone || !state.activeSlice || !state.activeTask)
118
168
  return;
119
- if (state.phase !== "executing")
120
- return;
169
+ // Write checkpoint for ALL phases, not just "executing" — discuss, research,
170
+ // and planning also carry in-memory state (user answers, gate verification)
171
+ // that would be lost on compaction (#4258).
172
+ // if (state.phase !== "executing") return;
121
173
  const sliceDir = resolveSlicePath(basePath, state.activeMilestone.id, state.activeSlice.id);
122
174
  if (!sliceDir)
123
175
  return;
@@ -240,7 +292,7 @@ export function registerHooks(pi) {
240
292
  pi.on("tool_call", async (event, ctx) => {
241
293
  if (!isAutoActive())
242
294
  return;
243
- safetyRecordToolCall(event.toolName, event.input);
295
+ safetyRecordToolCall(event.toolCallId, event.toolName, event.input);
244
296
  // Destructive command classification (warn only, never block)
245
297
  if (isToolCallEventType("bash", event)) {
246
298
  const classification = classifyCommand(event.input.command);
@@ -1,6 +1,6 @@
1
1
  // GSD Extension — Unified Cache Invalidation
2
2
  //
3
- // Three module-scoped caches exist across the GSD extension:
3
+ // Three module-scoped read caches exist across the GSD extension:
4
4
  // 1. State cache (state.ts) — memoized deriveState() result
5
5
  // 2. Path cache (paths.ts) — directory listing results (readdirSync)
6
6
  // 3. Parse cache (files.ts) — parsed markdown file results
@@ -8,20 +8,31 @@
8
8
  // After any file write that changes .gsd/ contents, all three must be
9
9
  // invalidated together to prevent stale reads. This module provides a
10
10
  // single function that clears all three atomically.
11
+ //
12
+ // NOTE: The DB `artifacts` table is NOT included here. Earlier versions
13
+ // called clearArtifacts() as part of this bundle (#793), intending to
14
+ // force deriveState() to re-parse from disk when files were edited
15
+ // out-of-band. But invalidateAllCaches() fires on every post-unit pass,
16
+ // so bundling a DESTRUCTIVE `DELETE FROM artifacts` with routine cache
17
+ // invalidation meant every row written by saveArtifactToDb / writeAndStore
18
+ // was wiped within seconds — leaving the milestone completed on disk but
19
+ // the `artifacts` table empty and the agent looping on "file exists but
20
+ // DB record missing" recovery calls. If a call site genuinely needs the
21
+ // artifact table cleared after an out-of-band file mutation, it should
22
+ // invoke clearArtifacts() from gsd-db.js explicitly — do not add it back
23
+ // here.
11
24
  import { invalidateStateCache } from './state.js';
12
25
  import { clearPathCache } from './paths.js';
13
26
  import { clearParseCache } from './files.js';
14
- import { clearArtifacts } from './gsd-db.js';
15
27
  /**
16
- * Invalidate all GSD runtime caches in one call.
28
+ * Invalidate all GSD runtime read caches in one call.
17
29
  *
18
30
  * Call this after file writes, milestone transitions, merge reconciliation,
19
31
  * or any operation that changes .gsd/ contents on disk. Forgetting to clear
20
- * any single cache causes stale reads (see #431, #793).
32
+ * any single cache causes stale reads (see #431).
21
33
  */
22
34
  export function invalidateAllCaches() {
23
35
  invalidateStateCache();
24
36
  clearPathCache();
25
37
  clearParseCache();
26
- clearArtifacts();
27
38
  }
@@ -4,7 +4,7 @@ import { join } from "node:path";
4
4
  import { loadRegistry } from "../workflow-templates.js";
5
5
  import { resolveProjectRoot } from "../worktree.js";
6
6
  const gsdHome = process.env.GSD_HOME || join(homedir(), ".gsd");
7
- export const GSD_COMMAND_DESCRIPTION = "GSD — Get Shit Done: /gsd help|start|templates|next|auto|stop|pause|status|widget|visualize|queue|quick|discuss|capture|triage|dispatch|history|undo|undo-task|reset-slice|rate|skip|export|cleanup|model|mode|prefs|config|keys|hooks|run-hook|skill-health|doctor|logs|forensics|changelog|migrate|remote|steer|knowledge|new-milestone|parallel|cmux|park|unpark|init|setup|inspect|extensions|update|fast|mcp|rethink|codebase|notifications";
7
+ export const GSD_COMMAND_DESCRIPTION = "GSD — Get Shit Done: /gsd help|start|templates|next|auto|stop|pause|status|widget|visualize|queue|quick|discuss|capture|triage|dispatch|history|undo|undo-task|reset-slice|rate|skip|export|cleanup|model|mode|prefs|config|keys|hooks|run-hook|skill-health|doctor|logs|forensics|changelog|migrate|remote|steer|knowledge|new-milestone|parallel|cmux|park|unpark|init|setup|inspect|extensions|update|fast|mcp|rethink|codebase|notifications|ship|do|session-report|backlog|pr-branch|add-tests|language";
8
8
  export const TOP_LEVEL_SUBCOMMANDS = [
9
9
  { cmd: "help", desc: "Categorized command reference with descriptions" },
10
10
  { cmd: "next", desc: "Explicit step mode (same as /gsd)" },
@@ -62,6 +62,13 @@ export const TOP_LEVEL_SUBCOMMANDS = [
62
62
  { cmd: "rethink", desc: "Conversational project reorganization — reorder, park, discard, add milestones" },
63
63
  { cmd: "workflow", desc: "Custom workflow lifecycle (new, run, list, validate, pause, resume)" },
64
64
  { cmd: "codebase", desc: "Generate, refresh, and inspect the codebase map cache (.gsd/CODEBASE.md)" },
65
+ { cmd: "ship", desc: "Create PR from milestone artifacts and open for review" },
66
+ { cmd: "do", desc: "Route freeform text to the right GSD command" },
67
+ { cmd: "session-report", desc: "Session cost, tokens, and work summary" },
68
+ { cmd: "backlog", desc: "Manage backlog items (add, promote, remove, list)" },
69
+ { cmd: "pr-branch", desc: "Create clean PR branch filtering .gsd/ commits" },
70
+ { cmd: "add-tests", desc: "Generate tests for completed slices" },
71
+ { cmd: "language", desc: "Set or clear the global response language (e.g. /gsd language Chinese)" },
65
72
  ];
66
73
  const NESTED_COMPLETIONS = {
67
74
  auto: [
@@ -231,6 +238,29 @@ const NESTED_COMPLETIONS = {
231
238
  { cmd: "stats", desc: "Show file count, description coverage, and generation time" },
232
239
  { cmd: "help", desc: "Show usage and available subcommands" },
233
240
  ],
241
+ ship: [
242
+ { cmd: "--dry-run", desc: "Preview PR without creating" },
243
+ { cmd: "--draft", desc: "Open as draft PR" },
244
+ { cmd: "--base", desc: "Override target branch (default: main)" },
245
+ { cmd: "--force", desc: "Ship even with pending tasks" },
246
+ ],
247
+ "session-report": [
248
+ { cmd: "--json", desc: "Machine-readable JSON output" },
249
+ { cmd: "--save", desc: "Save report to .gsd/reports/" },
250
+ ],
251
+ backlog: [
252
+ { cmd: "add", desc: "Add item to backlog" },
253
+ { cmd: "promote", desc: "Promote backlog item to active slice" },
254
+ { cmd: "remove", desc: "Remove backlog item" },
255
+ ],
256
+ "pr-branch": [
257
+ { cmd: "--dry-run", desc: "Preview what would be filtered" },
258
+ { cmd: "--name", desc: "Custom branch name" },
259
+ ],
260
+ language: [
261
+ { cmd: "off", desc: "Clear the language preference (revert to default)" },
262
+ { cmd: "clear", desc: "Alias for off — clear the language preference" },
263
+ ],
234
264
  };
235
265
  function filterOptions(partial, options, prefix = "") {
236
266
  const normalizedPrefix = prefix ? `${prefix} ` : "";
@@ -1,6 +1,6 @@
1
1
  import { computeProgressScore, formatProgressLine } from "../../progress-score.js";
2
2
  import { getGlobalGSDPreferencesPath, getProjectGSDPreferencesPath } from "../../preferences.js";
3
- import { ensurePreferencesFile, handlePrefs, handlePrefsMode, handlePrefsWizard } from "../../commands-prefs-wizard.js";
3
+ import { ensurePreferencesFile, handlePrefs, handlePrefsMode, handlePrefsWizard, handleLanguage } from "../../commands-prefs-wizard.js";
4
4
  import { runEnvironmentChecks } from "../../doctor-environment.js";
5
5
  import { deriveState } from "../../state.js";
6
6
  import { handleCmux } from "../../commands-cmux.js";
@@ -339,6 +339,10 @@ export async function handleCoreCommand(trimmed, ctx, pi) {
339
339
  await handlePrefs(trimmed.replace(/^prefs\s*/, "").trim(), ctx);
340
340
  return true;
341
341
  }
342
+ if (trimmed === "language" || trimmed.startsWith("language ")) {
343
+ await handleLanguage(trimmed.replace(/^language\s*/, "").trim(), ctx);
344
+ return true;
345
+ }
342
346
  if (trimmed === "cmux" || trimmed.startsWith("cmux ")) {
343
347
  await handleCmux(trimmed.replace(/^cmux\s*/, "").trim(), ctx);
344
348
  return true;
@@ -8,6 +8,9 @@ import { handleExport } from "../../export.js";
8
8
  import { handleHistory } from "../../history.js";
9
9
  import { handleUndo } from "../../undo.js";
10
10
  import { handleRemote } from "../../../remote-questions/mod.js";
11
+ import { handleShip } from "../../commands-ship.js";
12
+ import { handleSessionReport } from "../../commands-session-report.js";
13
+ import { handlePrBranch } from "../../commands-pr-branch.js";
11
14
  import { projectRoot } from "../context.js";
12
15
  export async function handleOpsCommand(trimmed, ctx, pi) {
13
16
  if (trimmed === "init") {
@@ -213,5 +216,27 @@ Examples:
213
216
  await handleCodebase(trimmed.replace(/^codebase\s*/, "").trim(), ctx, pi);
214
217
  return true;
215
218
  }
219
+ if (trimmed === "ship" || trimmed.startsWith("ship ")) {
220
+ await handleShip(trimmed.replace(/^ship\s*/, "").trim(), ctx, pi);
221
+ return true;
222
+ }
223
+ if (trimmed === "session-report" || trimmed.startsWith("session-report ")) {
224
+ await handleSessionReport(trimmed.replace(/^session-report\s*/, "").trim(), ctx);
225
+ return true;
226
+ }
227
+ if (trimmed === "pr-branch" || trimmed.startsWith("pr-branch ")) {
228
+ await handlePrBranch(trimmed.replace(/^pr-branch\s*/, "").trim(), ctx);
229
+ return true;
230
+ }
231
+ if (trimmed === "add-tests" || trimmed.startsWith("add-tests ")) {
232
+ const { handleAddTests } = await import("../../commands-add-tests.js");
233
+ await handleAddTests(trimmed.replace(/^add-tests\s*/, "").trim(), ctx, pi);
234
+ return true;
235
+ }
236
+ if (trimmed === "extract-learnings" || trimmed.startsWith("extract-learnings ")) {
237
+ const { handleExtractLearnings } = await import("../../commands-extract-learnings.js");
238
+ await handleExtractLearnings(trimmed.replace(/^extract-learnings\s*/, "").trim(), ctx, pi);
239
+ return true;
240
+ }
216
241
  return false;
217
242
  }
@@ -25,6 +25,61 @@ const WORKFLOW_USAGE = [
25
25
  " pause — Pause custom workflow auto-mode",
26
26
  " resume — Resume paused custom workflow auto-mode",
27
27
  ].join("\n");
28
+ function splitWorkflowRunArgs(input) {
29
+ const tokens = [];
30
+ let current = "";
31
+ let quote = null;
32
+ let escapeNext = false;
33
+ for (const ch of input) {
34
+ if (escapeNext) {
35
+ current += ch;
36
+ escapeNext = false;
37
+ continue;
38
+ }
39
+ if (ch === "\\") {
40
+ escapeNext = true;
41
+ continue;
42
+ }
43
+ if (quote) {
44
+ if (ch === quote) {
45
+ quote = null;
46
+ }
47
+ else {
48
+ current += ch;
49
+ }
50
+ continue;
51
+ }
52
+ if (ch === '"' || ch === "'") {
53
+ quote = ch;
54
+ continue;
55
+ }
56
+ if (/\s/.test(ch)) {
57
+ if (current) {
58
+ tokens.push(current);
59
+ current = "";
60
+ }
61
+ continue;
62
+ }
63
+ current += ch;
64
+ }
65
+ if (escapeNext)
66
+ current += "\\";
67
+ if (current)
68
+ tokens.push(current);
69
+ return tokens;
70
+ }
71
+ export function parseWorkflowRunArgs(args) {
72
+ const parts = splitWorkflowRunArgs(args);
73
+ const defName = parts[0] ?? "";
74
+ const overrides = {};
75
+ for (let i = 1; i < parts.length; i++) {
76
+ const eqIdx = parts[i].indexOf("=");
77
+ if (eqIdx > 0) {
78
+ overrides[parts[i].slice(0, eqIdx)] = parts[i].slice(eqIdx + 1);
79
+ }
80
+ }
81
+ return { defName, overrides };
82
+ }
28
83
  async function handleCustomWorkflow(sub, ctx, pi) {
29
84
  // Bare `/gsd workflow` — show usage
30
85
  if (!sub) {
@@ -43,15 +98,7 @@ async function handleCustomWorkflow(sub, ctx, pi) {
43
98
  ctx.ui.notify("Usage: /gsd workflow run <name> [param=value ...]", "warning");
44
99
  return true;
45
100
  }
46
- const parts = args.split(/\s+/);
47
- const defName = parts[0];
48
- const overrides = {};
49
- for (let i = 1; i < parts.length; i++) {
50
- const eqIdx = parts[i].indexOf("=");
51
- if (eqIdx > 0) {
52
- overrides[parts[i].slice(0, eqIdx)] = parts[i].slice(eqIdx + 1);
53
- }
54
- }
101
+ const { defName, overrides } = parseWorkflowRunArgs(args);
55
102
  try {
56
103
  const base = projectRoot();
57
104
  const runDir = createRun(base, defName, Object.keys(overrides).length > 0 ? overrides : undefined);
@@ -146,6 +193,18 @@ async function handleCustomWorkflow(sub, ctx, pi) {
146
193
  return true;
147
194
  }
148
195
  export async function handleWorkflowCommand(trimmed, ctx, pi) {
196
+ // ── /gsd do — natural language routing (must be early to route to other commands) ──
197
+ if (trimmed === "do" || trimmed.startsWith("do ")) {
198
+ const { handleDo } = await import("../../commands-do.js");
199
+ await handleDo(trimmed.replace(/^do\s*/, "").trim(), ctx, pi);
200
+ return true;
201
+ }
202
+ // ── Backlog management ──
203
+ if (trimmed === "backlog" || trimmed.startsWith("backlog ")) {
204
+ const { handleBacklog } = await import("../../commands-backlog.js");
205
+ await handleBacklog(trimmed.replace(/^backlog\s*/, "").trim(), ctx, pi);
206
+ return true;
207
+ }
149
208
  // ── Custom workflow commands (`/gsd workflow ...`) ──
150
209
  if (trimmed === "workflow" || trimmed.startsWith("workflow ")) {
151
210
  const sub = trimmed.slice("workflow".length).trim();
@@ -0,0 +1,111 @@
1
+ /**
2
+ * GSD Command — /gsd add-tests
3
+ *
4
+ * Generates tests for a completed slice by dispatching an LLM prompt
5
+ * with implementation context (summaries, changed files, test patterns).
6
+ */
7
+ import { existsSync, readFileSync, readdirSync } from "node:fs";
8
+ import { join } from "node:path";
9
+ import { deriveState } from "./state.js";
10
+ import { gsdRoot, resolveSliceFile } from "./paths.js";
11
+ import { loadPrompt } from "./prompt-loader.js";
12
+ function findLastCompletedSlice(basePath, milestoneId) {
13
+ // Scan disk for slices that have a SUMMARY.md (indicating completion)
14
+ const slicesDir = join(gsdRoot(basePath), "milestones", milestoneId, "slices");
15
+ if (!existsSync(slicesDir))
16
+ return null;
17
+ try {
18
+ const entries = readdirSync(slicesDir, { withFileTypes: true })
19
+ .filter((e) => e.isDirectory() && /^S\d+$/.test(e.name))
20
+ .sort((a, b) => b.name.localeCompare(a.name)); // reverse order — latest first
21
+ for (const entry of entries) {
22
+ const summaryPath = join(slicesDir, entry.name, `${entry.name}-SUMMARY.md`);
23
+ if (existsSync(summaryPath))
24
+ return entry.name;
25
+ }
26
+ }
27
+ catch {
28
+ // non-fatal
29
+ }
30
+ return null;
31
+ }
32
+ function readSliceSummary(basePath, milestoneId, sliceId) {
33
+ const summaryPath = resolveSliceFile(basePath, milestoneId, sliceId, "SUMMARY");
34
+ if (summaryPath && existsSync(summaryPath)) {
35
+ const content = readFileSync(summaryPath, "utf-8");
36
+ const titleMatch = content.match(/^#\s+(.+)/m);
37
+ return { title: titleMatch?.[1] ?? sliceId, content };
38
+ }
39
+ return { title: sliceId, content: "(no summary available)" };
40
+ }
41
+ function detectTestPatterns(basePath) {
42
+ const patterns = [];
43
+ // Check for common test configs
44
+ const checks = [
45
+ { file: "jest.config.ts", name: "Jest" },
46
+ { file: "jest.config.js", name: "Jest" },
47
+ { file: "vitest.config.ts", name: "Vitest" },
48
+ { file: "vitest.config.js", name: "Vitest" },
49
+ { file: ".mocharc.yml", name: "Mocha" },
50
+ ];
51
+ for (const check of checks) {
52
+ if (existsSync(join(basePath, check.file))) {
53
+ patterns.push(`Framework: ${check.name} (${check.file})`);
54
+ }
55
+ }
56
+ // Look for existing test files to infer patterns
57
+ const testDirs = ["tests", "test", "src/__tests__", "__tests__"];
58
+ for (const dir of testDirs) {
59
+ const fullDir = join(basePath, dir);
60
+ if (existsSync(fullDir)) {
61
+ try {
62
+ const files = readdirSync(fullDir).filter((f) => f.endsWith(".test.ts") || f.endsWith(".spec.ts") || f.endsWith(".test.js"));
63
+ if (files.length > 0) {
64
+ patterns.push(`Test directory: ${dir}/ (${files.length} test files)`);
65
+ // Read first test file for patterns
66
+ const samplePath = join(fullDir, files[0]);
67
+ const sample = readFileSync(samplePath, "utf-8").slice(0, 500);
68
+ patterns.push(`Sample pattern from ${files[0]}:\n${sample}`);
69
+ break;
70
+ }
71
+ }
72
+ catch {
73
+ // non-fatal
74
+ }
75
+ }
76
+ }
77
+ return patterns.length > 0 ? patterns.join("\n") : "No test framework detected. Use Node.js built-in test runner.";
78
+ }
79
+ export async function handleAddTests(args, ctx, pi) {
80
+ const basePath = process.cwd();
81
+ const state = await deriveState(basePath);
82
+ if (!state.activeMilestone) {
83
+ ctx.ui.notify("No active milestone.", "warning");
84
+ return;
85
+ }
86
+ const milestoneId = state.activeMilestone.id;
87
+ // Determine target
88
+ const targetId = args.trim() || findLastCompletedSlice(basePath, milestoneId);
89
+ if (!targetId) {
90
+ ctx.ui.notify("No completed slices found. Specify a slice ID: /gsd add-tests S03", "warning");
91
+ return;
92
+ }
93
+ // Gather context
94
+ const summary = readSliceSummary(basePath, milestoneId, targetId);
95
+ const testPatterns = detectTestPatterns(basePath);
96
+ ctx.ui.notify(`Generating tests for ${targetId}: "${summary.title}"...`, "info");
97
+ try {
98
+ const prompt = loadPrompt("add-tests", {
99
+ sliceId: targetId,
100
+ sliceTitle: summary.title,
101
+ sliceSummary: summary.content,
102
+ existingTestPatterns: testPatterns,
103
+ workingDirectory: basePath,
104
+ });
105
+ pi.sendMessage({ customType: "gsd-add-tests", content: prompt, display: false }, { triggerTurn: true });
106
+ }
107
+ catch (err) {
108
+ const msg = err instanceof Error ? err.message : String(err);
109
+ ctx.ui.notify(`Failed to dispatch test generation: ${msg}`, "error");
110
+ }
111
+ }