gsd-pi 2.73.1-dev.d987996 → 2.74.0-dev.0306a2e

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 (535) hide show
  1. package/dist/cli-web-branch.d.ts +4 -3
  2. package/dist/cli-web-branch.js +10 -7
  3. package/dist/cli.js +184 -206
  4. package/dist/headless-query.js +4 -1
  5. package/dist/help-text.js +23 -0
  6. package/dist/logo.d.ts +1 -1
  7. package/dist/logo.js +1 -1
  8. package/dist/onboarding.js +59 -53
  9. package/dist/resource-loader.js +2 -2
  10. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +68 -4
  11. package/dist/resources/extensions/gsd/activity-log.js +16 -0
  12. package/dist/resources/extensions/gsd/auto/detect-stuck.js +11 -4
  13. package/dist/resources/extensions/gsd/auto/loop.js +147 -10
  14. package/dist/resources/extensions/gsd/auto/phases.js +173 -13
  15. package/dist/resources/extensions/gsd/auto/session.js +10 -0
  16. package/dist/resources/extensions/gsd/auto-dispatch.js +22 -4
  17. package/dist/resources/extensions/gsd/auto-model-selection.js +105 -16
  18. package/dist/resources/extensions/gsd/auto-post-unit.js +254 -15
  19. package/dist/resources/extensions/gsd/auto-prompts.js +12 -0
  20. package/dist/resources/extensions/gsd/auto-start.js +23 -6
  21. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +13 -0
  22. package/dist/resources/extensions/gsd/auto-unit-closeout.js +18 -0
  23. package/dist/resources/extensions/gsd/auto-verification.js +186 -3
  24. package/dist/resources/extensions/gsd/auto.js +65 -12
  25. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +30 -8
  26. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +41 -2
  27. package/dist/resources/extensions/gsd/commands/catalog.js +26 -1
  28. package/dist/resources/extensions/gsd/commands/handlers/ops.js +25 -0
  29. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +68 -9
  30. package/dist/resources/extensions/gsd/commands-add-tests.js +111 -0
  31. package/dist/resources/extensions/gsd/commands-backlog.js +140 -0
  32. package/dist/resources/extensions/gsd/commands-do.js +79 -0
  33. package/dist/resources/extensions/gsd/commands-extract-learnings.js +225 -0
  34. package/dist/resources/extensions/gsd/commands-handlers.js +8 -2
  35. package/dist/resources/extensions/gsd/commands-maintenance.js +6 -6
  36. package/dist/resources/extensions/gsd/commands-pr-branch.js +180 -0
  37. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +1 -1
  38. package/dist/resources/extensions/gsd/commands-session-report.js +82 -0
  39. package/dist/resources/extensions/gsd/commands-ship.js +187 -0
  40. package/dist/resources/extensions/gsd/db-writer.js +3 -5
  41. package/dist/resources/extensions/gsd/docs/preferences-reference.md +15 -2
  42. package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +144 -0
  43. package/dist/resources/extensions/gsd/ecosystem/loader.js +145 -0
  44. package/dist/resources/extensions/gsd/git-service.js +49 -1
  45. package/dist/resources/extensions/gsd/graph-context.js +157 -0
  46. package/dist/resources/extensions/gsd/gsd-db.js +581 -2
  47. package/dist/resources/extensions/gsd/guided-flow.js +23 -0
  48. package/dist/resources/extensions/gsd/index.js +15 -2
  49. package/dist/resources/extensions/gsd/init-wizard.js +1 -0
  50. package/dist/resources/extensions/gsd/journal.js +27 -0
  51. package/dist/resources/extensions/gsd/md-importer.js +3 -4
  52. package/dist/resources/extensions/gsd/memory-store.js +19 -51
  53. package/dist/resources/extensions/gsd/metrics.js +19 -0
  54. package/dist/resources/extensions/gsd/milestone-validation-gates.js +13 -12
  55. package/dist/resources/extensions/gsd/native-git-bridge.js +7 -4
  56. package/dist/resources/extensions/gsd/notification-widget.js +2 -2
  57. package/dist/resources/extensions/gsd/parallel-orchestrator.js +33 -1
  58. package/dist/resources/extensions/gsd/preferences-models.js +63 -3
  59. package/dist/resources/extensions/gsd/preferences-types.js +2 -0
  60. package/dist/resources/extensions/gsd/preferences-validation.js +130 -2
  61. package/dist/resources/extensions/gsd/preferences.js +26 -0
  62. package/dist/resources/extensions/gsd/prompts/add-tests.md +35 -0
  63. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +12 -2
  64. package/dist/resources/extensions/gsd/state.js +66 -15
  65. package/dist/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  66. package/dist/resources/extensions/gsd/tools/complete-slice.js +20 -0
  67. package/dist/resources/extensions/gsd/tools/validate-milestone.js +39 -4
  68. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +3 -14
  69. package/dist/resources/extensions/gsd/triage-resolution.js +2 -5
  70. package/dist/resources/extensions/gsd/unit-ownership.js +1 -1
  71. package/dist/resources/extensions/gsd/uok/audit-toggle.js +7 -0
  72. package/dist/resources/extensions/gsd/uok/audit.js +40 -0
  73. package/dist/resources/extensions/gsd/uok/contracts.js +1 -0
  74. package/dist/resources/extensions/gsd/uok/execution-graph.js +179 -0
  75. package/dist/resources/extensions/gsd/uok/flags.js +29 -0
  76. package/dist/resources/extensions/gsd/uok/gate-runner.js +109 -0
  77. package/dist/resources/extensions/gsd/uok/gitops.js +53 -0
  78. package/dist/resources/extensions/gsd/uok/kernel.js +80 -0
  79. package/dist/resources/extensions/gsd/uok/loop-adapter.js +133 -0
  80. package/dist/resources/extensions/gsd/uok/model-policy.js +66 -0
  81. package/dist/resources/extensions/gsd/uok/plan-v2.js +132 -0
  82. package/dist/resources/extensions/gsd/workflow-logger.js +22 -0
  83. package/dist/resources/extensions/gsd/workflow-manifest.js +8 -69
  84. package/dist/resources/extensions/gsd/workflow-migration.js +21 -22
  85. package/dist/resources/extensions/gsd/workflow-projections.js +4 -1
  86. package/dist/resources/extensions/gsd/workflow-reconcile.js +14 -11
  87. package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
  88. package/dist/tsconfig.extensions.tsbuildinfo +1 -0
  89. package/dist/update-check.d.ts +1 -0
  90. package/dist/update-check.js +13 -5
  91. package/dist/update-cmd.js +4 -3
  92. package/dist/web/standalone/.next/BUILD_ID +1 -1
  93. package/dist/web/standalone/.next/app-path-routes-manifest.json +13 -13
  94. package/dist/web/standalone/.next/build-manifest.json +2 -2
  95. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  96. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  97. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  98. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  99. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  100. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  102. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  103. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  104. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  105. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  106. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  107. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  108. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  109. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  110. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  112. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  113. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +1 -1
  114. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +1 -1
  115. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
  116. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
  117. package/dist/web/standalone/.next/server/app/index.html +1 -1
  118. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  119. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  120. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  121. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  122. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  123. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  124. package/dist/web/standalone/.next/server/app-paths-manifest.json +13 -13
  125. package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
  126. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  127. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  128. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  129. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  130. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  131. package/package.json +3 -3
  132. package/packages/daemon/package.json +2 -2
  133. package/packages/mcp-server/dist/index.d.ts +3 -0
  134. package/packages/mcp-server/dist/index.d.ts.map +1 -1
  135. package/packages/mcp-server/dist/index.js +3 -0
  136. package/packages/mcp-server/dist/index.js.map +1 -1
  137. package/packages/mcp-server/dist/readers/graph.d.ts +87 -0
  138. package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -0
  139. package/packages/mcp-server/dist/readers/graph.js +655 -0
  140. package/packages/mcp-server/dist/readers/graph.js.map +1 -0
  141. package/packages/mcp-server/dist/readers/index.d.ts +2 -0
  142. package/packages/mcp-server/dist/readers/index.d.ts.map +1 -1
  143. package/packages/mcp-server/dist/readers/index.js +1 -0
  144. package/packages/mcp-server/dist/readers/index.js.map +1 -1
  145. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  146. package/packages/mcp-server/dist/server.js +65 -0
  147. package/packages/mcp-server/dist/server.js.map +1 -1
  148. package/packages/mcp-server/package.json +2 -2
  149. package/packages/mcp-server/src/index.ts +15 -0
  150. package/packages/mcp-server/src/readers/graph.test.ts +604 -0
  151. package/packages/mcp-server/src/readers/graph.ts +855 -0
  152. package/packages/mcp-server/src/readers/index.ts +12 -0
  153. package/packages/mcp-server/src/server.ts +83 -0
  154. package/packages/mcp-server/tsconfig.json +1 -0
  155. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -0
  156. package/packages/native/package.json +2 -2
  157. package/packages/native/tsconfig.tsbuildinfo +1 -0
  158. package/packages/pi-agent-core/package.json +1 -1
  159. package/packages/pi-agent-core/tsconfig.json +1 -0
  160. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -0
  161. package/packages/pi-ai/dist/index.d.ts +2 -9
  162. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  163. package/packages/pi-ai/dist/index.js +2 -9
  164. package/packages/pi-ai/dist/index.js.map +1 -1
  165. package/packages/pi-ai/dist/models/capability-patches.d.ts +19 -0
  166. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -0
  167. package/packages/pi-ai/dist/models/capability-patches.js +36 -0
  168. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -0
  169. package/packages/pi-ai/dist/{models.custom.d.ts → models/custom.d.ts} +1 -1
  170. package/packages/pi-ai/dist/models/custom.d.ts.map +1 -0
  171. package/packages/pi-ai/dist/{models.custom.js → models/custom.js} +4 -4
  172. package/packages/pi-ai/dist/models/custom.js.map +1 -0
  173. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts +1482 -0
  174. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts.map +1 -0
  175. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js +1484 -0
  176. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js.map +1 -0
  177. package/packages/pi-ai/dist/models/generated/anthropic.d.ts +377 -0
  178. package/packages/pi-ai/dist/models/generated/anthropic.d.ts.map +1 -0
  179. package/packages/pi-ai/dist/models/generated/anthropic.js +379 -0
  180. package/packages/pi-ai/dist/models/generated/anthropic.js.map +1 -0
  181. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts +700 -0
  182. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts.map +1 -0
  183. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js +702 -0
  184. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js.map +1 -0
  185. package/packages/pi-ai/dist/models/generated/cerebras.d.ts +71 -0
  186. package/packages/pi-ai/dist/models/generated/cerebras.d.ts.map +1 -0
  187. package/packages/pi-ai/dist/models/generated/cerebras.js +73 -0
  188. package/packages/pi-ai/dist/models/generated/cerebras.js.map +1 -0
  189. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts +590 -0
  190. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts.map +1 -0
  191. package/packages/pi-ai/dist/models/generated/github-copilot.js +444 -0
  192. package/packages/pi-ai/dist/models/generated/github-copilot.js.map +1 -0
  193. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts +156 -0
  194. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts.map +1 -0
  195. package/packages/pi-ai/dist/models/generated/google-antigravity.js +158 -0
  196. package/packages/pi-ai/dist/models/generated/google-antigravity.js.map +1 -0
  197. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts +105 -0
  198. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts.map +1 -0
  199. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js +107 -0
  200. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js.map +1 -0
  201. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts +207 -0
  202. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts.map +1 -0
  203. package/packages/pi-ai/dist/models/generated/google-vertex.js +209 -0
  204. package/packages/pi-ai/dist/models/generated/google-vertex.js.map +1 -0
  205. package/packages/pi-ai/dist/models/generated/google.d.ts +462 -0
  206. package/packages/pi-ai/dist/models/generated/google.d.ts.map +1 -0
  207. package/packages/pi-ai/dist/models/generated/google.js +464 -0
  208. package/packages/pi-ai/dist/models/generated/google.js.map +1 -0
  209. package/packages/pi-ai/dist/models/generated/groq.d.ts +309 -0
  210. package/packages/pi-ai/dist/models/generated/groq.d.ts.map +1 -0
  211. package/packages/pi-ai/dist/models/generated/groq.js +311 -0
  212. package/packages/pi-ai/dist/models/generated/groq.js.map +1 -0
  213. package/packages/pi-ai/dist/models/generated/huggingface.d.ts +383 -0
  214. package/packages/pi-ai/dist/models/generated/huggingface.d.ts.map +1 -0
  215. package/packages/pi-ai/dist/models/generated/huggingface.js +347 -0
  216. package/packages/pi-ai/dist/models/generated/huggingface.js.map +1 -0
  217. package/packages/pi-ai/dist/{models.generated.d.ts → models/generated/index.d.ts} +1 -1
  218. package/packages/pi-ai/dist/{models.generated.d.ts.map → models/generated/index.d.ts.map} +1 -1
  219. package/packages/pi-ai/dist/models/generated/index.js +51 -0
  220. package/packages/pi-ai/dist/models/generated/index.js.map +1 -0
  221. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts +37 -0
  222. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts.map +1 -0
  223. package/packages/pi-ai/dist/models/generated/kimi-coding.js +39 -0
  224. package/packages/pi-ai/dist/models/generated/kimi-coding.js.map +1 -0
  225. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts +105 -0
  226. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts.map +1 -0
  227. package/packages/pi-ai/dist/models/generated/minimax-cn.js +107 -0
  228. package/packages/pi-ai/dist/models/generated/minimax-cn.js.map +1 -0
  229. package/packages/pi-ai/dist/models/generated/minimax.d.ts +105 -0
  230. package/packages/pi-ai/dist/models/generated/minimax.d.ts.map +1 -0
  231. package/packages/pi-ai/dist/models/generated/minimax.js +107 -0
  232. package/packages/pi-ai/dist/models/generated/minimax.js.map +1 -0
  233. package/packages/pi-ai/dist/models/generated/mistral.d.ts +445 -0
  234. package/packages/pi-ai/dist/models/generated/mistral.d.ts.map +1 -0
  235. package/packages/pi-ai/dist/models/generated/mistral.js +447 -0
  236. package/packages/pi-ai/dist/models/generated/mistral.js.map +1 -0
  237. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +139 -0
  238. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -0
  239. package/packages/pi-ai/dist/models/generated/openai-codex.js +141 -0
  240. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -0
  241. package/packages/pi-ai/dist/models/generated/openai.d.ts +700 -0
  242. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -0
  243. package/packages/pi-ai/dist/models/generated/openai.js +702 -0
  244. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -0
  245. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts +122 -0
  246. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts.map +1 -0
  247. package/packages/pi-ai/dist/models/generated/opencode-go.js +124 -0
  248. package/packages/pi-ai/dist/models/generated/opencode-go.js.map +1 -0
  249. package/packages/pi-ai/dist/models/generated/opencode.d.ts +530 -0
  250. package/packages/pi-ai/dist/models/generated/opencode.d.ts.map +1 -0
  251. package/packages/pi-ai/dist/models/generated/opencode.js +532 -0
  252. package/packages/pi-ai/dist/models/generated/opencode.js.map +1 -0
  253. package/packages/pi-ai/dist/models/generated/openrouter.d.ts +4270 -0
  254. package/packages/pi-ai/dist/models/generated/openrouter.d.ts.map +1 -0
  255. package/packages/pi-ai/dist/models/generated/openrouter.js +4272 -0
  256. package/packages/pi-ai/dist/models/generated/openrouter.js.map +1 -0
  257. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts +2604 -0
  258. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts.map +1 -0
  259. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js +2606 -0
  260. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js.map +1 -0
  261. package/packages/pi-ai/dist/models/generated/xai.d.ts +411 -0
  262. package/packages/pi-ai/dist/models/generated/xai.d.ts.map +1 -0
  263. package/packages/pi-ai/dist/models/generated/xai.js +413 -0
  264. package/packages/pi-ai/dist/models/generated/xai.js.map +1 -0
  265. package/packages/pi-ai/dist/models/generated/zai.d.ts +276 -0
  266. package/packages/pi-ai/dist/models/generated/zai.d.ts.map +1 -0
  267. package/packages/pi-ai/dist/models/generated/zai.js +239 -0
  268. package/packages/pi-ai/dist/models/generated/zai.js.map +1 -0
  269. package/packages/pi-ai/dist/models/index.d.ts +27 -0
  270. package/packages/pi-ai/dist/models/index.d.ts.map +1 -0
  271. package/packages/pi-ai/dist/models/index.js +80 -0
  272. package/packages/pi-ai/dist/models/index.js.map +1 -0
  273. package/packages/pi-ai/dist/models.d.ts +1 -36
  274. package/packages/pi-ai/dist/models.d.ts.map +1 -1
  275. package/packages/pi-ai/dist/models.generated.test.js +1 -2
  276. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  277. package/packages/pi-ai/dist/models.js +3 -112
  278. package/packages/pi-ai/dist/models.js.map +1 -1
  279. package/packages/pi-ai/dist/models.test.js +6 -5
  280. package/packages/pi-ai/dist/models.test.js.map +1 -1
  281. package/packages/pi-ai/dist/utils/overflow.d.ts.map +1 -1
  282. package/packages/pi-ai/dist/utils/overflow.js +12 -0
  283. package/packages/pi-ai/dist/utils/overflow.js.map +1 -1
  284. package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts +2 -0
  285. package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts.map +1 -0
  286. package/packages/pi-ai/dist/utils/tests/overflow.test.js +50 -0
  287. package/packages/pi-ai/dist/utils/tests/overflow.test.js.map +1 -0
  288. package/packages/pi-ai/package.json +1 -1
  289. package/packages/pi-ai/scripts/generate-models.ts +74 -40
  290. package/packages/pi-ai/src/index.ts +5 -9
  291. package/packages/pi-ai/src/models/capability-patches.ts +40 -0
  292. package/packages/pi-ai/src/{models.custom.ts → models/custom.ts} +4 -4
  293. package/packages/pi-ai/src/models/generated/amazon-bedrock.ts +1486 -0
  294. package/packages/pi-ai/src/models/generated/anthropic.ts +381 -0
  295. package/packages/pi-ai/src/models/generated/azure-openai-responses.ts +704 -0
  296. package/packages/pi-ai/src/models/generated/cerebras.ts +75 -0
  297. package/packages/pi-ai/src/models/generated/github-copilot.ts +446 -0
  298. package/packages/pi-ai/src/models/generated/google-antigravity.ts +160 -0
  299. package/packages/pi-ai/src/models/generated/google-gemini-cli.ts +109 -0
  300. package/packages/pi-ai/src/models/generated/google-vertex.ts +211 -0
  301. package/packages/pi-ai/src/models/generated/google.ts +466 -0
  302. package/packages/pi-ai/src/models/generated/groq.ts +313 -0
  303. package/packages/pi-ai/src/models/generated/huggingface.ts +349 -0
  304. package/packages/pi-ai/src/models/generated/index.ts +52 -0
  305. package/packages/pi-ai/src/models/generated/kimi-coding.ts +41 -0
  306. package/packages/pi-ai/src/models/generated/minimax-cn.ts +109 -0
  307. package/packages/pi-ai/src/models/generated/minimax.ts +109 -0
  308. package/packages/pi-ai/src/models/generated/mistral.ts +449 -0
  309. package/packages/pi-ai/src/models/generated/openai-codex.ts +143 -0
  310. package/packages/pi-ai/src/models/generated/openai.ts +704 -0
  311. package/packages/pi-ai/src/models/generated/opencode-go.ts +126 -0
  312. package/packages/pi-ai/src/models/generated/opencode.ts +534 -0
  313. package/packages/pi-ai/src/models/generated/openrouter.ts +4274 -0
  314. package/packages/pi-ai/src/models/generated/vercel-ai-gateway.ts +2608 -0
  315. package/packages/pi-ai/src/models/generated/xai.ts +415 -0
  316. package/packages/pi-ai/src/models/generated/zai.ts +241 -0
  317. package/packages/pi-ai/src/models/index.ts +106 -0
  318. package/packages/pi-ai/src/models.generated.test.ts +1 -2
  319. package/packages/pi-ai/src/models.test.ts +6 -5
  320. package/packages/pi-ai/src/models.ts +3 -153
  321. package/packages/pi-ai/src/utils/overflow.ts +14 -1
  322. package/packages/pi-ai/src/utils/tests/overflow.test.ts +58 -0
  323. package/packages/pi-ai/tsconfig.json +1 -0
  324. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -0
  325. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +721 -8
  326. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  327. package/packages/pi-coding-agent/dist/core/compaction/utils.js +5 -5
  328. package/packages/pi-coding-agent/dist/core/compaction/utils.js.map +1 -1
  329. package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts +2 -0
  330. package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts.map +1 -0
  331. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js +45 -0
  332. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js.map +1 -0
  333. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts +2 -0
  334. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts.map +1 -0
  335. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js +52 -0
  336. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js.map +1 -0
  337. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  338. package/packages/pi-coding-agent/dist/core/model-registry.js +2 -2
  339. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  340. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +12 -2
  341. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  342. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +65 -28
  343. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
  344. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts +2 -1
  345. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -1
  346. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js +9 -3
  347. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js.map +1 -1
  348. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts +2 -0
  349. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts.map +1 -0
  350. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +52 -0
  351. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -0
  352. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  353. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +305 -20
  354. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  355. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts +2 -0
  356. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts.map +1 -0
  357. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +38 -0
  358. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -0
  359. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +13 -0
  360. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  361. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +59 -6
  362. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  363. package/packages/pi-coding-agent/package.json +1 -1
  364. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +884 -8
  365. package/packages/pi-coding-agent/src/core/compaction/utils.ts +5 -5
  366. package/packages/pi-coding-agent/src/core/compaction-utils.test.ts +50 -0
  367. package/packages/pi-coding-agent/src/core/model-registry-env-fallback.test.ts +59 -0
  368. package/packages/pi-coding-agent/src/core/model-registry.ts +2 -1
  369. package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +78 -32
  370. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +73 -0
  371. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.ts +9 -3
  372. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +381 -39
  373. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +44 -0
  374. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +79 -6
  375. package/packages/pi-coding-agent/src/types/ambient-modules.d.ts +69 -0
  376. package/packages/pi-coding-agent/tsconfig.json +3 -2
  377. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -0
  378. package/packages/pi-tui/dist/__tests__/tui.test.js +60 -1
  379. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  380. package/packages/pi-tui/dist/tui.d.ts +8 -0
  381. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  382. package/packages/pi-tui/dist/tui.js +32 -3
  383. package/packages/pi-tui/dist/tui.js.map +1 -1
  384. package/packages/pi-tui/package.json +1 -1
  385. package/packages/pi-tui/src/__tests__/tui.test.ts +76 -1
  386. package/packages/pi-tui/src/tui.ts +31 -3
  387. package/packages/pi-tui/tsconfig.json +1 -0
  388. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -0
  389. package/packages/rpc-client/package.json +1 -1
  390. package/packages/rpc-client/tsconfig.json +1 -0
  391. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -0
  392. package/pkg/package.json +1 -1
  393. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +107 -5
  394. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +111 -2
  395. package/src/resources/extensions/gsd/activity-log.ts +21 -0
  396. package/src/resources/extensions/gsd/auto/detect-stuck.ts +12 -4
  397. package/src/resources/extensions/gsd/auto/loop-deps.ts +10 -0
  398. package/src/resources/extensions/gsd/auto/loop.ts +159 -10
  399. package/src/resources/extensions/gsd/auto/phases.ts +213 -13
  400. package/src/resources/extensions/gsd/auto/session.ts +10 -0
  401. package/src/resources/extensions/gsd/auto-dispatch.ts +26 -10
  402. package/src/resources/extensions/gsd/auto-model-selection.ts +151 -16
  403. package/src/resources/extensions/gsd/auto-post-unit.ts +278 -16
  404. package/src/resources/extensions/gsd/auto-prompts.ts +13 -0
  405. package/src/resources/extensions/gsd/auto-start.ts +30 -6
  406. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +17 -0
  407. package/src/resources/extensions/gsd/auto-unit-closeout.ts +25 -1
  408. package/src/resources/extensions/gsd/auto-verification.ts +225 -3
  409. package/src/resources/extensions/gsd/auto.ts +72 -16
  410. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +38 -8
  411. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +52 -2
  412. package/src/resources/extensions/gsd/commands/catalog.ts +26 -1
  413. package/src/resources/extensions/gsd/commands/handlers/ops.ts +25 -0
  414. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +74 -9
  415. package/src/resources/extensions/gsd/commands-add-tests.ts +137 -0
  416. package/src/resources/extensions/gsd/commands-backlog.ts +182 -0
  417. package/src/resources/extensions/gsd/commands-do.ts +109 -0
  418. package/src/resources/extensions/gsd/commands-extract-learnings.ts +304 -0
  419. package/src/resources/extensions/gsd/commands-handlers.ts +8 -2
  420. package/src/resources/extensions/gsd/commands-maintenance.ts +6 -6
  421. package/src/resources/extensions/gsd/commands-pr-branch.ts +234 -0
  422. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +1 -1
  423. package/src/resources/extensions/gsd/commands-session-report.ts +101 -0
  424. package/src/resources/extensions/gsd/commands-ship.ts +219 -0
  425. package/src/resources/extensions/gsd/db-writer.ts +3 -5
  426. package/src/resources/extensions/gsd/docs/preferences-reference.md +15 -2
  427. package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +228 -0
  428. package/src/resources/extensions/gsd/ecosystem/loader.ts +201 -0
  429. package/src/resources/extensions/gsd/git-service.ts +68 -0
  430. package/src/resources/extensions/gsd/graph-context.ts +212 -0
  431. package/src/resources/extensions/gsd/gsd-db.ts +788 -3
  432. package/src/resources/extensions/gsd/guided-flow.ts +32 -0
  433. package/src/resources/extensions/gsd/index.ts +18 -2
  434. package/src/resources/extensions/gsd/init-wizard.ts +3 -2
  435. package/src/resources/extensions/gsd/journal.ts +30 -0
  436. package/src/resources/extensions/gsd/md-importer.ts +3 -5
  437. package/src/resources/extensions/gsd/memory-store.ts +31 -62
  438. package/src/resources/extensions/gsd/metrics.ts +26 -0
  439. package/src/resources/extensions/gsd/milestone-validation-gates.ts +13 -14
  440. package/src/resources/extensions/gsd/native-git-bridge.ts +11 -12
  441. package/src/resources/extensions/gsd/notification-widget.ts +2 -2
  442. package/src/resources/extensions/gsd/parallel-orchestrator.ts +40 -1
  443. package/src/resources/extensions/gsd/preferences-models.ts +61 -3
  444. package/src/resources/extensions/gsd/preferences-types.ts +44 -0
  445. package/src/resources/extensions/gsd/preferences-validation.ts +130 -2
  446. package/src/resources/extensions/gsd/preferences.ts +28 -0
  447. package/src/resources/extensions/gsd/prompts/add-tests.md +35 -0
  448. package/src/resources/extensions/gsd/session-lock.ts +14 -2
  449. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +20 -1
  450. package/src/resources/extensions/gsd/state.ts +80 -17
  451. package/src/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  452. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +9 -5
  453. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +20 -0
  454. package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +53 -0
  455. package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +7 -3
  456. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +51 -2
  457. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +6 -2
  458. package/src/resources/extensions/gsd/tests/commands-backlog.test.ts +158 -0
  459. package/src/resources/extensions/gsd/tests/commands-do.test.ts +127 -0
  460. package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +340 -0
  461. package/src/resources/extensions/gsd/tests/commands-pr-branch.test.ts +68 -0
  462. package/src/resources/extensions/gsd/tests/commands-session-report.test.ts +82 -0
  463. package/src/resources/extensions/gsd/tests/commands-ship.test.ts +71 -0
  464. package/src/resources/extensions/gsd/tests/commands-workflow-custom.test.ts +14 -0
  465. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +142 -0
  466. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  467. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  468. package/src/resources/extensions/gsd/tests/completed-at-reconcile.test.ts +42 -0
  469. package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -2
  470. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +3 -2
  471. package/src/resources/extensions/gsd/tests/derive-state-helpers.test.ts +68 -8
  472. package/src/resources/extensions/gsd/tests/derive-state.test.ts +3 -3
  473. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +154 -0
  474. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +10 -7
  475. package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +137 -1
  476. package/src/resources/extensions/gsd/tests/graph-context.test.ts +337 -0
  477. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
  478. package/src/resources/extensions/gsd/tests/health-widget.test.ts +1 -1
  479. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +4 -2
  480. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +68 -1
  481. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -2
  482. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -3
  483. package/src/resources/extensions/gsd/tests/model-isolation.test.ts +91 -2
  484. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +140 -0
  485. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  486. package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +2 -1
  487. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +40 -1
  488. package/src/resources/extensions/gsd/tests/preferences.test.ts +47 -0
  489. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +1 -1
  490. package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +180 -0
  491. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +5 -7
  492. package/src/resources/extensions/gsd/tests/token-profile.test.ts +9 -6
  493. package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +101 -0
  494. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +85 -0
  495. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +69 -0
  496. package/src/resources/extensions/gsd/tests/uok-flags.test.ts +39 -0
  497. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +70 -0
  498. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +85 -0
  499. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +35 -0
  500. package/src/resources/extensions/gsd/tests/uok-model-policy.test.ts +89 -0
  501. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +167 -0
  502. package/src/resources/extensions/gsd/tests/uok-preferences.test.ts +42 -0
  503. package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +179 -0
  504. package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +39 -0
  505. package/src/resources/extensions/gsd/tests/workflow-logger-wiring.test.ts +223 -0
  506. package/src/resources/extensions/gsd/tools/complete-slice.ts +26 -0
  507. package/src/resources/extensions/gsd/tools/validate-milestone.ts +48 -3
  508. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +3 -11
  509. package/src/resources/extensions/gsd/triage-resolution.ts +2 -7
  510. package/src/resources/extensions/gsd/types.ts +14 -1
  511. package/src/resources/extensions/gsd/unit-ownership.ts +2 -2
  512. package/src/resources/extensions/gsd/uok/audit-toggle.ts +9 -0
  513. package/src/resources/extensions/gsd/uok/audit.ts +51 -0
  514. package/src/resources/extensions/gsd/uok/contracts.ts +135 -0
  515. package/src/resources/extensions/gsd/uok/execution-graph.ts +241 -0
  516. package/src/resources/extensions/gsd/uok/flags.ts +45 -0
  517. package/src/resources/extensions/gsd/uok/gate-runner.ts +146 -0
  518. package/src/resources/extensions/gsd/uok/gitops.ts +75 -0
  519. package/src/resources/extensions/gsd/uok/kernel.ts +105 -0
  520. package/src/resources/extensions/gsd/uok/loop-adapter.ts +162 -0
  521. package/src/resources/extensions/gsd/uok/model-policy.ts +112 -0
  522. package/src/resources/extensions/gsd/uok/plan-v2.ts +156 -0
  523. package/src/resources/extensions/gsd/workflow-logger.ts +27 -1
  524. package/src/resources/extensions/gsd/workflow-manifest.ts +9 -104
  525. package/src/resources/extensions/gsd/workflow-migration.ts +21 -29
  526. package/src/resources/extensions/gsd/workflow-projections.ts +8 -1
  527. package/src/resources/extensions/gsd/workflow-reconcile.ts +15 -15
  528. package/src/resources/extensions/ttsr/ttsr-manager.ts +10 -5
  529. package/packages/pi-ai/dist/models.custom.d.ts.map +0 -1
  530. package/packages/pi-ai/dist/models.custom.js.map +0 -1
  531. package/packages/pi-ai/dist/models.generated.js +0 -14343
  532. package/packages/pi-ai/dist/models.generated.js.map +0 -1
  533. package/packages/pi-ai/src/models.generated.ts +0 -14345
  534. /package/dist/web/standalone/.next/static/{cGmbVq2su4f9tMpgIkG8u → tqdo0yKKYz6fJXQnIgbdx}/_buildManifest.js +0 -0
  535. /package/dist/web/standalone/.next/static/{cGmbVq2su4f9tMpgIkG8u → tqdo0yKKYz6fJXQnIgbdx}/_ssgManifest.js +0 -0
