gsd-pi 2.65.0 → 2.66.0-dev.6c91c1f

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 (458) hide show
  1. package/dist/mcp-server.js +6 -2
  2. package/dist/resources/extensions/browser-tools/capture.js +20 -1
  3. package/dist/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +93 -0
  4. package/dist/resources/extensions/gsd/auto/finalize-timeout.js +2 -0
  5. package/dist/resources/extensions/gsd/auto/loop.js +2 -2
  6. package/dist/resources/extensions/gsd/auto/phases.js +48 -5
  7. package/dist/resources/extensions/gsd/auto/run-unit.js +13 -2
  8. package/dist/resources/extensions/gsd/auto/session.js +4 -0
  9. package/dist/resources/extensions/gsd/auto/types.js +2 -0
  10. package/dist/resources/extensions/gsd/auto-dashboard.js +2 -1
  11. package/dist/resources/extensions/gsd/auto-dispatch.js +99 -9
  12. package/dist/resources/extensions/gsd/auto-model-selection.js +7 -5
  13. package/dist/resources/extensions/gsd/auto-post-unit.js +17 -6
  14. package/dist/resources/extensions/gsd/auto-prompts.js +24 -0
  15. package/dist/resources/extensions/gsd/auto-recovery.js +40 -22
  16. package/dist/resources/extensions/gsd/auto-start.js +175 -12
  17. package/dist/resources/extensions/gsd/auto-tool-tracking.js +10 -0
  18. package/dist/resources/extensions/gsd/auto-worktree.js +29 -7
  19. package/dist/resources/extensions/gsd/auto.js +21 -15
  20. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +17 -4
  21. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +10 -0
  22. package/dist/resources/extensions/gsd/bootstrap/query-tools.js +6 -4
  23. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +5 -1
  24. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +11 -3
  25. package/dist/resources/extensions/gsd/bootstrap/system-context.js +3 -1
  26. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +31 -1
  27. package/dist/resources/extensions/gsd/commands/context.js +8 -1
  28. package/dist/resources/extensions/gsd/commands/handlers/core.js +23 -2
  29. package/dist/resources/extensions/gsd/commands-extensions.js +1 -1
  30. package/dist/resources/extensions/gsd/config-overlay.js +312 -0
  31. package/dist/resources/extensions/gsd/db-writer.js +13 -3
  32. package/dist/resources/extensions/gsd/detection.js +1 -1
  33. package/dist/resources/extensions/gsd/dispatch-guard.js +2 -1
  34. package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -0
  35. package/dist/resources/extensions/gsd/doctor.js +2 -1
  36. package/dist/resources/extensions/gsd/files.js +17 -0
  37. package/dist/resources/extensions/gsd/gitignore.js +1 -0
  38. package/dist/resources/extensions/gsd/gsd-db.js +47 -4
  39. package/dist/resources/extensions/gsd/guided-flow.js +220 -29
  40. package/dist/resources/extensions/gsd/index.js +1 -1
  41. package/dist/resources/extensions/gsd/json-persistence.js +5 -2
  42. package/dist/resources/extensions/gsd/md-importer.js +14 -7
  43. package/dist/resources/extensions/gsd/notification-overlay.js +1 -1
  44. package/dist/resources/extensions/gsd/notification-widget.js +2 -1
  45. package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +1 -1
  46. package/dist/resources/extensions/gsd/parallel-orchestrator.js +17 -11
  47. package/dist/resources/extensions/gsd/pre-execution-checks.js +26 -5
  48. package/dist/resources/extensions/gsd/preferences-types.js +3 -0
  49. package/dist/resources/extensions/gsd/preferences-validation.js +45 -1
  50. package/dist/resources/extensions/gsd/preferences.js +9 -2
  51. package/dist/resources/extensions/gsd/preparation.js +1092 -0
  52. package/dist/resources/extensions/gsd/prompt-validation.js +67 -0
  53. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +3 -3
  54. package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  55. package/dist/resources/extensions/gsd/prompts/discuss-prepared.md +424 -0
  56. package/dist/resources/extensions/gsd/prompts/discuss.md +2 -0
  57. package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +6 -1
  58. package/dist/resources/extensions/gsd/prompts/guided-discuss-slice.md +5 -4
  59. package/dist/resources/extensions/gsd/prompts/parallel-research-slices.md +23 -0
  60. package/dist/resources/extensions/gsd/prompts/queue.md +2 -0
  61. package/dist/resources/extensions/gsd/prompts/rethink.md +2 -1
  62. package/dist/resources/extensions/gsd/prompts/system.md +2 -2
  63. package/dist/resources/extensions/gsd/prompts/validate-milestone.md +56 -23
  64. package/dist/resources/extensions/gsd/quick.js +19 -15
  65. package/dist/resources/extensions/gsd/reactive-graph.js +12 -0
  66. package/dist/resources/extensions/gsd/roadmap-slices.js +24 -5
  67. package/dist/resources/extensions/gsd/safety/content-validator.js +3 -3
  68. package/dist/resources/extensions/gsd/session-lock.js +23 -1
  69. package/dist/resources/extensions/gsd/state.js +115 -28
  70. package/dist/resources/extensions/gsd/templates/context-enhanced.md +138 -0
  71. package/dist/resources/extensions/gsd/tools/complete-milestone.js +15 -3
  72. package/dist/resources/extensions/gsd/tools/complete-slice.js +27 -6
  73. package/dist/resources/extensions/gsd/tools/complete-task.js +31 -7
  74. package/dist/resources/extensions/gsd/tools/plan-milestone.js +7 -5
  75. package/dist/resources/extensions/gsd/tools/reassess-roadmap.js +5 -2
  76. package/dist/resources/extensions/gsd/tools/reopen-milestone.js +119 -0
  77. package/dist/resources/extensions/gsd/tools/reopen-slice.js +30 -0
  78. package/dist/resources/extensions/gsd/tools/reopen-task.js +18 -0
  79. package/dist/resources/extensions/gsd/triage-resolution.js +33 -16
  80. package/dist/resources/extensions/gsd/undo.js +3 -2
  81. package/dist/resources/extensions/gsd/workflow-events.js +1 -0
  82. package/dist/resources/extensions/gsd/workflow-logger.js +1 -1
  83. package/dist/resources/extensions/gsd/workflow-projections.js +7 -9
  84. package/dist/resources/extensions/gsd/workflow-reconcile.js +100 -9
  85. package/dist/resources/extensions/gsd/workflow-templates.js +11 -2
  86. package/dist/resources/extensions/gsd/worktree-manager.js +5 -2
  87. package/dist/resources/extensions/gsd/worktree.js +9 -0
  88. package/dist/resources/extensions/shared/interview-ui.js +1 -1
  89. package/dist/resources/extensions/subagent/agents.js +19 -5
  90. package/dist/web/standalone/.next/BUILD_ID +1 -1
  91. package/dist/web/standalone/.next/app-path-routes-manifest.json +11 -11
  92. package/dist/web/standalone/.next/build-manifest.json +4 -4
  93. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  94. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  95. package/dist/web/standalone/.next/required-server-files.json +3 -3
  96. package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
  97. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  98. package/dist/web/standalone/.next/server/app/_global-error.html +2 -2
  99. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  100. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  102. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  103. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  104. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  105. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  106. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  107. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  108. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  109. package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
  110. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
  111. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  112. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +3 -3
  113. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  114. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  115. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  116. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  117. package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
  118. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  119. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
  120. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  121. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
  122. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  123. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
  124. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  125. package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
  126. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  127. package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
  128. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  129. package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
  130. package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
  131. package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
  132. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  133. package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
  134. package/dist/web/standalone/.next/server/app/api/experimental/route.js +2 -2
  135. package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -1
  136. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  137. package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
  138. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  139. package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
  140. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  141. package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
  142. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  143. package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
  144. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  145. package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
  146. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  147. package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
  148. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  149. package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
  150. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  151. package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
  152. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  153. package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
  154. package/dist/web/standalone/.next/server/app/api/notifications/route.js +2 -2
  155. package/dist/web/standalone/.next/server/app/api/notifications/route_client-reference-manifest.js +1 -1
  156. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  157. package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
  158. package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
  159. package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
  160. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  161. package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  162. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  163. package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
  164. package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +2 -2
  165. package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
  166. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  167. package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
  168. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  169. package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
  170. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  171. package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
  172. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  173. package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
  174. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  175. package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
  176. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  177. package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
  178. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  179. package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
  180. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  181. package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
  182. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  183. package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
  184. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +2 -2
  185. package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
  186. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
  187. package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
  188. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
  189. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
  190. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +4 -4
  191. package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
  192. package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js +1 -1
  193. package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
  194. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  195. package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
  196. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  197. package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  198. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  199. package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
  200. package/dist/web/standalone/.next/server/app/index.html +1 -1
  201. package/dist/web/standalone/.next/server/app/index.rsc +4 -4
  202. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  203. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
  204. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  205. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +3 -3
  206. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  207. package/dist/web/standalone/.next/server/app/page.js +2 -2
  208. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  209. package/dist/web/standalone/.next/server/app-paths-manifest.json +11 -11
  210. package/dist/web/standalone/.next/server/chunks/6897.js +1 -1
  211. package/dist/web/standalone/.next/server/chunks/7471.js +3 -3
  212. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  213. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  214. package/dist/web/standalone/.next/server/middleware.js +2 -2
  215. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  216. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  217. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  218. package/dist/web/standalone/.next/server/pages/500.html +2 -2
  219. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  220. package/dist/web/standalone/.next/static/chunks/6502.8874bcae249c02e1.js +9 -0
  221. package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-2f24283c162b6ab3.js → page-f2a7482d42a5614b.js} +1 -1
  222. package/dist/web/standalone/.next/static/chunks/app/{layout-9ecfd95f343793f0.js → layout-a16c7a7ecdf0c2cf.js} +1 -1
  223. package/dist/web/standalone/.next/static/chunks/app/page-0c485498795110d6.js +1 -0
  224. package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +1 -0
  225. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +1 -0
  226. package/dist/web/standalone/.next/static/chunks/{webpack-a1c1e452c6b32d04.js → webpack-9fed74684e1c5bb1.js} +1 -1
  227. package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
  228. package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
  229. package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
  230. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
  231. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
  232. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
  233. package/dist/web/standalone/server.js +1 -1
  234. package/package.json +1 -1
  235. package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
  236. package/packages/pi-coding-agent/dist/core/retry-handler.js +30 -19
  237. package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
  238. package/packages/pi-coding-agent/dist/core/retry-handler.test.js +51 -0
  239. package/packages/pi-coding-agent/dist/core/retry-handler.test.js.map +1 -1
  240. package/packages/pi-coding-agent/dist/core/sdk.js +9 -9
  241. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  242. package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.d.ts +2 -1
  243. package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.d.ts.map +1 -1
  244. package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.js +10 -1
  245. package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.js.map +1 -1
  246. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +1 -0
  247. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  248. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +20 -5
  249. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  250. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +15 -1
  251. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  252. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +18 -0
  253. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  254. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  255. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +4 -0
  256. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  257. package/packages/pi-coding-agent/package.json +1 -1
  258. package/packages/pi-coding-agent/src/core/retry-handler.test.ts +80 -0
  259. package/packages/pi-coding-agent/src/core/retry-handler.ts +37 -25
  260. package/packages/pi-coding-agent/src/core/sdk.ts +9 -9
  261. package/packages/pi-coding-agent/src/modes/interactive/components/provider-manager.ts +10 -0
  262. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +20 -4
  263. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +27 -0
  264. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +16 -1
  265. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +5 -0
  266. package/packages/pi-tui/dist/components/image.d.ts +2 -0
  267. package/packages/pi-tui/dist/components/image.d.ts.map +1 -1
  268. package/packages/pi-tui/dist/components/image.js +4 -0
  269. package/packages/pi-tui/dist/components/image.js.map +1 -1
  270. package/packages/pi-tui/dist/components/image.test.d.ts +6 -0
  271. package/packages/pi-tui/dist/components/image.test.d.ts.map +1 -0
  272. package/packages/pi-tui/dist/components/image.test.js +32 -0
  273. package/packages/pi-tui/dist/components/image.test.js.map +1 -0
  274. package/packages/pi-tui/src/components/image.test.ts +36 -0
  275. package/packages/pi-tui/src/components/image.ts +5 -0
  276. package/pkg/package.json +1 -1
  277. package/src/resources/extensions/browser-tools/capture.ts +19 -1
  278. package/src/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +93 -0
  279. package/src/resources/extensions/gsd/auto/finalize-timeout.ts +3 -0
  280. package/src/resources/extensions/gsd/auto/loop.ts +2 -2
  281. package/src/resources/extensions/gsd/auto/phases.ts +68 -3
  282. package/src/resources/extensions/gsd/auto/run-unit.ts +12 -2
  283. package/src/resources/extensions/gsd/auto/session.ts +4 -0
  284. package/src/resources/extensions/gsd/auto/types.ts +5 -0
  285. package/src/resources/extensions/gsd/auto-dashboard.ts +2 -1
  286. package/src/resources/extensions/gsd/auto-dispatch.ts +110 -9
  287. package/src/resources/extensions/gsd/auto-model-selection.ts +7 -5
  288. package/src/resources/extensions/gsd/auto-post-unit.ts +16 -6
  289. package/src/resources/extensions/gsd/auto-prompts.ts +31 -0
  290. package/src/resources/extensions/gsd/auto-recovery.ts +29 -23
  291. package/src/resources/extensions/gsd/auto-start.ts +188 -10
  292. package/src/resources/extensions/gsd/auto-tool-tracking.ts +10 -0
  293. package/src/resources/extensions/gsd/auto-worktree.ts +28 -7
  294. package/src/resources/extensions/gsd/auto.ts +19 -8
  295. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +16 -4
  296. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +10 -0
  297. package/src/resources/extensions/gsd/bootstrap/query-tools.ts +5 -4
  298. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +4 -1
  299. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +11 -3
  300. package/src/resources/extensions/gsd/bootstrap/system-context.ts +3 -1
  301. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +36 -1
  302. package/src/resources/extensions/gsd/commands/context.ts +7 -1
  303. package/src/resources/extensions/gsd/commands/handlers/core.ts +26 -2
  304. package/src/resources/extensions/gsd/commands-extensions.ts +1 -1
  305. package/src/resources/extensions/gsd/config-overlay.ts +331 -0
  306. package/src/resources/extensions/gsd/db-writer.ts +11 -3
  307. package/src/resources/extensions/gsd/detection.ts +1 -1
  308. package/src/resources/extensions/gsd/dispatch-guard.ts +2 -1
  309. package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -0
  310. package/src/resources/extensions/gsd/doctor.ts +2 -1
  311. package/src/resources/extensions/gsd/files.ts +19 -0
  312. package/src/resources/extensions/gsd/gitignore.ts +1 -0
  313. package/src/resources/extensions/gsd/gsd-db.ts +46 -4
  314. package/src/resources/extensions/gsd/guided-flow.ts +254 -30
  315. package/src/resources/extensions/gsd/index.ts +1 -0
  316. package/src/resources/extensions/gsd/json-persistence.ts +6 -3
  317. package/src/resources/extensions/gsd/md-importer.ts +13 -6
  318. package/src/resources/extensions/gsd/notification-overlay.ts +1 -1
  319. package/src/resources/extensions/gsd/notification-widget.ts +2 -1
  320. package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +1 -1
  321. package/src/resources/extensions/gsd/parallel-orchestrator.ts +19 -11
  322. package/src/resources/extensions/gsd/pre-execution-checks.ts +32 -7
  323. package/src/resources/extensions/gsd/preferences-types.ts +25 -0
  324. package/src/resources/extensions/gsd/preferences-validation.ts +45 -1
  325. package/src/resources/extensions/gsd/preferences.ts +9 -2
  326. package/src/resources/extensions/gsd/preparation.ts +1419 -0
  327. package/src/resources/extensions/gsd/prompt-validation.ts +88 -0
  328. package/src/resources/extensions/gsd/prompts/complete-milestone.md +3 -3
  329. package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  330. package/src/resources/extensions/gsd/prompts/discuss-prepared.md +424 -0
  331. package/src/resources/extensions/gsd/prompts/discuss.md +2 -0
  332. package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +6 -1
  333. package/src/resources/extensions/gsd/prompts/guided-discuss-slice.md +5 -4
  334. package/src/resources/extensions/gsd/prompts/parallel-research-slices.md +23 -0
  335. package/src/resources/extensions/gsd/prompts/queue.md +2 -0
  336. package/src/resources/extensions/gsd/prompts/rethink.md +2 -1
  337. package/src/resources/extensions/gsd/prompts/system.md +2 -2
  338. package/src/resources/extensions/gsd/prompts/validate-milestone.md +56 -23
  339. package/src/resources/extensions/gsd/quick.ts +20 -15
  340. package/src/resources/extensions/gsd/reactive-graph.ts +18 -0
  341. package/src/resources/extensions/gsd/roadmap-slices.ts +21 -5
  342. package/src/resources/extensions/gsd/safety/content-validator.ts +3 -3
  343. package/src/resources/extensions/gsd/session-lock.ts +17 -1
  344. package/src/resources/extensions/gsd/state.ts +115 -26
  345. package/src/resources/extensions/gsd/templates/context-enhanced.md +138 -0
  346. package/src/resources/extensions/gsd/tests/adversarial-review-fixes.test.ts +223 -0
  347. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +33 -2
  348. package/src/resources/extensions/gsd/tests/auto-remediate-slice-status.test.ts +56 -0
  349. package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +41 -0
  350. package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +72 -0
  351. package/src/resources/extensions/gsd/tests/complete-task-normalize-lists.test.ts +54 -0
  352. package/src/resources/extensions/gsd/tests/defer-milestone-stamp.test.ts +30 -0
  353. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +4 -3
  354. package/src/resources/extensions/gsd/tests/discuss-incremental-persistence.test.ts +36 -0
  355. package/src/resources/extensions/gsd/tests/discuss-slice-structured-questions.test.ts +46 -0
  356. package/src/resources/extensions/gsd/tests/dispatch-guard-closed-status.test.ts +33 -0
  357. package/src/resources/extensions/gsd/tests/dispatcher-stuck-planning.test.ts +37 -0
  358. package/src/resources/extensions/gsd/tests/error-success-mask.test.ts +37 -0
  359. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +125 -0
  360. package/src/resources/extensions/gsd/tests/find-missing-summaries-closed.test.ts +48 -0
  361. package/src/resources/extensions/gsd/tests/format-shortcut.test.ts +69 -0
  362. package/src/resources/extensions/gsd/tests/frontmatter-parse-noise.test.ts +42 -0
  363. package/src/resources/extensions/gsd/tests/gitignore-bg-shell.test.ts +38 -0
  364. package/src/resources/extensions/gsd/tests/guided-flow-state-rebuild.test.ts +103 -0
  365. package/src/resources/extensions/gsd/tests/import-done-milestones.test.ts +42 -0
  366. package/src/resources/extensions/gsd/tests/integration/auto-recovery.test.ts +11 -9
  367. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +4 -2
  368. package/src/resources/extensions/gsd/tests/integration/state-machine-live-validation.test.ts +28 -30
  369. package/src/resources/extensions/gsd/tests/integration/test-isolation.ts +53 -0
  370. package/src/resources/extensions/gsd/tests/integration-prepared-discussion.test.ts +525 -0
  371. package/src/resources/extensions/gsd/tests/isolation-none-branch-guard.test.ts +62 -0
  372. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +11 -10
  373. package/src/resources/extensions/gsd/tests/needs-remediation-revalidation.test.ts +48 -0
  374. package/src/resources/extensions/gsd/tests/note-captures-executed.test.ts +46 -0
  375. package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +189 -0
  376. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +77 -0
  377. package/src/resources/extensions/gsd/tests/phantom-ghost-detection.test.ts +55 -0
  378. package/src/resources/extensions/gsd/tests/phantom-milestone-default-queued.test.ts +39 -0
  379. package/src/resources/extensions/gsd/tests/pre-exec-backtick-strip.test.ts +68 -0
  380. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +284 -20
  381. package/src/resources/extensions/gsd/tests/pre-execution-fail-closed.test.ts +2 -2
  382. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +2 -2
  383. package/src/resources/extensions/gsd/tests/preparation.test.ts +1211 -0
  384. package/src/resources/extensions/gsd/tests/project-root-cwd-crash.test.ts +53 -0
  385. package/src/resources/extensions/gsd/tests/projection-no-plan-overwrite.test.ts +83 -0
  386. package/src/resources/extensions/gsd/tests/prompt-builder.test.ts +669 -0
  387. package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +7 -4
  388. package/src/resources/extensions/gsd/tests/prompt-step-ordering.test.ts +85 -0
  389. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +2 -1
  390. package/src/resources/extensions/gsd/tests/query-tools-db-open.test.ts +47 -0
  391. package/src/resources/extensions/gsd/tests/queued-discuss-fast-path.test.ts +107 -0
  392. package/src/resources/extensions/gsd/tests/reactive-graph.test.ts +45 -0
  393. package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +63 -0
  394. package/src/resources/extensions/gsd/tests/rogue-file-detection.test.ts +4 -5
  395. package/src/resources/extensions/gsd/tests/run-uat-replay-cap.test.ts +51 -0
  396. package/src/resources/extensions/gsd/tests/show-config-command.test.ts +56 -0
  397. package/src/resources/extensions/gsd/tests/skip-slice-state-rebuild.test.ts +31 -0
  398. package/src/resources/extensions/gsd/tests/skipped-validation-completion.test.ts +39 -0
  399. package/src/resources/extensions/gsd/tests/slice-sequence-insert.test.ts +51 -0
  400. package/src/resources/extensions/gsd/tests/smart-entry-complete.test.ts +1 -1
  401. package/src/resources/extensions/gsd/tests/stale-lockfile-recovery.test.ts +36 -0
  402. package/src/resources/extensions/gsd/tests/stale-queued-milestone.test.ts +147 -0
  403. package/src/resources/extensions/gsd/tests/stale-worktree-cwd.test.ts +13 -0
  404. package/src/resources/extensions/gsd/tests/stash-pop-gsd-conflict.test.ts +21 -0
  405. package/src/resources/extensions/gsd/tests/stash-queued-context-files.test.ts +21 -0
  406. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +6 -7
  407. package/src/resources/extensions/gsd/tests/status-db-open.test.ts +47 -0
  408. package/src/resources/extensions/gsd/tests/stuck-detection-coverage.test.ts +1 -0
  409. package/src/resources/extensions/gsd/tests/subagent-agent-discovery.test.ts +47 -0
  410. package/src/resources/extensions/gsd/tests/symlink-extension-discovery.test.ts +125 -0
  411. package/src/resources/extensions/gsd/tests/sync-worktree-skip-current.test.ts +65 -0
  412. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +29 -1
  413. package/src/resources/extensions/gsd/tests/triage-resolution.test.ts +2 -1
  414. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +3 -4
  415. package/src/resources/extensions/gsd/tests/verification-operational-gate.test.ts +15 -0
  416. package/src/resources/extensions/gsd/tests/verify-artifact-tightened.test.ts +89 -0
  417. package/src/resources/extensions/gsd/tests/wave1-critical-regressions.test.ts +49 -0
  418. package/src/resources/extensions/gsd/tests/wave2-events-regressions.test.ts +48 -0
  419. package/src/resources/extensions/gsd/tests/wave3-session-regressions.test.ts +47 -0
  420. package/src/resources/extensions/gsd/tests/wave4-write-safety-regressions.test.ts +70 -0
  421. package/src/resources/extensions/gsd/tests/wave5-consistency-regressions.test.ts +165 -0
  422. package/src/resources/extensions/gsd/tests/worker-model-override.test.ts +48 -0
  423. package/src/resources/extensions/gsd/tests/workflow-logger-audit.test.ts +6 -3
  424. package/src/resources/extensions/gsd/tests/worktree-expected-warnings.test.ts +38 -0
  425. package/src/resources/extensions/gsd/tests/worktree-integration.test.ts +16 -0
  426. package/src/resources/extensions/gsd/tests/worktree-main-branch.test.ts +20 -0
  427. package/src/resources/extensions/gsd/tests/worktree-sync-milestones.test.ts +16 -17
  428. package/src/resources/extensions/gsd/tests/worktree-sync-tasks.test.ts +13 -9
  429. package/src/resources/extensions/gsd/tests/worktree.test.ts +26 -9
  430. package/src/resources/extensions/gsd/tests/write-gate.test.ts +127 -2
  431. package/src/resources/extensions/gsd/tests/zero-slice-roadmap-guided.test.ts +19 -0
  432. package/src/resources/extensions/gsd/tools/complete-milestone.ts +13 -3
  433. package/src/resources/extensions/gsd/tools/complete-slice.ts +26 -6
  434. package/src/resources/extensions/gsd/tools/complete-task.ts +29 -7
  435. package/src/resources/extensions/gsd/tools/plan-milestone.ts +11 -9
  436. package/src/resources/extensions/gsd/tools/reassess-roadmap.ts +5 -2
  437. package/src/resources/extensions/gsd/tools/reopen-milestone.ts +152 -0
  438. package/src/resources/extensions/gsd/tools/reopen-slice.ts +27 -0
  439. package/src/resources/extensions/gsd/tools/reopen-task.ts +17 -0
  440. package/src/resources/extensions/gsd/triage-resolution.ts +37 -17
  441. package/src/resources/extensions/gsd/types.ts +4 -0
  442. package/src/resources/extensions/gsd/undo.ts +3 -2
  443. package/src/resources/extensions/gsd/workflow-events.ts +5 -3
  444. package/src/resources/extensions/gsd/workflow-logger.ts +1 -1
  445. package/src/resources/extensions/gsd/workflow-projections.ts +7 -8
  446. package/src/resources/extensions/gsd/workflow-reconcile.ts +109 -8
  447. package/src/resources/extensions/gsd/workflow-templates.ts +11 -2
  448. package/src/resources/extensions/gsd/worktree-manager.ts +4 -2
  449. package/src/resources/extensions/gsd/worktree.ts +10 -0
  450. package/src/resources/extensions/shared/interview-ui.ts +1 -1
  451. package/src/resources/extensions/shared/tests/interview-notes-loop.test.ts +8 -10
  452. package/src/resources/extensions/subagent/agents.ts +30 -6
  453. package/dist/web/standalone/.next/static/chunks/6502.7593d7797a4b3999.js +0 -9
  454. package/dist/web/standalone/.next/static/chunks/app/page-62be3b5fa91e4c8f.js +0 -1
  455. package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +0 -1
  456. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +0 -1
  457. /package/dist/web/standalone/.next/static/{MRM3OSYIAa4HMDqVGQ9nt → _X1i-S7l1jZfb7lmIZozb}/_buildManifest.js +0 -0
  458. /package/dist/web/standalone/.next/static/{MRM3OSYIAa4HMDqVGQ9nt → _X1i-S7l1jZfb7lmIZozb}/_ssgManifest.js +0 -0
