gsd-pi 2.50.0-dev.d210a87 → 2.51.0-dev.7d435fe

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 (302) hide show
  1. package/README.md +4 -4
  2. package/dist/headless-events.d.ts +18 -0
  3. package/dist/headless-events.js +36 -0
  4. package/dist/headless-types.d.ts +28 -0
  5. package/dist/headless-types.js +7 -0
  6. package/dist/headless.d.ts +8 -3
  7. package/dist/headless.js +47 -16
  8. package/dist/help-text.js +16 -5
  9. package/dist/onboarding.js +5 -4
  10. package/dist/remote-questions-config.js +1 -1
  11. package/dist/resources/extensions/async-jobs/async-bash-tool.js +29 -17
  12. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +18 -19
  13. package/dist/resources/extensions/gsd/auto-dispatch.js +18 -0
  14. package/dist/resources/extensions/gsd/auto-start.js +2 -0
  15. package/dist/resources/extensions/gsd/auto-timers.js +24 -2
  16. package/dist/resources/extensions/gsd/auto-tool-tracking.js +25 -7
  17. package/dist/resources/extensions/gsd/auto-worktree.js +21 -0
  18. package/dist/resources/extensions/gsd/auto.js +4 -2
  19. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +95 -69
  20. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +12 -2
  21. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +1 -1
  22. package/dist/resources/extensions/gsd/claude-import.js +60 -9
  23. package/dist/resources/extensions/gsd/commands/handlers/auto.js +69 -6
  24. package/dist/resources/extensions/gsd/commands-config.js +10 -5
  25. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +1 -1
  26. package/dist/resources/extensions/gsd/detection.js +6 -6
  27. package/dist/resources/extensions/gsd/docs/preferences-reference.md +3 -3
  28. package/dist/resources/extensions/gsd/error-classifier.js +105 -0
  29. package/dist/resources/extensions/gsd/gitignore.js +7 -7
  30. package/dist/resources/extensions/gsd/gsd-db.js +298 -45
  31. package/dist/resources/extensions/gsd/init-wizard.js +2 -2
  32. package/dist/resources/extensions/gsd/key-manager.js +7 -16
  33. package/dist/resources/extensions/gsd/memory-store.js +28 -13
  34. package/dist/resources/extensions/gsd/milestone-actions.js +19 -0
  35. package/dist/resources/extensions/gsd/preferences-models.js +1 -13
  36. package/dist/resources/extensions/gsd/preferences.js +13 -13
  37. package/dist/resources/extensions/gsd/prompts/system.md +1 -1
  38. package/dist/resources/extensions/gsd/provider-error-pause.js +0 -44
  39. package/dist/resources/extensions/gsd/rule-registry.js +1 -1
  40. package/dist/resources/extensions/gsd/service-tier.js +13 -2
  41. package/dist/resources/extensions/gsd/state.js +21 -2
  42. package/dist/resources/extensions/gsd/tools/complete-milestone.js +3 -10
  43. package/dist/resources/extensions/gsd/tools/complete-slice.js +3 -17
  44. package/dist/resources/extensions/gsd/tools/complete-task.js +7 -18
  45. package/dist/resources/extensions/gsd/tools/plan-milestone.js +26 -17
  46. package/dist/resources/extensions/gsd/tools/plan-slice.js +25 -14
  47. package/dist/resources/extensions/gsd/tools/plan-task.js +21 -11
  48. package/dist/resources/extensions/gsd/tools/reassess-roadmap.js +47 -37
  49. package/dist/resources/extensions/gsd/tools/replan-slice.js +49 -38
  50. package/dist/resources/extensions/gsd/tools/validate-milestone.js +23 -16
  51. package/dist/resources/extensions/gsd/workflow-logger.js +0 -1
  52. package/dist/resources/extensions/remote-questions/config.js +1 -1
  53. package/dist/resources/extensions/remote-questions/remote-command.js +1 -1
  54. package/dist/resources/extensions/search-the-web/native-search.js +1 -1
  55. package/dist/resources/extensions/search-the-web/provider.js +1 -1
  56. package/dist/web/standalone/.next/BUILD_ID +1 -1
  57. package/dist/web/standalone/.next/app-path-routes-manifest.json +21 -21
  58. package/dist/web/standalone/.next/build-manifest.json +3 -3
  59. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  60. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  61. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  62. package/dist/web/standalone/.next/server/app/_global-error.html +2 -2
  63. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  64. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  65. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  66. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  67. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  68. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  69. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  70. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  71. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  72. package/dist/web/standalone/.next/server/app/_not-found.rsc +2 -2
  73. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
  74. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  75. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
  76. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  77. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  78. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
  79. package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
  80. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
  81. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
  82. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
  83. package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
  84. package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
  85. package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
  86. package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
  87. package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
  88. package/dist/web/standalone/.next/server/app/api/experimental/route.js +1 -1
  89. package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -1
  90. package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
  91. package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
  92. package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
  93. package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
  94. package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
  95. package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
  96. package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
  97. package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
  98. package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
  99. package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
  100. package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
  101. package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  102. package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
  103. package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +1 -1
  104. package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
  105. package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
  106. package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
  107. package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
  108. package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
  109. package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
  110. package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
  111. package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
  112. package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
  113. package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
  114. package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
  115. package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
  116. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
  117. package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
  118. package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
  119. package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
  120. package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  121. package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
  122. package/dist/web/standalone/.next/server/app/index.html +1 -1
  123. package/dist/web/standalone/.next/server/app/index.rsc +2 -2
  124. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  125. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +2 -2
  126. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  127. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +2 -2
  128. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -2
  129. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  130. package/dist/web/standalone/.next/server/app-paths-manifest.json +21 -21
  131. package/dist/web/standalone/.next/server/chunks/2229.js +2 -2
  132. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  133. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  134. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  135. package/dist/web/standalone/.next/server/pages/500.html +2 -2
  136. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  137. package/dist/web/standalone/.next/static/chunks/4024.21054f459af5cc78.js +9 -0
  138. package/dist/web/standalone/.next/static/chunks/{webpack-cfc9a116e6450a6b.js → webpack-024d82be84800e52.js} +1 -1
  139. package/dist/web/standalone/.next/static/css/a58ef8a151aa0493.css +1 -0
  140. package/dist/wizard.js +4 -1
  141. package/package.json +2 -2
  142. package/packages/pi-ai/dist/models.d.ts +14 -3
  143. package/packages/pi-ai/dist/models.d.ts.map +1 -1
  144. package/packages/pi-ai/dist/models.js +53 -10
  145. package/packages/pi-ai/dist/models.js.map +1 -1
  146. package/packages/pi-ai/dist/models.test.js +102 -1
  147. package/packages/pi-ai/dist/models.test.js.map +1 -1
  148. package/packages/pi-ai/dist/types.d.ts +30 -0
  149. package/packages/pi-ai/dist/types.d.ts.map +1 -1
  150. package/packages/pi-ai/dist/types.js.map +1 -1
  151. package/packages/pi-ai/src/models.test.ts +114 -1
  152. package/packages/pi-ai/src/models.ts +70 -13
  153. package/packages/pi-ai/src/types.ts +31 -0
  154. package/packages/pi-coding-agent/dist/cli/args.d.ts +2 -0
  155. package/packages/pi-coding-agent/dist/cli/args.d.ts.map +1 -1
  156. package/packages/pi-coding-agent/dist/cli/args.js +3 -0
  157. package/packages/pi-coding-agent/dist/cli/args.js.map +1 -1
  158. package/packages/pi-coding-agent/dist/core/bash-executor.d.ts.map +1 -1
  159. package/packages/pi-coding-agent/dist/core/bash-executor.js +5 -1
  160. package/packages/pi-coding-agent/dist/core/bash-executor.js.map +1 -1
  161. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  162. package/packages/pi-coding-agent/dist/core/model-registry.js +9 -4
  163. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  164. package/packages/pi-coding-agent/dist/core/tools/bash-spawn-windows.test.d.ts +19 -0
  165. package/packages/pi-coding-agent/dist/core/tools/bash-spawn-windows.test.d.ts.map +1 -0
  166. package/packages/pi-coding-agent/dist/core/tools/bash-spawn-windows.test.js +83 -0
  167. package/packages/pi-coding-agent/dist/core/tools/bash-spawn-windows.test.js.map +1 -0
  168. package/packages/pi-coding-agent/dist/core/tools/bash.d.ts.map +1 -1
  169. package/packages/pi-coding-agent/dist/core/tools/bash.js +5 -1
  170. package/packages/pi-coding-agent/dist/core/tools/bash.js.map +1 -1
  171. package/packages/pi-coding-agent/dist/index.d.ts +1 -1
  172. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  173. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  174. package/packages/pi-coding-agent/dist/main.d.ts.map +1 -1
  175. package/packages/pi-coding-agent/dist/main.js +5 -3
  176. package/packages/pi-coding-agent/dist/main.js.map +1 -1
  177. package/packages/pi-coding-agent/dist/modes/index.d.ts +1 -1
  178. package/packages/pi-coding-agent/dist/modes/index.d.ts.map +1 -1
  179. package/packages/pi-coding-agent/dist/modes/index.js.map +1 -1
  180. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  181. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +0 -2
  182. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  183. package/packages/pi-coding-agent/dist/modes/rpc/rpc-client.d.ts +28 -1
  184. package/packages/pi-coding-agent/dist/modes/rpc/rpc-client.d.ts.map +1 -1
  185. package/packages/pi-coding-agent/dist/modes/rpc/rpc-client.js +49 -0
  186. package/packages/pi-coding-agent/dist/modes/rpc/rpc-client.js.map +1 -1
  187. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.d.ts +1 -1
  188. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
  189. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js +114 -6
  190. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js.map +1 -1
  191. package/packages/pi-coding-agent/dist/modes/rpc/rpc-protocol-v2.test.d.ts +9 -0
  192. package/packages/pi-coding-agent/dist/modes/rpc/rpc-protocol-v2.test.d.ts.map +1 -0
  193. package/packages/pi-coding-agent/dist/modes/rpc/rpc-protocol-v2.test.js +831 -0
  194. package/packages/pi-coding-agent/dist/modes/rpc/rpc-protocol-v2.test.js.map +1 -0
  195. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts +66 -0
  196. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts.map +1 -1
  197. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.js.map +1 -1
  198. package/packages/pi-coding-agent/dist/utils/shell.d.ts.map +1 -1
  199. package/packages/pi-coding-agent/dist/utils/shell.js +0 -1
  200. package/packages/pi-coding-agent/dist/utils/shell.js.map +1 -1
  201. package/packages/pi-coding-agent/package.json +1 -1
  202. package/packages/pi-coding-agent/src/cli/args.ts +4 -0
  203. package/packages/pi-coding-agent/src/core/bash-executor.ts +5 -1
  204. package/packages/pi-coding-agent/src/core/model-registry.ts +10 -3
  205. package/packages/pi-coding-agent/src/core/tools/bash-spawn-windows.test.ts +101 -0
  206. package/packages/pi-coding-agent/src/core/tools/bash.ts +5 -1
  207. package/packages/pi-coding-agent/src/index.ts +3 -0
  208. package/packages/pi-coding-agent/src/main.ts +5 -3
  209. package/packages/pi-coding-agent/src/modes/index.ts +8 -1
  210. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +0 -2
  211. package/packages/pi-coding-agent/src/modes/rpc/rpc-client.ts +54 -1
  212. package/packages/pi-coding-agent/src/modes/rpc/rpc-mode.ts +124 -6
  213. package/packages/pi-coding-agent/src/modes/rpc/rpc-protocol-v2.test.ts +971 -0
  214. package/packages/pi-coding-agent/src/modes/rpc/rpc-types.ts +61 -4
  215. package/packages/pi-coding-agent/src/utils/shell.ts +0 -1
  216. package/pkg/package.json +1 -1
  217. package/src/resources/extensions/async-jobs/async-bash-tool.ts +22 -11
  218. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +19 -20
  219. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +21 -0
  220. package/src/resources/extensions/gsd/auto-dispatch.ts +19 -0
  221. package/src/resources/extensions/gsd/auto-start.ts +2 -0
  222. package/src/resources/extensions/gsd/auto-timers.ts +25 -1
  223. package/src/resources/extensions/gsd/auto-tool-tracking.ts +30 -6
  224. package/src/resources/extensions/gsd/auto-worktree.ts +21 -0
  225. package/src/resources/extensions/gsd/auto.ts +5 -2
  226. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +115 -72
  227. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +11 -2
  228. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +1 -1
  229. package/src/resources/extensions/gsd/claude-import.ts +58 -9
  230. package/src/resources/extensions/gsd/commands/handlers/auto.ts +73 -6
  231. package/src/resources/extensions/gsd/commands-config.ts +11 -5
  232. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +1 -1
  233. package/src/resources/extensions/gsd/detection.ts +6 -6
  234. package/src/resources/extensions/gsd/docs/preferences-reference.md +3 -3
  235. package/src/resources/extensions/gsd/error-classifier.ts +139 -0
  236. package/src/resources/extensions/gsd/gitignore.ts +7 -7
  237. package/src/resources/extensions/gsd/gsd-db.ts +355 -63
  238. package/src/resources/extensions/gsd/init-wizard.ts +2 -2
  239. package/src/resources/extensions/gsd/key-manager.ts +7 -16
  240. package/src/resources/extensions/gsd/memory-store.ts +29 -18
  241. package/src/resources/extensions/gsd/milestone-actions.ts +17 -0
  242. package/src/resources/extensions/gsd/preferences-models.ts +1 -13
  243. package/src/resources/extensions/gsd/preferences.ts +12 -13
  244. package/src/resources/extensions/gsd/prompts/system.md +1 -1
  245. package/src/resources/extensions/gsd/provider-error-pause.ts +0 -57
  246. package/src/resources/extensions/gsd/rule-registry.ts +1 -1
  247. package/src/resources/extensions/gsd/service-tier.ts +14 -2
  248. package/src/resources/extensions/gsd/state.ts +22 -2
  249. package/src/resources/extensions/gsd/tests/auto-milestone-target.test.ts +61 -0
  250. package/src/resources/extensions/gsd/tests/claude-import-marketplace-discovery.test.ts +191 -0
  251. package/src/resources/extensions/gsd/tests/claude-import-tui.test.ts +1 -1
  252. package/src/resources/extensions/gsd/tests/commands-config.test.ts +24 -0
  253. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  254. package/src/resources/extensions/gsd/tests/complete-task-rollback-evidence.test.ts +106 -0
  255. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  256. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +35 -7
  257. package/src/resources/extensions/gsd/tests/detection.test.ts +1 -1
  258. package/src/resources/extensions/gsd/tests/doctor-git.test.ts +4 -4
  259. package/src/resources/extensions/gsd/tests/doctor-proactive.test.ts +1 -1
  260. package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +2 -2
  261. package/src/resources/extensions/gsd/tests/empty-db-reconciliation.test.ts +79 -0
  262. package/src/resources/extensions/gsd/tests/git-service.test.ts +1 -1
  263. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
  264. package/src/resources/extensions/gsd/tests/idle-watchdog-stall-override.test.ts +125 -0
  265. package/src/resources/extensions/gsd/tests/init-wizard.test.ts +1 -1
  266. package/src/resources/extensions/gsd/tests/interactive-tool-idle-exemption.test.ts +119 -0
  267. package/src/resources/extensions/gsd/tests/key-manager.test.ts +16 -1
  268. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -1
  269. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -2
  270. package/src/resources/extensions/gsd/tests/none-mode-gates.test.ts +7 -7
  271. package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +85 -0
  272. package/src/resources/extensions/gsd/tests/preferences-worktree-sync.test.ts +91 -0
  273. package/src/resources/extensions/gsd/tests/preferences.test.ts +1 -1
  274. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +77 -70
  275. package/src/resources/extensions/gsd/tests/remediation-completion-guard.test.ts +110 -0
  276. package/src/resources/extensions/gsd/tests/remote-questions.test.ts +29 -0
  277. package/src/resources/extensions/gsd/tests/terminated-transient.test.ts +42 -31
  278. package/src/resources/extensions/gsd/tests/token-cost-display.test.ts +2 -2
  279. package/src/resources/extensions/gsd/tests/vacuous-truth-slices.test.ts +115 -0
  280. package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +90 -0
  281. package/src/resources/extensions/gsd/tests/workflow-logger.test.ts +81 -1
  282. package/src/resources/extensions/gsd/tests/worktree-preferences-sync.test.ts +130 -0
  283. package/src/resources/extensions/gsd/tools/complete-milestone.ts +3 -14
  284. package/src/resources/extensions/gsd/tools/complete-slice.ts +3 -21
  285. package/src/resources/extensions/gsd/tools/complete-task.ts +9 -22
  286. package/src/resources/extensions/gsd/tools/plan-milestone.ts +28 -18
  287. package/src/resources/extensions/gsd/tools/plan-slice.ts +28 -16
  288. package/src/resources/extensions/gsd/tools/plan-task.ts +24 -12
  289. package/src/resources/extensions/gsd/tools/reassess-roadmap.ts +54 -42
  290. package/src/resources/extensions/gsd/tools/replan-slice.ts +53 -40
  291. package/src/resources/extensions/gsd/tools/validate-milestone.ts +26 -20
  292. package/src/resources/extensions/gsd/workflow-logger.ts +0 -1
  293. package/src/resources/extensions/remote-questions/config.ts +1 -1
  294. package/src/resources/extensions/remote-questions/remote-command.ts +1 -1
  295. package/src/resources/extensions/search-the-web/native-search.ts +1 -1
  296. package/src/resources/extensions/search-the-web/provider.ts +1 -1
  297. package/dist/web/standalone/.next/static/chunks/4024.9ad5def014d90ce4.js +0 -9
  298. package/dist/web/standalone/.next/static/css/de141508b083f922.css +0 -1
  299. /package/dist/resources/extensions/gsd/templates/{preferences.md → PREFERENCES.md} +0 -0
  300. /package/dist/web/standalone/.next/static/{yJIyd5cXPNpmXTv18ZlyC → RqOU-jOv9uZ1Q03P6L6nn}/_buildManifest.js +0 -0
  301. /package/dist/web/standalone/.next/static/{yJIyd5cXPNpmXTv18ZlyC → RqOU-jOv9uZ1Q03P6L6nn}/_ssgManifest.js +0 -0
  302. /package/src/resources/extensions/gsd/templates/{preferences.md → PREFERENCES.md} +0 -0