@@ -2,9 +2,10 @@
2
2
 
3
3
  import type { ExtensionAPI, ExtensionCommandContext } from "@gsd/pi-coding-agent";
4
4
 
5
- import { registerGSDCommand } from "../commands.js";
6
5
  import { registerExitCommand } from "../exit-command.js";
7
6
  import { registerWorktreeCommand } from "../worktree-command.js";
7
+ import type { GSDEcosystemBeforeAgentStartHandler } from "../ecosystem/gsd-extension-api.js";
8
+ import { loadEcosystemExtensions } from "../ecosystem/loader.js";
8
9
  import { registerDbTools } from "./db-tools.js";
9
10
  import { registerDynamicTools } from "./dynamic-tools.js";
10
11
  import { registerJournalTools } from "./journal-tools.js";
@@ -12,6 +13,7 @@ import { registerQueryTools } from "./query-tools.js";
12
13
  import { registerHooks } from "./register-hooks.js";
13
14
  import { registerShortcuts } from "./register-shortcuts.js";
14
15
  import { writeCrashLog } from "./crash-log.js";
16
+ import { logWarning } from "../workflow-logger.js";
15
17
 
16
18
  export { writeCrashLog } from "./crash-log.js";
17
19
 
@@ -58,12 +60,17 @@ function installEpipeGuard(): void {
58
60
  }
