gsd-pi 2.33.0-dev.bafba33 → 2.33.1-dev.235d83a

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