@@ -125,6 +125,9 @@ export function getActiveMemoriesRanked(limit = 30): Memory[] {
125
125
  /**
126
126
  * Generate the next memory ID: MEM + zero-padded 3-digit from MAX(seq).
127
127
  * Returns MEM001 if no memories exist.
128
+ *
129
+ * NOTE: For race-safe creation, prefer createMemory() which inserts with a
130
+ * placeholder ID then updates to the seq-derived ID atomically.
128
131
  */
129
132
  export function nextMemoryId(): string {
130
133
  if (!isDbAvailable()) return 'MEM001';
@@ -147,7 +150,9 @@ export function nextMemoryId(): string {
147
150
  // ─── Mutation Functions ─────────────────────────────────────────────────────
148
151
 
149
152
  /**
150
- * Insert a new memory with auto-assigned ID.
153
+ * Insert a new memory with a race-safe auto-assigned ID.
154
+ * Uses AUTOINCREMENT seq to derive the ID after insert, avoiding
155
+ * the read-then-write race in concurrent scenarios (e.g. worktrees).
151
156
  * Returns the assigned ID, or null on failure.
152
157
  */
153
158
  export function createMemory(fields: {
@@ -162,13 +167,14 @@ export function createMemory(fields: {
162
167
  if (!adapter) return null;
163
168
 
164
169
  try {
165
- const id = nextMemoryId();
166
170
  const now = new Date().toISOString();
171
+ // Insert with a temporary placeholder ID — seq is auto-assigned
172
+ const placeholder = `_TMP_${Date.now()}_${Math.random().toString(36).slice(2, 8)}`;
167
173
  adapter.prepare(
168
174
  `INSERT INTO memories (id, category, content, confidence, source_unit_type, source_unit_id, created_at, updated_at)
169
175
  VALUES (:id, :category, :content, :confidence, :source_unit_type, :source_unit_id, :created_at, :updated_at)`,
170
176
  ).run({
171
- ':id': id,
177
+ ':id': placeholder,
172
178
  ':category': fields.category,
173
179
  ':content': fields.content,
174
180
  ':confidence': fields.confidence ?? 0.8,
@@ -177,7 +183,16 @@ export function createMemory(fields: {
177
183
  ':created_at': now,
178
184
  ':updated_at': now,
179
185
  });
180
- return id;
186
+ // Derive the real ID from the assigned seq
187
+ const row = adapter.prepare('SELECT seq FROM memories WHERE id = :id').get({ ':id': placeholder });
188
+ if (!row) return placeholder; // fallback — should not happen
189
+ const seq = row['seq'] as number;
190
+ const realId = `MEM${String(seq).padStart(3, '0')}`;
191
+ adapter.prepare('UPDATE memories SET id = :real_id WHERE id = :placeholder').run({
192
+ ':real_id': realId,
193
+ ':placeholder': placeholder,
194
+ });
195
+ return realId;
181
196
  } catch {
182
197
  return null;
183
198
  }
@@ -331,20 +346,16 @@ export function enforceMemoryCap(max = 50): void {
331
346
  if (count <= max) return;
332
347
 
333
348
  const excess = count - max;
334
- // Find the IDs of the lowest-ranked active memories
335
- const rows = adapter.prepare(
336
- `SELECT id FROM memories
337
- WHERE superseded_by IS NULL
338
- ORDER BY (confidence * (1.0 + hit_count * 0.1)) ASC
339
- LIMIT :limit`,
340
- ).all({ ':limit': excess });
341
-
342
- const now = new Date().toISOString();
343
- for (const row of rows) {
344
- adapter.prepare(
345
- 'UPDATE memories SET superseded_by = :reason, updated_at = :now WHERE id = :id',
346
- ).run({ ':reason': 'CAP_EXCEEDED', ':now': now, ':id': row['id'] as string });
347
- }
349
+ // Batch update: supersede lowest-ranked active memories in a single statement
350
+ adapter.prepare(
351
+ `UPDATE memories SET superseded_by = 'CAP_EXCEEDED', updated_at = :now
352
+ WHERE id IN (
353
+ SELECT id FROM memories
354
+ WHERE superseded_by IS NULL
355
+ ORDER BY (confidence * (1.0 + hit_count * 0.1)) ASC
356
+ LIMIT :limit
357
+ )`,
358
+ ).run({ ':now': new Date().toISOString(), ':limit': excess });
348
359
  } catch {
349
360
  // non-fatal
350
361
  }
@@ -20,6 +20,7 @@ import {
20
20
  } from "./paths.js";
21
21
  import { invalidateAllCaches } from "./cache.js";
22
22
  import { loadQueueOrder, saveQueueOrder } from "./queue-order.js";
23
+ import { isDbAvailable, updateMilestoneStatus } from "./gsd-db.js";
23
24
 
24
25
  // ─── Park ──────────────────────────────────────────────────────────────────
25
26
 
@@ -52,6 +53,14 @@ export function parkMilestone(basePath: string, milestoneId: string, reason: str
52
53
  ].join("\n");
53
54
 
54
55
  writeFileSync(parkedPath, content, "utf-8");
56
+ // Sync DB status so deriveStateFromDb also skips this milestone (#2694)
57
+ if (isDbAvailable()) {
58
+ try {
59
+ updateMilestoneStatus(milestoneId, "parked");
60
+ } catch (err) {
61
+ process.stderr.write(`gsd: parkMilestone DB sync failed for ${milestoneId}: ${(err as Error).message}\n`);
62
+ }
63
+ }
55
64
  invalidateAllCaches();
56
65
  return true;
57
66
  }
@@ -70,6 +79,14 @@ export function unparkMilestone(basePath: string, milestoneId: string): boolean
70
79
  if (!existsSync(parkedPath)) return false; // not parked
71
80
 
72
81
  unlinkSync(parkedPath);
82
+ // Sync DB status so deriveStateFromDb picks up the unparked milestone (#2694)
83
+ if (isDbAvailable()) {
84
+ try {
85
+ updateMilestoneStatus(milestoneId, "active");
86
+ } catch (err) {
87
+ process.stderr.write(`gsd: unparkMilestone DB sync failed for ${milestoneId}: ${(err as Error).message}\n`);
88
+ }
89
+ }
73
90
  invalidateAllCaches();
74
91
  return true;
75
92
  }
@@ -125,18 +125,6 @@ export function getNextFallbackModel(
125
125
  }
126
126
  }
127
127
 
128
- /**
129
- * Detect whether an error message indicates a transient network error
130
- * (worth retrying the same model) vs a permanent provider error
131
- * (auth failure, quota exceeded, etc. -- should fall back immediately).
132
- */
133
- export function isTransientNetworkError(errorMsg: string): boolean {
134
- if (!errorMsg) return false;
135
- const hasNetworkSignal = /network|ECONNRESET|ETIMEDOUT|ECONNREFUSED|socket hang up|fetch failed|connection.*reset|dns/i.test(errorMsg);
136
- const hasPermanentSignal = /auth|unauthorized|forbidden|invalid.*key|quota|billing/i.test(errorMsg);
137
- return hasNetworkSignal && !hasPermanentSignal;
138
- }
139
-
140
128
  /**
141
129
  * Validate a model ID string.
142
130
  * Returns true if the ID looks like a valid model identifier.
@@ -308,7 +296,7 @@ export function resolveContextSelection(): import("./types.js").ContextSelection
308
296
  }
309
297
 
310
298
  /**
311
- * Resolve the search provider preference from preferences.md.
299
+ * Resolve the search provider preference from PREFERENCES.md.
312
300
  * Returns undefined if not configured (caller falls back to existing behavior).
313
301
  */
314
302
  export function resolveSearchProviderFromPreferences(): GSDPreferences["search_provider"] | undefined {
@@ -68,7 +68,6 @@ export {
68
68
  resolveModelForUnit,
69
69
  resolveModelWithFallbacksForUnit,
70
70
  getNextFallbackModel,
71
- isTransientNetworkError,
72
71
  validateModelId,
73
72
  updatePreferencesModels,
74
73
  resolveDynamicRoutingConfig,
@@ -87,7 +86,7 @@ function gsdHome(): string {
87
86
  }
88
87
 
89
88
  function globalPreferencesPath(): string {
90
- return join(gsdHome(), "preferences.md");
89
+ return join(gsdHome(), "PREFERENCES.md");
91
90
  }
92
91
 
93
92
  function legacyGlobalPreferencesPath(): string {
@@ -95,15 +94,15 @@ function legacyGlobalPreferencesPath(): string {
95
94
  }
96
95
 
97
96
  function projectPreferencesPath(): string {
98
- return join(gsdRoot(process.cwd()), "preferences.md");
97
+ return join(gsdRoot(process.cwd()), "PREFERENCES.md");
99
98
  }
100
- // Bootstrap in gitignore.ts historically created PREFERENCES.md (uppercase) by mistake.
101
- // Check uppercase as a fallback so those files aren't silently ignored.
102
- function globalPreferencesPathUppercase(): string {
103
- return join(gsdHome(), "PREFERENCES.md");
99
+ // Legacy: older versions used lowercase preferences.md.
100
+ // Check lowercase as a fallback so those files aren't silently ignored.
101
+ function globalPreferencesPathLegacy(): string {
102
+ return join(gsdHome(), "preferences.md");
104
103
  }
105
- function projectPreferencesPathUppercase(): string {
106
- return join(gsdRoot(process.cwd()), "PREFERENCES.md");
104
+ function projectPreferencesPathLegacy(): string {
105
+ return join(gsdRoot(process.cwd()), "preferences.md");
107
106
  }
108
107
 
109
108
  export function getGlobalGSDPreferencesPath(): string {
@@ -122,13 +121,13 @@ export function getProjectGSDPreferencesPath(): string {
122
121
 
123
122
  export function loadGlobalGSDPreferences(): LoadedGSDPreferences | null {
124
123
  return loadPreferencesFile(globalPreferencesPath(), "global")
125
- ?? loadPreferencesFile(globalPreferencesPathUppercase(), "global")
124
+ ?? loadPreferencesFile(globalPreferencesPathLegacy(), "global")
126
125
  ?? loadPreferencesFile(legacyGlobalPreferencesPath(), "global");
127
126
  }
128
127
 
129
128
  export function loadProjectGSDPreferences(): LoadedGSDPreferences | null {
130
129
  return loadPreferencesFile(projectPreferencesPath(), "project")
131
- ?? loadPreferencesFile(projectPreferencesPathUppercase(), "project");
130
+ ?? loadPreferencesFile(projectPreferencesPathLegacy(), "project");
132
131
  }
133
132
 
134
133
  export function loadEffectiveGSDPreferences(): LoadedGSDPreferences | null {
@@ -223,7 +222,7 @@ export function parsePreferencesMarkdown(content: string): GSDPreferences | null
223
222
 
224
223
  if (!_warnedUnrecognizedFormat) {
225
224
  _warnedUnrecognizedFormat = true;
226
- console.warn("[parsePreferencesMarkdown] preferences.md exists but uses an unrecognized format — skipping.");
225
+ console.warn("[parsePreferencesMarkdown] PREFERENCES.md exists but uses an unrecognized format — skipping.");
227
226
  }
228
227
  return null;
229
228
  }
@@ -502,7 +501,7 @@ export function resolvePreDispatchHooks(): PreDispatchHookConfig[] {
502
501
  * Resolve the effective git isolation mode from preferences.
503
502
  * Returns "none" (default), "worktree", or "branch".
504
503
  *
505
- * Default is "none" so GSD works out of the box without preferences.md.
504
+ * Default is "none" so GSD works out of the box without PREFERENCES.md.
506
505
  * Worktree isolation requires explicit opt-in because it depends on git
507
506
  * branch infrastructure that must be set up before use.
508
507
  */
@@ -92,7 +92,7 @@ Titles live inside file content (headings, frontmatter), not in file or director
92
92
 
93
93
  ### Isolation Model
94
94
 
95
- Auto-mode supports three isolation modes (configured in `.gsd/preferences.md` under `taskIsolation.mode`):
95
+ Auto-mode supports three isolation modes (configured in `.gsd/PREFERENCES.md` under `taskIsolation.mode`):
96
96
 
97
97
  - **worktree** (default): Work happens in `.gsd/worktrees/<MID>/`, a full git worktree on the `milestone/<MID>` branch. Each worktree has its own working copy and `.gsd/` directory. Squash-merged back to the integration branch on milestone completion.
98
98
  - **branch**: Work happens in the project root on a `milestone/<MID>` branch. No worktree directory — files are checked out in-place.
@@ -2,63 +2,6 @@ export type ProviderErrorPauseUI = {
2
2
  notify(message: string, level?: "info" | "warning" | "error" | "success"): void;
3
3
  };
4
4
 
5
- /**
6
- * Classify a provider error as transient (auto-resume) or permanent (manual resume).
7
- *
8
- * Transient: rate limits, server errors (500/502/503), overloaded, internal errors.
9
- * These are expected to self-resolve and should auto-resume after a delay.
10
- *
11
- * Permanent: auth errors, invalid API key, billing issues.
12
- * These require user intervention and should pause indefinitely.
13
- */
14
- export function classifyProviderError(errorMsg: string): {
15
- isTransient: boolean;
16
- isRateLimit: boolean;
17
- suggestedDelayMs: number;
18
- } {
19
- const isRateLimit = /rate.?limit|too many requests|429/i.test(errorMsg);
20
- const isServerError = /internal server error|500|502|503|overloaded|server_error|api_error|service.?unavailable/i.test(errorMsg);
21
-
22
- // Connection/process errors — transient, auto-resume after brief backoff (#2309).
23
- // These indicate the process was killed, the connection was reset, or a network
24
- // blip occurred. They are NOT permanent failures.
25
- const isConnectionError = /terminated|connection.?reset|connection.?refused|other side closed|fetch failed|network.?(?:is\s+)?unavailable|ECONNREFUSED|ECONNRESET|EPIPE/i.test(errorMsg);
26
-
27
- // Permanent errors — never auto-resume
28
- const isPermanent = /auth|unauthorized|forbidden|invalid.*key|invalid.*api|billing|quota exceeded|account/i.test(errorMsg);
29
-
30
- if (isPermanent && !isRateLimit) {
31
- return { isTransient: false, isRateLimit: false, suggestedDelayMs: 0 };
32
- }
33
-
34
- if (isRateLimit) {
35
- // Try to extract retry-after from the message
36
- const resetMatch = errorMsg.match(/reset in (\d+)s/i);
37
- const delayMs = resetMatch ? Number(resetMatch[1]) * 1000 : 60_000; // default 60s for rate limits
38
- return { isTransient: true, isRateLimit: true, suggestedDelayMs: delayMs };
39
- }
40
-
41
- if (isServerError) {
42
- return { isTransient: true, isRateLimit: false, suggestedDelayMs: 30_000 }; // 30s for server errors
43
- }
44
-
45
- if (isConnectionError) {
46
- return { isTransient: true, isRateLimit: false, suggestedDelayMs: 15_000 }; // 15s for connection errors
47
- }
48
-
49
- // Stream-truncation JSON parse errors — transient (#2572).
50
- // When the API stream is cut mid-chunk, pi tries to reassemble the partial
51
- // tool-call JSON and gets a SyntaxError. This is the downstream symptom of
52
- // a connection drop — same root cause as ECONNRESET, one layer up.
53
- const isMalformedStream = /Unexpected end of JSON|Unexpected token.*JSON|Expected double-quoted property name|SyntaxError.*JSON/i.test(errorMsg);
54
- if (isMalformedStream) {
55
- return { isTransient: true, isRateLimit: false, suggestedDelayMs: 15_000 }; // 15s, same as connection errors
56
- }
57
-
58
- // Unknown error — treat as permanent (user reviews)
59
- return { isTransient: false, isRateLimit: false, suggestedDelayMs: 0 };
60
- }
61
-
62
5
  /**
63
6
  * Pause auto-mode due to a provider error.
64
7
  *
@@ -524,7 +524,7 @@ export class RuleRegistry {
524
524
  formatHookStatus(): string {
525
525
  const entries = this.getHookStatus();
526
526
  if (entries.length === 0) {
527
- return "No hooks configured. Add post_unit_hooks or pre_dispatch_hooks to .gsd/preferences.md";
527
+ return "No hooks configured. Add post_unit_hooks or pre_dispatch_hooks to .gsd/PREFERENCES.md";
528
528
  }
529
529
 
530
530
  const lines: string[] = ["Configured Hooks:", ""];
@@ -27,15 +27,27 @@ const SERVICE_TIER_SCOPE_NOTE = "Only affects gpt-5.4 models, regardless of prov
27
27
 
28
28
  // ─── Gating ──────────────────────────────────────────────────────────────────
29
29
 
30
+ /**
31
+ * Model ID prefixes (bare, without provider) that support OpenAI service tiers.
32
+ *
33
+ * This list is the fallback for callers that only have a model ID string.
34
+ * The authoritative source of truth is `model.capabilities.supportsServiceTier`
35
+ * (set via CAPABILITY_PATCHES in packages/pi-ai/src/models.ts). When callers
36
+ * have access to the full Model object, prefer reading capabilities directly.
37
+ *
38
+ * See: https://github.com/gsd-build/gsd-2/issues/2546
39
+ */
40
+ const SERVICE_TIER_MODEL_PREFIXES = ["gpt-5.4"] as const;
41
+
30
42
  /**
31
43
  * Returns true when the given model ID supports OpenAI service tiers.
32
- * Currently only gpt-5.4 variants qualify.
44
+ * Reads from SERVICE_TIER_MODEL_PREFIXES update that list, not this function.
33
45
  */
34
46
  export function supportsServiceTier(modelId: string): boolean {
35
47
  if (!modelId) return false;
36
48
  // Strip provider prefix if present (e.g. "openai/gpt-5.4" → "gpt-5.4")
37
49
  const bare = modelId.includes("/") ? modelId.split("/").pop()! : modelId;
38
- return bare.startsWith("gpt-5.4");
50
+ return SERVICE_TIER_MODEL_PREFIXES.some((prefix) => bare.startsWith(prefix));
39
51
  }
40
52
 
41
53
  // ─── Status Formatting ───────────────────────────────────────────────────────
@@ -211,7 +211,24 @@ export async function deriveState(basePath: string): Promise<GSDState> {
211
211
 
212
212
  // Dual-path: try DB-backed derivation first when hierarchy tables are populated
213
213
  if (isDbAvailable()) {
214
- const dbMilestones = getAllMilestones();
214
+ let dbMilestones = getAllMilestones();
215
+
216
+ // Disk→DB reconciliation (#2631): when the milestones table is empty
217
+ // (e.g. failed initial migration per #2529), the reconciliation code
218
+ // inside deriveStateFromDb is unreachable. Populate from disk here so
219
+ // the DB path activates correctly.
220
+ if (dbMilestones.length === 0) {
221
+ const diskIds = findMilestoneIds(basePath);
222
+ let synced = false;
223
+ for (const diskId of diskIds) {
224
+ if (!isGhostMilestone(basePath, diskId)) {
225
+ insertMilestone({ id: diskId, status: 'active' });
226
+ synced = true;
227
+ }
228
+ }
229
+ if (synced) dbMilestones = getAllMilestones();
230
+ }
231
+
215
232
  if (dbMilestones.length > 0) {
216
233
  const stopDbTimer = debugTime("derive-state-db");
217
234
  result = await deriveStateFromDb(basePath);
@@ -562,7 +579,10 @@ export async function deriveStateFromDb(basePath: string): Promise<GSDState> {
562
579
  }
563
580
 
564
581
  // ── All slices done → validating/completing ─────────────────────────
565
- const allSlicesDone = activeMilestoneSlices.every(s => isStatusDone(s.status));
582
+ // Guard: [].every() === true (vacuous truth). Without the length check,
583
+ // an empty slice array causes a premature phase transition to
584
+ // validating-milestone. See: https://github.com/gsd-build/gsd-2/issues/2667
585
+ const allSlicesDone = activeMilestoneSlices.length > 0 && activeMilestoneSlices.every(s => isStatusDone(s.status));
566
586
  if (allSlicesDone) {
567
587
  const validationFile = resolveMilestoneFile(basePath, activeMilestone.id, "VALIDATION");
568
588
  const validationContent = validationFile ? await loadFile(validationFile) : null;
@@ -0,0 +1,61 @@
1
+ import { describe, it } from "node:test";
2
+ import assert from "node:assert/strict";
3
+
4
+ import { parseMilestoneTarget } from "../commands/handlers/auto.js";
5
+
6
+ describe("parseMilestoneTarget", () => {
7
+ it("extracts a simple milestone ID", () => {
8
+ const result = parseMilestoneTarget("auto M016");
9
+ assert.equal(result.milestoneId, "M016");
10
+ assert.equal(result.rest, "auto");
11
+ });
12
+
13
+ it("extracts a milestone ID with unique suffix", () => {
14
+ const result = parseMilestoneTarget("auto M001-a3b4c5 --verbose");
15
+ assert.equal(result.milestoneId, "M001-a3b4c5");
16
+ assert.equal(result.rest, "auto --verbose");
17
+ });
18
+
19
+ it("returns null when no milestone ID is present", () => {
20
+ const result = parseMilestoneTarget("auto --verbose");
21
+ assert.equal(result.milestoneId, null);
22
+ assert.equal(result.rest, "auto --verbose");
23
+ });
24
+
25
+ it("extracts milestone ID with flags in any order", () => {
26
+ const result = parseMilestoneTarget("auto --verbose M003 --debug");
27
+ assert.equal(result.milestoneId, "M003");
28
+ assert.equal(result.rest, "auto --verbose --debug");
29
+ });
30
+
31
+ it("returns null for plain 'auto'", () => {
32
+ const result = parseMilestoneTarget("auto");
33
+ assert.equal(result.milestoneId, null);
34
+ assert.equal(result.rest, "auto");
35
+ });
36
+
37
+ it("extracts from 'next' command", () => {
38
+ const result = parseMilestoneTarget("next M012");
39
+ assert.equal(result.milestoneId, "M012");
40
+ assert.equal(result.rest, "next");
41
+ });
42
+
43
+ it("handles milestone ID at the start of input", () => {
44
+ const result = parseMilestoneTarget("M007");
45
+ assert.equal(result.milestoneId, "M007");
46
+ assert.equal(result.rest, "");
47
+ });
48
+
49
+ it("picks the first milestone ID when multiple appear", () => {
50
+ // Edge case: user accidentally types two. First one wins.
51
+ const result = parseMilestoneTarget("auto M001 M002");
52
+ assert.equal(result.milestoneId, "M001");
53
+ // M002 remains in rest since only the first match is removed
54
+ assert.ok(result.rest.includes("M002"));
55
+ });
56
+
57
+ it("does not match bare numbers without M prefix", () => {
58
+ const result = parseMilestoneTarget("auto 016");
59
+ assert.equal(result.milestoneId, null);
60
+ });
61
+ });
@@ -0,0 +1,191 @@
1
+ /**
2
+ * Portable tests for marketplace discovery in claude-import.
3
+ *
4
+ * Validates that categorizePluginRoots correctly discovers marketplace repos
5
+ * nested inside container directories (the Claude Code convention), and that
6
+ * discoverClaudePlugins recognizes .claude-plugin/plugin.json in addition to
7
+ * package.json.
8
+ *
9
+ * Uses temp-dir fixtures — no real marketplace repos required.
10
+ *
11
+ * Fixes: https://github.com/gsd-build/gsd-2/issues/2717
12
+ */
13
+
14
+ import { describe, it, beforeEach, afterEach } from "node:test";
15
+ import assert from "node:assert/strict";
16
+ import { existsSync, mkdirSync, mkdtempSync, rmSync, writeFileSync } from "node:fs";
17
+ import { tmpdir } from "node:os";
18
+ import { join } from "node:path";
19
+ import { categorizePluginRoots } from "../claude-import.js";
20
+
21
+ describe("categorizePluginRoots", () => {
22
+ let tmpDir: string;
23
+
24
+ beforeEach(() => {
25
+ tmpDir = mkdtempSync(join(tmpdir(), "gsd-mktplace-test-"));
26
+ });
27
+
28
+ afterEach(() => {
29
+ rmSync(tmpDir, { recursive: true, force: true });
30
+ });
31
+
32
+ it("should detect a direct marketplace root", () => {
33
+ // Root itself has .claude-plugin/marketplace.json
34
+ mkdirSync(join(tmpDir, ".claude-plugin"), { recursive: true });
35
+ writeFileSync(
36
+ join(tmpDir, ".claude-plugin", "marketplace.json"),
37
+ JSON.stringify({ name: "direct", plugins: [] })
38
+ );
39
+
40
+ const { marketplaces, flat } = categorizePluginRoots([tmpDir]);
41
+
42
+ assert.equal(marketplaces.length, 1);
43
+ assert.equal(marketplaces[0], tmpDir);
44
+ assert.equal(flat.length, 0);
45
+ });
46
+
47
+ it("should discover marketplace repos nested one level inside a container directory", () => {
48
+ // Simulate ~/.claude/plugins/marketplaces/ with two marketplace subdirs
49
+ const mktA = join(tmpDir, "marketplace-a");
50
+ const mktB = join(tmpDir, "marketplace-b");
51
+
52
+ mkdirSync(join(mktA, ".claude-plugin"), { recursive: true });
53
+ writeFileSync(
54
+ join(mktA, ".claude-plugin", "marketplace.json"),
55
+ JSON.stringify({ name: "a", plugins: [] })
56
+ );
57
+
58
+ mkdirSync(join(mktB, ".claude-plugin"), { recursive: true });
59
+ writeFileSync(
60
+ join(mktB, ".claude-plugin", "marketplace.json"),
61
+ JSON.stringify({ name: "b", plugins: [] })
62
+ );
63
+
64
+ const { marketplaces, flat } = categorizePluginRoots([tmpDir]);
65
+
66
+ assert.equal(marketplaces.length, 2);
67
+ assert.ok(marketplaces.includes(mktA));
68
+ assert.ok(marketplaces.includes(mktB));
69
+ assert.equal(flat.length, 0);
70
+ });
71
+
72
+ it("should fall back to flat when no child is a marketplace", () => {
73
+ // Container with no marketplace subdirs
74
+ mkdirSync(join(tmpDir, "some-dir"), { recursive: true });
75
+
76
+ const { marketplaces, flat } = categorizePluginRoots([tmpDir]);
77
+
78
+ assert.equal(marketplaces.length, 0);
79
+ assert.equal(flat.length, 1);
80
+ assert.equal(flat[0], tmpDir);
81
+ });
82
+
83
+ it("should handle a mix of direct marketplace and container roots", () => {
84
+ // Root A is a direct marketplace
85
+ const directRoot = join(tmpDir, "direct");
86
+ mkdirSync(join(directRoot, ".claude-plugin"), { recursive: true });
87
+ writeFileSync(
88
+ join(directRoot, ".claude-plugin", "marketplace.json"),
89
+ JSON.stringify({ name: "direct", plugins: [] })
90
+ );
91
+
92
+ // Root B is a container with a child marketplace
93
+ const container = join(tmpDir, "container");
94
+ const child = join(container, "child-marketplace");
95
+ mkdirSync(join(child, ".claude-plugin"), { recursive: true });
96
+ writeFileSync(
97
+ join(child, ".claude-plugin", "marketplace.json"),
98
+ JSON.stringify({ name: "child", plugins: [] })
99
+ );
100
+
101
+ // Root C has nothing
102
+ const emptyRoot = join(tmpDir, "empty");
103
+ mkdirSync(emptyRoot, { recursive: true });
104
+
105
+ const { marketplaces, flat } = categorizePluginRoots([
106
+ directRoot,
107
+ container,
108
+ emptyRoot,
109
+ ]);
110
+
111
+ assert.equal(marketplaces.length, 2);
112
+ assert.ok(marketplaces.includes(directRoot));
113
+ assert.ok(marketplaces.includes(child));
114
+ assert.equal(flat.length, 1);
115
+ assert.equal(flat[0], emptyRoot);
116
+ });
117
+
118
+ it("should not duplicate when the same marketplace appears via multiple roots", () => {
119
+ // Direct reference AND container reference to the same marketplace
120
+ const mkt = join(tmpDir, "mkt");
121
+ mkdirSync(join(mkt, ".claude-plugin"), { recursive: true });
122
+ writeFileSync(
123
+ join(mkt, ".claude-plugin", "marketplace.json"),
124
+ JSON.stringify({ name: "mkt", plugins: [] })
125
+ );
126
+
127
+ const { marketplaces } = categorizePluginRoots([mkt, tmpDir]);
128
+
129
+ assert.equal(marketplaces.length, 1);
130
+ assert.equal(marketplaces[0], mkt);
131
+ });
132
+
133
+ it("should skip .git and node_modules subdirectories", () => {
134
+ // Put a marketplace.json inside .git — should be ignored
135
+ mkdirSync(join(tmpDir, ".git", ".claude-plugin"), { recursive: true });
136
+ writeFileSync(
137
+ join(tmpDir, ".git", ".claude-plugin", "marketplace.json"),
138
+ JSON.stringify({ name: "hidden", plugins: [] })
139
+ );
140
+
141
+ const { marketplaces, flat } = categorizePluginRoots([tmpDir]);
142
+
143
+ assert.equal(marketplaces.length, 0);
144
+ assert.equal(flat.length, 1);
145
+ });
146
+
147
+ it("should handle non-existent root gracefully", () => {
148
+ const missing = join(tmpDir, "does-not-exist");
149
+ // categorizePluginRoots receives paths from uniqueExistingDirs, but
150
+ // be defensive — it should not crash on a missing root
151
+ const { marketplaces, flat } = categorizePluginRoots([missing]);
152
+
153
+ assert.equal(marketplaces.length, 0);
154
+ assert.equal(flat.length, 1); // falls through to flat
155
+ });
156
+ });
157
+
158
+ describe("discoverClaudePlugins — Claude plugin.json recognition", () => {
159
+ let tmpDir: string;
160
+
161
+ beforeEach(() => {
162
+ tmpDir = mkdtempSync(join(tmpdir(), "gsd-plugin-disc-"));
163
+ });
164
+
165
+ afterEach(() => {
166
+ rmSync(tmpDir, { recursive: true, force: true });
167
+ });
168
+
169
+ it("should discover a plugin with .claude-plugin/plugin.json (no package.json)", async () => {
170
+ // Simulate a cached Claude marketplace plugin
171
+ const pluginDir = join(tmpDir, "my-plugin");
172
+ mkdirSync(join(pluginDir, ".claude-plugin"), { recursive: true });
173
+ mkdirSync(join(pluginDir, "skills", "my-skill"), { recursive: true });
174
+ writeFileSync(
175
+ join(pluginDir, ".claude-plugin", "plugin.json"),
176
+ JSON.stringify({ name: "my-plugin", version: "1.0.0", description: "Test plugin" })
177
+ );
178
+ writeFileSync(join(pluginDir, "skills", "my-skill", "SKILL.md"), "# My Skill");
179
+
180
+ // Import discoverClaudePlugins dynamically since it depends on getClaudeSearchRoots
181
+ // which uses hardcoded paths. Instead, test the flat-path discovery logic directly
182
+ // by checking that the plugin.json file is recognized.
183
+ const claudePluginPath = join(pluginDir, ".claude-plugin", "plugin.json");
184
+ assert.ok(existsSync(claudePluginPath), "Claude plugin.json should exist");
185
+
186
+ // The fix ensures walkDirs checks for .claude-plugin/plugin.json in addition
187
+ // to package.json. We verify the file structure is correct for discovery.
188
+ const pkgPath = join(pluginDir, "package.json");
189
+ assert.ok(!existsSync(pkgPath), "package.json should NOT exist — this is a Claude plugin");
190
+ });
191
+ });
@@ -126,7 +126,7 @@ describe(
126
126
 
127
127
  before(() => {
128
128
  tempDir = mkdtempSync(join(tmpdir(), 'gsd-tui-test-'));
129
- prefsPath = join(tempDir, 'preferences.md');
129
+ prefsPath = join(tempDir, 'PREFERENCES.md');
130
130
  prefs = { version: 1 };
131
131
  });
132
132
 
@@ -0,0 +1,24 @@
1
+ import test from "node:test";
2
+ import assert from "node:assert/strict";
3
+ import { readFileSync } from "node:fs";
4
+ import { dirname, join } from "node:path";
5
+ import { fileURLToPath } from "node:url";
6
+
7
+ const __filename = fileURLToPath(import.meta.url);
8
+ const __dirname = dirname(__filename);
9
+
10
+ test("commands-config source-level: tool key lookup skips empty api_key entries", () => {
11
+ const source = readFileSync(join(__dirname, "..", "commands-config.ts"), "utf-8");
12
+ assert.ok(
13
+ source.includes('getCredentialsForProvider(providerId)'),
14
+ "commands-config should read the full credential list",
15
+ );
16
+ assert.ok(
17
+ source.includes('c.type === "api_key" && c.key'),
18
+ "commands-config should require a non-empty api_key when resolving stored tool keys",
19
+ );
20
+ assert.ok(
21
+ !source.includes("auth.get(tool.id)"),
22
+ "commands-config should not rely on auth.get(tool.id), which can return an empty shadowing entry",
23
+ );
24
+ });