59
61
 
60
62
  export function registerGsdExtension(pi: ExtensionAPI): void {
61
- registerGSDCommand(pi);
63
+ // Note: registerGSDCommand is called by index.ts before this function,
64
+ // so we intentionally skip it here to avoid double-registration.
62
65
  registerWorktreeCommand(pi);
63
66
  registerExitCommand(pi);
64
67
 
65
68
  installEpipeGuard();
66
69
 
70
+ // Ecosystem handlers captured by the GSDExtensionAPI wrapper for the
71
+ // GSD-owned `before_agent_start` dispatch step (#3338).
72
+ const ecosystemHandlers: GSDEcosystemBeforeAgentStartHandler[] = [];
73
+
67
74
  pi.registerCommand("kill", {
68
75
  description: "Exit GSD immediately (no cleanup)",
69
76
  handler: async (_args: string, _ctx: ExtensionCommandContext) => {
@@ -71,10 +78,33 @@ export function registerGsdExtension(pi: ExtensionAPI): void {
71
78
  },
72
79
  });
73
80
 
74
- registerDynamicTools(pi);
75
- registerDbTools(pi);
76
- registerJournalTools(pi);
77
- registerQueryTools(pi);
78
- registerShortcuts(pi);
79
- registerHooks(pi);
81
+ // Wrap non-critical registrations individually so one failure
82
+ // doesn't prevent the others from loading.
83
+ const nonCriticalRegistrations: Array<[string, () => void]> = [
84
+ ["dynamic-tools", () => registerDynamicTools(pi)],
85
+ ["db-tools", () => registerDbTools(pi)],
86
+ ["journal-tools", () => registerJournalTools(pi)],
87
+ ["query-tools", () => registerQueryTools(pi)],
88
+ ["shortcuts", () => registerShortcuts(pi)],
89
+ ["hooks", () => registerHooks(pi, ecosystemHandlers)],
90
+ ["ecosystem", () => {
91
+ void loadEcosystemExtensions(pi, ecosystemHandlers).catch((err) => {
92
+ logWarning(
93
+ "ecosystem",
94
+ `loader failed: ${err instanceof Error ? err.message : String(err)}`,
95
+ );
96
+ });
97
+ }],
98
+ ];
99
+
100
+ for (const [name, register] of nonCriticalRegistrations) {
101
+ try {
102
+ register();
103
+ } catch (err) {
104
+ logWarning(
105
+ "bootstrap",
106
+ `Failed to register ${name}: ${err instanceof Error ? err.message : String(err)}`,
107
+ );
108
+ }
109
+ }
80
110
  }
@@ -3,6 +3,10 @@ import { join } from "node:path";
3
3
  import type { ExtensionAPI, ExtensionContext } from "@gsd/pi-coding-agent";
4
4
  import { isToolCallEventType } from "@gsd/pi-coding-agent";
5
5
 
6
+ import type { GSDEcosystemBeforeAgentStartHandler } from "../ecosystem/gsd-extension-api.js";
7
+ import { updateSnapshot } from "../ecosystem/gsd-extension-api.js";
8
+ import { getEcosystemReadyPromise } from "../ecosystem/loader.js";
9
+
6
10
  import { buildMilestoneFileName, resolveMilestonePath, resolveSliceFile, resolveSlicePath } from "../paths.js";
7
11
  import { buildBeforeAgentStartResult } from "./system-context.js";
8
12
  import { handleAgentEnd } from "./agent-end-recovery.js";
@@ -35,7 +39,10 @@ async function syncServiceTierStatus(ctx: ExtensionContext): Promise<void> {
35
39
  ctx.ui.setStatus("gsd-fast", formatServiceTierFooterStatus(getEffectiveServiceTier(), ctx.model?.id));
36
40
  }
37
41
 
38
- export function registerHooks(pi: ExtensionAPI): void {
42
+ export function registerHooks(
43
+ pi: ExtensionAPI,
44
+ ecosystemHandlers: GSDEcosystemBeforeAgentStartHandler[],
45
+ ): void {
39
46
  pi.on("session_start", async (_event, ctx) => {
40
47
  initNotificationStore(process.cwd());
41
48
  installNotifyInterceptor(ctx);
@@ -93,7 +100,50 @@ export function registerHooks(pi: ExtensionAPI): void {
93
100
  });
94
101
 
95
102
  pi.on("before_agent_start", async (event, ctx: ExtensionContext) => {
96
- return buildBeforeAgentStartResult(event, ctx);
103
+ // Wait for ecosystem loader to finish (no-op after first turn).
104
+ await getEcosystemReadyPromise();
105
+
106
+ // GSD's own context injection (existing behavior — unchanged).
107
+ const gsdResult = await buildBeforeAgentStartResult(event, ctx);
108
+
109
+ // Refresh the snapshot used by ecosystem getPhase()/getActiveUnit().
110
+ // deriveState has its own ~100ms cache so this is cheap on repeat calls.
111
+ try {
112
+ const state = await deriveState(process.cwd());
113
+ updateSnapshot(state);
114
+ } catch {
115
+ updateSnapshot(null);
116
+ }
117
+
118
+ // Chain ecosystem handlers using pi's runner.ts chaining protocol:
119
+ // each handler sees the systemPrompt mutated by prior handlers.
120
+ let currentSystemPrompt = gsdResult?.systemPrompt ?? event.systemPrompt;
121
+ // `any` because pi's BeforeAgentStartEventResult.message uses an internal
122
+ // CustomMessage type that's not re-exported (see ecosystem/gsd-extension-api.ts).
123
+ let lastMessage: any = gsdResult?.message;
124
+
125
+ for (const handler of ecosystemHandlers) {
126
+ try {
127
+ const r = await handler(
128
+ { ...event, systemPrompt: currentSystemPrompt },
129
+ ctx,
130
+ );
131
+ if (r?.systemPrompt !== undefined) currentSystemPrompt = r.systemPrompt;
132
+ if (r?.message) lastMessage = r.message;
133
+ } catch (err) {
134
+ safetyLogWarning(
135
+ "ecosystem",
136
+ `before_agent_start handler failed: ${err instanceof Error ? err.message : String(err)}`,
137
+ );
138
+ }
139
+ }
140
+
141
+ // Compose result. Return undefined if nothing changed (preserves runner contract).
142
+ if (currentSystemPrompt === event.systemPrompt && !lastMessage) return undefined;
143
+ return {
144
+ systemPrompt: currentSystemPrompt !== event.systemPrompt ? currentSystemPrompt : undefined,
145
+ message: lastMessage,
146
+ };
97
147
  });
98
148
 
99
149
  pi.on("agent_end", async (event, ctx: ExtensionContext) => {
@@ -15,7 +15,7 @@ export interface GsdCommandDefinition {
15
15
  type CompletionMap = Record<string, readonly GsdCommandDefinition[]>;
16
16
 
17
17
  export const GSD_COMMAND_DESCRIPTION =
18
- "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";
18
+ "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";
19
19
 
20
20
  export const TOP_LEVEL_SUBCOMMANDS: readonly GsdCommandDefinition[] = [
21
21
  { cmd: "help", desc: "Categorized command reference with descriptions" },
@@ -74,6 +74,12 @@ export const TOP_LEVEL_SUBCOMMANDS: readonly GsdCommandDefinition[] = [
74
74
  { cmd: "rethink", desc: "Conversational project reorganization — reorder, park, discard, add milestones" },
75
75
  { cmd: "workflow", desc: "Custom workflow lifecycle (new, run, list, validate, pause, resume)" },
76
76
  { cmd: "codebase", desc: "Generate, refresh, and inspect the codebase map cache (.gsd/CODEBASE.md)" },
77
+ { cmd: "ship", desc: "Create PR from milestone artifacts and open for review" },
78
+ { cmd: "do", desc: "Route freeform text to the right GSD command" },
79
+ { cmd: "session-report", desc: "Session cost, tokens, and work summary" },
80
+ { cmd: "backlog", desc: "Manage backlog items (add, promote, remove, list)" },
81
+ { cmd: "pr-branch", desc: "Create clean PR branch filtering .gsd/ commits" },
82
+ { cmd: "add-tests", desc: "Generate tests for completed slices" },
77
83
  ];
78
84
 
79
85
  const NESTED_COMPLETIONS: CompletionMap = {
@@ -244,6 +250,25 @@ const NESTED_COMPLETIONS: CompletionMap = {
244
250
  { cmd: "stats", desc: "Show file count, description coverage, and generation time" },
245
251
  { cmd: "help", desc: "Show usage and available subcommands" },
246
252
  ],
253
+ ship: [
254
+ { cmd: "--dry-run", desc: "Preview PR without creating" },
255
+ { cmd: "--draft", desc: "Open as draft PR" },
256
+ { cmd: "--base", desc: "Override target branch (default: main)" },
257
+ { cmd: "--force", desc: "Ship even with pending tasks" },
258
+ ],
259
+ "session-report": [
260
+ { cmd: "--json", desc: "Machine-readable JSON output" },
261
+ { cmd: "--save", desc: "Save report to .gsd/reports/" },
262
+ ],
263
+ backlog: [
264
+ { cmd: "add", desc: "Add item to backlog" },
265
+ { cmd: "promote", desc: "Promote backlog item to active slice" },
266
+ { cmd: "remove", desc: "Remove backlog item" },
267
+ ],
268
+ "pr-branch": [
269
+ { cmd: "--dry-run", desc: "Preview what would be filtered" },
270
+ { cmd: "--name", desc: "Custom branch name" },
271
+ ],
247
272
  };
248
273
 
249
274
  function filterOptions(
@@ -11,6 +11,9 @@ import { handleExport } from "../../export.js";
11
11
  import { handleHistory } from "../../history.js";
12
12
  import { handleUndo } from "../../undo.js";
13
13
  import { handleRemote } from "../../../remote-questions/mod.js";
14
+ import { handleShip } from "../../commands-ship.js";
15
+ import { handleSessionReport } from "../../commands-session-report.js";
16
+ import { handlePrBranch } from "../../commands-pr-branch.js";
14
17
  import { projectRoot } from "../context.js";
15
18
 
16
19
  export async function handleOpsCommand(trimmed: string, ctx: ExtensionCommandContext, pi: ExtensionAPI): Promise<boolean> {
@@ -216,5 +219,27 @@ Examples:
216
219
  await handleCodebase(trimmed.replace(/^codebase\s*/, "").trim(), ctx, pi);
217
220
  return true;
218
221
  }
222
+ if (trimmed === "ship" || trimmed.startsWith("ship ")) {
223
+ await handleShip(trimmed.replace(/^ship\s*/, "").trim(), ctx, pi);
224
+ return true;
225
+ }
226
+ if (trimmed === "session-report" || trimmed.startsWith("session-report ")) {
227
+ await handleSessionReport(trimmed.replace(/^session-report\s*/, "").trim(), ctx);
228
+ return true;
229
+ }
230
+ if (trimmed === "pr-branch" || trimmed.startsWith("pr-branch ")) {
231
+ await handlePrBranch(trimmed.replace(/^pr-branch\s*/, "").trim(), ctx);
232
+ return true;
233
+ }
234
+ if (trimmed === "add-tests" || trimmed.startsWith("add-tests ")) {
235
+ const { handleAddTests } = await import("../../commands-add-tests.js");
236
+ await handleAddTests(trimmed.replace(/^add-tests\s*/, "").trim(), ctx, pi);
237
+ return true;
238
+ }
239
+ if (trimmed === "extract-learnings" || trimmed.startsWith("extract-learnings ")) {
240
+ const { handleExtractLearnings } = await import("../../commands-extract-learnings.js");
241
+ await handleExtractLearnings(trimmed.replace(/^extract-learnings\s*/, "").trim(), ctx, pi);
242
+ return true;
243
+ }
219
244
  return false;
220
245
  }
@@ -38,6 +38,67 @@ const WORKFLOW_USAGE = [
38
38
  " resume — Resume paused custom workflow auto-mode",
39
39
  ].join("\n");
40
40
 
41
+ function splitWorkflowRunArgs(input: string): string[] {
42
+ const tokens: string[] = [];
43
+ let current = "";
44
+ let quote: '"' | "'" | null = null;
45
+ let escapeNext = false;
46
+
47
+ for (const ch of input) {
48
+ if (escapeNext) {
49
+ current += ch;
50
+ escapeNext = false;
51
+ continue;
52
+ }
53
+
54
+ if (ch === "\\") {
55
+ escapeNext = true;
56
+ continue;
57
+ }
58
+
59
+ if (quote) {
60
+ if (ch === quote) {
61
+ quote = null;
62
+ } else {
63
+ current += ch;
64
+ }
65
+ continue;
66
+ }
67
+
68
+ if (ch === '"' || ch === "'") {
69
+ quote = ch;
70
+ continue;
71
+ }
72
+
73
+ if (/\s/.test(ch)) {
74
+ if (current) {
75
+ tokens.push(current);
76
+ current = "";
77
+ }
78
+ continue;
79
+ }
80
+
81
+ current += ch;
82
+ }
83
+
84
+ if (escapeNext) current += "\\";
85
+ if (current) tokens.push(current);
86
+ return tokens;
87
+ }
88
+
89
+ export function parseWorkflowRunArgs(args: string): { defName: string; overrides: Record<string, string> } {
90
+ const parts = splitWorkflowRunArgs(args);
91
+ const defName = parts[0] ?? "";
92
+ const overrides: Record<string, string> = {};
93
+ for (let i = 1; i < parts.length; i++) {
94
+ const eqIdx = parts[i].indexOf("=");
95
+ if (eqIdx > 0) {
96
+ overrides[parts[i].slice(0, eqIdx)] = parts[i].slice(eqIdx + 1);
97
+ }
98
+ }
99
+ return { defName, overrides };
100
+ }
101
+
41
102
  async function handleCustomWorkflow(
42
103
  sub: string,
43
104
  ctx: ExtensionCommandContext,
@@ -62,15 +123,7 @@ async function handleCustomWorkflow(
62
123
  ctx.ui.notify("Usage: /gsd workflow run <name> [param=value ...]", "warning");
63
124
  return true;
64
125
  }
65
- const parts = args.split(/\s+/);
66
- const defName = parts[0];
67
- const overrides: Record<string, string> = {};
68
- for (let i = 1; i < parts.length; i++) {
69
- const eqIdx = parts[i].indexOf("=");
70
- if (eqIdx > 0) {
71
- overrides[parts[i].slice(0, eqIdx)] = parts[i].slice(eqIdx + 1);
72
- }
73
- }
126
+ const { defName, overrides } = parseWorkflowRunArgs(args);
74
127
  try {
75
128
  const base = projectRoot();
76
129
  const runDir = createRun(base, defName, Object.keys(overrides).length > 0 ? overrides : undefined);
@@ -168,6 +221,18 @@ async function handleCustomWorkflow(
168
221
  }
169
222
 
170
223
  export async function handleWorkflowCommand(trimmed: string, ctx: ExtensionCommandContext, pi: ExtensionAPI): Promise<boolean> {
224
+ // ── /gsd do — natural language routing (must be early to route to other commands) ──
225
+ if (trimmed === "do" || trimmed.startsWith("do ")) {
226
+ const { handleDo } = await import("../../commands-do.js");
227
+ await handleDo(trimmed.replace(/^do\s*/, "").trim(), ctx, pi);
228
+ return true;
229
+ }
230
+ // ── Backlog management ──
231
+ if (trimmed === "backlog" || trimmed.startsWith("backlog ")) {
232
+ const { handleBacklog } = await import("../../commands-backlog.js");
233
+ await handleBacklog(trimmed.replace(/^backlog\s*/, "").trim(), ctx, pi);
234
+ return true;
235
+ }
171
236
  // ── Custom workflow commands (`/gsd workflow ...`) ──
172
237
  if (trimmed === "workflow" || trimmed.startsWith("workflow ")) {
173
238
  const sub = trimmed.slice("workflow".length).trim();
@@ -0,0 +1,137 @@
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
+
8
+ import type { ExtensionAPI, ExtensionCommandContext } from "@gsd/pi-coding-agent";
9
+
10
+ import { existsSync, readFileSync, readdirSync } from "node:fs";
11
+ import { join } from "node:path";
12
+
13
+ import { deriveState } from "./state.js";
14
+ import { gsdRoot, resolveSliceFile } from "./paths.js";
15
+ import { loadPrompt } from "./prompt-loader.js";
16
+
17
+ function findLastCompletedSlice(basePath: string, milestoneId: string): string | null {
18
+ // Scan disk for slices that have a SUMMARY.md (indicating completion)
19
+ const slicesDir = join(gsdRoot(basePath), "milestones", milestoneId, "slices");
20
+ if (!existsSync(slicesDir)) return null;
21
+
22
+ try {
23
+ const entries = readdirSync(slicesDir, { withFileTypes: true })
24
+ .filter((e) => e.isDirectory() && /^S\d+$/.test(e.name))
25
+ .sort((a, b) => b.name.localeCompare(a.name)); // reverse order — latest first
26
+
27
+ for (const entry of entries) {
28
+ const summaryPath = join(slicesDir, entry.name, `${entry.name}-SUMMARY.md`);
29
+ if (existsSync(summaryPath)) return entry.name;
30
+ }
31
+ } catch {
32
+ // non-fatal
33
+ }
34
+ return null;
35
+ }
36
+
37
+ function readSliceSummary(basePath: string, milestoneId: string, sliceId: string): { title: string; content: string } {
38
+ const summaryPath = resolveSliceFile(basePath, milestoneId, sliceId, "SUMMARY");
39
+ if (summaryPath && existsSync(summaryPath)) {
40
+ const content = readFileSync(summaryPath, "utf-8");
41
+ const titleMatch = content.match(/^#\s+(.+)/m);
42
+ return { title: titleMatch?.[1] ?? sliceId, content };
43
+ }
44
+ return { title: sliceId, content: "(no summary available)" };
45
+ }
46
+
47
+ function detectTestPatterns(basePath: string): string {
48
+ const patterns: string[] = [];
49
+
50
+ // Check for common test configs
51
+ const checks = [
52
+ { file: "jest.config.ts", name: "Jest" },
53
+ { file: "jest.config.js", name: "Jest" },
54
+ { file: "vitest.config.ts", name: "Vitest" },
55
+ { file: "vitest.config.js", name: "Vitest" },
56
+ { file: ".mocharc.yml", name: "Mocha" },
57
+ ];
58
+
59
+ for (const check of checks) {
60
+ if (existsSync(join(basePath, check.file))) {
61
+ patterns.push(`Framework: ${check.name} (${check.file})`);
62
+ }
63
+ }
64
+
65
+ // Look for existing test files to infer patterns
66
+ const testDirs = ["tests", "test", "src/__tests__", "__tests__"];
67
+ for (const dir of testDirs) {
68
+ const fullDir = join(basePath, dir);
69
+ if (existsSync(fullDir)) {
70
+ try {
71
+ const files = readdirSync(fullDir).filter((f) => f.endsWith(".test.ts") || f.endsWith(".spec.ts") || f.endsWith(".test.js"));
72
+ if (files.length > 0) {
73
+ patterns.push(`Test directory: ${dir}/ (${files.length} test files)`);
74
+ // Read first test file for patterns
75
+ const samplePath = join(fullDir, files[0]);
76
+ const sample = readFileSync(samplePath, "utf-8").slice(0, 500);
77
+ patterns.push(`Sample pattern from ${files[0]}:\n${sample}`);
78
+ break;
79
+ }
80
+ } catch {
81
+ // non-fatal
82
+ }
83
+ }
84
+ }
85
+
86
+ return patterns.length > 0 ? patterns.join("\n") : "No test framework detected. Use Node.js built-in test runner.";
87
+ }
88
+
89
+ export async function handleAddTests(
90
+ args: string,
91
+ ctx: ExtensionCommandContext,
92
+ pi: ExtensionAPI,
93
+ ): Promise<void> {
94
+ const basePath = process.cwd();
95
+ const state = await deriveState(basePath);
96
+
97
+ if (!state.activeMilestone) {
98
+ ctx.ui.notify("No active milestone.", "warning");
99
+ return;
100
+ }
101
+
102
+ const milestoneId = state.activeMilestone.id;
103
+
104
+ // Determine target
105
+ const targetId = args.trim() || findLastCompletedSlice(basePath, milestoneId);
106
+ if (!targetId) {
107
+ ctx.ui.notify(
108
+ "No completed slices found. Specify a slice ID: /gsd add-tests S03",
109
+ "warning",
110
+ );
111
+ return;
112
+ }
113
+
114
+ // Gather context
115
+ const summary = readSliceSummary(basePath, milestoneId, targetId);
116
+ const testPatterns = detectTestPatterns(basePath);
117
+
118
+ ctx.ui.notify(`Generating tests for ${targetId}: "${summary.title}"...`, "info");
119
+
120
+ try {
121
+ const prompt = loadPrompt("add-tests", {
122
+ sliceId: targetId,
123
+ sliceTitle: summary.title,
124
+ sliceSummary: summary.content,
125
+ existingTestPatterns: testPatterns,
126
+ workingDirectory: basePath,
127
+ });
128
+
129
+ pi.sendMessage(
130
+ { customType: "gsd-add-tests", content: prompt, display: false },
131
+ { triggerTurn: true },
132
+ );
133
+ } catch (err) {
134
+ const msg = err instanceof Error ? err.message : String(err);
135
+ ctx.ui.notify(`Failed to dispatch test generation: ${msg}`, "error");
136
+ }
137
+ }
@@ -0,0 +1,182 @@
1
+ /**
2
+ * GSD Command — /gsd backlog
3
+ *
4
+ * Structured backlog management with 999.x numbering.
5
+ * Items stored in .gsd/BACKLOG.md as markdown checklist.
6
+ * Items can be promoted to active slices via add-slice.
7
+ */
8
+
9
+ import type { ExtensionAPI, ExtensionCommandContext } from "@gsd/pi-coding-agent";
10
+
11
+ import { existsSync, readFileSync, writeFileSync, mkdirSync } from "node:fs";
12
+ import { join, dirname } from "node:path";
13
+
14
+ import { gsdRoot } from "./paths.js";
15
+
16
+ interface BacklogItem {
17
+ id: string;
18
+ title: string;
19
+ done: boolean;
20
+ note: string;
21
+ }
22
+
23
+ function backlogPath(basePath: string): string {
24
+ return join(gsdRoot(basePath), "BACKLOG.md");
25
+ }
26
+
27
+ function parseBacklog(basePath: string): BacklogItem[] {
28
+ const filePath = backlogPath(basePath);
29
+ if (!existsSync(filePath)) return [];
30
+
31
+ const content = readFileSync(filePath, "utf-8");
32
+ const items: BacklogItem[] = [];
33
+
34
+ for (const line of content.split("\n")) {
35
+ const match = line.match(/^- \[([ x])\] (999\.\d+) — (.+?)(?:\s*\((.+)\))?$/);
36
+ if (match) {
37
+ items.push({
38
+ id: match[2],
39
+ title: match[3].trim(),
40
+ done: match[1] === "x",
41
+ note: match[4] ?? "",
42
+ });
43
+ }
44
+ }
45
+
46
+ return items;
47
+ }
48
+
49
+ function writeBacklog(basePath: string, items: BacklogItem[]): void {
50
+ const filePath = backlogPath(basePath);
51
+ mkdirSync(dirname(filePath), { recursive: true });
52
+ const lines = ["# Backlog\n"];
53
+ for (const item of items) {
54
+ const check = item.done ? "x" : " ";
55
+ const note = item.note ? ` (${item.note})` : "";
56
+ lines.push(`- [${check}] ${item.id} — ${item.title}${note}`);
57
+ }
58
+ lines.push(""); // trailing newline
59
+ writeFileSync(filePath, lines.join("\n"), "utf-8");
60
+ }
61
+
62
+ function nextBacklogId(items: BacklogItem[]): string {
63
+ let maxNum = 0;
64
+ for (const item of items) {
65
+ const match = item.id.match(/^999\.(\d+)$/);
66
+ if (match) {
67
+ const num = parseInt(match[1], 10);
68
+ if (num > maxNum) maxNum = num;
69
+ }
70
+ }
71
+ return `999.${maxNum + 1}`;
72
+ }
73
+
74
+ async function listBacklog(basePath: string, ctx: ExtensionCommandContext): Promise<void> {
75
+ const items = parseBacklog(basePath);
76
+ if (items.length === 0) {
77
+ ctx.ui.notify("Backlog is empty. Add items with /gsd backlog add <title>", "info");
78
+ return;
79
+ }
80
+
81
+ const lines = ["Backlog:\n"];
82
+ for (const item of items) {
83
+ const status = item.done ? "✓" : "○";
84
+ const note = item.note ? ` (${item.note})` : "";
85
+ lines.push(` ${status} ${item.id} — ${item.title}${note}`);
86
+ }
87
+ const pending = items.filter((i) => !i.done).length;
88
+ lines.push(`\n${pending} pending, ${items.length - pending} promoted/done`);
89
+ ctx.ui.notify(lines.join("\n"), "info");
90
+ }
91
+
92
+ async function addBacklogItem(basePath: string, title: string, ctx: ExtensionCommandContext): Promise<void> {
93
+ if (!title) {
94
+ ctx.ui.notify("Usage: /gsd backlog add <title>", "warning");
95
+ return;
96
+ }
97
+
98
+ const items = parseBacklog(basePath);
99
+ const id = nextBacklogId(items);
100
+ const date = new Date().toISOString().slice(0, 10);
101
+
102
+ items.push({ id, title: title.replace(/^['"]|['"]$/g, ""), done: false, note: `added ${date}` });
103
+ writeBacklog(basePath, items);
104
+
105
+ ctx.ui.notify(`Added ${id}: "${title}"`, "success");
106
+ }
107
+
108
+ async function promoteBacklogItem(
109
+ basePath: string,
110
+ itemId: string,
111
+ ctx: ExtensionCommandContext,
112
+ pi: ExtensionAPI,
113
+ ): Promise<void> {
114
+ if (!itemId) {
115
+ ctx.ui.notify("Usage: /gsd backlog promote <id>\nExample: /gsd backlog promote 999.1", "warning");
116
+ return;
117
+ }
118
+
119
+ const items = parseBacklog(basePath);
120
+ const item = items.find((i) => i.id === itemId);
121
+
122
+ if (!item) {
123
+ ctx.ui.notify(`Backlog item ${itemId} not found.`, "warning");
124
+ return;
125
+ }
126
+
127
+ if (item.done) {
128
+ ctx.ui.notify(`${itemId} is already promoted/done.`, "info");
129
+ return;
130
+ }
131
+
132
+ // Promote — currently requires single-writer engine (not yet available)
133
+ // Mark as promoted in backlog for now; slice creation will be available with the engine.
134
+ item.done = true;
135
+ item.note = `promoted ${new Date().toISOString().slice(0, 10)}`;
136
+ writeBacklog(basePath, items);
137
+ ctx.ui.notify(`Promoted ${itemId}: "${item.title}" — add it to the roadmap manually or wait for engine slice commands.`, "info");
138
+ }
139
+
140
+ async function removeBacklogItem(basePath: string, itemId: string, ctx: ExtensionCommandContext): Promise<void> {
141
+ if (!itemId) {
142
+ ctx.ui.notify("Usage: /gsd backlog remove <id>", "warning");
143
+ return;
144
+ }
145
+
146
+ const items = parseBacklog(basePath);
147
+ const idx = items.findIndex((i) => i.id === itemId);
148
+
149
+ if (idx === -1) {
150
+ ctx.ui.notify(`Backlog item ${itemId} not found.`, "warning");
151
+ return;
152
+ }
153
+
154
+ const removed = items.splice(idx, 1)[0];
155
+ writeBacklog(basePath, items);
156
+ ctx.ui.notify(`Removed ${removed.id}: "${removed.title}"`, "success");
157
+ }
158
+
159
+ export async function handleBacklog(
160
+ args: string,
161
+ ctx: ExtensionCommandContext,
162
+ pi: ExtensionAPI,
163
+ ): Promise<void> {
164
+ const basePath = process.cwd();
165
+ const parts = args.trim().split(/\s+/);
166
+ const sub = parts[0] ?? "";
167
+ const rest = parts.slice(1).join(" ");
168
+
169
+ switch (sub) {
170
+ case "":
171
+ return listBacklog(basePath, ctx);
172
+ case "add":
173
+ return addBacklogItem(basePath, rest, ctx);
174
+ case "promote":
175
+ return promoteBacklogItem(basePath, rest.trim(), ctx, pi);
176
+ case "remove":
177
+ return removeBacklogItem(basePath, rest.trim(), ctx);
178
+ default:
179
+ // Treat as implicit add
180
+ return addBacklogItem(basePath, args, ctx);
181
+ }
182
+ }