gsd-pi 2.33.1 → 2.34.0-dev.bbb5216

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 (828) hide show
  1. package/dist/bundled-resource-path.d.ts +8 -0
  2. package/dist/bundled-resource-path.js +14 -0
  3. package/dist/cli.js +16 -0
  4. package/dist/headless-query.js +6 -6
  5. package/dist/resource-loader.d.ts +2 -0
  6. package/dist/resource-loader.js +39 -4
  7. package/dist/resources/extensions/ask-user-questions.js +217 -0
  8. package/dist/resources/extensions/async-jobs/async-bash-tool.js +180 -0
  9. package/dist/resources/extensions/async-jobs/await-tool.js +90 -0
  10. package/dist/resources/extensions/async-jobs/cancel-job-tool.js +28 -0
  11. package/dist/resources/extensions/async-jobs/index.js +119 -0
  12. package/dist/resources/extensions/async-jobs/job-manager.js +159 -0
  13. package/dist/resources/extensions/aws-auth/index.js +138 -0
  14. package/dist/resources/extensions/bg-shell/bg-shell-command.js +182 -0
  15. package/dist/resources/extensions/bg-shell/bg-shell-lifecycle.js +343 -0
  16. package/dist/resources/extensions/bg-shell/bg-shell-tool.js +831 -0
  17. package/dist/resources/extensions/bg-shell/index.js +41 -0
  18. package/dist/resources/extensions/bg-shell/interaction.js +160 -0
  19. package/dist/resources/extensions/bg-shell/output-formatter.js +245 -0
  20. package/dist/resources/extensions/bg-shell/overlay.js +378 -0
  21. package/dist/resources/extensions/bg-shell/process-manager.js +413 -0
  22. package/dist/resources/extensions/bg-shell/readiness-detector.js +109 -0
  23. package/dist/resources/extensions/bg-shell/types.js +96 -0
  24. package/dist/resources/extensions/bg-shell/utilities.js +50 -0
  25. package/dist/resources/extensions/browser-tools/capture.js +179 -0
  26. package/dist/resources/extensions/browser-tools/core.js +899 -0
  27. package/dist/resources/extensions/browser-tools/{evaluate-helpers.ts → evaluate-helpers.js} +0 -1
  28. package/dist/resources/extensions/browser-tools/index.js +123 -0
  29. package/dist/resources/extensions/browser-tools/lifecycle.js +222 -0
  30. package/dist/resources/extensions/browser-tools/refs.js +254 -0
  31. package/dist/resources/extensions/browser-tools/settle.js +173 -0
  32. package/dist/resources/extensions/browser-tools/state.js +126 -0
  33. package/dist/resources/extensions/browser-tools/tools/action-cache.js +179 -0
  34. package/dist/resources/extensions/browser-tools/tools/assertions.js +320 -0
  35. package/dist/resources/extensions/browser-tools/tools/codegen.js +242 -0
  36. package/dist/resources/extensions/browser-tools/tools/device.js +162 -0
  37. package/dist/resources/extensions/browser-tools/tools/extract.js +191 -0
  38. package/dist/resources/extensions/browser-tools/tools/forms.js +710 -0
  39. package/dist/resources/extensions/browser-tools/tools/injection-detect.js +178 -0
  40. package/dist/resources/extensions/browser-tools/tools/inspection.js +426 -0
  41. package/dist/resources/extensions/browser-tools/tools/intent.js +556 -0
  42. package/dist/resources/extensions/browser-tools/tools/interaction.js +776 -0
  43. package/dist/resources/extensions/browser-tools/tools/navigation.js +208 -0
  44. package/dist/resources/extensions/browser-tools/tools/network-mock.js +194 -0
  45. package/dist/resources/extensions/browser-tools/tools/pages.js +280 -0
  46. package/dist/resources/extensions/browser-tools/tools/pdf.js +74 -0
  47. package/dist/resources/extensions/browser-tools/tools/refs.js +485 -0
  48. package/dist/resources/extensions/browser-tools/tools/screenshot.js +87 -0
  49. package/dist/resources/extensions/browser-tools/tools/session.js +375 -0
  50. package/dist/resources/extensions/browser-tools/tools/state-persistence.js +180 -0
  51. package/dist/resources/extensions/browser-tools/tools/visual-diff.js +174 -0
  52. package/dist/resources/extensions/browser-tools/tools/wait.js +201 -0
  53. package/dist/resources/extensions/browser-tools/tools/zoom.js +90 -0
  54. package/dist/resources/extensions/browser-tools/utils.js +490 -0
  55. package/dist/resources/extensions/context7/index.js +337 -0
  56. package/dist/resources/extensions/get-secrets-from-user.js +492 -0
  57. package/dist/resources/extensions/google-search/index.js +373 -0
  58. package/dist/resources/extensions/gsd/activity-log.js +146 -0
  59. package/dist/resources/extensions/gsd/atomic-write.js +38 -0
  60. package/dist/resources/extensions/gsd/auto/session.js +182 -0
  61. package/dist/resources/extensions/gsd/auto-budget.js +30 -0
  62. package/dist/resources/extensions/gsd/auto-dashboard.js +429 -0
  63. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +185 -0
  64. package/dist/resources/extensions/gsd/auto-dispatch.js +353 -0
  65. package/dist/resources/extensions/gsd/auto-loop.js +956 -0
  66. package/dist/resources/extensions/gsd/auto-model-selection.js +133 -0
  67. package/dist/resources/extensions/gsd/auto-observability.js +56 -0
  68. package/dist/resources/extensions/gsd/auto-post-unit.js +385 -0
  69. package/dist/resources/extensions/gsd/auto-prompts.js +1153 -0
  70. package/dist/resources/extensions/gsd/auto-recovery.js +512 -0
  71. package/dist/resources/extensions/gsd/auto-start.js +414 -0
  72. package/dist/resources/extensions/gsd/auto-supervisor.js +47 -0
  73. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +183 -0
  74. package/dist/resources/extensions/gsd/auto-timers.js +180 -0
  75. package/dist/resources/extensions/gsd/auto-tool-tracking.js +50 -0
  76. package/dist/resources/extensions/gsd/auto-unit-closeout.js +30 -0
  77. package/dist/resources/extensions/gsd/auto-verification.js +171 -0
  78. package/dist/resources/extensions/gsd/auto-worktree-sync.js +167 -0
  79. package/dist/resources/extensions/gsd/auto-worktree.js +830 -0
  80. package/dist/resources/extensions/gsd/auto.js +765 -0
  81. package/dist/resources/extensions/gsd/{cache.ts → cache.js} +5 -7
  82. package/dist/resources/extensions/gsd/captures.js +354 -0
  83. package/dist/resources/extensions/gsd/claude-import.js +540 -0
  84. package/dist/resources/extensions/gsd/collision-diagnostics.js +226 -0
  85. package/dist/resources/extensions/gsd/commands-bootstrap.js +223 -0
  86. package/dist/resources/extensions/gsd/commands-config.js +89 -0
  87. package/dist/resources/extensions/gsd/commands-extensions.js +259 -0
  88. package/dist/resources/extensions/gsd/commands-handlers.js +310 -0
  89. package/dist/resources/extensions/gsd/commands-inspect.js +70 -0
  90. package/dist/resources/extensions/gsd/commands-logs.js +468 -0
  91. package/dist/resources/extensions/gsd/commands-maintenance.js +185 -0
  92. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +660 -0
  93. package/dist/resources/extensions/gsd/commands-workflow-templates.js +408 -0
  94. package/dist/resources/extensions/gsd/commands.js +1024 -0
  95. package/dist/resources/extensions/gsd/complexity-classifier.js +257 -0
  96. package/dist/resources/extensions/gsd/{constants.ts → constants.js} +0 -6
  97. package/dist/resources/extensions/gsd/context-budget.js +187 -0
  98. package/dist/resources/extensions/gsd/context-store.js +165 -0
  99. package/dist/resources/extensions/gsd/crash-recovery.js +110 -0
  100. package/dist/resources/extensions/gsd/dashboard-overlay.js +562 -0
  101. package/dist/resources/extensions/gsd/db-writer.js +298 -0
  102. package/dist/resources/extensions/gsd/debug-logger.js +161 -0
  103. package/dist/resources/extensions/gsd/detection.js +373 -0
  104. package/dist/resources/extensions/gsd/diff-context.js +168 -0
  105. package/dist/resources/extensions/gsd/dispatch-guard.js +75 -0
  106. package/dist/resources/extensions/gsd/docs/preferences-reference.md +25 -18
  107. package/dist/resources/extensions/gsd/doctor-checks.js +562 -0
  108. package/dist/resources/extensions/gsd/doctor-environment.js +429 -0
  109. package/dist/resources/extensions/gsd/doctor-format.js +71 -0
  110. package/dist/resources/extensions/gsd/doctor-proactive.js +239 -0
  111. package/dist/resources/extensions/gsd/doctor-providers.js +292 -0
  112. package/dist/resources/extensions/gsd/doctor-types.js +12 -0
  113. package/dist/resources/extensions/gsd/doctor.js +672 -0
  114. package/dist/resources/extensions/gsd/error-utils.js +6 -0
  115. package/dist/resources/extensions/gsd/{errors.ts → errors.js} +6 -11
  116. package/dist/resources/extensions/gsd/exit-command.js +11 -0
  117. package/dist/resources/extensions/gsd/{export-html.ts → export-html.js} +482 -614
  118. package/dist/resources/extensions/gsd/export.js +268 -0
  119. package/dist/resources/extensions/gsd/file-watcher.js +76 -0
  120. package/dist/resources/extensions/gsd/files.js +937 -0
  121. package/dist/resources/extensions/gsd/forensics.js +511 -0
  122. package/dist/resources/extensions/gsd/{git-constants.ts → git-constants.js} +4 -5
  123. package/dist/resources/extensions/gsd/git-self-heal.js +113 -0
  124. package/dist/resources/extensions/gsd/git-service.js +460 -0
  125. package/dist/resources/extensions/gsd/{gitignore.ts → gitignore.js} +98 -125
  126. package/dist/resources/extensions/gsd/gsd-db.js +735 -0
  127. package/dist/resources/extensions/gsd/guided-flow-queue.js +366 -0
  128. package/dist/resources/extensions/gsd/guided-flow.js +1158 -0
  129. package/dist/resources/extensions/gsd/health-widget.js +141 -0
  130. package/dist/resources/extensions/gsd/history.js +118 -0
  131. package/dist/resources/extensions/gsd/index.js +1114 -0
  132. package/dist/resources/extensions/gsd/init-wizard.js +479 -0
  133. package/dist/resources/extensions/gsd/json-persistence.js +62 -0
  134. package/dist/resources/extensions/gsd/{jsonl-utils.ts → jsonl-utils.js} +10 -7
  135. package/dist/resources/extensions/gsd/key-manager.js +829 -0
  136. package/dist/resources/extensions/gsd/marketplace-discovery.js +356 -0
  137. package/dist/resources/extensions/gsd/md-importer.js +440 -0
  138. package/dist/resources/extensions/gsd/memory-extractor.js +295 -0
  139. package/dist/resources/extensions/gsd/memory-store.js +351 -0
  140. package/dist/resources/extensions/gsd/metrics.js +377 -0
  141. package/dist/resources/extensions/gsd/migrate/command.js +157 -0
  142. package/dist/resources/extensions/gsd/migrate/index.js +7 -0
  143. package/dist/resources/extensions/gsd/migrate/parser.js +268 -0
  144. package/dist/resources/extensions/gsd/migrate/parsers.js +477 -0
  145. package/dist/resources/extensions/gsd/migrate/preview.js +47 -0
  146. package/dist/resources/extensions/gsd/migrate/transformer.js +278 -0
  147. package/dist/resources/extensions/gsd/migrate/types.js +4 -0
  148. package/dist/resources/extensions/gsd/migrate/validator.js +41 -0
  149. package/dist/resources/extensions/gsd/migrate/writer.js +477 -0
  150. package/dist/resources/extensions/gsd/migrate-external.js +130 -0
  151. package/dist/resources/extensions/gsd/milestone-actions.js +111 -0
  152. package/dist/resources/extensions/gsd/{milestone-ids.ts → milestone-ids.js} +50 -63
  153. package/dist/resources/extensions/gsd/model-cost-table.js +48 -0
  154. package/dist/resources/extensions/gsd/model-router.js +187 -0
  155. package/dist/resources/extensions/gsd/namespaced-registry.js +322 -0
  156. package/dist/resources/extensions/gsd/namespaced-resolver.js +176 -0
  157. package/dist/resources/extensions/gsd/native-git-bridge.js +842 -0
  158. package/dist/resources/extensions/gsd/native-parser-bridge.js +156 -0
  159. package/dist/resources/extensions/gsd/notifications.js +58 -0
  160. package/dist/resources/extensions/gsd/observability-validator.js +398 -0
  161. package/dist/resources/extensions/gsd/parallel-eligibility.js +182 -0
  162. package/dist/resources/extensions/gsd/parallel-merge.js +121 -0
  163. package/dist/resources/extensions/gsd/parallel-orchestrator.js +687 -0
  164. package/dist/resources/extensions/gsd/paths.js +414 -0
  165. package/dist/resources/extensions/gsd/plugin-importer.js +254 -0
  166. package/dist/resources/extensions/gsd/post-unit-hooks.js +433 -0
  167. package/dist/resources/extensions/gsd/preferences-models.js +294 -0
  168. package/dist/resources/extensions/gsd/preferences-skills.js +154 -0
  169. package/dist/resources/extensions/gsd/preferences-types.js +73 -0
  170. package/dist/resources/extensions/gsd/preferences-validation.js +607 -0
  171. package/dist/resources/extensions/gsd/preferences.js +325 -0
  172. package/dist/resources/extensions/gsd/progress-score.js +102 -0
  173. package/dist/resources/extensions/gsd/prompt-cache-optimizer.js +150 -0
  174. package/dist/resources/extensions/gsd/prompt-compressor.js +393 -0
  175. package/dist/resources/extensions/gsd/prompt-loader.js +119 -0
  176. package/dist/resources/extensions/gsd/prompt-ordering.js +170 -0
  177. package/dist/resources/extensions/gsd/provider-error-pause.js +60 -0
  178. package/dist/resources/extensions/gsd/queue-order.js +178 -0
  179. package/dist/resources/extensions/gsd/queue-reorder-ui.js +234 -0
  180. package/dist/resources/extensions/gsd/quick.js +206 -0
  181. package/dist/resources/extensions/gsd/repo-identity.js +187 -0
  182. package/dist/resources/extensions/gsd/{reports.ts → reports.js} +146 -241
  183. package/dist/resources/extensions/gsd/{resource-version.ts → resource-version.js} +50 -53
  184. package/dist/resources/extensions/gsd/roadmap-slices.js +130 -0
  185. package/dist/resources/extensions/gsd/routing-history.js +210 -0
  186. package/dist/resources/extensions/gsd/safe-fs.js +52 -0
  187. package/dist/resources/extensions/gsd/semantic-chunker.js +254 -0
  188. package/dist/resources/extensions/gsd/session-forensics.js +427 -0
  189. package/dist/resources/extensions/gsd/session-lock.js +397 -0
  190. package/dist/resources/extensions/gsd/session-status-io.js +134 -0
  191. package/dist/resources/extensions/gsd/skill-discovery.js +121 -0
  192. package/dist/resources/extensions/gsd/skill-health.js +324 -0
  193. package/dist/resources/extensions/gsd/{skill-telemetry.ts → skill-telemetry.js} +58 -74
  194. package/dist/resources/extensions/gsd/state.js +653 -0
  195. package/dist/resources/extensions/gsd/structured-data-formatter.js +97 -0
  196. package/dist/resources/extensions/gsd/summary-distiller.js +212 -0
  197. package/dist/resources/extensions/gsd/templates/preferences.md +1 -0
  198. package/dist/resources/extensions/gsd/token-counter.js +54 -0
  199. package/dist/resources/extensions/gsd/triage-resolution.js +217 -0
  200. package/dist/resources/extensions/gsd/triage-ui.js +125 -0
  201. package/dist/resources/extensions/gsd/types.js +4 -0
  202. package/dist/resources/extensions/gsd/undo.js +205 -0
  203. package/dist/resources/extensions/gsd/unit-id.js +7 -0
  204. package/dist/resources/extensions/gsd/unit-runtime.js +131 -0
  205. package/dist/resources/extensions/gsd/validate-directory.js +143 -0
  206. package/dist/resources/extensions/gsd/verification-evidence.js +122 -0
  207. package/dist/resources/extensions/gsd/verification-gate.js +514 -0
  208. package/dist/resources/extensions/gsd/visualizer-data.js +612 -0
  209. package/dist/resources/extensions/gsd/visualizer-overlay.js +501 -0
  210. package/dist/resources/extensions/gsd/visualizer-views.js +893 -0
  211. package/dist/resources/extensions/gsd/workflow-templates.js +188 -0
  212. package/dist/resources/extensions/gsd/workspace-index.js +139 -0
  213. package/dist/resources/extensions/gsd/worktree-command-bootstrap.js +40 -0
  214. package/dist/resources/extensions/gsd/worktree-command.js +682 -0
  215. package/dist/resources/extensions/gsd/worktree-manager.js +358 -0
  216. package/dist/resources/extensions/gsd/worktree-resolver.js +344 -0
  217. package/dist/resources/extensions/gsd/worktree.js +199 -0
  218. package/dist/resources/extensions/mac-tools/index.js +768 -0
  219. package/dist/resources/extensions/mcp-client/index.js +363 -0
  220. package/dist/resources/extensions/package.json +3 -0
  221. package/dist/resources/extensions/remote-questions/config.js +70 -0
  222. package/dist/resources/extensions/remote-questions/discord-adapter.js +134 -0
  223. package/dist/resources/extensions/remote-questions/format.js +234 -0
  224. package/dist/resources/extensions/remote-questions/http-client.js +43 -0
  225. package/dist/resources/extensions/remote-questions/manager.js +156 -0
  226. package/dist/resources/extensions/remote-questions/{mod.ts → mod.js} +1 -10
  227. package/dist/resources/extensions/remote-questions/notify.js +89 -0
  228. package/dist/resources/extensions/remote-questions/remote-command.js +453 -0
  229. package/dist/resources/extensions/remote-questions/slack-adapter.js +123 -0
  230. package/dist/resources/extensions/remote-questions/status.js +25 -0
  231. package/dist/resources/extensions/remote-questions/store.js +70 -0
  232. package/dist/resources/extensions/remote-questions/telegram-adapter.js +123 -0
  233. package/dist/resources/extensions/remote-questions/types.js +5 -0
  234. package/dist/resources/extensions/search-the-web/cache.js +74 -0
  235. package/dist/resources/extensions/search-the-web/command-search-provider.js +79 -0
  236. package/dist/resources/extensions/search-the-web/format.js +161 -0
  237. package/dist/resources/extensions/search-the-web/http.js +178 -0
  238. package/dist/resources/extensions/search-the-web/index.js +41 -0
  239. package/dist/resources/extensions/search-the-web/native-search.js +166 -0
  240. package/dist/resources/extensions/search-the-web/provider.js +143 -0
  241. package/dist/resources/extensions/search-the-web/tavily.js +82 -0
  242. package/dist/resources/extensions/search-the-web/tool-fetch-page.js +452 -0
  243. package/dist/resources/extensions/search-the-web/tool-llm-context.js +455 -0
  244. package/dist/resources/extensions/search-the-web/tool-search.js +482 -0
  245. package/dist/resources/extensions/search-the-web/url-utils.js +121 -0
  246. package/dist/resources/extensions/shared/confirm-ui.js +96 -0
  247. package/dist/resources/extensions/shared/{format-utils.ts → format-utils.js} +85 -91
  248. package/dist/resources/extensions/shared/frontmatter.js +109 -0
  249. package/dist/resources/extensions/shared/interview-ui.js +569 -0
  250. package/dist/resources/extensions/shared/{mod.ts → mod.js} +2 -24
  251. package/dist/resources/extensions/shared/next-action-ui.js +168 -0
  252. package/dist/resources/extensions/shared/{path-display.ts → path-display.js} +2 -3
  253. package/dist/resources/extensions/shared/sanitize.js +17 -0
  254. package/dist/resources/extensions/shared/terminal.js +21 -0
  255. package/dist/resources/extensions/shared/ui.js +245 -0
  256. package/dist/resources/extensions/shared/wizard-ui.js +478 -0
  257. package/dist/resources/extensions/slash-commands/audit.js +72 -0
  258. package/dist/resources/extensions/slash-commands/clear.js +8 -0
  259. package/dist/resources/extensions/slash-commands/create-extension.js +264 -0
  260. package/dist/resources/extensions/slash-commands/create-slash-command.js +208 -0
  261. package/dist/resources/extensions/slash-commands/index.js +10 -0
  262. package/dist/resources/extensions/subagent/agents.js +103 -0
  263. package/dist/resources/extensions/subagent/index.js +905 -0
  264. package/dist/resources/extensions/subagent/isolation.js +384 -0
  265. package/dist/resources/extensions/subagent/worker-registry.js +73 -0
  266. package/dist/resources/extensions/ttsr/index.js +144 -0
  267. package/dist/resources/extensions/ttsr/rule-loader.js +70 -0
  268. package/dist/resources/extensions/ttsr/ttsr-manager.js +380 -0
  269. package/dist/resources/extensions/universal-config/discovery.js +94 -0
  270. package/dist/resources/extensions/universal-config/format.js +178 -0
  271. package/dist/resources/extensions/universal-config/index.js +99 -0
  272. package/dist/resources/extensions/universal-config/scanners.js +574 -0
  273. package/dist/resources/extensions/universal-config/tools.js +57 -0
  274. package/dist/resources/extensions/universal-config/types.js +8 -0
  275. package/dist/resources/extensions/voice/index.js +247 -0
  276. package/dist/startup-timings.d.ts +2 -0
  277. package/dist/startup-timings.js +22 -0
  278. package/dist/tool-bootstrap.js +59 -11
  279. package/dist/worktree-cli.js +7 -7
  280. package/package.json +1 -1
  281. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  282. package/packages/pi-coding-agent/dist/core/agent-session.js +9 -0
  283. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  284. package/packages/pi-coding-agent/dist/core/extensions/index.d.ts +1 -1
  285. package/packages/pi-coding-agent/dist/core/extensions/index.d.ts.map +1 -1
  286. package/packages/pi-coding-agent/dist/core/extensions/index.js +1 -1
  287. package/packages/pi-coding-agent/dist/core/extensions/index.js.map +1 -1
  288. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +1 -0
  289. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  290. package/packages/pi-coding-agent/dist/core/extensions/loader.js +31 -4
  291. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  292. package/packages/pi-coding-agent/dist/index.d.ts +1 -1
  293. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  294. package/packages/pi-coding-agent/dist/index.js +1 -1
  295. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  296. package/packages/pi-coding-agent/package.json +1 -1
  297. package/packages/pi-coding-agent/src/core/agent-session.ts +9 -0
  298. package/packages/pi-coding-agent/src/core/extensions/index.ts +1 -0
  299. package/packages/pi-coding-agent/src/core/extensions/loader.ts +35 -4
  300. package/packages/pi-coding-agent/src/index.ts +1 -0
  301. package/pkg/package.json +1 -1
  302. package/src/resources/extensions/bg-shell/index.ts +41 -46
  303. package/src/resources/extensions/browser-tools/index.ts +156 -67
  304. package/src/resources/extensions/gsd/auto/session.ts +47 -30
  305. package/src/resources/extensions/gsd/auto-dashboard.ts +28 -131
  306. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +6 -1
  307. package/src/resources/extensions/gsd/auto-dispatch.ts +135 -91
  308. package/src/resources/extensions/gsd/auto-loop.ts +1665 -0
  309. package/src/resources/extensions/gsd/auto-observability.ts +4 -2
  310. package/src/resources/extensions/gsd/auto-post-unit.ts +85 -228
  311. package/src/resources/extensions/gsd/auto-prompts.ts +138 -109
  312. package/src/resources/extensions/gsd/auto-recovery.ts +124 -118
  313. package/src/resources/extensions/gsd/auto-start.ts +440 -354
  314. package/src/resources/extensions/gsd/auto-supervisor.ts +5 -12
  315. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +8 -8
  316. package/src/resources/extensions/gsd/auto-timers.ts +3 -4
  317. package/src/resources/extensions/gsd/auto-verification.ts +76 -72
  318. package/src/resources/extensions/gsd/auto-worktree-sync.ts +204 -0
  319. package/src/resources/extensions/gsd/auto-worktree.ts +453 -133
  320. package/src/resources/extensions/gsd/auto.ts +516 -1189
  321. package/src/resources/extensions/gsd/captures.ts +10 -4
  322. package/src/resources/extensions/gsd/commands-bootstrap.ts +252 -0
  323. package/src/resources/extensions/gsd/commands.ts +39 -31
  324. package/src/resources/extensions/gsd/dispatch-guard.ts +13 -9
  325. package/src/resources/extensions/gsd/docs/preferences-reference.md +25 -18
  326. package/src/resources/extensions/gsd/doctor-checks.ts +3 -4
  327. package/src/resources/extensions/gsd/git-service.ts +32 -12
  328. package/src/resources/extensions/gsd/gitignore.ts +4 -2
  329. package/src/resources/extensions/gsd/gsd-db.ts +375 -180
  330. package/src/resources/extensions/gsd/guided-flow.ts +22 -11
  331. package/src/resources/extensions/gsd/index.ts +76 -163
  332. package/src/resources/extensions/gsd/post-unit-hooks.ts +13 -13
  333. package/src/resources/extensions/gsd/progress-score.ts +65 -200
  334. package/src/resources/extensions/gsd/quick.ts +121 -76
  335. package/src/resources/extensions/gsd/repo-identity.ts +56 -22
  336. package/src/resources/extensions/gsd/session-lock.ts +17 -0
  337. package/src/resources/extensions/gsd/templates/preferences.md +1 -0
  338. package/src/resources/extensions/gsd/tests/agent-end-retry.test.ts +32 -59
  339. package/src/resources/extensions/gsd/tests/all-milestones-complete-merge.test.ts +75 -27
  340. package/src/resources/extensions/gsd/tests/auto-budget-alerts.test.ts +1 -1
  341. package/src/resources/extensions/gsd/tests/auto-lock-creation.test.ts +37 -0
  342. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +1458 -0
  343. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +8 -162
  344. package/src/resources/extensions/gsd/tests/auto-secrets-gate.test.ts +2 -108
  345. package/src/resources/extensions/gsd/tests/auto-session-encapsulation.test.ts +1 -3
  346. package/src/resources/extensions/gsd/tests/auto-worktree-milestone-merge.test.ts +0 -3
  347. package/src/resources/extensions/gsd/tests/auto-worktree.test.ts +58 -0
  348. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +0 -55
  349. package/src/resources/extensions/gsd/tests/git-service.test.ts +16 -7
  350. package/src/resources/extensions/gsd/tests/headless-query.test.ts +22 -0
  351. package/src/resources/extensions/gsd/tests/milestone-transition-worktree.test.ts +8 -11
  352. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +4 -6
  353. package/src/resources/extensions/gsd/tests/quick-branch-lifecycle.test.ts +281 -0
  354. package/src/resources/extensions/gsd/tests/repo-identity-worktree.test.ts +71 -0
  355. package/src/resources/extensions/gsd/tests/run-uat.test.ts +3 -3
  356. package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +64 -0
  357. package/src/resources/extensions/gsd/tests/sidecar-queue.test.ts +181 -0
  358. package/src/resources/extensions/gsd/tests/stale-worktree-cwd.test.ts +0 -3
  359. package/src/resources/extensions/gsd/tests/token-profile.test.ts +6 -6
  360. package/src/resources/extensions/gsd/tests/triage-dispatch.test.ts +6 -6
  361. package/src/resources/extensions/gsd/tests/undo.test.ts +6 -0
  362. package/src/resources/extensions/gsd/tests/verification-evidence.test.ts +24 -26
  363. package/src/resources/extensions/gsd/tests/verification-gate.test.ts +7 -136
  364. package/src/resources/extensions/gsd/tests/worktree-db-integration.test.ts +205 -0
  365. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +442 -0
  366. package/src/resources/extensions/gsd/tests/worktree-e2e.test.ts +0 -3
  367. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +705 -0
  368. package/src/resources/extensions/gsd/tests/worktree-sync-milestones.test.ts +57 -106
  369. package/src/resources/extensions/gsd/tests/worktree.test.ts +5 -1
  370. package/src/resources/extensions/gsd/tests/write-gate.test.ts +43 -132
  371. package/src/resources/extensions/gsd/types.ts +90 -77
  372. package/src/resources/extensions/gsd/undo.ts +42 -46
  373. package/src/resources/extensions/gsd/unit-runtime.ts +14 -18
  374. package/src/resources/extensions/gsd/verification-evidence.ts +0 -2
  375. package/src/resources/extensions/gsd/verification-gate.ts +4 -16
  376. package/src/resources/extensions/gsd/worktree-command-bootstrap.ts +46 -0
  377. package/src/resources/extensions/gsd/worktree-command.ts +29 -11
  378. package/src/resources/extensions/gsd/worktree-manager.ts +2 -3
  379. package/src/resources/extensions/gsd/worktree-resolver.ts +485 -0
  380. package/src/resources/extensions/gsd/worktree.ts +7 -44
  381. package/src/resources/extensions/package.json +3 -0
  382. package/src/resources/extensions/search-the-web/command-search-provider.ts +1 -1
  383. package/src/resources/extensions/search-the-web/index.ts +35 -52
  384. package/src/resources/extensions/search-the-web/tavily.ts +1 -1
  385. package/dist/resources/extensions/ask-user-questions.ts +0 -290
  386. package/dist/resources/extensions/async-jobs/async-bash-tool.ts +0 -212
  387. package/dist/resources/extensions/async-jobs/await-tool.ts +0 -103
  388. package/dist/resources/extensions/async-jobs/cancel-job-tool.ts +0 -35
  389. package/dist/resources/extensions/async-jobs/index.ts +0 -141
  390. package/dist/resources/extensions/async-jobs/job-manager.ts +0 -211
  391. package/dist/resources/extensions/aws-auth/index.ts +0 -144
  392. package/dist/resources/extensions/bg-shell/bg-shell-command.ts +0 -219
  393. package/dist/resources/extensions/bg-shell/bg-shell-lifecycle.ts +0 -400
  394. package/dist/resources/extensions/bg-shell/bg-shell-tool.ts +0 -985
  395. package/dist/resources/extensions/bg-shell/index.ts +0 -59
  396. package/dist/resources/extensions/bg-shell/interaction.ts +0 -198
  397. package/dist/resources/extensions/bg-shell/output-formatter.ts +0 -279
  398. package/dist/resources/extensions/bg-shell/overlay.ts +0 -437
  399. package/dist/resources/extensions/bg-shell/process-manager.ts +0 -464
  400. package/dist/resources/extensions/bg-shell/readiness-detector.ts +0 -126
  401. package/dist/resources/extensions/bg-shell/types.ts +0 -303
  402. package/dist/resources/extensions/bg-shell/utilities.ts +0 -57
  403. package/dist/resources/extensions/browser-tools/capture.ts +0 -199
  404. package/dist/resources/extensions/browser-tools/core.ts +0 -1196
  405. package/dist/resources/extensions/browser-tools/index.ts +0 -71
  406. package/dist/resources/extensions/browser-tools/lifecycle.ts +0 -270
  407. package/dist/resources/extensions/browser-tools/refs.ts +0 -264
  408. package/dist/resources/extensions/browser-tools/settle.ts +0 -197
  409. package/dist/resources/extensions/browser-tools/state.ts +0 -408
  410. package/dist/resources/extensions/browser-tools/tools/action-cache.ts +0 -216
  411. package/dist/resources/extensions/browser-tools/tools/assertions.ts +0 -342
  412. package/dist/resources/extensions/browser-tools/tools/codegen.ts +0 -274
  413. package/dist/resources/extensions/browser-tools/tools/device.ts +0 -183
  414. package/dist/resources/extensions/browser-tools/tools/extract.ts +0 -229
  415. package/dist/resources/extensions/browser-tools/tools/forms.ts +0 -801
  416. package/dist/resources/extensions/browser-tools/tools/injection-detect.ts +0 -221
  417. package/dist/resources/extensions/browser-tools/tools/inspection.ts +0 -492
  418. package/dist/resources/extensions/browser-tools/tools/intent.ts +0 -614
  419. package/dist/resources/extensions/browser-tools/tools/interaction.ts +0 -865
  420. package/dist/resources/extensions/browser-tools/tools/navigation.ts +0 -232
  421. package/dist/resources/extensions/browser-tools/tools/network-mock.ts +0 -244
  422. package/dist/resources/extensions/browser-tools/tools/pages.ts +0 -303
  423. package/dist/resources/extensions/browser-tools/tools/pdf.ts +0 -92
  424. package/dist/resources/extensions/browser-tools/tools/refs.ts +0 -541
  425. package/dist/resources/extensions/browser-tools/tools/screenshot.ts +0 -101
  426. package/dist/resources/extensions/browser-tools/tools/session.ts +0 -400
  427. package/dist/resources/extensions/browser-tools/tools/state-persistence.ts +0 -202
  428. package/dist/resources/extensions/browser-tools/tools/visual-diff.ts +0 -209
  429. package/dist/resources/extensions/browser-tools/tools/wait.ts +0 -247
  430. package/dist/resources/extensions/browser-tools/tools/zoom.ts +0 -104
  431. package/dist/resources/extensions/browser-tools/utils.ts +0 -660
  432. package/dist/resources/extensions/context7/index.ts +0 -428
  433. package/dist/resources/extensions/get-secrets-from-user.ts +0 -607
  434. package/dist/resources/extensions/google-search/index.ts +0 -466
  435. package/dist/resources/extensions/gsd/activity-log.ts +0 -162
  436. package/dist/resources/extensions/gsd/atomic-write.ts +0 -35
  437. package/dist/resources/extensions/gsd/auto/session.ts +0 -236
  438. package/dist/resources/extensions/gsd/auto-budget.ts +0 -32
  439. package/dist/resources/extensions/gsd/auto-constants.ts +0 -6
  440. package/dist/resources/extensions/gsd/auto-dashboard.ts +0 -626
  441. package/dist/resources/extensions/gsd/auto-direct-dispatch.ts +0 -224
  442. package/dist/resources/extensions/gsd/auto-dispatch.ts +0 -409
  443. package/dist/resources/extensions/gsd/auto-idempotency.ts +0 -151
  444. package/dist/resources/extensions/gsd/auto-model-selection.ts +0 -179
  445. package/dist/resources/extensions/gsd/auto-observability.ts +0 -72
  446. package/dist/resources/extensions/gsd/auto-post-unit.ts +0 -618
  447. package/dist/resources/extensions/gsd/auto-prompts.ts +0 -1273
  448. package/dist/resources/extensions/gsd/auto-recovery.ts +0 -578
  449. package/dist/resources/extensions/gsd/auto-start.ts +0 -483
  450. package/dist/resources/extensions/gsd/auto-stuck-detection.ts +0 -221
  451. package/dist/resources/extensions/gsd/auto-supervisor.ts +0 -61
  452. package/dist/resources/extensions/gsd/auto-timeout-recovery.ts +0 -263
  453. package/dist/resources/extensions/gsd/auto-timers.ts +0 -224
  454. package/dist/resources/extensions/gsd/auto-tool-tracking.ts +0 -54
  455. package/dist/resources/extensions/gsd/auto-unit-closeout.ts +0 -48
  456. package/dist/resources/extensions/gsd/auto-verification.ts +0 -229
  457. package/dist/resources/extensions/gsd/auto-worktree.ts +0 -658
  458. package/dist/resources/extensions/gsd/auto.ts +0 -1834
  459. package/dist/resources/extensions/gsd/captures.ts +0 -427
  460. package/dist/resources/extensions/gsd/claude-import.ts +0 -656
  461. package/dist/resources/extensions/gsd/collision-diagnostics.ts +0 -332
  462. package/dist/resources/extensions/gsd/commands-config.ts +0 -102
  463. package/dist/resources/extensions/gsd/commands-extensions.ts +0 -328
  464. package/dist/resources/extensions/gsd/commands-handlers.ts +0 -395
  465. package/dist/resources/extensions/gsd/commands-inspect.ts +0 -91
  466. package/dist/resources/extensions/gsd/commands-logs.ts +0 -536
  467. package/dist/resources/extensions/gsd/commands-maintenance.ts +0 -206
  468. package/dist/resources/extensions/gsd/commands-prefs-wizard.ts +0 -780
  469. package/dist/resources/extensions/gsd/commands-workflow-templates.ts +0 -543
  470. package/dist/resources/extensions/gsd/commands.ts +0 -1149
  471. package/dist/resources/extensions/gsd/complexity-classifier.ts +0 -320
  472. package/dist/resources/extensions/gsd/context-budget.ts +0 -266
  473. package/dist/resources/extensions/gsd/context-store.ts +0 -195
  474. package/dist/resources/extensions/gsd/crash-recovery.ts +0 -121
  475. package/dist/resources/extensions/gsd/dashboard-overlay.ts +0 -681
  476. package/dist/resources/extensions/gsd/db-writer.ts +0 -360
  477. package/dist/resources/extensions/gsd/debug-logger.ts +0 -178
  478. package/dist/resources/extensions/gsd/detection.ts +0 -470
  479. package/dist/resources/extensions/gsd/diff-context.ts +0 -214
  480. package/dist/resources/extensions/gsd/dispatch-guard.ts +0 -81
  481. package/dist/resources/extensions/gsd/doctor-checks.ts +0 -612
  482. package/dist/resources/extensions/gsd/doctor-environment.ts +0 -497
  483. package/dist/resources/extensions/gsd/doctor-format.ts +0 -78
  484. package/dist/resources/extensions/gsd/doctor-proactive.ts +0 -292
  485. package/dist/resources/extensions/gsd/doctor-providers.ts +0 -343
  486. package/dist/resources/extensions/gsd/doctor-types.ts +0 -87
  487. package/dist/resources/extensions/gsd/doctor.ts +0 -722
  488. package/dist/resources/extensions/gsd/error-utils.ts +0 -6
  489. package/dist/resources/extensions/gsd/exit-command.ts +0 -18
  490. package/dist/resources/extensions/gsd/export.ts +0 -317
  491. package/dist/resources/extensions/gsd/file-watcher.ts +0 -97
  492. package/dist/resources/extensions/gsd/files.ts +0 -1058
  493. package/dist/resources/extensions/gsd/forensics.ts +0 -629
  494. package/dist/resources/extensions/gsd/git-self-heal.ts +0 -127
  495. package/dist/resources/extensions/gsd/git-service.ts +0 -580
  496. package/dist/resources/extensions/gsd/gsd-db.ts +0 -685
  497. package/dist/resources/extensions/gsd/guided-flow-queue.ts +0 -440
  498. package/dist/resources/extensions/gsd/guided-flow.ts +0 -1303
  499. package/dist/resources/extensions/gsd/health-widget.ts +0 -167
  500. package/dist/resources/extensions/gsd/history.ts +0 -143
  501. package/dist/resources/extensions/gsd/index.ts +0 -1390
  502. package/dist/resources/extensions/gsd/init-wizard.ts +0 -587
  503. package/dist/resources/extensions/gsd/json-persistence.ts +0 -67
  504. package/dist/resources/extensions/gsd/key-manager.ts +0 -996
  505. package/dist/resources/extensions/gsd/marketplace-discovery.ts +0 -508
  506. package/dist/resources/extensions/gsd/md-importer.ts +0 -527
  507. package/dist/resources/extensions/gsd/mechanical-completion.ts +0 -430
  508. package/dist/resources/extensions/gsd/memory-extractor.ts +0 -352
  509. package/dist/resources/extensions/gsd/memory-store.ts +0 -441
  510. package/dist/resources/extensions/gsd/metrics.ts +0 -532
  511. package/dist/resources/extensions/gsd/migrate/command.ts +0 -219
  512. package/dist/resources/extensions/gsd/migrate/index.ts +0 -42
  513. package/dist/resources/extensions/gsd/migrate/parser.ts +0 -323
  514. package/dist/resources/extensions/gsd/migrate/parsers.ts +0 -539
  515. package/dist/resources/extensions/gsd/migrate/preview.ts +0 -48
  516. package/dist/resources/extensions/gsd/migrate/transformer.ts +0 -346
  517. package/dist/resources/extensions/gsd/migrate/types.ts +0 -370
  518. package/dist/resources/extensions/gsd/migrate/validator.ts +0 -55
  519. package/dist/resources/extensions/gsd/migrate/writer.ts +0 -579
  520. package/dist/resources/extensions/gsd/migrate-external.ts +0 -140
  521. package/dist/resources/extensions/gsd/milestone-actions.ts +0 -126
  522. package/dist/resources/extensions/gsd/model-cost-table.ts +0 -65
  523. package/dist/resources/extensions/gsd/model-router.ts +0 -256
  524. package/dist/resources/extensions/gsd/namespaced-registry.ts +0 -467
  525. package/dist/resources/extensions/gsd/namespaced-resolver.ts +0 -307
  526. package/dist/resources/extensions/gsd/native-git-bridge.ts +0 -1041
  527. package/dist/resources/extensions/gsd/native-parser-bridge.ts +0 -267
  528. package/dist/resources/extensions/gsd/notifications.ts +0 -87
  529. package/dist/resources/extensions/gsd/observability-validator.ts +0 -429
  530. package/dist/resources/extensions/gsd/parallel-eligibility.ts +0 -233
  531. package/dist/resources/extensions/gsd/parallel-merge.ts +0 -157
  532. package/dist/resources/extensions/gsd/parallel-orchestrator.ts +0 -826
  533. package/dist/resources/extensions/gsd/paths.ts +0 -449
  534. package/dist/resources/extensions/gsd/plugin-importer.ts +0 -411
  535. package/dist/resources/extensions/gsd/post-unit-hooks.ts +0 -520
  536. package/dist/resources/extensions/gsd/preferences-models.ts +0 -329
  537. package/dist/resources/extensions/gsd/preferences-skills.ts +0 -169
  538. package/dist/resources/extensions/gsd/preferences-types.ts +0 -229
  539. package/dist/resources/extensions/gsd/preferences-validation.ts +0 -590
  540. package/dist/resources/extensions/gsd/preferences.ts +0 -416
  541. package/dist/resources/extensions/gsd/progress-score.ts +0 -273
  542. package/dist/resources/extensions/gsd/prompt-cache-optimizer.ts +0 -213
  543. package/dist/resources/extensions/gsd/prompt-compressor.ts +0 -508
  544. package/dist/resources/extensions/gsd/prompt-loader.ts +0 -130
  545. package/dist/resources/extensions/gsd/prompt-ordering.ts +0 -200
  546. package/dist/resources/extensions/gsd/provider-error-pause.ts +0 -88
  547. package/dist/resources/extensions/gsd/queue-order.ts +0 -230
  548. package/dist/resources/extensions/gsd/queue-reorder-ui.ts +0 -276
  549. package/dist/resources/extensions/gsd/quick.ts +0 -212
  550. package/dist/resources/extensions/gsd/repo-identity.ts +0 -169
  551. package/dist/resources/extensions/gsd/roadmap-slices.ts +0 -149
  552. package/dist/resources/extensions/gsd/routing-history.ts +0 -286
  553. package/dist/resources/extensions/gsd/safe-fs.ts +0 -47
  554. package/dist/resources/extensions/gsd/semantic-chunker.ts +0 -336
  555. package/dist/resources/extensions/gsd/session-forensics.ts +0 -537
  556. package/dist/resources/extensions/gsd/session-lock.ts +0 -426
  557. package/dist/resources/extensions/gsd/session-status-io.ts +0 -179
  558. package/dist/resources/extensions/gsd/skill-discovery.ts +0 -139
  559. package/dist/resources/extensions/gsd/skill-health.ts +0 -417
  560. package/dist/resources/extensions/gsd/state.ts +0 -727
  561. package/dist/resources/extensions/gsd/structured-data-formatter.ts +0 -144
  562. package/dist/resources/extensions/gsd/summary-distiller.ts +0 -258
  563. package/dist/resources/extensions/gsd/tests/activity-log.test.ts +0 -213
  564. package/dist/resources/extensions/gsd/tests/agent-end-retry.test.ts +0 -107
  565. package/dist/resources/extensions/gsd/tests/all-milestones-complete-merge.test.ts +0 -200
  566. package/dist/resources/extensions/gsd/tests/auto-budget-alerts.test.ts +0 -50
  567. package/dist/resources/extensions/gsd/tests/auto-dashboard.test.ts +0 -166
  568. package/dist/resources/extensions/gsd/tests/auto-dispatch-loop.test.ts +0 -691
  569. package/dist/resources/extensions/gsd/tests/auto-lock-creation.test.ts +0 -186
  570. package/dist/resources/extensions/gsd/tests/auto-preflight.test.ts +0 -40
  571. package/dist/resources/extensions/gsd/tests/auto-recovery.test.ts +0 -640
  572. package/dist/resources/extensions/gsd/tests/auto-reentrancy-guard.test.ts +0 -127
  573. package/dist/resources/extensions/gsd/tests/auto-secrets-gate.test.ts +0 -302
  574. package/dist/resources/extensions/gsd/tests/auto-session-encapsulation.test.ts +0 -257
  575. package/dist/resources/extensions/gsd/tests/auto-skip-loop.test.ts +0 -123
  576. package/dist/resources/extensions/gsd/tests/auto-worktree-milestone-merge.test.ts +0 -340
  577. package/dist/resources/extensions/gsd/tests/auto-worktree.test.ts +0 -167
  578. package/dist/resources/extensions/gsd/tests/budget-prediction.test.ts +0 -220
  579. package/dist/resources/extensions/gsd/tests/cache-staleness-regression.test.ts +0 -317
  580. package/dist/resources/extensions/gsd/tests/captures.test.ts +0 -438
  581. package/dist/resources/extensions/gsd/tests/claude-import-tui.test.ts +0 -351
  582. package/dist/resources/extensions/gsd/tests/collect-from-manifest.test.ts +0 -469
  583. package/dist/resources/extensions/gsd/tests/collision-diagnostics.test.ts +0 -705
  584. package/dist/resources/extensions/gsd/tests/commands-logs.test.ts +0 -241
  585. package/dist/resources/extensions/gsd/tests/complete-milestone.test.ts +0 -209
  586. package/dist/resources/extensions/gsd/tests/complexity-classifier.test.ts +0 -181
  587. package/dist/resources/extensions/gsd/tests/context-budget.test.ts +0 -352
  588. package/dist/resources/extensions/gsd/tests/context-compression.test.ts +0 -193
  589. package/dist/resources/extensions/gsd/tests/context-store.test.ts +0 -462
  590. package/dist/resources/extensions/gsd/tests/continue-here.test.ts +0 -285
  591. package/dist/resources/extensions/gsd/tests/cost-projection.test.ts +0 -134
  592. package/dist/resources/extensions/gsd/tests/crash-recovery.test.ts +0 -134
  593. package/dist/resources/extensions/gsd/tests/dashboard-budget.test.ts +0 -346
  594. package/dist/resources/extensions/gsd/tests/db-writer.test.ts +0 -602
  595. package/dist/resources/extensions/gsd/tests/debug-logger.test.ts +0 -185
  596. package/dist/resources/extensions/gsd/tests/derive-state-db.test.ts +0 -405
  597. package/dist/resources/extensions/gsd/tests/derive-state-deps.test.ts +0 -421
  598. package/dist/resources/extensions/gsd/tests/derive-state-draft.test.ts +0 -308
  599. package/dist/resources/extensions/gsd/tests/derive-state.test.ts +0 -788
  600. package/dist/resources/extensions/gsd/tests/detection.test.ts +0 -398
  601. package/dist/resources/extensions/gsd/tests/diff-context.test.ts +0 -136
  602. package/dist/resources/extensions/gsd/tests/discuss-prompt.test.ts +0 -15
  603. package/dist/resources/extensions/gsd/tests/dispatch-guard.test.ts +0 -128
  604. package/dist/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +0 -132
  605. package/dist/resources/extensions/gsd/tests/dispatch-stall-guard.test.ts +0 -126
  606. package/dist/resources/extensions/gsd/tests/doctor-environment.test.ts +0 -314
  607. package/dist/resources/extensions/gsd/tests/doctor-fixlevel.test.ts +0 -245
  608. package/dist/resources/extensions/gsd/tests/doctor-git.test.ts +0 -344
  609. package/dist/resources/extensions/gsd/tests/doctor-proactive.test.ts +0 -278
  610. package/dist/resources/extensions/gsd/tests/doctor-providers.test.ts +0 -298
  611. package/dist/resources/extensions/gsd/tests/doctor-runtime.test.ts +0 -302
  612. package/dist/resources/extensions/gsd/tests/doctor.test.ts +0 -652
  613. package/dist/resources/extensions/gsd/tests/draft-promotion.test.ts +0 -169
  614. package/dist/resources/extensions/gsd/tests/exit-command.test.ts +0 -50
  615. package/dist/resources/extensions/gsd/tests/export-html-all.test.ts +0 -105
  616. package/dist/resources/extensions/gsd/tests/export-html-enhancements.test.ts +0 -378
  617. package/dist/resources/extensions/gsd/tests/extension-selector-separator.test.ts +0 -144
  618. package/dist/resources/extensions/gsd/tests/feature-branch-lifecycle-integration.test.ts +0 -424
  619. package/dist/resources/extensions/gsd/tests/git-self-heal.test.ts +0 -131
  620. package/dist/resources/extensions/gsd/tests/git-service.test.ts +0 -1172
  621. package/dist/resources/extensions/gsd/tests/gsd-db.test.ts +0 -353
  622. package/dist/resources/extensions/gsd/tests/gsd-inspect.test.ts +0 -125
  623. package/dist/resources/extensions/gsd/tests/gsd-tools.test.ts +0 -326
  624. package/dist/resources/extensions/gsd/tests/headless-answers.test.ts +0 -340
  625. package/dist/resources/extensions/gsd/tests/headless-query.test.ts +0 -162
  626. package/dist/resources/extensions/gsd/tests/idle-recovery.test.ts +0 -485
  627. package/dist/resources/extensions/gsd/tests/in-flight-tool-tracking.test.ts +0 -32
  628. package/dist/resources/extensions/gsd/tests/init-wizard.test.ts +0 -197
  629. package/dist/resources/extensions/gsd/tests/integration/headless-command.ts +0 -534
  630. package/dist/resources/extensions/gsd/tests/integration-edge.test.ts +0 -228
  631. package/dist/resources/extensions/gsd/tests/integration-lifecycle.test.ts +0 -277
  632. package/dist/resources/extensions/gsd/tests/integration-mixed-milestones.test.ts +0 -523
  633. package/dist/resources/extensions/gsd/tests/key-manager.test.ts +0 -414
  634. package/dist/resources/extensions/gsd/tests/knowledge.test.ts +0 -161
  635. package/dist/resources/extensions/gsd/tests/loop-regression.test.ts +0 -877
  636. package/dist/resources/extensions/gsd/tests/manifest-status.test.ts +0 -283
  637. package/dist/resources/extensions/gsd/tests/marketplace-test-fixtures.ts +0 -91
  638. package/dist/resources/extensions/gsd/tests/md-importer.test.ts +0 -410
  639. package/dist/resources/extensions/gsd/tests/mechanical-completion.test.ts +0 -356
  640. package/dist/resources/extensions/gsd/tests/memory-extractor.test.ts +0 -180
  641. package/dist/resources/extensions/gsd/tests/memory-leak-guards.test.ts +0 -91
  642. package/dist/resources/extensions/gsd/tests/memory-store.test.ts +0 -345
  643. package/dist/resources/extensions/gsd/tests/metrics.test.ts +0 -253
  644. package/dist/resources/extensions/gsd/tests/migrate-command.test.ts +0 -369
  645. package/dist/resources/extensions/gsd/tests/migrate-parser.test.ts +0 -757
  646. package/dist/resources/extensions/gsd/tests/migrate-transformer.test.ts +0 -635
  647. package/dist/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +0 -414
  648. package/dist/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +0 -303
  649. package/dist/resources/extensions/gsd/tests/migrate-writer.test.ts +0 -398
  650. package/dist/resources/extensions/gsd/tests/milestone-transition-worktree.test.ts +0 -147
  651. package/dist/resources/extensions/gsd/tests/model-cost-table.test.ts +0 -69
  652. package/dist/resources/extensions/gsd/tests/model-isolation.test.ts +0 -157
  653. package/dist/resources/extensions/gsd/tests/model-router.test.ts +0 -167
  654. package/dist/resources/extensions/gsd/tests/must-have-parser.test.ts +0 -291
  655. package/dist/resources/extensions/gsd/tests/namespaced-registry.test.ts +0 -1027
  656. package/dist/resources/extensions/gsd/tests/namespaced-resolver.test.ts +0 -671
  657. package/dist/resources/extensions/gsd/tests/native-has-changes-cache.test.ts +0 -61
  658. package/dist/resources/extensions/gsd/tests/next-milestone-id.test.ts +0 -23
  659. package/dist/resources/extensions/gsd/tests/none-mode-gates.test.ts +0 -114
  660. package/dist/resources/extensions/gsd/tests/notifications.test.ts +0 -67
  661. package/dist/resources/extensions/gsd/tests/overrides.test.ts +0 -131
  662. package/dist/resources/extensions/gsd/tests/parallel-budget-atomicity.test.ts +0 -331
  663. package/dist/resources/extensions/gsd/tests/parallel-crash-recovery.test.ts +0 -298
  664. package/dist/resources/extensions/gsd/tests/parallel-merge.test.ts +0 -468
  665. package/dist/resources/extensions/gsd/tests/parallel-orchestration.test.ts +0 -685
  666. package/dist/resources/extensions/gsd/tests/parallel-worker-monitoring.test.ts +0 -171
  667. package/dist/resources/extensions/gsd/tests/parallel-workers-multi-milestone-e2e.test.ts +0 -354
  668. package/dist/resources/extensions/gsd/tests/park-edge-cases.test.ts +0 -276
  669. package/dist/resources/extensions/gsd/tests/park-milestone.test.ts +0 -401
  670. package/dist/resources/extensions/gsd/tests/parsers.test.ts +0 -1704
  671. package/dist/resources/extensions/gsd/tests/plan-milestone.test.ts +0 -133
  672. package/dist/resources/extensions/gsd/tests/plan-quality-validator.test.ts +0 -363
  673. package/dist/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +0 -42
  674. package/dist/resources/extensions/gsd/tests/plugin-importer-live.test.ts +0 -481
  675. package/dist/resources/extensions/gsd/tests/plugin-importer.test.ts +0 -1383
  676. package/dist/resources/extensions/gsd/tests/post-unit-hooks.test.ts +0 -337
  677. package/dist/resources/extensions/gsd/tests/preferences.test.ts +0 -276
  678. package/dist/resources/extensions/gsd/tests/progress-score.test.ts +0 -206
  679. package/dist/resources/extensions/gsd/tests/prompt-budget-enforcement.test.ts +0 -464
  680. package/dist/resources/extensions/gsd/tests/prompt-cache-optimizer.test.ts +0 -314
  681. package/dist/resources/extensions/gsd/tests/prompt-compressor.test.ts +0 -529
  682. package/dist/resources/extensions/gsd/tests/prompt-db.test.ts +0 -385
  683. package/dist/resources/extensions/gsd/tests/prompt-ordering.test.ts +0 -296
  684. package/dist/resources/extensions/gsd/tests/provider-errors.test.ts +0 -338
  685. package/dist/resources/extensions/gsd/tests/queue-draft-detection.test.ts +0 -126
  686. package/dist/resources/extensions/gsd/tests/queue-order.test.ts +0 -204
  687. package/dist/resources/extensions/gsd/tests/queue-reorder-e2e.test.ts +0 -282
  688. package/dist/resources/extensions/gsd/tests/reassess-detection.test.ts +0 -154
  689. package/dist/resources/extensions/gsd/tests/reassess-prompt.test.ts +0 -145
  690. package/dist/resources/extensions/gsd/tests/regex-hardening.test.ts +0 -281
  691. package/dist/resources/extensions/gsd/tests/remote-questions.test.ts +0 -642
  692. package/dist/resources/extensions/gsd/tests/remote-status.test.ts +0 -99
  693. package/dist/resources/extensions/gsd/tests/replan-slice.test.ts +0 -538
  694. package/dist/resources/extensions/gsd/tests/requirements.test.ts +0 -106
  695. package/dist/resources/extensions/gsd/tests/roadmap-parse-regression.test.ts +0 -358
  696. package/dist/resources/extensions/gsd/tests/roadmap-slices.test.ts +0 -66
  697. package/dist/resources/extensions/gsd/tests/routing-history.test.ts +0 -240
  698. package/dist/resources/extensions/gsd/tests/run-uat.test.ts +0 -416
  699. package/dist/resources/extensions/gsd/tests/secure-env-collect.test.ts +0 -185
  700. package/dist/resources/extensions/gsd/tests/semantic-chunker.test.ts +0 -426
  701. package/dist/resources/extensions/gsd/tests/session-lock-regression.test.ts +0 -216
  702. package/dist/resources/extensions/gsd/tests/session-lock.test.ts +0 -434
  703. package/dist/resources/extensions/gsd/tests/skill-lifecycle.test.ts +0 -126
  704. package/dist/resources/extensions/gsd/tests/smart-entry-draft.test.ts +0 -123
  705. package/dist/resources/extensions/gsd/tests/stale-worktree-cwd.test.ts +0 -142
  706. package/dist/resources/extensions/gsd/tests/stop-auto-remote.test.ts +0 -156
  707. package/dist/resources/extensions/gsd/tests/structured-data-formatter.test.ts +0 -365
  708. package/dist/resources/extensions/gsd/tests/summary-distiller.test.ts +0 -323
  709. package/dist/resources/extensions/gsd/tests/test-helpers.ts +0 -61
  710. package/dist/resources/extensions/gsd/tests/token-counter.test.ts +0 -129
  711. package/dist/resources/extensions/gsd/tests/token-optimization-benchmark.test.ts +0 -1272
  712. package/dist/resources/extensions/gsd/tests/token-optimization-prefs.test.ts +0 -164
  713. package/dist/resources/extensions/gsd/tests/token-profile.test.ts +0 -268
  714. package/dist/resources/extensions/gsd/tests/token-savings.test.ts +0 -366
  715. package/dist/resources/extensions/gsd/tests/triage-dispatch.test.ts +0 -340
  716. package/dist/resources/extensions/gsd/tests/triage-resolution.test.ts +0 -416
  717. package/dist/resources/extensions/gsd/tests/undo.test.ts +0 -136
  718. package/dist/resources/extensions/gsd/tests/unique-milestone-ids.test.ts +0 -219
  719. package/dist/resources/extensions/gsd/tests/unit-runtime.test.ts +0 -258
  720. package/dist/resources/extensions/gsd/tests/update-command.test.ts +0 -67
  721. package/dist/resources/extensions/gsd/tests/validate-directory.test.ts +0 -222
  722. package/dist/resources/extensions/gsd/tests/validate-milestone.test.ts +0 -375
  723. package/dist/resources/extensions/gsd/tests/verification-evidence.test.ts +0 -745
  724. package/dist/resources/extensions/gsd/tests/verification-gate.test.ts +0 -1208
  725. package/dist/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +0 -145
  726. package/dist/resources/extensions/gsd/tests/visualizer-data.test.ts +0 -446
  727. package/dist/resources/extensions/gsd/tests/visualizer-overlay.test.ts +0 -237
  728. package/dist/resources/extensions/gsd/tests/visualizer-views.test.ts +0 -718
  729. package/dist/resources/extensions/gsd/tests/worker-registry.test.ts +0 -148
  730. package/dist/resources/extensions/gsd/tests/workflow-templates.test.ts +0 -173
  731. package/dist/resources/extensions/gsd/tests/workspace-index.test.ts +0 -38
  732. package/dist/resources/extensions/gsd/tests/worktree-bugfix.test.ts +0 -120
  733. package/dist/resources/extensions/gsd/tests/worktree-e2e.test.ts +0 -244
  734. package/dist/resources/extensions/gsd/tests/worktree-integration.test.ts +0 -207
  735. package/dist/resources/extensions/gsd/tests/worktree-manager.test.ts +0 -141
  736. package/dist/resources/extensions/gsd/tests/worktree-post-create-hook.test.ts +0 -165
  737. package/dist/resources/extensions/gsd/tests/worktree-sync-milestones.test.ts +0 -206
  738. package/dist/resources/extensions/gsd/tests/worktree.test.ts +0 -171
  739. package/dist/resources/extensions/gsd/tests/write-gate.test.ts +0 -211
  740. package/dist/resources/extensions/gsd/token-counter.ts +0 -65
  741. package/dist/resources/extensions/gsd/triage-resolution.ts +0 -284
  742. package/dist/resources/extensions/gsd/triage-ui.ts +0 -175
  743. package/dist/resources/extensions/gsd/types.ts +0 -425
  744. package/dist/resources/extensions/gsd/undo.ts +0 -223
  745. package/dist/resources/extensions/gsd/unit-id.ts +0 -14
  746. package/dist/resources/extensions/gsd/unit-runtime.ts +0 -192
  747. package/dist/resources/extensions/gsd/validate-directory.ts +0 -164
  748. package/dist/resources/extensions/gsd/verification-evidence.ts +0 -188
  749. package/dist/resources/extensions/gsd/verification-gate.ts +0 -643
  750. package/dist/resources/extensions/gsd/visualizer-data.ts +0 -866
  751. package/dist/resources/extensions/gsd/visualizer-overlay.ts +0 -566
  752. package/dist/resources/extensions/gsd/visualizer-views.ts +0 -1171
  753. package/dist/resources/extensions/gsd/workflow-templates.ts +0 -241
  754. package/dist/resources/extensions/gsd/workspace-index.ts +0 -217
  755. package/dist/resources/extensions/gsd/worktree-command.ts +0 -807
  756. package/dist/resources/extensions/gsd/worktree-manager.ts +0 -449
  757. package/dist/resources/extensions/gsd/worktree.ts +0 -257
  758. package/dist/resources/extensions/mac-tools/index.ts +0 -852
  759. package/dist/resources/extensions/mcp-client/index.ts +0 -459
  760. package/dist/resources/extensions/remote-questions/config.ts +0 -83
  761. package/dist/resources/extensions/remote-questions/discord-adapter.ts +0 -148
  762. package/dist/resources/extensions/remote-questions/format.ts +0 -315
  763. package/dist/resources/extensions/remote-questions/http-client.ts +0 -76
  764. package/dist/resources/extensions/remote-questions/manager.ts +0 -184
  765. package/dist/resources/extensions/remote-questions/notify.ts +0 -90
  766. package/dist/resources/extensions/remote-questions/remote-command.ts +0 -457
  767. package/dist/resources/extensions/remote-questions/slack-adapter.ts +0 -141
  768. package/dist/resources/extensions/remote-questions/status.ts +0 -31
  769. package/dist/resources/extensions/remote-questions/store.ts +0 -81
  770. package/dist/resources/extensions/remote-questions/telegram-adapter.ts +0 -149
  771. package/dist/resources/extensions/remote-questions/types.ts +0 -102
  772. package/dist/resources/extensions/search-the-web/cache.ts +0 -78
  773. package/dist/resources/extensions/search-the-web/command-search-provider.ts +0 -101
  774. package/dist/resources/extensions/search-the-web/format.ts +0 -258
  775. package/dist/resources/extensions/search-the-web/http.ts +0 -238
  776. package/dist/resources/extensions/search-the-web/index.ts +0 -65
  777. package/dist/resources/extensions/search-the-web/native-search.ts +0 -193
  778. package/dist/resources/extensions/search-the-web/provider.ts +0 -148
  779. package/dist/resources/extensions/search-the-web/tavily.ts +0 -116
  780. package/dist/resources/extensions/search-the-web/tool-fetch-page.ts +0 -589
  781. package/dist/resources/extensions/search-the-web/tool-llm-context.ts +0 -608
  782. package/dist/resources/extensions/search-the-web/tool-search.ts +0 -649
  783. package/dist/resources/extensions/search-the-web/url-utils.ts +0 -125
  784. package/dist/resources/extensions/shared/confirm-ui.ts +0 -126
  785. package/dist/resources/extensions/shared/frontmatter.ts +0 -117
  786. package/dist/resources/extensions/shared/interview-ui.ts +0 -613
  787. package/dist/resources/extensions/shared/next-action-ui.ts +0 -212
  788. package/dist/resources/extensions/shared/sanitize.ts +0 -19
  789. package/dist/resources/extensions/shared/terminal.ts +0 -23
  790. package/dist/resources/extensions/shared/tests/format-utils.test.ts +0 -153
  791. package/dist/resources/extensions/shared/ui.ts +0 -400
  792. package/dist/resources/extensions/shared/wizard-ui.ts +0 -551
  793. package/dist/resources/extensions/slash-commands/audit.ts +0 -88
  794. package/dist/resources/extensions/slash-commands/clear.ts +0 -10
  795. package/dist/resources/extensions/slash-commands/create-extension.ts +0 -297
  796. package/dist/resources/extensions/slash-commands/create-slash-command.ts +0 -234
  797. package/dist/resources/extensions/slash-commands/index.ts +0 -12
  798. package/dist/resources/extensions/subagent/agents.ts +0 -126
  799. package/dist/resources/extensions/subagent/index.ts +0 -1121
  800. package/dist/resources/extensions/subagent/isolation.ts +0 -501
  801. package/dist/resources/extensions/subagent/worker-registry.ts +0 -99
  802. package/dist/resources/extensions/ttsr/index.ts +0 -168
  803. package/dist/resources/extensions/ttsr/rule-loader.ts +0 -74
  804. package/dist/resources/extensions/ttsr/ttsr-manager.ts +0 -456
  805. package/dist/resources/extensions/universal-config/discovery.ts +0 -104
  806. package/dist/resources/extensions/universal-config/format.ts +0 -191
  807. package/dist/resources/extensions/universal-config/index.ts +0 -120
  808. package/dist/resources/extensions/universal-config/scanners.ts +0 -642
  809. package/dist/resources/extensions/universal-config/tests/discovery.test.ts +0 -119
  810. package/dist/resources/extensions/universal-config/tests/format.test.ts +0 -127
  811. package/dist/resources/extensions/universal-config/tests/scanners.test.ts +0 -456
  812. package/dist/resources/extensions/universal-config/tools.ts +0 -60
  813. package/dist/resources/extensions/universal-config/types.ts +0 -135
  814. package/dist/resources/extensions/voice/index.ts +0 -272
  815. package/dist/resources/skills/create-gsd-extension/templates/extension-skeleton.ts +0 -51
  816. package/dist/resources/skills/create-gsd-extension/templates/stateful-tool-skeleton.ts +0 -143
  817. package/src/resources/extensions/gsd/auto-constants.ts +0 -6
  818. package/src/resources/extensions/gsd/auto-idempotency.ts +0 -151
  819. package/src/resources/extensions/gsd/auto-stuck-detection.ts +0 -221
  820. package/src/resources/extensions/gsd/mechanical-completion.ts +0 -430
  821. package/src/resources/extensions/gsd/tests/auto-dispatch-loop.test.ts +0 -691
  822. package/src/resources/extensions/gsd/tests/auto-reentrancy-guard.test.ts +0 -127
  823. package/src/resources/extensions/gsd/tests/auto-skip-loop.test.ts +0 -123
  824. package/src/resources/extensions/gsd/tests/dispatch-stall-guard.test.ts +0 -126
  825. package/src/resources/extensions/gsd/tests/loop-regression.test.ts +0 -877
  826. package/src/resources/extensions/gsd/tests/mechanical-completion.test.ts +0 -356
  827. package/src/resources/extensions/gsd/tests/progress-score.test.ts +0 -206
  828. package/src/resources/extensions/gsd/tests/session-lock.test.ts +0 -434