@@ -6,7 +6,7 @@ import { isToolCallEventType } from "@gsd/pi-coding-agent";
6
6
  import { buildMilestoneFileName, resolveMilestonePath, resolveSliceFile, resolveSlicePath } from "../paths.js";
7
7
  import { buildBeforeAgentStartResult } from "./system-context.js";
8
8
  import { handleAgentEnd } from "./agent-end-recovery.js";
9
- import { clearDiscussionFlowState, isDepthVerified, isQueuePhaseActive, markDepthVerified, resetWriteGateState, shouldBlockContextWrite, shouldBlockQueueExecution } from "./write-gate.js";
9
+ import { clearDiscussionFlowState, isDepthVerified, isDepthConfirmationAnswer, isQueuePhaseActive, markDepthVerified, resetWriteGateState, shouldBlockContextWrite, shouldBlockQueueExecution } from "./write-gate.js";
10
10
  import { isBlockedStateFile, isBashWriteToStateFile, BLOCKED_WRITE_ERROR } from "../write-intercept.js";
11
11
  import { cleanupQuickBranch } from "../quick.js";
12
12
  import { getDiscussionMilestoneId } from "../guided-flow.js";
@@ -108,7 +108,10 @@ export function registerHooks(pi: ExtensionAPI): void {
108
108
  });
109
109
 
110
110
  pi.on("session_before_compact", async () => {
111
- if (isAutoActive() || isAutoPaused()) {
111
+ // Only cancel compaction while auto-mode is actively running.
112
+ // Paused auto-mode should allow compaction — the user may be doing
113
+ // interactive work (#3165).
114
+ if (isAutoActive()) {
112
115
  return { cancel: true };
113
116
  }
114
117
  const basePath = process.cwd();
@@ -246,7 +249,12 @@ export function registerHooks(pi: ExtensionAPI): void {
246
249
  const questions: any[] = (event.input as any)?.questions ?? [];
247
250
  for (const question of questions) {
248
251
  if (typeof question.id === "string" && question.id.includes("depth_verification")) {
249
- markDepthVerified();
252
+ // Only unlock the gate if the user selected the first option (confirmation).
253
+ // Cross-references against the question's defined options to reject free-form "Other" text.
254
+ const answer = details.response?.answers?.[question.id];
255
+ if (isDepthConfirmationAnswer(answer?.selected, question.options)) {
256
+ markDepthVerified();
257
+ }
250
258
  break;
251
259
  }
252
260
  }
@@ -15,7 +15,7 @@ import { hasSkillSnapshot, detectNewSkills, formatSkillsXml } from "../skill-dis
15
15
  import { getActiveAutoWorktreeContext } from "../auto-worktree.js";
16
16
  import { getActiveWorktreeName, getWorktreeOriginalCwd } from "../worktree-command.js";
17
17
  import { deriveState } from "../state.js";
18
- import { formatOverridesSection, loadActiveOverrides, loadFile, parseContinue, parseSummary } from "../files.js";
18
+ import { formatOverridesSection, formatShortcut, loadActiveOverrides, loadFile, parseContinue, parseSummary } from "../files.js";
19
19
  import { toPosixPath } from "../../shared/mod.js";
20
20
  import { markCmuxPromptShown, shouldPromptToEnableCmux } from "../../cmux/index.js";
21
21
 
@@ -72,6 +72,8 @@ export async function buildBeforeAgentStartResult(
72
72
  const systemContent = loadPrompt("system", {
73
73
  bundledSkillsTable: buildBundledSkillsTable(),
74
74
  templatesDir: getTemplatesDir(),
75
+ shortcutDashboard: formatShortcut("Ctrl+Alt+G"),
76
+ shortcutShell: formatShortcut("Ctrl+Alt+B"),
75
77
  });
76
78
  const loadedPreferences = loadEffectiveGSDPreferences();
77
79
  if (shouldPromptToEnableCmux(loadedPreferences?.preferences)) {
@@ -54,6 +54,35 @@ export function markDepthVerified(): void {
54
54
  depthVerificationDone = true;
55
55
  }
56
56
 
57
+ /**
58
+ * Check whether a depth_verification answer confirms the discussion is complete.
59
+ * Uses structural validation: the selected answer must exactly match the first
60
+ * option label from the question definition (the confirmation option by convention).
61
+ * This rejects free-form "Other" text, decline options, and garbage input without
62
+ * coupling to any specific label substring.
63
+ *
64
+ * @param selected The answer's selected value from details.response.answers[id].selected
65
+ * @param options The question's options array from event.input.questions[n].options
66
+ */
67
+ export function isDepthConfirmationAnswer(
68
+ selected: unknown,
69
+ options?: Array<{ label?: string }>,
70
+ ): boolean {
71
+ const value = Array.isArray(selected) ? selected[0] : selected;
72
+ if (typeof value !== "string" || !value) return false;
73
+
74
+ // If options are available, structurally validate: selected must exactly match
75
+ // the first option (confirmation) label. Rejects free-form "Other" and decline options.
76
+ if (Array.isArray(options) && options.length > 0) {
77
+ const confirmLabel = options[0]?.label;
78
+ return typeof confirmLabel === "string" && value === confirmLabel;
79
+ }
80
+
81
+ // Fallback when options aren't available (e.g., older call sites):
82
+ // accept only if it contains "(Recommended)" — the prompt convention suffix.
83
+ return value.includes("(Recommended)");
84
+ }
85
+
57
86
  export function shouldBlockContextWrite(
58
87
  toolName: string,
59
88
  inputPath: string,
@@ -71,7 +100,13 @@ export function shouldBlockContextWrite(
71
100
 
72
101
  return {
73
102
  block: true,
74
- reason: `Blocked: Cannot write to milestone CONTEXT.md during discussion phase without depth verification. Call ask_user_questions with question id "depth_verification" first to confirm discussion depth before writing context.`,
103
+ reason: [
104
+ `HARD BLOCK: Cannot write to milestone CONTEXT.md without depth verification.`,
105
+ `This is a mechanical gate — you MUST NOT proceed, retry, or rationalize past this block.`,
106
+ `Required action: call ask_user_questions with question id containing "depth_verification".`,
107
+ `The user MUST select the "(Recommended)" confirmation option to unlock this gate.`,
108
+ `If the user declines, cancels, or the tool fails, you must re-ask — not bypass.`,
109
+ ].join(" "),
75
110
  };
76
111
  }
77
112
 
@@ -13,7 +13,13 @@ export interface GsdDispatchContext {
13
13
  }
14
14
 
15
15
  export function projectRoot(): string {
16
- const cwd = process.cwd();
16
+ let cwd: string;
17
+ try {
18
+ cwd = process.cwd();
19
+ } catch {
20
+ // cwd directory was deleted (e.g. worktree teardown) — fall back to HOME (#3598)
21
+ cwd = process.env.HOME ?? "/";
22
+ }
17
23
  const root = resolveProjectRoot(cwd);
18
24
  if (root !== cwd) {
19
25
  assertSafeDirectory(cwd);
@@ -8,6 +8,7 @@ import { runEnvironmentChecks } from "../../doctor-environment.js";
8
8
  import { deriveState } from "../../state.js";
9
9
  import { handleCmux } from "../../commands-cmux.js";
10
10
  import { projectRoot } from "../context.js";
11
+ import { formatShortcut } from "../../files.js";
11
12
 
12
13
  export function showHelp(ctx: ExtensionCommandContext): void {
13
14
  const lines = [
@@ -24,12 +25,12 @@ export function showHelp(ctx: ExtensionCommandContext): void {
24
25
  " /gsd new-milestone Create milestone from headless context (used by gsd headless)",
25
26
  "",
26
27
  "VISIBILITY",
27
- " /gsd status Show progress dashboard (Ctrl+Alt+G)",
28
+ ` /gsd status Show progress dashboard (${formatShortcut("Ctrl+Alt+G")})`,
28
29
  " /gsd visualize Interactive 10-tab TUI (progress, timeline, deps, metrics, health, agent, changes, knowledge, captures, export)",
29
30
  " /gsd queue Show queued/dispatched units and execution order",
30
31
  " /gsd history View execution history [--cost] [--phase] [--model] [N]",
31
32
  " /gsd changelog Show categorized release notes [version]",
32
- " /gsd notifications View persistent notification history [clear|tail|filter] (Ctrl+Alt+N)",
33
+ ` /gsd notifications View persistent notification history [clear|tail|filter] (${formatShortcut("Ctrl+Alt+N")})`,
33
34
  "",
34
35
  "COURSE CORRECTION",
35
36
  " /gsd steer <desc> Apply user override to active work",
@@ -52,6 +53,7 @@ export function showHelp(ctx: ExtensionCommandContext): void {
52
53
  " /gsd cmux Manage cmux integration [status|on|off|notifications|sidebar|splits|browser]",
53
54
  " /gsd config Set API keys for external tools",
54
55
  " /gsd keys API key manager [list|add|remove|test|rotate|doctor]",
56
+ " /gsd show-config Show effective configuration (models, routing, toggles)",
55
57
  " /gsd hooks Show post-unit hook configuration",
56
58
  " /gsd extensions Manage extensions [list|enable|disable|info]",
57
59
  " /gsd fast Toggle OpenAI service tier [on|off|flex|status]",
@@ -71,6 +73,9 @@ export function showHelp(ctx: ExtensionCommandContext): void {
71
73
 
72
74
  export async function handleStatus(ctx: ExtensionCommandContext): Promise<void> {
73
75
  const basePath = projectRoot();
76
+ // Open DB in cold sessions so status uses DB-backed state, not filesystem fallback (#3385)
77
+ const { ensureDbOpen } = await import("../../bootstrap/dynamic-tools.js");
78
+ await ensureDbOpen();
74
79
  const state = await deriveState(basePath);
75
80
 
76
81
  if (state.registry.length === 0) {
@@ -214,6 +219,25 @@ export async function handleCoreCommand(trimmed: string, ctx: ExtensionCommandCo
214
219
  await handleCmux(trimmed.replace(/^cmux\s*/, "").trim(), ctx);
215
220
  return true;
216
221
  }
222
+ if (trimmed === "show-config") {
223
+ const { GSDConfigOverlay, formatConfigText } = await import("../../config-overlay.js");
224
+ const result = await ctx.ui.custom<void>(
225
+ (tui, theme, _kb, done) => new GSDConfigOverlay(tui, theme, () => done()),
226
+ {
227
+ overlay: true,
228
+ overlayOptions: {
229
+ width: "65%",
230
+ minWidth: 55,
231
+ maxHeight: "85%",
232
+ anchor: "center",
233
+ },
234
+ },
235
+ );
236
+ if (result === undefined) {
237
+ ctx.ui.notify(formatConfigText(), "info");
238
+ }
239
+ return true;
240
+ }
217
241
  if (trimmed === "setup" || trimmed.startsWith("setup ")) {
218
242
  await handleSetup(trimmed.replace(/^setup\s*/, "").trim(), ctx);
219
243
  return true;
@@ -105,7 +105,7 @@ function discoverManifests(): Map<string, ExtensionManifest> {
105
105
  const manifests = new Map<string, ExtensionManifest>();
106
106
  if (!existsSync(extDir)) return manifests;
107
107
  for (const entry of readdirSync(extDir, { withFileTypes: true })) {
108
- if (!entry.isDirectory()) continue;
108
+ if (!entry.isDirectory() && !entry.isSymbolicLink()) continue;
109
109
  const m = readManifest(join(extDir, entry.name));
110
110
  if (m) manifests.set(m.id, m);
111
111
  }
@@ -0,0 +1,331 @@
1
+ /**
2
+ * GSD Configuration Overlay
3
+ *
4
+ * Read-only TUI overlay showing the effective GSD configuration:
5
+ * token profile, model assignments, dynamic routing, git settings,
6
+ * budget, workflow toggles, and preference file sources.
7
+ * Opened via `/gsd show-config` or `/gsd config`.
8
+ */
9
+
10
+ import type { Theme } from "@gsd/pi-coding-agent";
11
+ import { matchesKey, Key, truncateToWidth } from "@gsd/pi-tui";
12
+
13
+ import {
14
+ loadEffectiveGSDPreferences,
15
+ loadGlobalGSDPreferences,
16
+ loadProjectGSDPreferences,
17
+ getGlobalGSDPreferencesPath,
18
+ getProjectGSDPreferencesPath,
19
+ resolveDynamicRoutingConfig,
20
+ resolveEffectiveProfile,
21
+ resolveModelWithFallbacksForUnit,
22
+ resolveAutoSupervisorConfig,
23
+ } from "./preferences.js";
24
+
25
+ // ─── Data Collection ──────────────────────────────────────────────────────
26
+
27
+ interface ConfigSection {
28
+ title: string;
29
+ rows: Array<{ label: string; value: string; accent?: boolean }>;
30
+ }
31
+
32
+ function collectConfigSections(): ConfigSection[] {
33
+ const sections: ConfigSection[] = [];
34
+
35
+ const globalPrefs = loadGlobalGSDPreferences();
36
+ const projectPrefs = loadProjectGSDPreferences();
37
+ const effective = loadEffectiveGSDPreferences();
38
+ const prefs = effective?.preferences;
39
+
40
+ // ─── Sources ─────────────────────────────────────────────────────────
41
+ sections.push({
42
+ title: "Sources",
43
+ rows: [
44
+ { label: "Global", value: globalPrefs ? globalPrefs.path : `(none) ${getGlobalGSDPreferencesPath()}` },
45
+ { label: "Project", value: projectPrefs ? projectPrefs.path : `(none) ${getProjectGSDPreferencesPath()}` },
46
+ ],
47
+ });
48
+
49
+ // ─── Profile ─────────────────────────────────────────────────────────
50
+ const profile = resolveEffectiveProfile();
51
+ const profileRows: ConfigSection["rows"] = [
52
+ { label: "Token profile", value: `${profile}${!prefs?.token_profile ? " (default)" : ""}`, accent: true },
53
+ ];
54
+ if (prefs?.mode) profileRows.push({ label: "Workflow mode", value: prefs.mode });
55
+ sections.push({ title: "Profile", rows: profileRows });
56
+
57
+ // ─── Models ──────────────────────────────────────────────────────────
58
+ const unitTypes: Array<[string, string]> = [
59
+ ["research", "research-milestone"],
60
+ ["planning", "plan-milestone"],
61
+ ["discuss", "discuss-milestone"],
62
+ ["execution", "execute-task"],
63
+ ["completion", "complete-slice"],
64
+ ["validation", "run-uat"],
65
+ ];
66
+
67
+ const modelRows: ConfigSection["rows"] = [];
68
+ for (const [label, unitType] of unitTypes) {
69
+ const resolved = resolveModelWithFallbacksForUnit(unitType);
70
+ if (resolved) {
71
+ let val = resolved.primary;
72
+ if (resolved.fallbacks.length > 0) {
73
+ val += ` \u2192 ${resolved.fallbacks.join(" \u2192 ")}`;
74
+ }
75
+ modelRows.push({ label, value: val });
76
+ } else {
77
+ modelRows.push({ label, value: "(inherit)" });
78
+ }
79
+ }
80
+
81
+ // subagent is a direct config key
82
+ const models = prefs?.models as Record<string, unknown> | undefined;
83
+ const subVal = models?.subagent;
84
+ if (subVal) {
85
+ const model = typeof subVal === "string" ? subVal : (subVal as { model?: string })?.model ?? "?";
86
+ modelRows.push({ label: "subagent", value: model });
87
+ } else {
88
+ modelRows.push({ label: "subagent", value: "(inherit)" });
89
+ }
90
+
91
+ sections.push({ title: "Models", rows: modelRows });
92
+
93
+ // ─── Dynamic Routing ─────────────────────────────────────────────────
94
+ const routing = resolveDynamicRoutingConfig();
95
+ const routingRows: ConfigSection["rows"] = [
96
+ { label: "Enabled", value: routing.enabled ? "yes" : "no", accent: routing.enabled },
97
+ ];
98
+ if (routing.enabled) {
99
+ routingRows.push({ label: "Escalate on fail", value: routing.escalate_on_failure !== false ? "yes" : "no" });
100
+ routingRows.push({ label: "Budget pressure", value: routing.budget_pressure !== false ? "yes" : "no" });
101
+ routingRows.push({ label: "Cross-provider", value: routing.cross_provider !== false ? "yes" : "no" });
102
+ if (routing.tier_models) {
103
+ const tm = routing.tier_models;
104
+ if (tm.light) routingRows.push({ label: "[L] light", value: tm.light });
105
+ if (tm.standard) routingRows.push({ label: "[S] standard", value: tm.standard });
106
+ if (tm.heavy) routingRows.push({ label: "[H] heavy", value: tm.heavy });
107
+ }
108
+ }
109
+ sections.push({ title: "Dynamic Routing", rows: routingRows });
110
+
111
+ // ─── Git ─────────────────────────────────────────────────────────────
112
+ if (prefs?.git) {
113
+ const g = prefs.git;
114
+ const gitRows: ConfigSection["rows"] = [];
115
+ if (g.isolation !== undefined) gitRows.push({ label: "Isolation", value: String(g.isolation) });
116
+ if (g.auto_push !== undefined) gitRows.push({ label: "Auto push", value: String(g.auto_push) });
117
+ if (g.push_branches !== undefined) gitRows.push({ label: "Push branches", value: String(g.push_branches) });
118
+ if (g.merge_strategy) gitRows.push({ label: "Merge strategy", value: g.merge_strategy });
119
+ if (g.main_branch) gitRows.push({ label: "Main branch", value: g.main_branch });
120
+ if (g.remote) gitRows.push({ label: "Remote", value: g.remote });
121
+ if (gitRows.length > 0) sections.push({ title: "Git", rows: gitRows });
122
+ }
123
+
124
+ // ─── Budget ──────────────────────────────────────────────────────────
125
+ if (prefs?.budget_ceiling !== undefined || prefs?.budget_enforcement) {
126
+ const budgetRows: ConfigSection["rows"] = [];
127
+ if (prefs.budget_ceiling !== undefined) budgetRows.push({ label: "Ceiling", value: `$${prefs.budget_ceiling}` });
128
+ if (prefs.budget_enforcement) budgetRows.push({ label: "Enforcement", value: String(prefs.budget_enforcement) });
129
+ sections.push({ title: "Budget", rows: budgetRows });
130
+ }
131
+
132
+ // ─── Auto Supervisor ─────────────────────────────────────────────────
133
+ if (prefs?.auto_supervisor) {
134
+ const sup = resolveAutoSupervisorConfig();
135
+ const supRows: ConfigSection["rows"] = [];
136
+ if (sup.model) supRows.push({ label: "Model", value: sup.model });
137
+ supRows.push({ label: "Soft timeout", value: `${sup.soft_timeout_minutes}m` });
138
+ supRows.push({ label: "Idle timeout", value: `${sup.idle_timeout_minutes}m` });
139
+ supRows.push({ label: "Hard timeout", value: `${sup.hard_timeout_minutes}m` });
140
+ sections.push({ title: "Auto Supervisor", rows: supRows });
141
+ }
142
+
143
+ // ─── Toggles ─────────────────────────────────────────────────────────
144
+ const toggleRows: ConfigSection["rows"] = [];
145
+ if (prefs?.phases) {
146
+ const p = prefs.phases;
147
+ if (p.skip_research) toggleRows.push({ label: "skip_research", value: "on" });
148
+ if (p.skip_reassess) toggleRows.push({ label: "skip_reassess", value: "on" });
149
+ if (p.skip_slice_research) toggleRows.push({ label: "skip_slice_research", value: "on" });
150
+ if (p.skip_milestone_validation) toggleRows.push({ label: "skip_milestone_validation", value: "on" });
151
+ if (p.require_slice_discussion) toggleRows.push({ label: "require_slice_discussion", value: "on" });
152
+ }
153
+ if (prefs?.uat_dispatch) toggleRows.push({ label: "uat_dispatch", value: "on" });
154
+ if (prefs?.auto_visualize) toggleRows.push({ label: "auto_visualize", value: "on" });
155
+ if (prefs?.auto_report === false) toggleRows.push({ label: "auto_report", value: "off" });
156
+ if (prefs?.show_token_cost) toggleRows.push({ label: "show_token_cost", value: "on" });
157
+ if (prefs?.forensics_dedup) toggleRows.push({ label: "forensics_dedup", value: "on" });
158
+ if (prefs?.unique_milestone_ids) toggleRows.push({ label: "unique_milestone_ids", value: "on" });
159
+ if (prefs?.service_tier) toggleRows.push({ label: "service_tier", value: prefs.service_tier });
160
+ if (prefs?.search_provider && prefs.search_provider !== "auto") toggleRows.push({ label: "search_provider", value: prefs.search_provider });
161
+ if (prefs?.context_selection) toggleRows.push({ label: "context_selection", value: prefs.context_selection });
162
+ if (prefs?.widget_mode && prefs.widget_mode !== "full") toggleRows.push({ label: "widget_mode", value: prefs.widget_mode });
163
+ if (prefs?.experimental?.rtk) toggleRows.push({ label: "experimental.rtk", value: "on" });
164
+ if (toggleRows.length > 0) sections.push({ title: "Toggles", rows: toggleRows });
165
+
166
+ // ─── Parallel ────────────────────────────────────────────────────────
167
+ if (prefs?.parallel) {
168
+ const pc = prefs.parallel;
169
+ const parallelRows: ConfigSection["rows"] = [];
170
+ if (pc.max_workers !== undefined) parallelRows.push({ label: "Max workers", value: String(pc.max_workers) });
171
+ if (pc.merge_strategy) parallelRows.push({ label: "Merge strategy", value: pc.merge_strategy });
172
+ if (pc.auto_merge) parallelRows.push({ label: "Auto merge", value: pc.auto_merge });
173
+ if (parallelRows.length > 0) sections.push({ title: "Parallel", rows: parallelRows });
174
+ }
175
+
176
+ // ─── Hooks ───────────────────────────────────────────────────────────
177
+ const postHooks = prefs?.post_unit_hooks?.filter(h => h.enabled !== false) ?? [];
178
+ const preHooks = prefs?.pre_dispatch_hooks?.filter(h => h.enabled !== false) ?? [];
179
+ if (postHooks.length > 0 || preHooks.length > 0) {
180
+ const hookRows: ConfigSection["rows"] = [];
181
+ if (preHooks.length > 0) hookRows.push({ label: "Pre-dispatch", value: `${preHooks.length} active` });
182
+ if (postHooks.length > 0) hookRows.push({ label: "Post-unit", value: `${postHooks.length} active` });
183
+ sections.push({ title: "Hooks", rows: hookRows });
184
+ }
185
+
186
+ // ─── Warnings ────────────────────────────────────────────────────────
187
+ const warnings = [
188
+ ...(globalPrefs?.warnings ?? []),
189
+ ...(projectPrefs?.warnings ?? []),
190
+ ];
191
+ if (warnings.length > 0) {
192
+ sections.push({
193
+ title: "Warnings",
194
+ rows: warnings.map(w => ({ label: "\u26a0", value: w })),
195
+ });
196
+ }
197
+
198
+ return sections;
199
+ }
200
+
201
+ // ─── Plain Text Formatter (headless/RPC fallback) ─────────────────────────
202
+
203
+ export function formatConfigText(): string {
204
+ const sections = collectConfigSections();
205
+ const lines: string[] = ["GSD Configuration\n"];
206
+
207
+ let maxLabel = 0;
208
+ for (const section of sections) {
209
+ for (const row of section.rows) {
210
+ if (row.label.length > maxLabel) maxLabel = row.label.length;
211
+ }
212
+ }
213
+ const pad = Math.min(maxLabel + 2, 24);
214
+
215
+ for (const section of sections) {
216
+ lines.push("");
217
+ lines.push(section.title.toUpperCase());
218
+ for (const row of section.rows) {
219
+ lines.push(` ${row.label.padEnd(pad)}${row.value}`);
220
+ }
221
+ }
222
+
223
+ return lines.join("\n");
224
+ }
225
+
226
+ // ─── Overlay Class ────────────────────────────────────────────────────────
227
+
228
+ export class GSDConfigOverlay {
229
+ private tui: { requestRender: () => void };
230
+ private theme: Theme;
231
+ private onClose: () => void;
232
+ private sections: ConfigSection[];
233
+ private cachedLines?: string[];
234
+ private scrollOffset = 0;
235
+ private disposed = false;
236
+
237
+ constructor(
238
+ tui: { requestRender: () => void },
239
+ theme: Theme,
240
+ onClose: () => void,
241
+ ) {
242
+ this.tui = tui;
243
+ this.theme = theme;
244
+ this.onClose = onClose;
245
+ this.sections = collectConfigSections();
246
+ }
247
+
248
+ invalidate(): void {
249
+ this.cachedLines = undefined;
250
+ }
251
+
252
+ dispose(): void {
253
+ this.disposed = true;
254
+ }
255
+
256
+ handleInput(data: string): void {
257
+ if (matchesKey(data, Key.escape) || data === "q") {
258
+ this.dispose();
259
+ this.onClose();
260
+ return;
261
+ }
262
+ if (matchesKey(data, Key.down) || data === "j") {
263
+ this.scrollOffset++;
264
+ this.cachedLines = undefined;
265
+ this.tui.requestRender();
266
+ return;
267
+ }
268
+ if (matchesKey(data, Key.up) || data === "k") {
269
+ this.scrollOffset = Math.max(0, this.scrollOffset - 1);
270
+ this.cachedLines = undefined;
271
+ this.tui.requestRender();
272
+ return;
273
+ }
274
+ if (matchesKey(data, Key.pageDown)) {
275
+ this.scrollOffset += 10;
276
+ this.cachedLines = undefined;
277
+ this.tui.requestRender();
278
+ return;
279
+ }
280
+ if (matchesKey(data, Key.pageUp)) {
281
+ this.scrollOffset = Math.max(0, this.scrollOffset - 10);
282
+ this.cachedLines = undefined;
283
+ this.tui.requestRender();
284
+ return;
285
+ }
286
+ }
287
+
288
+ render(width: number): string[] {
289
+ if (this.cachedLines) return this.cachedLines;
290
+
291
+ const t = this.theme;
292
+ const w = Math.max(width, 50);
293
+ const allLines: string[] = [];
294
+
295
+ // Header
296
+ allLines.push(t.bold(t.fg("accent", " GSD Configuration ")));
297
+ allLines.push(t.fg("muted", "\u2500".repeat(w)));
298
+
299
+ // Find max label width for alignment
300
+ let maxLabel = 0;
301
+ for (const section of this.sections) {
302
+ for (const row of section.rows) {
303
+ if (row.label.length > maxLabel) maxLabel = row.label.length;
304
+ }
305
+ }
306
+ const labelPad = Math.min(maxLabel + 2, 24);
307
+
308
+ for (const section of this.sections) {
309
+ allLines.push("");
310
+ allLines.push(t.bold(t.fg("accent", ` ${section.title}`)));
311
+
312
+ for (const row of section.rows) {
313
+ const label = t.fg("muted", ` ${row.label.padEnd(labelPad)}`);
314
+ const value = row.accent ? t.bold(row.value) : row.value;
315
+ allLines.push(truncateToWidth(`${label}${value}`, w));
316
+ }
317
+ }
318
+
319
+ allLines.push("");
320
+ allLines.push(t.fg("muted", ` ${"\u2500".repeat(w - 4)}`));
321
+ allLines.push(t.fg("muted", " esc/q close \u2502 \u2191\u2193/jk scroll \u2502 /gsd prefs to edit"));
322
+
323
+ // Apply scroll
324
+ const maxScroll = Math.max(0, allLines.length - 20);
325
+ this.scrollOffset = Math.min(this.scrollOffset, maxScroll);
326
+ const visible = allLines.slice(this.scrollOffset);
327
+
328
+ this.cachedLines = visible;
329
+ return visible;
330
+ }
331
+ }
@@ -345,8 +345,12 @@ export async function saveRequirementToDb(
345
345
  await saveFile(filePath, md);
346
346
  } catch (diskErr) {
347
347
  logError('manifest', 'disk write failed, rolling back DB row', { fn: 'saveRequirementToDb', error: String((diskErr as Error).message) });
348
- const rollbackAdapter = db._getAdapter();
349
- rollbackAdapter?.prepare('DELETE FROM requirements WHERE id = :id').run({ ':id': id });
348
+ try {
349
+ const rollbackAdapter = db._getAdapter();
350
+ rollbackAdapter?.prepare('DELETE FROM requirements WHERE id = :id').run({ ':id': id });
351
+ } catch (rollbackErr) {
352
+ logError('manifest', 'SPLIT BRAIN: disk write failed AND DB rollback failed — DB has orphaned row', { fn: 'saveRequirementToDb', id, error: String((rollbackErr as Error).message) });
353
+ }
350
354
  throw diskErr;
351
355
  }
352
356
  invalidateStateCache();
@@ -466,7 +470,11 @@ export async function saveDecisionToDb(
466
470
  await saveFile(filePath, md);
467
471
  } catch (diskErr) {
468
472
  logError('manifest', 'disk write failed, rolling back DB row', { fn: 'saveDecisionToDb', error: String((diskErr as Error).message) });
469
- adapter?.prepare('DELETE FROM decisions WHERE id = :id').run({ ':id': id });
473
+ try {
474
+ adapter?.prepare('DELETE FROM decisions WHERE id = :id').run({ ':id': id });
475
+ } catch (rollbackErr) {
476
+ logError('manifest', 'SPLIT BRAIN: disk write failed AND DB rollback failed — DB has orphaned row', { fn: 'saveDecisionToDb', id, error: String((rollbackErr as Error).message) });
477
+ }
470
478
  throw diskErr;
471
479
  }
472
480
  // #2661: When a decision defers a slice, update the slice status in the DB
@@ -1114,7 +1114,7 @@ function resolveVersionCatalogAccessors(
1114
1114
  return accessors;
1115
1115
  }
1116
1116
 
1117
- function scanProjectFiles(basePath: string): string[] {
1117
+ export function scanProjectFiles(basePath: string): string[] {
1118
1118
  const files: string[] = [];
1119
1119
  const queue: Array<{ path: string; depth: number }> = [{ path: basePath, depth: 0 }];
1120
1120
 
@@ -5,6 +5,7 @@ import { findMilestoneIds } from "./guided-flow.js";
5
5
  import { parseUnitId } from "./unit-id.js";
6
6
  import { isDbAvailable, getMilestoneSlices } from "./gsd-db.js";
7
7
  import { parseRoadmap } from "./parsers-legacy.js";
8
+ import { isClosedStatus } from "./status-guards.js";
8
9
  import { readFileSync } from "node:fs";
9
10
 
10
11
  const SLICE_DISPATCH_TYPES = new Set([
@@ -57,7 +58,7 @@ export function getPriorSliceCompletionBlocker(
57
58
  if (rows.length > 0) {
58
59
  slices = rows.map((r) => ({
59
60
  id: r.id,
60
- done: r.status === "complete",
61
+ done: isClosedStatus(r.status),
61
62
  depends: r.depends ?? [],
62
63
  }));
63
64
  }
@@ -211,6 +211,7 @@ Setting `prefer_skills: []` does **not** disable skill discovery — it just mea
211
211
  - `budget_ceiling`: number — optional per-parallel-run budget ceiling.
212
212
  - `merge_strategy`: `"per-slice"` or `"per-milestone"` — when to merge worktree results back. Default: `"per-milestone"`.
213
213
  - `auto_merge`: `"auto"`, `"confirm"`, or `"manual"` — merge behavior after completion. `"auto"` merges immediately; `"confirm"` asks first; `"manual"` leaves branches for you. Default: `"confirm"`.
214
+ - `worker_model`: string — optional model override for parallel milestone workers. When set, workers use this model (e.g. `"claude-haiku-4-5"`) instead of inheriting the coordinator's model. Useful for cost savings on execution-heavy milestones.
214
215
 
215
216
  - `verification_commands`: string[] — shell commands to run as verification after task execution (e.g., `["npm test", "npm run lint"]`). Commands run in order; if any fails, the task is marked as needing fixes.
216
217
 
@@ -87,7 +87,8 @@ function validatePreferenceShape(preferences: GSDPreferences): string[] {
87
87
  return issues;
88
88
  }
89
89
 
90
- function buildStateMarkdown(state: Awaited<ReturnType<typeof deriveState>>): string {
90
+ /** Build STATE.md content from derived state. Exported for guided-flow pre-dispatch rebuild (#3475). */
91
+ export function buildStateMarkdown(state: Awaited<ReturnType<typeof deriveState>>): string {
91
92
  const lines: string[] = [];
92
93
  lines.push("# GSD State", "");
93
94
 
@@ -70,6 +70,25 @@ export function clearParseCache(): void {
70
70
  for (const cb of _cacheClearCallbacks) cb();
71
71
  }
72
72
 
73
+ // ─── Platform shortcuts ───────────────────────────────────────────────────
74
+
75
+ const IS_MAC = process.platform === "darwin";
76
+
77
+ /**
78
+ * Format a keyboard shortcut for the current OS.
79
+ * Input: modifier key combo like "Ctrl+Alt+G"
80
+ * Output: "⌃⌥G" on macOS, "Ctrl+Alt+G" on Windows/Linux.
81
+ */
82
+ export function formatShortcut(combo: string): string {
83
+ if (!IS_MAC) return combo;
84
+ return combo
85
+ .replace(/Ctrl\+Alt\+/i, "⌃⌥")
86
+ .replace(/Ctrl\+/i, "⌃")
87
+ .replace(/Alt\+/i, "⌥")
88
+ .replace(/Shift\+/i, "⇧")
89
+ .replace(/Cmd\+/i, "⌘");
90
+ }
91
+
73
92
  // ─── Helpers ───────────────────────────────────────────────────────────────
74
93
 
75
94
  /** Extract the text after a heading at a given level, up to the next heading of same or higher level. */
@@ -42,6 +42,7 @@ const BASELINE_PATTERNS = [
42
42
  // ── GSD state directory (symlink to external storage) ──
43
43
  ".gsd",
44
44
  ".gsd-id",
45
+ ".bg-shell/",
45
46
 
46
47
  // ── OS junk ──
47
48
  ".DS_Store",