@@ -1,1390 +0,0 @@
1
- /**
2
- * GSD Extension — /gsd
3
- *
4
- * One command, one wizard. Reads state from disk, shows contextual options,
5
- * dispatches through GSD-WORKFLOW.md. The LLM does the rest.
6
- *
7
- * Auto-mode: /gsd auto loops fresh sessions until milestone complete.
8
- *
9
- * Commands:
10
- * /gsd — contextual wizard (smart entry point)
11
- * /gsd auto — start auto-mode (fresh session per unit)
12
- * /gsd stop — stop auto-mode gracefully
13
- * /gsd status — progress dashboard
14
- *
15
- * Hooks:
16
- * before_agent_start — inject GSD system context for GSD projects
17
- * agent_end — auto-mode advancement
18
- * session_before_compact — save continue.md OR block during auto
19
- */
20
-
21
- import type {
22
- ExtensionAPI,
23
- ExtensionCommandContext,
24
- ExtensionContext,
25
- } from "@gsd/pi-coding-agent";
26
- import { createBashTool, createWriteTool, createReadTool, createEditTool, isToolCallEventType } from "@gsd/pi-coding-agent";
27
- import { Type } from "@sinclair/typebox";
28
-
29
- import { debugLog, debugTime } from "./debug-logger.js";
30
- import { registerGSDCommand } from "./commands.js";
31
- import { loadToolApiKeys } from "./commands-config.js";
32
- import { registerExitCommand } from "./exit-command.js";
33
- import { registerWorktreeCommand, getWorktreeOriginalCwd, getActiveWorktreeName } from "./worktree-command.js";
34
- import { getActiveAutoWorktreeContext } from "./auto-worktree.js";
35
- import { saveFile, formatContinue, loadFile, parseContinue, parseSummary, loadActiveOverrides, formatOverridesSection } from "./files.js";
36
- import { loadPrompt } from "./prompt-loader.js";
37
- import { deriveState } from "./state.js";
38
- import { isAutoActive, isAutoPaused, handleAgentEnd, pauseAuto, getAutoDashboardData, getAutoModeStartModel, markToolStart, markToolEnd } from "./auto.js";
39
- import { saveActivityLog } from "./activity-log.js";
40
- import { checkAutoStartAfterDiscuss, getDiscussionMilestoneId, findMilestoneIds, nextMilestoneId } from "./guided-flow.js";
41
- import { GSDDashboardOverlay } from "./dashboard-overlay.js";
42
- import {
43
- loadEffectiveGSDPreferences,
44
- renderPreferencesForSystemPrompt,
45
- resolveAllSkillReferences,
46
- resolveModelWithFallbacksForUnit,
47
- getNextFallbackModel,
48
- isTransientNetworkError,
49
- } from "./preferences.js";
50
- import { hasSkillSnapshot, detectNewSkills, formatSkillsXml } from "./skill-discovery.js";
51
- import {
52
- resolveSlicePath, resolveSliceFile, resolveTaskFile, resolveTaskFiles, resolveTasksDir,
53
- relSliceFile, relSlicePath, relTaskFile,
54
- buildSliceFileName, buildMilestoneFileName, gsdRoot, resolveMilestonePath,
55
- resolveGsdRootFile,
56
- } from "./paths.js";
57
- import { Key } from "@gsd/pi-tui";
58
- import { join } from "node:path";
59
- import { existsSync, readFileSync } from "node:fs";
60
- import { homedir } from "node:os";
61
- import { shortcutDesc } from "../shared/mod.js";
62
- import { Text } from "@gsd/pi-tui";
63
- import { pauseAutoForProviderError, classifyProviderError } from "./provider-error-pause.js";
64
- import { toPosixPath } from "../shared/mod.js";
65
- import { isParallelActive, shutdownParallel } from "./parallel-orchestrator.js";
66
- import { DEFAULT_BASH_TIMEOUT_SECS } from "./constants.js";
67
- import { getErrorMessage } from "./error-utils.js";
68
-
69
- /**
70
- * Ensure the GSD database is available, auto-initializing if needed.
71
- * Returns true if the DB is ready, false if initialization failed.
72
- */
73
- async function ensureDbAvailable(): Promise<boolean> {
74
- try {
75
- const db = await import("./gsd-db.js");
76
- if (db.isDbAvailable()) return true;
77
-
78
- // Auto-initialize: open (and create if needed) the DB at the standard path
79
- const gsdDir = gsdRoot(process.cwd());
80
- if (!existsSync(gsdDir)) return false; // No GSD project — can't create DB
81
- const dbPath = join(gsdDir, "gsd.db");
82
- return db.openDatabase(dbPath);
83
- } catch {
84
- return false;
85
- }
86
- }
87
-
88
- // ── Agent Instructions ────────────────────────────────────────────────────
89
- // Lightweight "always follow" files injected into every GSD agent session.
90
- // Global: ~/.gsd/agent-instructions.md Project: .gsd/agent-instructions.md
91
- // Both are loaded and concatenated (global first, project appends).
92
-
93
- function loadAgentInstructions(): string | null {
94
- const parts: string[] = [];
95
-
96
- const globalPath = join(homedir(), ".gsd", "agent-instructions.md");
97
- if (existsSync(globalPath)) {
98
- try {
99
- const content = readFileSync(globalPath, "utf-8").trim();
100
- if (content) parts.push(content);
101
- } catch { /* non-fatal — skip unreadable file */ }
102
- }
103
-
104
- const projectPath = join(process.cwd(), ".gsd", "agent-instructions.md");
105
- if (existsSync(projectPath)) {
106
- try {
107
- const content = readFileSync(projectPath, "utf-8").trim();
108
- if (content) parts.push(content);
109
- } catch { /* non-fatal — skip unreadable file */ }
110
- }
111
-
112
- if (parts.length === 0) return null;
113
- return parts.join("\n\n");
114
- }
115
-
116
- // ── Depth verification state ──────────────────────────────────────────────
117
- // Tracks which milestones have passed depth verification.
118
- // Single-milestone flows set '*' (wildcard). Multi-milestone flows set per-ID.
119
- const depthVerifiedMilestones = new Set<string>();
120
-
121
- // ── Queue phase tracking ──────────────────────────────────────────────────
122
- // When true, the LLM is in a queue flow writing CONTEXT.md files.
123
- // The write-gate applies during queue flows just like discussion flows.
124
- let activeQueuePhase = false;
125
-
126
- // ── Network error retry counters ──────────────────────────────────────────
127
- // Tracks per-model retry attempts for transient network errors.
128
- // Cleared when a model switch occurs or retries are exhausted.
129
- const networkRetryCounters = new Map<string, number>();
130
-
131
- // ── Transient error escalation ───────────────────────────────────────────
132
- // Tracks consecutive transient auto-resume attempts. Each attempt doubles
133
- // the delay. After MAX_TRANSIENT_AUTO_RESUMES consecutive failures, auto-mode
134
- // pauses indefinitely to avoid infinite rapid-fire retries (#1166).
135
- const MAX_TRANSIENT_AUTO_RESUMES = 5;
136
- let consecutiveTransientErrors = 0;
137
-
138
- export function isDepthVerified(): boolean {
139
- return depthVerifiedMilestones.has("*") || depthVerifiedMilestones.size > 0;
140
- }
141
-
142
- /** Check whether a specific milestone has passed depth verification. */
143
- export function isDepthVerifiedFor(milestoneId: string): boolean {
144
- // Wildcard means "all milestones verified" (single-milestone flow)
145
- if (depthVerifiedMilestones.has("*")) return true;
146
- return depthVerifiedMilestones.has(milestoneId);
147
- }
148
-
149
- /** Mark a specific milestone as depth-verified. */
150
- export function markDepthVerified(milestoneId: string): void {
151
- depthVerifiedMilestones.add(milestoneId);
152
- }
153
-
154
- /** Check whether a queue phase is active. */
155
- export function isQueuePhaseActive(): boolean {
156
- return activeQueuePhase;
157
- }
158
-
159
- /** Set the queue phase state — called from guided-flow-queue.ts on dispatch. */
160
- export function setQueuePhaseActive(active: boolean): void {
161
- activeQueuePhase = active;
162
- }
163
-
164
- // ── Write-gate: block CONTEXT.md writes during discussion without depth verification ──
165
- const MILESTONE_CONTEXT_RE = /M\d+(?:-[a-z0-9]{6})?-CONTEXT\.md$/;
166
-
167
- export function shouldBlockContextWrite(
168
- toolName: string,
169
- inputPath: string,
170
- milestoneId: string | null,
171
- depthVerified: boolean,
172
- queuePhaseActive?: boolean,
173
- ): { block: boolean; reason?: string } {
174
- if (toolName !== "write") return { block: false };
175
-
176
- // Gate applies during both discussion (milestoneId set) and queue (queuePhaseActive) flows
177
- const inDiscussion = milestoneId !== null;
178
- const inQueue = queuePhaseActive ?? false;
179
- if (!inDiscussion && !inQueue) return { block: false };
180
-
181
- if (!MILESTONE_CONTEXT_RE.test(inputPath)) return { block: false };
182
-
183
- // For discussion flows: check global depth verification (backward compat)
184
- if (inDiscussion && depthVerified) return { block: false };
185
-
186
- // For queue flows: extract milestone ID from the path and check per-milestone verification
187
- if (inQueue) {
188
- const pathMatch = inputPath.match(/\/(M\d+(?:-[a-z0-9]{6})?)-CONTEXT\.md$/);
189
- const targetMid = pathMatch?.[1];
190
- if (targetMid && depthVerifiedMilestones.has(targetMid)) return { block: false };
191
- // Wildcard passes all
192
- if (depthVerifiedMilestones.has("*")) return { block: false };
193
- }
194
-
195
- return {
196
- block: true,
197
- reason: `Blocked: Cannot write milestone CONTEXT.md without depth verification. ` +
198
- `Use ask_user_questions with a question id containing "depth_verification" first. ` +
199
- `For multi-milestone flows, include the milestone ID in the question id (e.g., "depth_verification_M001"). ` +
200
- `This ensures each milestone's context has been critically examined before being written.`,
201
- };
202
- }
203
-
204
- // ── ASCII logo ────────────────────────────────────────────────────────────
205
- const GSD_LOGO_LINES = [
206
- " ██████╗ ███████╗██████╗ ",
207
- " ██╔════╝ ██╔════╝██╔══██╗",
208
- " ██║ ███╗███████╗██║ ██║",
209
- " ██║ ██║╚════██║██║ ██║",
210
- " ╚██████╔╝███████║██████╔╝",
211
- " ╚═════╝ ╚══════╝╚═════╝ ",
212
- ];
213
-
214
- export default function (pi: ExtensionAPI) {
215
- registerGSDCommand(pi);
216
- registerWorktreeCommand(pi);
217
- registerExitCommand(pi);
218
-
219
- // ── EPIPE guard — prevent crash when stdout/stderr pipe closes unexpectedly ──
220
- // Node.js throws a fatal `Error: write EPIPE` when the parent process closes
221
- // its end of the stdio pipe (e.g. during shell/IPC teardown) while auto-mode
222
- // is still writing diagnostics. Catching this here gives auto-mode a clean
223
- // chance to persist state and pause instead of crashing (see issue #739).
224
- if (!process.listeners("uncaughtException").some(l => l.name === "_gsdEpipeGuard")) {
225
- const _gsdEpipeGuard = (err: Error): void => {
226
- if ((err as NodeJS.ErrnoException).code === "EPIPE") {
227
- // Pipe closed — nothing we can write; just exit cleanly
228
- process.exit(0);
229
- }
230
- // Re-throw anything that isn't EPIPE so real crashes still surface
231
- throw err;
232
- };
233
- process.on("uncaughtException", _gsdEpipeGuard);
234
- }
235
-
236
- // ── /kill — immediate exit (bypass cleanup) ─────────────────────────────
237
- pi.registerCommand("kill", {
238
- description: "Exit GSD immediately (no cleanup)",
239
- handler: async (_args: string, _ctx: ExtensionCommandContext) => {
240
- process.exit(0);
241
- },
242
- });
243
-
244
- // ── Dynamic-cwd bash tool with default timeout ────────────────────────
245
- // The built-in bash tool captures cwd at startup. This replacement uses
246
- // a spawnHook to read process.cwd() dynamically so that process.chdir()
247
- // (used by /worktree switch) propagates to shell commands.
248
- //
249
- // The upstream SDK's bash tool has no default timeout — if the LLM omits
250
- // the timeout parameter, commands run indefinitely, causing hangs on
251
- // Windows where process killing is unreliable (see #40). We wrap execute
252
- // to inject a 120-second default when no timeout is provided.
253
- const baseBash = createBashTool(process.cwd(), {
254
- spawnHook: (ctx) => ({ ...ctx, cwd: process.cwd() }),
255
- });
256
- const dynamicBash = {
257
- ...baseBash,
258
- execute: async (
259
- toolCallId: string,
260
- params: { command: string; timeout?: number },
261
- signal?: AbortSignal,
262
- onUpdate?: any,
263
- ctx?: any,
264
- ) => {
265
- const paramsWithTimeout = {
266
- ...params,
267
- timeout: params.timeout ?? DEFAULT_BASH_TIMEOUT_SECS,
268
- };
269
- return (baseBash as any).execute(toolCallId, paramsWithTimeout, signal, onUpdate, ctx);
270
- },
271
- };
272
- pi.registerTool(dynamicBash as any);
273
-
274
- // ── Dynamic-cwd file tools (write, read, edit) ────────────────────────
275
- // The built-in file tools capture cwd at startup. When process.chdir()
276
- // moves us into a worktree, relative paths still resolve against the
277
- // original launch directory. These replacements delegate to freshly-
278
- // created tools on each call so that process.cwd() is read dynamically.
279
- const baseWrite = createWriteTool(process.cwd());
280
- const dynamicWrite = {
281
- ...baseWrite,
282
- execute: async (
283
- toolCallId: string,
284
- params: { path: string; content: string },
285
- signal?: AbortSignal,
286
- onUpdate?: any,
287
- ctx?: any,
288
- ) => {
289
- const fresh = createWriteTool(process.cwd());
290
- return (fresh as any).execute(toolCallId, params, signal, onUpdate, ctx);
291
- },
292
- };
293
- pi.registerTool(dynamicWrite as any);
294
-
295
- const baseRead = createReadTool(process.cwd());
296
- const dynamicRead = {
297
- ...baseRead,
298
- execute: async (
299
- toolCallId: string,
300
- params: { path: string; offset?: number; limit?: number },
301
- signal?: AbortSignal,
302
- onUpdate?: any,
303
- ctx?: any,
304
- ) => {
305
- const fresh = createReadTool(process.cwd());
306
- return (fresh as any).execute(toolCallId, params, signal, onUpdate, ctx);
307
- },
308
- };
309
- pi.registerTool(dynamicRead as any);
310
-
311
- const baseEdit = createEditTool(process.cwd());
312
- const dynamicEdit = {
313
- ...baseEdit,
314
- execute: async (
315
- toolCallId: string,
316
- params: { path: string; oldText: string; newText: string },
317
- signal?: AbortSignal,
318
- onUpdate?: any,
319
- ctx?: any,
320
- ) => {
321
- const fresh = createEditTool(process.cwd());
322
- return (fresh as any).execute(toolCallId, params, signal, onUpdate, ctx);
323
- },
324
- };
325
- pi.registerTool(dynamicEdit as any);
326
-
327
- // ── Structured LLM tools — DB-first write path (R014) ──────────────────
328
-
329
- pi.registerTool({
330
- name: "gsd_save_decision",
331
- label: "Save Decision",
332
- description:
333
- "Record a project decision to the GSD database and regenerate DECISIONS.md. " +
334
- "Decision IDs are auto-assigned — never provide an ID manually.",
335
- promptSnippet: "Record a project decision to the GSD database (auto-assigns ID, regenerates DECISIONS.md)",
336
- promptGuidelines: [
337
- "Use gsd_save_decision when recording an architectural, pattern, library, or observability decision.",
338
- "Decision IDs are auto-assigned (D001, D002, ...) — never guess or provide an ID.",
339
- "All fields except revisable and when_context are required.",
340
- "The tool writes to the DB and regenerates .gsd/DECISIONS.md automatically.",
341
- ],
342
- parameters: Type.Object({
343
- scope: Type.String({ description: "Scope of the decision (e.g. 'architecture', 'library', 'observability')" }),
344
- decision: Type.String({ description: "What is being decided" }),
345
- choice: Type.String({ description: "The choice made" }),
346
- rationale: Type.String({ description: "Why this choice was made" }),
347
- revisable: Type.Optional(Type.String({ description: "Whether this can be revisited (default: 'Yes')" })),
348
- when_context: Type.Optional(Type.String({ description: "When/context for the decision (e.g. milestone ID)" })),
349
- }),
350
- async execute(_toolCallId, params, _signal, _onUpdate, _ctx) {
351
- // Ensure DB is available (auto-initialize if needed)
352
- if (!await ensureDbAvailable()) {
353
- return {
354
- content: [{ type: "text" as const, text: "Error: GSD database is not available. Cannot save decision." }],
355
- isError: true,
356
- details: { operation: "save_decision", error: "db_unavailable" },
357
- };
358
- }
359
-
360
- try {
361
- const { saveDecisionToDb } = await import("./db-writer.js");
362
- const { id } = await saveDecisionToDb(
363
- {
364
- scope: params.scope,
365
- decision: params.decision,
366
- choice: params.choice,
367
- rationale: params.rationale,
368
- revisable: params.revisable,
369
- when_context: params.when_context,
370
- },
371
- process.cwd(),
372
- );
373
- return {
374
- content: [{ type: "text" as const, text: `Saved decision ${id}` }],
375
- details: { operation: "save_decision", id },
376
- };
377
- } catch (err) {
378
- const msg = getErrorMessage(err);
379
- process.stderr.write(`gsd-db: gsd_save_decision tool failed: ${msg}\n`);
380
- return {
381
- content: [{ type: "text" as const, text: `Error saving decision: ${msg}` }],
382
- isError: true,
383
- details: { operation: "save_decision", error: msg },
384
- };
385
- }
386
- },
387
- });
388
-
389
- pi.registerTool({
390
- name: "gsd_update_requirement",
391
- label: "Update Requirement",
392
- description:
393
- "Update an existing requirement in the GSD database and regenerate REQUIREMENTS.md. " +
394
- "Provide the requirement ID (e.g. R001) and any fields to update.",
395
- promptSnippet: "Update an existing GSD requirement by ID (regenerates REQUIREMENTS.md)",
396
- promptGuidelines: [
397
- "Use gsd_update_requirement to change status, validation, notes, or other fields on an existing requirement.",
398
- "The id parameter is required — it must be an existing RXXX identifier.",
399
- "All other fields are optional — only provided fields are updated.",
400
- "The tool verifies the requirement exists before updating.",
401
- ],
402
- parameters: Type.Object({
403
- id: Type.String({ description: "The requirement ID (e.g. R001, R014)" }),
404
- status: Type.Optional(Type.String({ description: "New status (e.g. 'active', 'validated', 'deferred')" })),
405
- validation: Type.Optional(Type.String({ description: "Validation criteria or proof" })),
406
- notes: Type.Optional(Type.String({ description: "Additional notes" })),
407
- description: Type.Optional(Type.String({ description: "Updated description" })),
408
- primary_owner: Type.Optional(Type.String({ description: "Primary owning slice" })),
409
- supporting_slices: Type.Optional(Type.String({ description: "Supporting slices" })),
410
- }),
411
- async execute(_toolCallId, params, _signal, _onUpdate, _ctx) {
412
- // Ensure DB is available (auto-initialize if needed)
413
- if (!await ensureDbAvailable()) {
414
- return {
415
- content: [{ type: "text" as const, text: "Error: GSD database is not available. Cannot update requirement." }],
416
- isError: true,
417
- details: { operation: "update_requirement", id: params.id, error: "db_unavailable" },
418
- };
419
- }
420
-
421
- try {
422
- // Verify requirement exists
423
- const db = await import("./gsd-db.js");
424
- const existing = db.getRequirementById(params.id);
425
- if (!existing) {
426
- return {
427
- content: [{ type: "text" as const, text: `Error: Requirement ${params.id} not found.` }],
428
- isError: true,
429
- details: { operation: "update_requirement", id: params.id, error: "not_found" },
430
- };
431
- }
432
-
433
- const { updateRequirementInDb } = await import("./db-writer.js");
434
- const updates: Record<string, string | undefined> = {};
435
- if (params.status !== undefined) updates.status = params.status;
436
- if (params.validation !== undefined) updates.validation = params.validation;
437
- if (params.notes !== undefined) updates.notes = params.notes;
438
- if (params.description !== undefined) updates.description = params.description;
439
- if (params.primary_owner !== undefined) updates.primary_owner = params.primary_owner;
440
- if (params.supporting_slices !== undefined) updates.supporting_slices = params.supporting_slices;
441
-
442
- await updateRequirementInDb(params.id, updates, process.cwd());
443
-
444
- return {
445
- content: [{ type: "text" as const, text: `Updated requirement ${params.id}` }],
446
- details: { operation: "update_requirement", id: params.id },
447
- };
448
- } catch (err) {
449
- const msg = getErrorMessage(err);
450
- process.stderr.write(`gsd-db: gsd_update_requirement tool failed: ${msg}\n`);
451
- return {
452
- content: [{ type: "text" as const, text: `Error updating requirement: ${msg}` }],
453
- isError: true,
454
- details: { operation: "update_requirement", id: params.id, error: msg },
455
- };
456
- }
457
- },
458
- });
459
-
460
- pi.registerTool({
461
- name: "gsd_save_summary",
462
- label: "Save Summary",
463
- description:
464
- "Save a summary, research, context, or assessment artifact to the GSD database and write it to disk. " +
465
- "Computes the file path from milestone/slice/task IDs automatically.",
466
- promptSnippet: "Save a GSD artifact (summary/research/context/assessment) to DB and disk",
467
- promptGuidelines: [
468
- "Use gsd_save_summary to persist structured artifacts (SUMMARY, RESEARCH, CONTEXT, ASSESSMENT).",
469
- "milestone_id is required. slice_id and task_id are optional — they determine the file path.",
470
- "The tool computes the relative path automatically: milestones/M001/M001-SUMMARY.md, milestones/M001/slices/S01/S01-SUMMARY.md, etc.",
471
- "artifact_type must be one of: SUMMARY, RESEARCH, CONTEXT, ASSESSMENT.",
472
- ],
473
- parameters: Type.Object({
474
- milestone_id: Type.String({ description: "Milestone ID (e.g. M001)" }),
475
- slice_id: Type.Optional(Type.String({ description: "Slice ID (e.g. S01)" })),
476
- task_id: Type.Optional(Type.String({ description: "Task ID (e.g. T01)" })),
477
- artifact_type: Type.String({ description: "One of: SUMMARY, RESEARCH, CONTEXT, ASSESSMENT" }),
478
- content: Type.String({ description: "The full markdown content of the artifact" }),
479
- }),
480
- async execute(_toolCallId, params, _signal, _onUpdate, _ctx) {
481
- // Ensure DB is available (auto-initialize if needed)
482
- if (!await ensureDbAvailable()) {
483
- return {
484
- content: [{ type: "text" as const, text: "Error: GSD database is not available. Cannot save artifact." }],
485
- isError: true,
486
- details: { operation: "save_summary", error: "db_unavailable" },
487
- };
488
- }
489
-
490
- // Validate artifact_type
491
- const validTypes = ["SUMMARY", "RESEARCH", "CONTEXT", "ASSESSMENT"];
492
- if (!validTypes.includes(params.artifact_type)) {
493
- return {
494
- content: [{ type: "text" as const, text: `Error: Invalid artifact_type "${params.artifact_type}". Must be one of: ${validTypes.join(", ")}` }],
495
- isError: true,
496
- details: { operation: "save_summary", error: "invalid_artifact_type" },
497
- };
498
- }
499
-
500
- try {
501
- // Compute relative path from IDs
502
- let relativePath: string;
503
- if (params.task_id && params.slice_id) {
504
- relativePath = `milestones/${params.milestone_id}/slices/${params.slice_id}/tasks/${params.task_id}-${params.artifact_type}.md`;
505
- } else if (params.slice_id) {
506
- relativePath = `milestones/${params.milestone_id}/slices/${params.slice_id}/${params.slice_id}-${params.artifact_type}.md`;
507
- } else {
508
- relativePath = `milestones/${params.milestone_id}/${params.milestone_id}-${params.artifact_type}.md`;
509
- }
510
-
511
- const { saveArtifactToDb } = await import("./db-writer.js");
512
- await saveArtifactToDb(
513
- {
514
- path: relativePath,
515
- artifact_type: params.artifact_type,
516
- content: params.content,
517
- milestone_id: params.milestone_id,
518
- slice_id: params.slice_id,
519
- task_id: params.task_id,
520
- },
521
- process.cwd(),
522
- );
523
-
524
- return {
525
- content: [{ type: "text" as const, text: `Saved ${params.artifact_type} artifact to ${relativePath}` }],
526
- details: { operation: "save_summary", path: relativePath, artifact_type: params.artifact_type },
527
- };
528
- } catch (err) {
529
- const msg = getErrorMessage(err);
530
- process.stderr.write(`gsd-db: gsd_save_summary tool failed: ${msg}\n`);
531
- return {
532
- content: [{ type: "text" as const, text: `Error saving artifact: ${msg}` }],
533
- isError: true,
534
- details: { operation: "save_summary", error: msg },
535
- };
536
- }
537
- },
538
- });
539
-
540
- // ── gsd_generate_milestone_id — canonical milestone ID generation ──────
541
- // The LLM cannot generate random suffixes for unique_milestone_ids on its
542
- // own. This tool calls back into the TS code that owns ID generation,
543
- // ensuring the preference is always respected and IDs are always valid.
544
- //
545
- // Reservation set: tracks IDs returned by this tool but not yet persisted
546
- // to disk, preventing duplicate M001 when called multiple times (#961).
547
- const reservedMilestoneIds = new Set<string>();
548
-
549
- pi.registerTool({
550
- name: "gsd_generate_milestone_id",
551
- label: "Generate Milestone ID",
552
- description:
553
- "Generate the next milestone ID for a new GSD milestone. " +
554
- "Scans existing milestones on disk and respects the unique_milestone_ids preference. " +
555
- "Always use this tool when creating a new milestone — never invent milestone IDs manually.",
556
- promptSnippet: "Generate a valid milestone ID (respects unique_milestone_ids preference)",
557
- promptGuidelines: [
558
- "ALWAYS call gsd_generate_milestone_id before creating a new milestone directory or writing milestone files.",
559
- "Never invent or hardcode milestone IDs like M001, M002 — always use this tool.",
560
- "Call it once per milestone you need to create. For multi-milestone projects, call it once for each milestone in sequence.",
561
- "The tool returns the correct format based on project preferences (e.g. M001 or M001-r5jzab).",
562
- ],
563
- parameters: Type.Object({}),
564
- async execute(_toolCallId, _params, _signal, _onUpdate, _ctx) {
565
- try {
566
- const basePath = process.cwd();
567
- const existingIds = findMilestoneIds(basePath);
568
- const uniqueEnabled = !!loadEffectiveGSDPreferences()?.preferences?.unique_milestone_ids;
569
- // Combine on-disk IDs with previously reserved (but not yet persisted) IDs
570
- const allIds = [...new Set([...existingIds, ...reservedMilestoneIds])];
571
- const newId = nextMilestoneId(allIds, uniqueEnabled);
572
- reservedMilestoneIds.add(newId);
573
- return {
574
- content: [{ type: "text" as const, text: newId }],
575
- details: { operation: "generate_milestone_id", id: newId, existingCount: existingIds.length, reservedCount: reservedMilestoneIds.size, uniqueEnabled },
576
- };
577
- } catch (err) {
578
- const msg = getErrorMessage(err);
579
- return {
580
- content: [{ type: "text" as const, text: `Error generating milestone ID: ${msg}` }],
581
- isError: true,
582
- details: { operation: "generate_milestone_id", error: msg },
583
- };
584
- }
585
- },
586
- });
587
-
588
- // ── session_start: render branded GSD header + load tool keys + remote status ──
589
- pi.on("session_start", async (_event, ctx) => {
590
- // Clear depth verification and queue phase state from any prior session
591
- depthVerifiedMilestones.clear();
592
- activeQueuePhase = false;
593
-
594
- // Theme access throws in RPC mode (no TUI) — header is decorative, skip it
595
- try {
596
- const theme = ctx.ui.theme;
597
- const version = process.env.GSD_VERSION || "0.0.0";
598
-
599
- const logoText = GSD_LOGO_LINES.map((line) => theme.fg("accent", line)).join("\n");
600
- const titleLine = ` ${theme.bold("Get Shit Done")} ${theme.fg("dim", `v${version}`)}`;
601
-
602
- const headerContent = `${logoText}\n${titleLine}`;
603
- ctx.ui.setHeader((_ui, _theme) => new Text(headerContent, 1, 0));
604
- } catch {
605
- // RPC mode — no TUI, skip header rendering
606
- }
607
-
608
- // Load tool API keys from auth.json into environment
609
- loadToolApiKeys();
610
-
611
- // Always-on health widget — ambient system health signal below the editor
612
- try {
613
- const { initHealthWidget } = await import("./health-widget.js");
614
- initHealthWidget(ctx);
615
- } catch { /* non-fatal — widget is best-effort */ }
616
-
617
- // Notify remote questions status if configured
618
- try {
619
- const [{ getRemoteConfigStatus }, { getLatestPromptSummary }] = await Promise.all([
620
- import("../remote-questions/config.js"),
621
- import("../remote-questions/status.js"),
622
- ]);
623
- const status = getRemoteConfigStatus();
624
- const latest = getLatestPromptSummary();
625
- if (!status.includes("not configured")) {
626
- const suffix = latest ? `\nLast remote prompt: ${latest.id} (${latest.status})` : "";
627
- ctx.ui.notify(`${status}${suffix}`, status.includes("disabled") ? "warning" : "info");
628
- }
629
- } catch {
630
- // Remote questions module not available — ignore
631
- }
632
- });
633
-
634
- // ── Ctrl+Alt+G shortcut — GSD dashboard overlay ────────────────────────
635
- pi.registerShortcut(Key.ctrlAlt("g"), {
636
- description: shortcutDesc("Open GSD dashboard", "/gsd status"),
637
- handler: async (ctx) => {
638
- // Only show if .gsd/ exists
639
- if (!existsSync(gsdRoot(process.cwd()))) {
640
- ctx.ui.notify("No .gsd/ directory found. Run /gsd to start.", "info");
641
- return;
642
- }
643
-
644
- const result = await ctx.ui.custom<void>(
645
- (tui, theme, _kb, done) => {
646
- return new GSDDashboardOverlay(tui, theme, () => done());
647
- },
648
- {
649
- overlay: true,
650
- overlayOptions: {
651
- width: "90%",
652
- minWidth: 80,
653
- maxHeight: "92%",
654
- anchor: "center",
655
- },
656
- },
657
- );
658
-
659
- // Fallback for RPC mode where ctx.ui.custom() returns undefined.
660
- if (result === undefined) {
661
- const { fireStatusViaCommand } = await import("./commands.js");
662
- await fireStatusViaCommand(ctx);
663
- }
664
- },
665
- });
666
-
667
- // ── before_agent_start: inject GSD contract into true system prompt ─────
668
- pi.on("before_agent_start", async (event, ctx: ExtensionContext) => {
669
- if (!existsSync(gsdRoot(process.cwd()))) return;
670
-
671
- const stopContextTimer = debugTime("context-inject");
672
- const systemContent = loadPrompt("system");
673
- const loadedPreferences = loadEffectiveGSDPreferences();
674
- let preferenceBlock = "";
675
- if (loadedPreferences) {
676
- const cwd = process.cwd();
677
- const report = resolveAllSkillReferences(loadedPreferences.preferences, cwd);
678
- preferenceBlock = `\n\n${renderPreferencesForSystemPrompt(loadedPreferences.preferences, report.resolutions)}`;
679
-
680
- // Emit warnings for unresolved skill references
681
- if (report.warnings.length > 0) {
682
- ctx.ui.notify(
683
- `GSD skill preferences: ${report.warnings.length} unresolved skill${report.warnings.length === 1 ? "" : "s"}: ${report.warnings.join(", ")}`,
684
- "warning",
685
- );
686
- }
687
- }
688
-
689
- // Load project knowledge if available
690
- let knowledgeBlock = "";
691
- const knowledgePath = resolveGsdRootFile(process.cwd(), "KNOWLEDGE");
692
- if (existsSync(knowledgePath)) {
693
- try {
694
- const content = readFileSync(knowledgePath, "utf-8").trim();
695
- if (content) {
696
- knowledgeBlock = `\n\n[PROJECT KNOWLEDGE — Rules, patterns, and lessons learned]\n\n${content}`;
697
- }
698
- } catch {
699
- // File read error — skip knowledge injection
700
- }
701
- }
702
-
703
- // Inject auto-learned project memories
704
- let memoryBlock = "";
705
- try {
706
- const { getActiveMemoriesRanked, formatMemoriesForPrompt } = await import("./memory-store.js");
707
- const memories = getActiveMemoriesRanked(30);
708
- if (memories.length > 0) {
709
- const formatted = formatMemoriesForPrompt(memories, 2000);
710
- if (formatted) {
711
- memoryBlock = `\n\n${formatted}`;
712
- }
713
- }
714
- } catch { /* non-fatal */ }
715
-
716
- // Detect skills installed during this auto-mode session
717
- let newSkillsBlock = "";
718
- if (hasSkillSnapshot()) {
719
- const newSkills = detectNewSkills();
720
- if (newSkills.length > 0) {
721
- newSkillsBlock = formatSkillsXml(newSkills);
722
- }
723
- }
724
-
725
- // Load agent instructions (global + project)
726
- let agentInstructionsBlock = "";
727
- const agentInstructions = loadAgentInstructions();
728
- if (agentInstructions) {
729
- agentInstructionsBlock = `\n\n## Agent Instructions\n\nThe following instructions were provided by the user and must be followed in every session:\n\n${agentInstructions}`;
730
- }
731
-
732
- const injection = await buildGuidedExecuteContextInjection(event.prompt, process.cwd());
733
-
734
- // Worktree context — override the static CWD in the system prompt
735
- let worktreeBlock = "";
736
- const worktreeName = getActiveWorktreeName();
737
- const worktreeMainCwd = getWorktreeOriginalCwd();
738
- const autoWorktree = getActiveAutoWorktreeContext();
739
- if (worktreeName && worktreeMainCwd) {
740
- worktreeBlock = [
741
- "",
742
- "",
743
- "[WORKTREE CONTEXT — OVERRIDES CURRENT WORKING DIRECTORY ABOVE]",
744
- `IMPORTANT: Ignore the "Current working directory" shown earlier in this prompt.`,
745
- `The actual current working directory is: ${toPosixPath(process.cwd())}`,
746
- "",
747
- `You are working inside a GSD worktree.`,
748
- `- Worktree name: ${worktreeName}`,
749
- `- Worktree path (this is the real cwd): ${toPosixPath(process.cwd())}`,
750
- `- Main project: ${toPosixPath(worktreeMainCwd)}`,
751
- `- Branch: worktree/${worktreeName}`,
752
- "",
753
- "All file operations, bash commands, and GSD state resolve against the worktree path above.",
754
- "Use /worktree merge to merge changes back. Use /worktree return to switch back to the main tree.",
755
- ].join("\n");
756
- } else if (autoWorktree) {
757
- worktreeBlock = [
758
- "",
759
- "",
760
- "[WORKTREE CONTEXT — OVERRIDES CURRENT WORKING DIRECTORY ABOVE]",
761
- `IMPORTANT: Ignore the "Current working directory" shown earlier in this prompt.`,
762
- `The actual current working directory is: ${toPosixPath(process.cwd())}`,
763
- "",
764
- "You are working inside a GSD auto-worktree.",
765
- `- Milestone worktree: ${autoWorktree.worktreeName}`,
766
- `- Worktree path (this is the real cwd): ${toPosixPath(process.cwd())}`,
767
- `- Main project: ${toPosixPath(autoWorktree.originalBase)}`,
768
- `- Branch: ${autoWorktree.branch}`,
769
- "",
770
- "All file operations, bash commands, and GSD state resolve against the worktree path above.",
771
- "Write every .gsd artifact in the worktree path above, never in the main project tree.",
772
- ].join("\n");
773
- }
774
-
775
- const fullSystem = `${event.systemPrompt}\n\n[SYSTEM CONTEXT — GSD]\n\n${systemContent}${preferenceBlock}${agentInstructionsBlock}${knowledgeBlock}${memoryBlock}${newSkillsBlock}${worktreeBlock}`;
776
- stopContextTimer({
777
- systemPromptSize: fullSystem.length,
778
- injectionSize: injection?.length ?? 0,
779
- hasPreferences: preferenceBlock.length > 0,
780
- hasNewSkills: newSkillsBlock.length > 0,
781
- });
782
-
783
- return {
784
- systemPrompt: fullSystem,
785
- ...(injection
786
- ? {
787
- message: {
788
- customType: "gsd-guided-context",
789
- content: injection,
790
- display: false,
791
- },
792
- }
793
- : {}),
794
- };
795
- });
796
-
797
- // ── agent_end: auto-mode advancement or auto-start after discuss ───────────
798
- pi.on("agent_end", async (event, ctx: ExtensionContext) => {
799
- // Clean up quick-task branch if one just completed (#1269)
800
- try {
801
- const { cleanupQuickBranch } = await import("./quick.js");
802
- cleanupQuickBranch();
803
- } catch { /* non-fatal */ }
804
-
805
- // If discuss phase just finished, start auto-mode
806
- if (checkAutoStartAfterDiscuss()) {
807
- depthVerifiedMilestones.clear();
808
- activeQueuePhase = false;
809
- return;
810
- }
811
-
812
- // If auto-mode is already running, advance to next unit
813
- if (!isAutoActive()) return;
814
-
815
- // If the agent was aborted (user pressed Escape) or hit a provider
816
- // error (fetch failure, rate limit, etc.), pause auto-mode instead of
817
- // advancing. This preserves the conversation so the user can inspect
818
- // what happened, interact with the agent, or resume.
819
- const lastMsg = event.messages[event.messages.length - 1];
820
- if (lastMsg && "stopReason" in lastMsg && lastMsg.stopReason === "aborted") {
821
- await pauseAuto(ctx, pi);
822
- return;
823
- }
824
- if (lastMsg && "stopReason" in lastMsg && lastMsg.stopReason === "error") {
825
- const errorDetail =
826
- "errorMessage" in lastMsg && lastMsg.errorMessage
827
- ? `: ${lastMsg.errorMessage}`
828
- : "";
829
-
830
- const errorMsg = ("errorMessage" in lastMsg && lastMsg.errorMessage) ? String(lastMsg.errorMessage) : "";
831
-
832
- // ── Transient network error retry ──────────────────────────────────
833
- // Before falling back to a different model, retry the current model
834
- // for transient network errors (connection reset, timeout, DNS, etc.).
835
- // This prevents providers with occasional network flakiness from being
836
- // immediately abandoned in favor of fallback models (#941).
837
- if (isTransientNetworkError(errorMsg)) {
838
- const currentModelId = ctx.model?.id ?? "unknown";
839
- const retryKey = `network-retry:${currentModelId}`;
840
- const maxRetries = 2;
841
- const currentRetries = networkRetryCounters.get(retryKey) ?? 0;
842
-
843
- if (currentRetries < maxRetries) {
844
- networkRetryCounters.set(retryKey, currentRetries + 1);
845
- const attempt = currentRetries + 1;
846
- const delayMs = attempt * 3000; // 3s, 6s backoff
847
- ctx.ui.notify(
848
- `Network error on ${currentModelId}${errorDetail}. Retry ${attempt}/${maxRetries} in ${delayMs / 1000}s...`,
849
- "warning",
850
- );
851
- setTimeout(() => {
852
- pi.sendMessage(
853
- { customType: "gsd-auto-timeout-recovery", content: "Continue execution — retrying after transient network error.", display: false },
854
- { triggerTurn: true },
855
- );
856
- }, delayMs);
857
- return;
858
- }
859
- // Retries exhausted — clear counter and fall through to fallback logic
860
- networkRetryCounters.delete(retryKey);
861
- ctx.ui.notify(
862
- `Network retries exhausted for ${currentModelId}. Attempting model fallback.`,
863
- "warning",
864
- );
865
- }
866
-
867
- const dash = getAutoDashboardData();
868
- if (dash.currentUnit) {
869
- const modelConfig = resolveModelWithFallbacksForUnit(dash.currentUnit.type);
870
- if (modelConfig && modelConfig.fallbacks.length > 0) {
871
- const availableModels = ctx.modelRegistry.getAvailable();
872
- const currentModelId = ctx.model?.id;
873
-
874
- const nextModelId = getNextFallbackModel(currentModelId, modelConfig);
875
-
876
- if (nextModelId) {
877
- // Clear any network retry counters when switching models
878
- networkRetryCounters.clear();
879
-
880
- let modelToSet;
881
- const slashIdx = nextModelId.indexOf("/");
882
- if (slashIdx !== -1) {
883
- const provider = nextModelId.substring(0, slashIdx);
884
- const id = nextModelId.substring(slashIdx + 1);
885
- modelToSet = availableModels.find(
886
- m => m.provider.toLowerCase() === provider.toLowerCase()
887
- && m.id.toLowerCase() === id.toLowerCase()
888
- );
889
- } else {
890
- const currentProvider = ctx.model?.provider;
891
- const exactProviderMatch = availableModels.find(
892
- m => m.id === nextModelId && m.provider === currentProvider
893
- );
894
- modelToSet = exactProviderMatch ?? availableModels.find(m => m.id === nextModelId);
895
- }
896
-
897
- if (modelToSet) {
898
- const ok = await pi.setModel(modelToSet, { persist: false });
899
- if (ok) {
900
- ctx.ui.notify(`Model error${errorDetail}. Switched to fallback: ${nextModelId} and resuming.`, "warning");
901
- // Trigger a generic "Continue execution" to resume the task since the previous attempt failed
902
- pi.sendMessage(
903
- { customType: "gsd-auto-timeout-recovery", content: "Continue execution.", display: false },
904
- { triggerTurn: true }
905
- );
906
- return;
907
- }
908
- }
909
- }
910
- }
911
- }
912
-
913
- // ── Session model recovery (#1065) ──────────────────────────────────
914
- // Before pausing, attempt to restore the model captured at auto-mode
915
- // start. This prevents cross-session model leakage: when fallback
916
- // chains are exhausted (or absent), the session retries with the model
917
- // the user originally chose instead of reading (possibly stale) global
918
- // preferences that another concurrent session may have modified.
919
- const sessionModel = getAutoModeStartModel();
920
- if (sessionModel) {
921
- const currentModelId = ctx.model?.id;
922
- const currentProvider = ctx.model?.provider;
923
- // Only attempt recovery if the current model diverged from the session model
924
- if (currentModelId !== sessionModel.id || currentProvider !== sessionModel.provider) {
925
- const availableModels = ctx.modelRegistry.getAvailable();
926
- const startModel = availableModels.find(
927
- m => m.provider === sessionModel.provider && m.id === sessionModel.id,
928
- );
929
- if (startModel) {
930
- const ok = await pi.setModel(startModel, { persist: false });
931
- if (ok) {
932
- networkRetryCounters.clear();
933
- ctx.ui.notify(
934
- `Model error${errorDetail}. Restored session model: ${sessionModel.provider}/${sessionModel.id} and resuming.`,
935
- "warning",
936
- );
937
- pi.sendMessage(
938
- { customType: "gsd-auto-timeout-recovery", content: "Continue execution.", display: false },
939
- { triggerTurn: true },
940
- );
941
- return;
942
- }
943
- }
944
- }
945
- }
946
-
947
- // Classify the error: transient (auto-resume) vs permanent (manual resume)
948
- const classification = classifyProviderError(errorMsg);
949
-
950
- // Extract explicit retry-after from the message or response metadata
951
- const explicitRetryAfterMs = ("retryAfterMs" in lastMsg && typeof lastMsg.retryAfterMs === "number")
952
- ? lastMsg.retryAfterMs
953
- : undefined;
954
- let retryAfterMs = explicitRetryAfterMs ?? classification.suggestedDelayMs;
955
-
956
- // ── Escalating backoff for repeated transient errors ──────────────
957
- // Each consecutive transient auto-resume doubles the delay. After
958
- // MAX_TRANSIENT_AUTO_RESUMES consecutive failures, treat as permanent
959
- // to avoid infinite rapid-fire retries (#1166).
960
- let effectiveTransient = classification.isTransient;
961
- if (classification.isTransient) {
962
- consecutiveTransientErrors++;
963
- if (consecutiveTransientErrors > MAX_TRANSIENT_AUTO_RESUMES) {
964
- effectiveTransient = false;
965
- ctx.ui.notify(
966
- `${consecutiveTransientErrors} consecutive transient errors. Pausing indefinitely — resume manually with /gsd auto.`,
967
- "error",
968
- );
969
- consecutiveTransientErrors = 0;
970
- } else {
971
- // Escalate: base delay × 2^(consecutive-1) → 30s, 60s, 120s, 240s, 480s
972
- retryAfterMs = retryAfterMs * 2 ** (consecutiveTransientErrors - 1);
973
- }
974
- }
975
-
976
- await pauseAutoForProviderError(ctx.ui, errorDetail, () => pauseAuto(ctx, pi), {
977
- isRateLimit: classification.isRateLimit,
978
- isTransient: effectiveTransient,
979
- retryAfterMs,
980
- resume: () => {
981
- pi.sendMessage(
982
- { customType: "gsd-auto-timeout-recovery", content: "Continue execution \u2014 provider error recovery delay elapsed.", display: false },
983
- { triggerTurn: true },
984
- );
985
- },
986
- });
987
- return;
988
- }
989
-
990
- try {
991
- networkRetryCounters.clear(); // Clear network retry state on successful unit completion
992
- consecutiveTransientErrors = 0; // Reset escalating backoff on success
993
- await handleAgentEnd(ctx, pi);
994
- } catch (err) {
995
- // Safety net: if handleAgentEnd throws despite its internal try-catch,
996
- // ensure auto-mode stops gracefully instead of silently stalling (#381).
997
- const message = getErrorMessage(err);
998
- ctx.ui.notify(
999
- `Auto-mode error in agent_end handler: ${message}. Stopping auto-mode.`,
1000
- "error",
1001
- );
1002
- try {
1003
- await pauseAuto(ctx, pi);
1004
- } catch {
1005
- // Last resort — at least log
1006
- }
1007
- }
1008
- });
1009
-
1010
- // ── session_before_compact ────────────────────────────────────────────────
1011
- pi.on("session_before_compact", async (_event, _ctx: ExtensionContext) => {
1012
- // Block compaction during auto-mode — each unit is a fresh session
1013
- // Also block during paused state — context is valuable for the user
1014
- if (isAutoActive() || isAutoPaused()) {
1015
- return { cancel: true };
1016
- }
1017
-
1018
- const basePath = process.cwd();
1019
- const state = await deriveState(basePath);
1020
-
1021
- // Only save continue.md if we're actively executing a task
1022
- if (!state.activeMilestone || !state.activeSlice || !state.activeTask) return;
1023
- if (state.phase !== "executing") return;
1024
-
1025
- const sDir = resolveSlicePath(basePath, state.activeMilestone.id, state.activeSlice.id);
1026
- if (!sDir) return;
1027
-
1028
- // Check for existing continue file (new naming or legacy)
1029
- const existingFile = resolveSliceFile(basePath, state.activeMilestone.id, state.activeSlice.id, "CONTINUE");
1030
- if (existingFile && await loadFile(existingFile)) return;
1031
- const legacyContinue = join(sDir, "continue.md");
1032
- if (await loadFile(legacyContinue)) return;
1033
-
1034
- const continuePath = join(sDir, buildSliceFileName(state.activeSlice.id, "CONTINUE"));
1035
-
1036
- const continueData = {
1037
- frontmatter: {
1038
- milestone: state.activeMilestone.id,
1039
- slice: state.activeSlice.id,
1040
- task: state.activeTask.id,
1041
- step: 0,
1042
- totalSteps: 0,
1043
- status: "compacted" as const,
1044
- savedAt: new Date().toISOString(),
1045
- },
1046
- completedWork: `Task ${state.activeTask.id} (${state.activeTask.title}) was in progress when compaction occurred.`,
1047
- remainingWork: "Check the task plan for remaining steps.",
1048
- decisions: "Check task summary files for prior decisions.",
1049
- context: "Session was auto-compacted by Pi. Resume with /gsd.",
1050
- nextAction: `Resume task ${state.activeTask.id}: ${state.activeTask.title}.`,
1051
- };
1052
-
1053
- await saveFile(continuePath, formatContinue(continueData));
1054
- });
1055
-
1056
- // ── session_shutdown: save activity log on Ctrl+C / SIGTERM ─────────────
1057
- pi.on("session_shutdown", async (_event, ctx: ExtensionContext) => {
1058
- if (isParallelActive()) {
1059
- try {
1060
- await shutdownParallel(process.cwd());
1061
- } catch { /* best-effort */ }
1062
- }
1063
-
1064
- // Auto-commit dirty work in CLI-spawned worktrees so nothing is lost.
1065
- // The CLI sets GSD_CLI_WORKTREE when launched with -w.
1066
- const cliWorktree = process.env.GSD_CLI_WORKTREE;
1067
- if (cliWorktree) {
1068
- try {
1069
- const { autoCommitCurrentBranch } = await import("./worktree.js");
1070
- const msg = autoCommitCurrentBranch(process.cwd(), "session-end", cliWorktree);
1071
- if (msg) {
1072
- ctx.ui.notify(`Auto-committed worktree ${cliWorktree} before exit.`, "info");
1073
- }
1074
- } catch { /* best-effort */ }
1075
- }
1076
-
1077
- if (!isAutoActive() && !isAutoPaused()) return;
1078
-
1079
- // Save the current session — the lock file stays on disk
1080
- // so the next /gsd auto knows it was interrupted
1081
- const dash = getAutoDashboardData();
1082
- if (dash.currentUnit) {
1083
- saveActivityLog(ctx, dash.basePath, dash.currentUnit.type, dash.currentUnit.id);
1084
- }
1085
- });
1086
-
1087
- // ── tool_call: block CONTEXT.md writes without depth verification ──
1088
- // Active during both discussion flows (pendingAutoStart set) and
1089
- // queue flows (activeQueuePhase set). For multi-milestone queue flows,
1090
- // each milestone must pass its own depth verification before its
1091
- // CONTEXT.md can be written.
1092
- pi.on("tool_call", async (event) => {
1093
- if (!isToolCallEventType("write", event)) return;
1094
- const result = shouldBlockContextWrite(
1095
- event.toolName,
1096
- event.input.path,
1097
- getDiscussionMilestoneId(),
1098
- isDepthVerified(),
1099
- activeQueuePhase,
1100
- );
1101
- if (result.block) return result;
1102
- });
1103
-
1104
- // ── tool_result: persist discussion exchanges & detect depth gate ──────
1105
- // Handles both discussion flows and queue flows. For queue flows,
1106
- // depth verification question IDs may include milestone IDs
1107
- // (e.g., "depth_verification_M001") for per-milestone gating.
1108
- pi.on("tool_result", async (event) => {
1109
- if (event.toolName !== "ask_user_questions") return;
1110
-
1111
- const milestoneId = getDiscussionMilestoneId();
1112
- // Queue flows don't set pendingAutoStart, so milestoneId may be null.
1113
- // Depth gate detection still applies — it sets per-milestone flags.
1114
- const inQueue = activeQueuePhase;
1115
-
1116
- const details = event.details as any;
1117
- if (details?.cancelled || !details?.response) return;
1118
-
1119
- // ── Depth gate detection ──────────────────────────────────────────
1120
- // Supports two patterns:
1121
- // 1. "depth_verification" — wildcard, marks all milestones verified
1122
- // 2. "depth_verification_M001" — per-milestone verification
1123
- const questions: any[] = (event.input as any)?.questions ?? [];
1124
- for (const q of questions) {
1125
- if (typeof q.id === "string" && q.id.includes("depth_verification")) {
1126
- // Extract milestone ID from question ID if present
1127
- const midMatch = q.id.match(/depth_verification[_-](M\d+(?:-[a-z0-9]{6})?)/i);
1128
- if (midMatch) {
1129
- depthVerifiedMilestones.add(midMatch[1]);
1130
- } else {
1131
- // Wildcard — all milestones verified (backward compat for single-milestone)
1132
- depthVerifiedMilestones.add("*");
1133
- }
1134
- break;
1135
- }
1136
- }
1137
-
1138
- // Discussion persistence only applies when in a discussion flow with a known milestone
1139
- if (!milestoneId) return;
1140
-
1141
- // ── Persist exchange to DISCUSSION.md ──────────────────────────────
1142
- const basePath = process.cwd();
1143
- const milestoneDir = resolveMilestonePath(basePath, milestoneId);
1144
- if (!milestoneDir) return;
1145
-
1146
- const fileName = buildMilestoneFileName(milestoneId, "DISCUSSION");
1147
- const discussionPath = join(milestoneDir, fileName);
1148
- const timestamp = new Date().toISOString();
1149
-
1150
- // Format exchange as markdown
1151
- const lines: string[] = [`## Exchange — ${timestamp}`, ""];
1152
-
1153
- for (const q of questions) {
1154
- lines.push(`### ${q.header ?? "Question"}`);
1155
- lines.push("");
1156
- lines.push(q.question ?? "");
1157
- if (Array.isArray(q.options)) {
1158
- lines.push("");
1159
- for (const opt of q.options) {
1160
- lines.push(`- **${opt.label}** — ${opt.description ?? ""}`);
1161
- }
1162
- }
1163
-
1164
- // Append user response for this question
1165
- const answer = details.response?.answers?.[q.id];
1166
- if (answer) {
1167
- lines.push("");
1168
- const selected = Array.isArray(answer.selected) ? answer.selected.join(", ") : answer.selected;
1169
- lines.push(`**Selected:** ${selected}`);
1170
- if (answer.notes) {
1171
- lines.push(`**Notes:** ${answer.notes}`);
1172
- }
1173
- }
1174
- lines.push("");
1175
- }
1176
-
1177
- lines.push("---", "");
1178
-
1179
- const newBlock = lines.join("\n");
1180
- const existing = await loadFile(discussionPath) ?? `# ${milestoneId} Discussion Log\n\n`;
1181
- await saveFile(discussionPath, existing + newBlock);
1182
- });
1183
-
1184
- // ── tool_execution_start/end: track in-flight tools for idle detection ──
1185
- pi.on("tool_execution_start", async (event) => {
1186
- if (!isAutoActive()) return;
1187
- markToolStart(event.toolCallId);
1188
- });
1189
-
1190
- pi.on("tool_execution_end", async (event) => {
1191
- markToolEnd(event.toolCallId);
1192
- });
1193
- }
1194
-
1195
- async function buildGuidedExecuteContextInjection(prompt: string, basePath: string): Promise<string | null> {
1196
- const executeMatch = prompt.match(/Execute the next task:\s+(T\d+)\s+\("([^"]+)"\)\s+in slice\s+(S\d+)\s+of milestone\s+(M\d+(?:-[a-z0-9]{6})?)/i);
1197
- if (executeMatch) {
1198
- const [, taskId, taskTitle, sliceId, milestoneId] = executeMatch;
1199
- return buildTaskExecutionContextInjection(basePath, milestoneId, sliceId, taskId, taskTitle);
1200
- }
1201
-
1202
- const resumeMatch = prompt.match(/Resume interrupted work\.[\s\S]*?slice\s+(S\d+)\s+of milestone\s+(M\d+(?:-[a-z0-9]{6})?)/i);
1203
- if (resumeMatch) {
1204
- const [, sliceId, milestoneId] = resumeMatch;
1205
- const state = await deriveState(basePath);
1206
- if (
1207
- state.activeMilestone?.id === milestoneId &&
1208
- state.activeSlice?.id === sliceId &&
1209
- state.activeTask
1210
- ) {
1211
- return buildTaskExecutionContextInjection(
1212
- basePath,
1213
- milestoneId,
1214
- sliceId,
1215
- state.activeTask.id,
1216
- state.activeTask.title,
1217
- );
1218
- }
1219
- }
1220
-
1221
- return null;
1222
- }
1223
-
1224
- async function buildTaskExecutionContextInjection(
1225
- basePath: string,
1226
- milestoneId: string,
1227
- sliceId: string,
1228
- taskId: string,
1229
- taskTitle: string,
1230
- ): Promise<string> {
1231
- const taskPlanPath = resolveTaskFile(basePath, milestoneId, sliceId, taskId, "PLAN");
1232
- const taskPlanRelPath = relTaskFile(basePath, milestoneId, sliceId, taskId, "PLAN");
1233
- const taskPlanContent = taskPlanPath ? await loadFile(taskPlanPath) : null;
1234
- const taskPlanInline = taskPlanContent
1235
- ? [
1236
- "## Inlined Task Plan (authoritative local execution contract)",
1237
- `Source: \`${taskPlanRelPath}\``,
1238
- "",
1239
- taskPlanContent.trim(),
1240
- ].join("\n")
1241
- : [
1242
- "## Inlined Task Plan (authoritative local execution contract)",
1243
- `Task plan not found at dispatch time. Read \`${taskPlanRelPath}\` before executing.`,
1244
- ].join("\n");
1245
-
1246
- const slicePlanPath = resolveSliceFile(basePath, milestoneId, sliceId, "PLAN");
1247
- const slicePlanRelPath = relSliceFile(basePath, milestoneId, sliceId, "PLAN");
1248
- const slicePlanContent = slicePlanPath ? await loadFile(slicePlanPath) : null;
1249
- const slicePlanExcerpt = extractSliceExecutionExcerpt(slicePlanContent, slicePlanRelPath);
1250
-
1251
- const priorTaskLines = await buildCarryForwardLines(basePath, milestoneId, sliceId, taskId);
1252
- const resumeSection = await buildResumeSection(basePath, milestoneId, sliceId);
1253
-
1254
- const activeOverrides = await loadActiveOverrides(basePath);
1255
- const overridesSection = formatOverridesSection(activeOverrides);
1256
-
1257
- return [
1258
- "[GSD Guided Execute Context]",
1259
- "Use this injected context as startup context for guided task execution. Treat the inlined task plan as the authoritative local execution contract. Use source artifacts to verify details and run checks.",
1260
- overridesSection, "",
1261
- "",
1262
- resumeSection,
1263
- "",
1264
- "## Carry-Forward Context",
1265
- ...priorTaskLines,
1266
- "",
1267
- taskPlanInline,
1268
- "",
1269
- slicePlanExcerpt,
1270
- "",
1271
- "## Backing Source Artifacts",
1272
- `- Slice plan: \`${slicePlanRelPath}\``,
1273
- `- Task plan source: \`${taskPlanRelPath}\``,
1274
- ].join("\n");
1275
- }
1276
-
1277
- async function buildCarryForwardLines(
1278
- basePath: string,
1279
- milestoneId: string,
1280
- sliceId: string,
1281
- taskId: string,
1282
- ): Promise<string[]> {
1283
- const tDir = resolveTasksDir(basePath, milestoneId, sliceId);
1284
- if (!tDir) return ["- No prior task summaries in this slice."];
1285
-
1286
- const currentNum = parseInt(taskId.replace(/^T/, ""), 10);
1287
- const sRel = relSlicePath(basePath, milestoneId, sliceId);
1288
- const summaryFiles = resolveTaskFiles(tDir, "SUMMARY")
1289
- .filter((file) => parseInt(file.replace(/^T/, ""), 10) < currentNum)
1290
- .sort();
1291
-
1292
- if (summaryFiles.length === 0) return ["- No prior task summaries in this slice."];
1293
-
1294
- const lines = await Promise.all(summaryFiles.map(async (file) => {
1295
- const absPath = join(tDir, file);
1296
- const content = await loadFile(absPath);
1297
- const relPath = `${sRel}/tasks/${file}`;
1298
- if (!content) return `- \`${relPath}\``;
1299
-
1300
- const summary = parseSummary(content);
1301
- const provided = summary.frontmatter.provides.slice(0, 2).join("; ");
1302
- const decisions = summary.frontmatter.key_decisions.slice(0, 2).join("; ");
1303
- const patterns = summary.frontmatter.patterns_established.slice(0, 2).join("; ");
1304
- const diagnostics = extractMarkdownSection(content, "Diagnostics");
1305
-
1306
- const parts = [summary.title || relPath];
1307
- if (summary.oneLiner) parts.push(summary.oneLiner);
1308
- if (provided) parts.push(`provides: ${provided}`);
1309
- if (decisions) parts.push(`decisions: ${decisions}`);
1310
- if (patterns) parts.push(`patterns: ${patterns}`);
1311
- if (diagnostics) parts.push(`diagnostics: ${oneLine(diagnostics)}`);
1312
-
1313
- return `- \`${relPath}\` — ${parts.join(" | ")}`;
1314
- }));
1315
-
1316
- return lines;
1317
- }
1318
-
1319
- async function buildResumeSection(basePath: string, milestoneId: string, sliceId: string): Promise<string> {
1320
- const continueFile = resolveSliceFile(basePath, milestoneId, sliceId, "CONTINUE");
1321
- const legacyDir = resolveSlicePath(basePath, milestoneId, sliceId);
1322
- const legacyPath = legacyDir ? join(legacyDir, "continue.md") : null;
1323
- const continueContent = continueFile ? await loadFile(continueFile) : null;
1324
- const legacyContent = !continueContent && legacyPath ? await loadFile(legacyPath) : null;
1325
- const resolvedContent = continueContent ?? legacyContent;
1326
- const resolvedRelPath = continueContent
1327
- ? relSliceFile(basePath, milestoneId, sliceId, "CONTINUE")
1328
- : (legacyPath ? `${relSlicePath(basePath, milestoneId, sliceId)}/continue.md` : null);
1329
-
1330
- if (!resolvedContent || !resolvedRelPath) {
1331
- return ["## Resume State", "- No continue file present. Start from the top of the task plan."].join("\n");
1332
- }
1333
-
1334
- const cont = parseContinue(resolvedContent);
1335
- const lines = [
1336
- "## Resume State",
1337
- `Source: \`${resolvedRelPath}\``,
1338
- `- Status: ${cont.frontmatter.status || "in_progress"}`,
1339
- ];
1340
-
1341
- if (cont.frontmatter.step && cont.frontmatter.totalSteps) {
1342
- lines.push(`- Progress: step ${cont.frontmatter.step} of ${cont.frontmatter.totalSteps}`);
1343
- }
1344
- if (cont.completedWork) lines.push(`- Completed: ${oneLine(cont.completedWork)}`);
1345
- if (cont.remainingWork) lines.push(`- Remaining: ${oneLine(cont.remainingWork)}`);
1346
- if (cont.decisions) lines.push(`- Decisions: ${oneLine(cont.decisions)}`);
1347
- if (cont.nextAction) lines.push(`- Next action: ${oneLine(cont.nextAction)}`);
1348
-
1349
- return lines.join("\n");
1350
- }
1351
-
1352
- function extractSliceExecutionExcerpt(content: string | null, relPath: string): string {
1353
- if (!content) {
1354
- return [
1355
- "## Slice Plan Excerpt",
1356
- `Slice plan not found at dispatch time. Read \`${relPath}\` before running slice-level verification.`,
1357
- ].join("\n");
1358
- }
1359
-
1360
- const lines = content.split("\n");
1361
- const goalLine = lines.find((line) => line.startsWith("**Goal:**"))?.trim();
1362
- const demoLine = lines.find((line) => line.startsWith("**Demo:**"))?.trim();
1363
- const verification = extractMarkdownSection(content, "Verification");
1364
- const observability = extractMarkdownSection(content, "Observability / Diagnostics");
1365
-
1366
- const parts = ["## Slice Plan Excerpt", `Source: \`${relPath}\``];
1367
- if (goalLine) parts.push(goalLine);
1368
- if (demoLine) parts.push(demoLine);
1369
- if (verification) parts.push("", "### Slice Verification", verification.trim());
1370
- if (observability) parts.push("", "### Slice Observability / Diagnostics", observability.trim());
1371
- return parts.join("\n");
1372
- }
1373
-
1374
- function extractMarkdownSection(content: string, heading: string): string | null {
1375
- const match = new RegExp(`^## ${escapeRegExp(heading)}\\s*$`, "m").exec(content);
1376
- if (!match) return null;
1377
- const start = match.index + match[0].length;
1378
- const rest = content.slice(start);
1379
- const nextHeading = rest.match(/^##\s+/m);
1380
- const end = nextHeading?.index ?? rest.length;
1381
- return rest.slice(0, end).trim();
1382
- }
1383
-
1384
- function escapeRegExp(value: string): string {
1385
- return value.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
1386
- }
1387
-
1388
- function oneLine(text: string): string {
1389
- return text.replace(/\s+/g, " ").trim();
1390
- }