gsd-pi 2.63.0-dev.026d309 → 2.63.0-dev.351157b

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 (312) hide show
  1. package/README.md +46 -134
  2. package/dist/cli.js +44 -6
  3. package/dist/help-text.js +4 -1
  4. package/dist/onboarding.js +15 -8
  5. package/dist/resource-loader.js +18 -3
  6. package/dist/resources/extensions/cmux/index.js +21 -12
  7. package/dist/resources/extensions/gsd/auto/finalize-timeout.js +40 -0
  8. package/dist/resources/extensions/gsd/auto/loop.js +4 -0
  9. package/dist/resources/extensions/gsd/auto/phases.js +123 -22
  10. package/dist/resources/extensions/gsd/auto/session.js +8 -0
  11. package/dist/resources/extensions/gsd/auto-dashboard.js +9 -3
  12. package/dist/resources/extensions/gsd/auto-post-unit.js +45 -10
  13. package/dist/resources/extensions/gsd/auto-prompts.js +25 -0
  14. package/dist/resources/extensions/gsd/auto-recovery.js +15 -7
  15. package/dist/resources/extensions/gsd/auto-start.js +10 -21
  16. package/dist/resources/extensions/gsd/auto-tool-tracking.js +17 -0
  17. package/dist/resources/extensions/gsd/auto-worktree.js +13 -7
  18. package/dist/resources/extensions/gsd/auto.js +19 -2
  19. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +73 -60
  20. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +13 -0
  21. package/dist/resources/extensions/gsd/bootstrap/query-tools.js +85 -0
  22. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +3 -0
  23. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +9 -1
  24. package/dist/resources/extensions/gsd/bootstrap/sanitize-complete-milestone.js +54 -0
  25. package/dist/resources/extensions/gsd/commands-handlers.js +9 -4
  26. package/dist/resources/extensions/gsd/constants.js +42 -0
  27. package/dist/resources/extensions/gsd/db-writer.js +72 -4
  28. package/dist/resources/extensions/gsd/forensics.js +20 -4
  29. package/dist/resources/extensions/gsd/gsd-db.js +64 -17
  30. package/dist/resources/extensions/gsd/guided-flow.js +19 -0
  31. package/dist/resources/extensions/gsd/metrics.js +27 -1
  32. package/dist/resources/extensions/gsd/native-git-bridge.js +5 -3
  33. package/dist/resources/extensions/gsd/preferences-types.js +1 -0
  34. package/dist/resources/extensions/gsd/preferences.js +7 -2
  35. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +2 -0
  36. package/dist/resources/extensions/gsd/prompts/complete-slice.md +2 -0
  37. package/dist/resources/extensions/gsd/prompts/doctor-heal.md +1 -0
  38. package/dist/resources/extensions/gsd/prompts/forensics.md +2 -0
  39. package/dist/resources/extensions/gsd/prompts/reassess-roadmap.md +2 -0
  40. package/dist/resources/extensions/gsd/prompts/system.md +1 -0
  41. package/dist/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
  42. package/dist/resources/extensions/gsd/roadmap-mutations.js +1 -1
  43. package/dist/resources/extensions/gsd/roadmap-slices.js +9 -5
  44. package/dist/resources/extensions/gsd/slice-parallel-conflict.js +67 -0
  45. package/dist/resources/extensions/gsd/slice-parallel-eligibility.js +51 -0
  46. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +378 -0
  47. package/dist/resources/extensions/gsd/state.js +74 -14
  48. package/dist/resources/extensions/gsd/status-guards.js +11 -0
  49. package/dist/resources/extensions/gsd/tools/complete-milestone.js +17 -12
  50. package/dist/resources/extensions/gsd/tools/complete-slice.js +40 -26
  51. package/dist/resources/extensions/gsd/tools/complete-task.js +12 -12
  52. package/dist/resources/extensions/gsd/tools/plan-milestone.js +33 -25
  53. package/dist/resources/extensions/gsd/tools/plan-slice.js +5 -8
  54. package/dist/resources/extensions/gsd/workflow-projections.js +21 -5
  55. package/dist/resources/extensions/gsd/worktree-manager.js +82 -29
  56. package/dist/resources/extensions/gsd/worktree-resolver.js +4 -3
  57. package/dist/resources/extensions/mcp-client/auth.js +101 -0
  58. package/dist/resources/extensions/mcp-client/index.js +10 -1
  59. package/dist/resources/extensions/ollama/index.js +6 -12
  60. package/dist/resources/extensions/ollama/model-capabilities.js +37 -34
  61. package/dist/resources/extensions/ollama/ndjson-stream.js +54 -0
  62. package/dist/resources/extensions/ollama/ollama-chat-provider.js +380 -0
  63. package/dist/resources/extensions/ollama/ollama-client.js +23 -32
  64. package/dist/resources/extensions/ollama/ollama-discovery.js +2 -7
  65. package/dist/resources/extensions/ollama/ollama-tool.js +62 -0
  66. package/dist/resources/extensions/ollama/thinking-parser.js +104 -0
  67. package/dist/web/standalone/.next/BUILD_ID +1 -1
  68. package/dist/web/standalone/.next/app-path-routes-manifest.json +14 -14
  69. package/dist/web/standalone/.next/build-manifest.json +2 -2
  70. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  71. package/dist/web/standalone/.next/server/app/_global-error.html +2 -2
  72. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  73. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  74. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  75. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  76. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  77. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  78. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  79. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  80. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  81. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  82. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  83. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  85. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  86. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  87. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  88. package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
  89. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  90. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
  91. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  92. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
  93. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  94. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
  95. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  96. package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
  97. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  98. package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
  99. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  100. package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
  101. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  102. package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
  103. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  104. package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
  105. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  106. package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
  107. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  108. package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
  109. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  110. package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
  111. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  112. package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
  113. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  114. package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
  115. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  116. package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
  117. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  118. package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
  119. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  120. package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
  121. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  122. package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
  123. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  124. package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
  125. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  126. package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
  127. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  128. package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
  129. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  130. package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
  131. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  132. package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
  133. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  134. package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
  135. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  136. package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
  137. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  138. package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
  139. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  140. package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
  141. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
  142. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
  143. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +2 -2
  144. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
  145. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  146. package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
  147. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  148. package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
  149. package/dist/web/standalone/.next/server/app/index.html +1 -1
  150. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  151. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  152. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  153. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  154. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  155. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  156. package/dist/web/standalone/.next/server/app-paths-manifest.json +14 -14
  157. package/dist/web/standalone/.next/server/chunks/6897.js +12 -0
  158. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  159. package/dist/web/standalone/.next/server/pages/500.html +2 -2
  160. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  161. package/package.json +1 -1
  162. package/packages/pi-agent-core/dist/agent-loop.d.ts +8 -0
  163. package/packages/pi-agent-core/dist/agent-loop.d.ts.map +1 -1
  164. package/packages/pi-agent-core/dist/agent-loop.js +50 -0
  165. package/packages/pi-agent-core/dist/agent-loop.js.map +1 -1
  166. package/packages/pi-agent-core/src/agent-loop.test.ts +221 -5
  167. package/packages/pi-agent-core/src/agent-loop.ts +53 -0
  168. package/packages/pi-ai/dist/types.d.ts +16 -1
  169. package/packages/pi-ai/dist/types.d.ts.map +1 -1
  170. package/packages/pi-ai/dist/types.js.map +1 -1
  171. package/packages/pi-ai/src/types.ts +18 -1
  172. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +9 -0
  173. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
  174. package/packages/pi-coding-agent/dist/core/auth-storage.js +50 -1
  175. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  176. package/packages/pi-coding-agent/dist/core/auth-storage.test.js +41 -0
  177. package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
  178. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +7 -0
  179. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  180. package/packages/pi-coding-agent/dist/core/extensions/loader.js +31 -4
  181. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  182. package/packages/pi-coding-agent/dist/core/extensions/loader.test.js +28 -1
  183. package/packages/pi-coding-agent/dist/core/extensions/loader.test.js.map +1 -1
  184. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +2 -0
  185. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
  186. package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
  187. package/packages/pi-coding-agent/dist/core/model-registry.d.ts +1 -0
  188. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  189. package/packages/pi-coding-agent/dist/core/model-registry.js +1 -0
  190. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  191. package/packages/pi-coding-agent/dist/core/model-resolver.js +3 -3
  192. package/packages/pi-coding-agent/dist/core/model-resolver.js.map +1 -1
  193. package/packages/pi-coding-agent/dist/core/resource-loader.d.ts +23 -1
  194. package/packages/pi-coding-agent/dist/core/resource-loader.d.ts.map +1 -1
  195. package/packages/pi-coding-agent/dist/core/resource-loader.js +80 -56
  196. package/packages/pi-coding-agent/dist/core/resource-loader.js.map +1 -1
  197. package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
  198. package/packages/pi-coding-agent/dist/core/sdk.js +10 -0
  199. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  200. package/packages/pi-coding-agent/src/core/auth-storage.test.ts +53 -0
  201. package/packages/pi-coding-agent/src/core/auth-storage.ts +66 -1
  202. package/packages/pi-coding-agent/src/core/extensions/loader.test.ts +39 -1
  203. package/packages/pi-coding-agent/src/core/extensions/loader.ts +34 -4
  204. package/packages/pi-coding-agent/src/core/extensions/types.ts +2 -0
  205. package/packages/pi-coding-agent/src/core/model-registry.ts +2 -0
  206. package/packages/pi-coding-agent/src/core/model-resolver.ts +3 -3
  207. package/packages/pi-coding-agent/src/core/resource-loader.ts +89 -56
  208. package/packages/pi-coding-agent/src/core/sdk.ts +11 -0
  209. package/src/resources/extensions/cmux/index.ts +18 -12
  210. package/src/resources/extensions/gsd/auto/finalize-timeout.ts +46 -0
  211. package/src/resources/extensions/gsd/auto/loop.ts +5 -0
  212. package/src/resources/extensions/gsd/auto/phases.ts +156 -34
  213. package/src/resources/extensions/gsd/auto/session.ts +9 -0
  214. package/src/resources/extensions/gsd/auto-dashboard.ts +11 -3
  215. package/src/resources/extensions/gsd/auto-post-unit.ts +53 -12
  216. package/src/resources/extensions/gsd/auto-prompts.ts +21 -0
  217. package/src/resources/extensions/gsd/auto-recovery.ts +9 -8
  218. package/src/resources/extensions/gsd/auto-start.ts +11 -20
  219. package/src/resources/extensions/gsd/auto-tool-tracking.ts +19 -0
  220. package/src/resources/extensions/gsd/auto-worktree.ts +14 -6
  221. package/src/resources/extensions/gsd/auto.ts +22 -1
  222. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +74 -60
  223. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +15 -0
  224. package/src/resources/extensions/gsd/bootstrap/query-tools.ts +98 -0
  225. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +4 -0
  226. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +9 -1
  227. package/src/resources/extensions/gsd/bootstrap/sanitize-complete-milestone.ts +57 -0
  228. package/src/resources/extensions/gsd/commands-handlers.ts +10 -4
  229. package/src/resources/extensions/gsd/constants.ts +44 -0
  230. package/src/resources/extensions/gsd/db-writer.ts +78 -4
  231. package/src/resources/extensions/gsd/forensics.ts +21 -5
  232. package/src/resources/extensions/gsd/gsd-db.ts +64 -17
  233. package/src/resources/extensions/gsd/guided-flow.ts +22 -0
  234. package/src/resources/extensions/gsd/metrics.ts +28 -1
  235. package/src/resources/extensions/gsd/native-git-bridge.ts +5 -3
  236. package/src/resources/extensions/gsd/preferences-types.ts +3 -0
  237. package/src/resources/extensions/gsd/preferences.ts +9 -2
  238. package/src/resources/extensions/gsd/prompts/complete-milestone.md +2 -0
  239. package/src/resources/extensions/gsd/prompts/complete-slice.md +2 -0
  240. package/src/resources/extensions/gsd/prompts/doctor-heal.md +1 -0
  241. package/src/resources/extensions/gsd/prompts/forensics.md +2 -0
  242. package/src/resources/extensions/gsd/prompts/reassess-roadmap.md +2 -0
  243. package/src/resources/extensions/gsd/prompts/system.md +1 -0
  244. package/src/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
  245. package/src/resources/extensions/gsd/roadmap-mutations.ts +1 -1
  246. package/src/resources/extensions/gsd/roadmap-slices.ts +10 -5
  247. package/src/resources/extensions/gsd/slice-parallel-conflict.ts +86 -0
  248. package/src/resources/extensions/gsd/slice-parallel-eligibility.ts +73 -0
  249. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +477 -0
  250. package/src/resources/extensions/gsd/state.ts +67 -12
  251. package/src/resources/extensions/gsd/status-guards.ts +13 -0
  252. package/src/resources/extensions/gsd/tests/artifact-corruption-2630.test.ts +288 -0
  253. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +34 -13
  254. package/src/resources/extensions/gsd/tests/cmux.test.ts +58 -0
  255. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +51 -0
  256. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +140 -0
  257. package/src/resources/extensions/gsd/tests/complete-task.test.ts +39 -0
  258. package/src/resources/extensions/gsd/tests/dashboard-model-label-ordering.test.ts +107 -0
  259. package/src/resources/extensions/gsd/tests/db-access-guardrails.test.ts +109 -0
  260. package/src/resources/extensions/gsd/tests/db-path-worktree-symlink.test.ts +13 -9
  261. package/src/resources/extensions/gsd/tests/db-writer.test.ts +134 -0
  262. package/src/resources/extensions/gsd/tests/deferred-slice-dispatch.test.ts +203 -0
  263. package/src/resources/extensions/gsd/tests/discuss-tool-scoping.test.ts +130 -0
  264. package/src/resources/extensions/gsd/tests/doctor-fix-flag.test.ts +92 -0
  265. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +116 -0
  266. package/src/resources/extensions/gsd/tests/forensics-stuck-loops.test.ts +103 -0
  267. package/src/resources/extensions/gsd/tests/insert-slice-no-wipe.test.ts +88 -0
  268. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +27 -7
  269. package/src/resources/extensions/gsd/tests/integration/idle-recovery.test.ts +34 -0
  270. package/src/resources/extensions/gsd/tests/metrics.test.ts +116 -1
  271. package/src/resources/extensions/gsd/tests/milestone-status-tool.test.ts +201 -0
  272. package/src/resources/extensions/gsd/tests/plan-milestone-title.test.ts +2 -1
  273. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +82 -18
  274. package/src/resources/extensions/gsd/tests/preferences.test.ts +10 -0
  275. package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +25 -0
  276. package/src/resources/extensions/gsd/tests/roadmap-slices.test.ts +69 -0
  277. package/src/resources/extensions/gsd/tests/shared-wal.test.ts +30 -0
  278. package/src/resources/extensions/gsd/tests/slice-context-injection.test.ts +50 -0
  279. package/src/resources/extensions/gsd/tests/slice-parallel-conflict.test.ts +92 -0
  280. package/src/resources/extensions/gsd/tests/slice-parallel-eligibility.test.ts +95 -0
  281. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +83 -0
  282. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +103 -0
  283. package/src/resources/extensions/gsd/tests/tool-param-optionality.test.ts +349 -0
  284. package/src/resources/extensions/gsd/tests/worktree-health-dispatch.test.ts +35 -2
  285. package/src/resources/extensions/gsd/tests/worktree-health-monorepo.test.ts +73 -0
  286. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +34 -0
  287. package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +1 -1
  288. package/src/resources/extensions/gsd/tests/worktree-teardown-safety.test.ts +148 -0
  289. package/src/resources/extensions/gsd/tools/complete-milestone.ts +34 -20
  290. package/src/resources/extensions/gsd/tools/complete-slice.ts +41 -26
  291. package/src/resources/extensions/gsd/tools/complete-task.ts +12 -12
  292. package/src/resources/extensions/gsd/tools/plan-milestone.ts +55 -30
  293. package/src/resources/extensions/gsd/tools/plan-slice.ts +13 -8
  294. package/src/resources/extensions/gsd/types.ts +44 -22
  295. package/src/resources/extensions/gsd/workflow-projections.ts +23 -5
  296. package/src/resources/extensions/gsd/worktree-manager.ts +76 -28
  297. package/src/resources/extensions/gsd/worktree-resolver.ts +4 -3
  298. package/src/resources/extensions/mcp-client/auth.ts +149 -0
  299. package/src/resources/extensions/mcp-client/index.ts +16 -1
  300. package/src/resources/extensions/ollama/index.ts +6 -14
  301. package/src/resources/extensions/ollama/model-capabilities.ts +41 -34
  302. package/src/resources/extensions/ollama/ndjson-stream.ts +63 -0
  303. package/src/resources/extensions/ollama/ollama-chat-provider.ts +459 -0
  304. package/src/resources/extensions/ollama/ollama-client.ts +30 -30
  305. package/src/resources/extensions/ollama/ollama-discovery.ts +5 -8
  306. package/src/resources/extensions/ollama/ollama-tool.ts +69 -0
  307. package/src/resources/extensions/ollama/tests/ollama-discovery.test.ts +0 -27
  308. package/src/resources/extensions/ollama/thinking-parser.ts +116 -0
  309. package/src/resources/extensions/ollama/types.ts +23 -0
  310. package/dist/web/standalone/.next/server/chunks/2229.js +0 -12
  311. /package/dist/web/standalone/.next/static/{TTlAguZQ5vR9EOv6G8cel → QmuF-eAbuU_2MQ03t38qr}/_buildManifest.js +0 -0
  312. /package/dist/web/standalone/.next/static/{TTlAguZQ5vR9EOv6G8cel → QmuF-eAbuU_2MQ03t38qr}/_ssgManifest.js +0 -0
@@ -0,0 +1,88 @@
1
+ import test from 'node:test';
2
+ import assert from 'node:assert/strict';
3
+
4
+ import { openDatabase, closeDatabase, insertMilestone, insertSlice, getSlice } from '../gsd-db.ts';
5
+
6
+ test('insertSlice with minimal args does not wipe populated fields', (t) => {
7
+ t.after(() => { try { closeDatabase(); } catch { /* noop */ } });
8
+ openDatabase(":memory:");
9
+
10
+ insertMilestone({ id: 'M001', title: 'Milestone', status: 'active' });
11
+
12
+ // First insert: full data
13
+ insertSlice({
14
+ id: 'S01',
15
+ milestoneId: 'M001',
16
+ title: 'Auth flow',
17
+ status: 'in-progress',
18
+ risk: 'high',
19
+ demo: 'Login page renders.',
20
+ sequence: 3,
21
+ planning: {
22
+ goal: 'Secure authentication',
23
+ successCriteria: 'All tests pass',
24
+ proofLevel: 'integration',
25
+ integrationClosure: 'Fully integrated',
26
+ observabilityImpact: 'Metrics available',
27
+ },
28
+ });
29
+
30
+ const before = getSlice('M001', 'S01');
31
+ assert.ok(before, 'slice should exist after first insert');
32
+ assert.equal(before.title, 'Auth flow');
33
+ assert.equal(before.demo, 'Login page renders.');
34
+ assert.equal(before.risk, 'high');
35
+
36
+ // Second insert: minimal "ensure exists" call (mirrors complete-task.ts usage)
37
+ insertSlice({ id: 'S01', milestoneId: 'M001' });
38
+
39
+ const after = getSlice('M001', 'S01');
40
+ assert.ok(after, 'slice should still exist after second insert');
41
+
42
+ // These must NOT be wiped to empty strings
43
+ assert.equal(after.title, 'Auth flow', 'title must survive minimal re-insert');
44
+ assert.equal(after.demo, 'Login page renders.', 'demo must survive minimal re-insert');
45
+ assert.equal(after.risk, 'high', 'risk must survive minimal re-insert');
46
+ assert.equal(after.sequence, 3, 'sequence must survive minimal re-insert');
47
+
48
+ // Planning fields must also survive
49
+ assert.equal(after.goal, 'Secure authentication', 'goal must survive minimal re-insert');
50
+ assert.equal(after.success_criteria, 'All tests pass', 'success_criteria must survive');
51
+ assert.equal(after.proof_level, 'integration', 'proof_level must survive');
52
+ assert.equal(after.integration_closure, 'Fully integrated', 'integration_closure must survive');
53
+ assert.equal(after.observability_impact, 'Metrics available', 'observability_impact must survive');
54
+ });
55
+
56
+ test('insertSlice ON CONFLICT preserves completed status', (t) => {
57
+ t.after(() => { try { closeDatabase(); } catch { /* noop */ } });
58
+ openDatabase(":memory:");
59
+
60
+ insertMilestone({ id: 'M001', title: 'Milestone', status: 'active' });
61
+
62
+ insertSlice({ id: 'S01', milestoneId: 'M001', title: 'Done slice', status: 'complete' });
63
+
64
+ // Re-insert with pending status (default) should NOT overwrite complete
65
+ insertSlice({ id: 'S01', milestoneId: 'M001' });
66
+
67
+ const after = getSlice('M001', 'S01');
68
+ assert.ok(after);
69
+ assert.equal(after.status, 'complete', 'completed status must not be overwritten');
70
+ });
71
+
72
+ test('insertSlice ON CONFLICT allows explicit updates to non-empty values', (t) => {
73
+ t.after(() => { try { closeDatabase(); } catch { /* noop */ } });
74
+ openDatabase(":memory:");
75
+
76
+ insertMilestone({ id: 'M001', title: 'Milestone', status: 'active' });
77
+
78
+ insertSlice({ id: 'S01', milestoneId: 'M001', title: 'Original', demo: 'Old demo', risk: 'low' });
79
+
80
+ // Explicit update with real values should overwrite
81
+ insertSlice({ id: 'S01', milestoneId: 'M001', title: 'Updated', demo: 'New demo', risk: 'high' });
82
+
83
+ const after = getSlice('M001', 'S01');
84
+ assert.ok(after);
85
+ assert.equal(after.title, 'Updated', 'explicit title update should apply');
86
+ assert.equal(after.demo, 'New demo', 'explicit demo update should apply');
87
+ assert.equal(after.risk, 'high', 'explicit risk update should apply');
88
+ });
@@ -1246,7 +1246,7 @@ describe('git-service', async () => {
1246
1246
  test('nativeAddAllWithExclusions: symlinked .gsd fallback', () => {
1247
1247
  // When .gsd is a symlink, git rejects `:!.gsd/...` pathspecs with
1248
1248
  // "fatal: pathspec '...' is beyond a symbolic link". The fix falls
1249
- // back to plain `git add -A`, which respects .gitignore.
1249
+ // back to `git add -u` (tracked files only), NOT `git add -A`.
1250
1250
  const repo = initTempRepo();
1251
1251
 
1252
1252
  // Create the real .gsd directory outside the repo, then symlink it
@@ -1258,11 +1258,18 @@ describe('git-service', async () => {
1258
1258
  // Symlink .gsd -> external directory
1259
1259
  symlinkSync(externalGsd, join(repo, ".gsd"));
1260
1260
 
1261
- // Add .gitignore so git add -A fallback skips .gsd/
1261
+ // Add .gitignore so .gsd/ is ignored
1262
1262
  writeFileSync(join(repo, ".gitignore"), ".gsd\n");
1263
1263
 
1264
- // Create a real file that should be staged
1264
+ // Create a tracked file and commit it, then modify it
1265
1265
  createFile(repo, "src/app.ts", "export const x = 1;");
1266
+ run("git add -A", repo);
1267
+ run('git commit -m "add app"', repo);
1268
+ writeFileSync(join(repo, "src/app.ts"), "export const x = 2;");
1269
+
1270
+ // Create an untracked file simulating large data (NOT in .gitignore)
1271
+ // This is the key scenario: large untracked dirs that git add -A would traverse
1272
+ createFile(repo, "data/large-model.bin", "pretend this is 10GB");
1266
1273
 
1267
1274
  // nativeAddAllWithExclusions should NOT throw despite .gsd being a symlink
1268
1275
  let threw = false;
@@ -1274,9 +1281,15 @@ describe('git-service', async () => {
1274
1281
  }
1275
1282
  assert.ok(!threw, "nativeAddAllWithExclusions does not throw with symlinked .gsd");
1276
1283
 
1277
- // Verify the real file was staged
1284
+ // Verify the tracked modified file was staged
1278
1285
  const staged = run("git diff --cached --name-only", repo);
1279
- assert.ok(staged.includes("src/app.ts"), "real file staged despite symlinked .gsd");
1286
+ assert.ok(staged.includes("src/app.ts"), "modified tracked file staged despite symlinked .gsd");
1287
+
1288
+ // CRITICAL: untracked files must NOT be staged — the symlink fallback
1289
+ // should use `git add -u` (tracked only), not `git add -A` (all files).
1290
+ // Using `git add -A` on a repo with large untracked data dirs hangs. (#1977)
1291
+ assert.ok(!staged.includes("data/large-model.bin"),
1292
+ "symlink fallback must not stage untracked files (would hang on large repos)");
1280
1293
  assert.ok(!staged.includes(".gsd"), ".gsd content not staged");
1281
1294
 
1282
1295
  rmSync(repo, { recursive: true, force: true });
@@ -1435,13 +1448,20 @@ describe('git-service', async () => {
1435
1448
  run('git add .gitignore', repo);
1436
1449
  run('git commit -m "add gitignore"', repo);
1437
1450
 
1451
+ // Pre-commit a tracked source file so git add -u can stage modifications.
1452
+ // The symlink fallback uses git add -u (tracked files only), so the file
1453
+ // must be tracked before the autoCommit scenario runs.
1454
+ createFile(repo, "src/feature.ts", "export const feature = true;");
1455
+ run('git add src/feature.ts', repo);
1456
+ run('git commit -m "add feature"', repo);
1457
+
1438
1458
  // Simulate new milestone artifacts created during execution
1439
1459
  writeFileSync(join(externalGsd, "milestones", "M009", "M009-SUMMARY.md"), "# M009 Summary");
1440
1460
  writeFileSync(join(externalGsd, "milestones", "M009", "S01-SUMMARY.md"), "# S01 Summary");
1441
1461
  writeFileSync(join(externalGsd, "milestones", "M009", "T01-VERIFY.json"), '{"passed":true}');
1442
1462
 
1443
- // Also create a normal source file change
1444
- createFile(repo, "src/feature.ts", "export const feature = true;");
1463
+ // Modify the tracked source file — git add -u will stage this change
1464
+ writeFileSync(join(repo, "src/feature.ts"), "export const feature = false; // updated");
1445
1465
 
1446
1466
  const svc = new GitServiceImpl(repo);
1447
1467
  const msg = svc.autoCommit("complete-milestone", "M009");
@@ -357,3 +357,37 @@ test('writeBlockerPlaceholder: does NOT update DB for non-execute-task types', a
357
357
  cleanup(base);
358
358
  }
359
359
  });
360
+
361
+ test('writeBlockerPlaceholder: updates DB slice status for complete-slice (#2653)', async () => {
362
+ const base = createFixtureBase();
363
+ try {
364
+ const { openDatabase, closeDatabase, insertMilestone, insertSlice, getSlice, isDbAvailable } =
365
+ await import("../../gsd-db.ts");
366
+
367
+ const dbPath = join(base, ".gsd", "gsd.db");
368
+ mkdirSync(join(base, ".gsd", "milestones", "M001", "slices", "S01"), { recursive: true });
369
+
370
+ openDatabase(dbPath);
371
+ try {
372
+ insertMilestone({ id: "M001", title: "Test", status: "active" });
373
+ insertSlice({ id: "S01", milestoneId: "M001", title: "Slice", status: "active" });
374
+
375
+ // complete-slice blocker should update slice DB status to "complete"
376
+ writeBlockerPlaceholder("complete-slice", "M001/S01", base, "context exhaustion recovery");
377
+
378
+ const slice = getSlice("M001", "S01");
379
+ assert.equal(slice?.status, "complete",
380
+ "writeBlockerPlaceholder must update DB slice status to 'complete' for complete-slice so dispatch guard unblocks downstream (#2653)");
381
+
382
+ // Verify the full chain works: verifyExpectedArtifact should return true
383
+ // (requires both UAT file and DB status = complete)
384
+ // Note: the placeholder writes a SUMMARY file, but complete-slice also needs UAT.
385
+ // The placeholder itself doesn't write UAT, so artifact verification may still fail
386
+ // for complete-slice — but the DB status is now correct, breaking the circular dep.
387
+ } finally {
388
+ if (isDbAvailable()) closeDatabase();
389
+ }
390
+ } finally {
391
+ cleanup(base);
392
+ }
393
+ });
@@ -6,7 +6,7 @@
6
6
 
7
7
  import test from "node:test";
8
8
  import assert from "node:assert/strict";
9
- import { mkdtempSync, mkdirSync, readFileSync, rmSync } from "node:fs";
9
+ import { mkdtempSync, mkdirSync, readFileSync, rmSync, writeFileSync } from "node:fs";
10
10
  import { join } from "node:path";
11
11
  import { tmpdir } from "node:os";
12
12
  import {
@@ -382,3 +382,118 @@ test("snapshotUnitMetrics counts toolCall blocks correctly (#1713)", () => {
382
382
  rmSync(tmpBase, { recursive: true, force: true });
383
383
  }
384
384
  });
385
+
386
+ // ── #1943 — Duplicate metrics entries from idle watchdog ──────────────────────
387
+
388
+ test("#1943 initMetrics deduplicates entries loaded from a corrupted disk ledger", () => {
389
+ const tmpBase = mkdtempSync(join(tmpdir(), "gsd-metrics-dedup-load-"));
390
+ mkdirSync(join(tmpBase, ".gsd"), { recursive: true });
391
+
392
+ try {
393
+ resetMetrics();
394
+
395
+ // Simulate a corrupted metrics.json with duplicate entries on disk
396
+ // (same type+id+startedAt but different finishedAt — idle watchdog pattern)
397
+ const corruptedLedger: MetricsLedger = {
398
+ version: 1,
399
+ projectStartedAt: 1700000000000,
400
+ units: [
401
+ makeUnit({ type: "research-slice", id: "M009/S02", startedAt: 1774011016218, finishedAt: 1774011031218, cost: 1.50, tokens: { input: 6600000, output: 100000, cacheRead: 0, cacheWrite: 0, total: 6700000 } }),
402
+ makeUnit({ type: "research-slice", id: "M009/S02", startedAt: 1774011016218, finishedAt: 1774011046218, cost: 1.55, tokens: { input: 6800000, output: 110000, cacheRead: 0, cacheWrite: 0, total: 6910000 } }),
403
+ makeUnit({ type: "research-slice", id: "M009/S02", startedAt: 1774011016218, finishedAt: 1774011061218, cost: 1.60, tokens: { input: 7000000, output: 120000, cacheRead: 0, cacheWrite: 0, total: 7120000 } }),
404
+ makeUnit({ type: "research-slice", id: "M009/S02", startedAt: 1774011016218, finishedAt: 1774011076218, cost: 1.65, tokens: { input: 7200000, output: 130000, cacheRead: 0, cacheWrite: 0, total: 7330000 } }),
405
+ // A different unit — should be preserved
406
+ makeUnit({ type: "execute-task", id: "M001/S01/T01", startedAt: 1774012000000, finishedAt: 1774012060000, cost: 0.50 }),
407
+ ],
408
+ };
409
+ writeFileSync(
410
+ join(tmpBase, ".gsd", "metrics.json"),
411
+ JSON.stringify(corruptedLedger, null, 2),
412
+ );
413
+
414
+ // Load the corrupted ledger — duplicates should be collapsed on load
415
+ initMetrics(tmpBase);
416
+ const ledger = getLedger();
417
+ assert.ok(ledger);
418
+
419
+ // The 4 entries with identical (type, id, startedAt) should collapse to 1,
420
+ // keeping the latest (highest finishedAt). Plus the 1 different unit = 2 total.
421
+ assert.equal(
422
+ ledger!.units.length, 2,
423
+ `expected 2 entries after dedup (1 collapsed group + 1 unique), got ${ledger!.units.length}`,
424
+ );
425
+
426
+ // The surviving duplicate should be the one with the latest finishedAt
427
+ const researchEntry = ledger!.units.find(u => u.type === "research-slice");
428
+ assert.ok(researchEntry);
429
+ assert.equal(researchEntry!.finishedAt, 1774011076218, "should keep the latest finishedAt");
430
+ assert.equal(researchEntry!.cost, 1.65, "should keep the latest cost");
431
+
432
+ // The on-disk file should also be deduplicated
433
+ const diskRaw = readFileSync(join(tmpBase, ".gsd", "metrics.json"), "utf-8");
434
+ const diskLedger: MetricsLedger = JSON.parse(diskRaw);
435
+ assert.equal(diskLedger.units.length, 2, "disk should also have deduplicated entries");
436
+ } finally {
437
+ resetMetrics();
438
+ rmSync(tmpBase, { recursive: true, force: true });
439
+ }
440
+ });
441
+
442
+ test("#1943 getProjectTotals reports correct cost after dedup (no 35% inflation)", () => {
443
+ // Simulate the exact scenario from the issue: 20 entries for a single dispatch
444
+ // with monotonically increasing token counts and 15s-apart finishedAt values
445
+ const startedAt = 1774011016218;
446
+ const baseCost = 1.50;
447
+ const duplicateUnits: UnitMetrics[] = [];
448
+
449
+ for (let i = 0; i < 20; i++) {
450
+ duplicateUnits.push(makeUnit({
451
+ type: "research-slice",
452
+ id: "M009/S02",
453
+ startedAt,
454
+ finishedAt: startedAt + (i + 1) * 15000,
455
+ cost: baseCost + i * 0.05,
456
+ toolCalls: 0,
457
+ tokens: {
458
+ input: 6600000 + i * 200000,
459
+ output: 100000 + i * 10000,
460
+ cacheRead: 0,
461
+ cacheWrite: 0,
462
+ total: 6700000 + i * 210000,
463
+ },
464
+ }));
465
+ }
466
+
467
+ // Without dedup, getProjectTotals would sum all 20 entries' costs
468
+ const rawTotals = getProjectTotals(duplicateUnits);
469
+ // With dedup (only last entry should count), cost should be the last entry's cost
470
+ const lastEntryCost = duplicateUnits[duplicateUnits.length - 1].cost;
471
+
472
+ // This test documents the bug: raw totals inflate cost by summing duplicates
473
+ assert.ok(
474
+ rawTotals.cost > lastEntryCost * 2,
475
+ "raw totals with duplicates inflate cost (bug demonstration)",
476
+ );
477
+
478
+ // After loading through initMetrics (which should dedup), totals should be correct
479
+ const tmpBase = mkdtempSync(join(tmpdir(), "gsd-metrics-cost-inflation-"));
480
+ mkdirSync(join(tmpBase, ".gsd"), { recursive: true });
481
+ try {
482
+ resetMetrics();
483
+ writeFileSync(
484
+ join(tmpBase, ".gsd", "metrics.json"),
485
+ JSON.stringify({ version: 1, projectStartedAt: 1700000000000, units: duplicateUnits }, null, 2),
486
+ );
487
+ initMetrics(tmpBase);
488
+ const ledger = getLedger()!;
489
+ const dedupedTotals = getProjectTotals(ledger.units);
490
+ assert.equal(ledger.units.length, 1, "20 duplicates should collapse to 1 entry");
491
+ assert.equal(
492
+ dedupedTotals.cost, lastEntryCost,
493
+ `deduped cost should be ${lastEntryCost}, not ${dedupedTotals.cost}`,
494
+ );
495
+ } finally {
496
+ resetMetrics();
497
+ rmSync(tmpBase, { recursive: true, force: true });
498
+ }
499
+ });
@@ -0,0 +1,201 @@
1
+ // GSD2 — Tests for gsd_milestone_status read-only query tool
2
+
3
+ import test from "node:test";
4
+ import assert from "node:assert/strict";
5
+ import { mkdirSync, rmSync } from "node:fs";
6
+ import { join } from "node:path";
7
+ import { tmpdir } from "node:os";
8
+ import { randomUUID } from "node:crypto";
9
+
10
+ import { registerQueryTools } from "../bootstrap/query-tools.ts";
11
+ import {
12
+ openDatabase,
13
+ closeDatabase,
14
+ _getAdapter,
15
+ } from "../gsd-db.ts";
16
+
17
+ // ─── Helpers ──────────────────────────────────────────────────────────────────
18
+
19
+ function makeMockPi() {
20
+ const tools: any[] = [];
21
+ return {
22
+ registerTool: (tool: any) => tools.push(tool),
23
+ tools,
24
+ } as any;
25
+ }
26
+
27
+ function makeTmpBase(): string {
28
+ const base = join(tmpdir(), `gsd-query-tool-test-${randomUUID()}`);
29
+ mkdirSync(join(base, ".gsd"), { recursive: true });
30
+ return base;
31
+ }
32
+
33
+ function cleanup(base: string): void {
34
+ try { rmSync(base, { recursive: true, force: true }); } catch { /* swallow */ }
35
+ }
36
+
37
+ function openTestDb(base: string): void {
38
+ openDatabase(join(base, ".gsd", "gsd.db"));
39
+ }
40
+
41
+ async function executeToolInDir(tool: any, params: Record<string, unknown>, dir: string) {
42
+ const originalCwd = process.cwd();
43
+ try {
44
+ process.chdir(dir);
45
+ return await tool.execute("test-call-id", params, undefined, undefined, undefined);
46
+ } finally {
47
+ process.chdir(originalCwd);
48
+ }
49
+ }
50
+
51
+ // ─── Seed helpers ─────────────────────────────────────────────────────────────
52
+
53
+ function seedMilestone(milestoneId: string, title: string, status = "active"): void {
54
+ const db = _getAdapter();
55
+ if (!db) throw new Error("DB not open");
56
+ db.prepare(
57
+ "INSERT OR REPLACE INTO milestones (id, title, status, created_at) VALUES (?, ?, ?, ?)",
58
+ ).run(milestoneId, title, status, new Date().toISOString());
59
+ }
60
+
61
+ function seedSlice(milestoneId: string, sliceId: string, status: string): void {
62
+ const db = _getAdapter();
63
+ if (!db) throw new Error("DB not open");
64
+ db.prepare(
65
+ "INSERT OR REPLACE INTO slices (milestone_id, id, title, status, created_at) VALUES (?, ?, ?, ?, ?)",
66
+ ).run(milestoneId, sliceId, `Slice ${sliceId}`, status, new Date().toISOString());
67
+ }
68
+
69
+ function seedTask(milestoneId: string, sliceId: string, taskId: string, status: string): void {
70
+ const db = _getAdapter();
71
+ if (!db) throw new Error("DB not open");
72
+ db.prepare(
73
+ "INSERT OR REPLACE INTO tasks (milestone_id, slice_id, id, title, status) VALUES (?, ?, ?, ?, ?)",
74
+ ).run(milestoneId, sliceId, taskId, `Task ${taskId}`, status);
75
+ }
76
+
77
+ // ─── Registration ─────────────────────────────────────────────────────────────
78
+
79
+ test("registerQueryTools registers gsd_milestone_status tool", () => {
80
+ const pi = makeMockPi();
81
+ registerQueryTools(pi);
82
+ assert.equal(pi.tools.length, 1, "Should register exactly one tool");
83
+ assert.equal(pi.tools[0].name, "gsd_milestone_status");
84
+ });
85
+
86
+ test("gsd_milestone_status has promptGuidelines mentioning prohibited alternatives", () => {
87
+ const pi = makeMockPi();
88
+ registerQueryTools(pi);
89
+ const tool = pi.tools[0];
90
+ assert.ok(Array.isArray(tool.promptGuidelines), "promptGuidelines must be an array");
91
+ assert.ok(tool.promptGuidelines.length >= 1, "Must have at least one guideline");
92
+ const joined = tool.promptGuidelines.join(" ");
93
+ assert.match(joined, /sqlite3|better-sqlite3/, "Guidelines must mention prohibited alternatives");
94
+ });
95
+
96
+ // ─── Happy path: milestone with slices and tasks ──────────────────────────────
97
+
98
+ test("gsd_milestone_status returns milestone metadata and slice statuses", async () => {
99
+ const base = makeTmpBase();
100
+ try {
101
+ openTestDb(base);
102
+ seedMilestone("M001", "Test Milestone");
103
+ seedSlice("M001", "S01", "complete");
104
+ seedSlice("M001", "S02", "active");
105
+ seedTask("M001", "S01", "T01", "done");
106
+ seedTask("M001", "S01", "T02", "done");
107
+ seedTask("M001", "S02", "T01", "pending");
108
+
109
+ const pi = makeMockPi();
110
+ registerQueryTools(pi);
111
+ const tool = pi.tools[0];
112
+
113
+ const result = await executeToolInDir(tool, { milestoneId: "M001" }, base);
114
+ const parsed = JSON.parse(result.content[0].text);
115
+
116
+ assert.equal(parsed.milestoneId, "M001");
117
+ assert.equal(parsed.title, "Test Milestone");
118
+ assert.equal(parsed.status, "active");
119
+ assert.equal(parsed.sliceCount, 2);
120
+ assert.equal(parsed.slices.length, 2);
121
+
122
+ const s01 = parsed.slices.find((s: any) => s.id === "S01");
123
+ assert.ok(s01, "S01 should be in slices");
124
+ assert.equal(s01.status, "complete");
125
+ assert.equal(s01.taskCounts.total, 2);
126
+ assert.equal(s01.taskCounts.done, 2);
127
+
128
+ const s02 = parsed.slices.find((s: any) => s.id === "S02");
129
+ assert.ok(s02, "S02 should be in slices");
130
+ assert.equal(s02.status, "active");
131
+ assert.equal(s02.taskCounts.pending, 1);
132
+ } finally {
133
+ closeDatabase();
134
+ cleanup(base);
135
+ }
136
+ });
137
+
138
+ // ─── Milestone with no slices ─────────────────────────────────────────────────
139
+
140
+ test("gsd_milestone_status returns empty slices array for milestone with no slices", async () => {
141
+ const base = makeTmpBase();
142
+ try {
143
+ openTestDb(base);
144
+ seedMilestone("M002", "Empty Milestone");
145
+
146
+ const pi = makeMockPi();
147
+ registerQueryTools(pi);
148
+ const tool = pi.tools[0];
149
+
150
+ const result = await executeToolInDir(tool, { milestoneId: "M002" }, base);
151
+ const parsed = JSON.parse(result.content[0].text);
152
+
153
+ assert.equal(parsed.milestoneId, "M002");
154
+ assert.equal(parsed.sliceCount, 0);
155
+ assert.deepEqual(parsed.slices, []);
156
+ } finally {
157
+ closeDatabase();
158
+ cleanup(base);
159
+ }
160
+ });
161
+
162
+ // ─── Missing milestone ────────────────────────────────────────────────────────
163
+
164
+ test("gsd_milestone_status returns not-found for missing milestone", async () => {
165
+ const base = makeTmpBase();
166
+ try {
167
+ openTestDb(base);
168
+
169
+ const pi = makeMockPi();
170
+ registerQueryTools(pi);
171
+ const tool = pi.tools[0];
172
+
173
+ const result = await executeToolInDir(tool, { milestoneId: "M999" }, base);
174
+ assert.match(result.content[0].text, /M999.*not found/i);
175
+ assert.equal(result.details.found, false);
176
+ } finally {
177
+ closeDatabase();
178
+ cleanup(base);
179
+ }
180
+ });
181
+
182
+ // ─── DB unavailable ───────────────────────────────────────────────────────────
183
+
184
+ test("gsd_milestone_status handles missing DB gracefully", async () => {
185
+ // Create a directory without .gsd/ to ensure ensureDbOpen has nothing to open
186
+ const base = join(tmpdir(), `gsd-no-db-${randomUUID()}`);
187
+ mkdirSync(base, { recursive: true });
188
+ closeDatabase(); // ensure no prior DB is open
189
+ try {
190
+ const pi = makeMockPi();
191
+ registerQueryTools(pi);
192
+ const tool = pi.tools[0];
193
+
194
+ const result = await executeToolInDir(tool, { milestoneId: "M001" }, base);
195
+ assert.match(result.content[0].text, /GSD database is not available/);
196
+ assert.equal(result.details.error, "db_unavailable");
197
+ } finally {
198
+ closeDatabase();
199
+ cleanup(base);
200
+ }
201
+ });
@@ -38,8 +38,9 @@ test("upsertMilestonePlanning updates title when DB row pre-exists with empty ti
38
38
 
39
39
  // Step 3: upsertMilestonePlanning should update the title
40
40
  upsertMilestonePlanning("M099", {
41
+ title: "My Important Milestone",
41
42
  vision: "Test vision",
42
- }, "My Important Milestone");
43
+ });
43
44
  const afterUpsert = getMilestone("M099");
44
45
  assert.ok(afterUpsert);
45
46
  assert.equal(
@@ -4,7 +4,7 @@ import { mkdtempSync, mkdirSync, rmSync, readFileSync, existsSync, writeFileSync
4
4
  import { join } from 'node:path';
5
5
  import { tmpdir } from 'node:os';
6
6
 
7
- import { openDatabase, closeDatabase, getMilestone, getMilestoneSlices, updateSliceStatus } from '../gsd-db.ts';
7
+ import { openDatabase, closeDatabase, getMilestone, getMilestoneSlices, getSlice, updateSliceStatus, deleteSlice, insertMilestone } from '../gsd-db.ts';
8
8
  import { handlePlanMilestone } from '../tools/plan-milestone.ts';
9
9
  import { parseRoadmap } from '../parsers-legacy.ts';
10
10
 
@@ -198,33 +198,97 @@ test('handlePlanMilestone reruns idempotently and updates existing planning stat
198
198
  }
199
199
  });
200
200
 
201
- // Regression: #2960 plan-milestone must refuse to overwrite completed slices
202
- test('handlePlanMilestone refuses to re-plan a milestone with completed slices (#2960)', async () => {
201
+ test('handlePlanMilestone preserves completed slice status on re-plan (#2558)', async () => {
203
202
  const base = makeTmpBase();
204
203
  const dbPath = join(base, '.gsd', 'gsd.db');
205
204
  openDatabase(dbPath);
206
205
 
207
206
  try {
208
- // First plan succeeds
207
+ // Initial plan — both slices start as "pending"
209
208
  const first = await handlePlanMilestone(validParams(), base);
210
- assert.ok(!('error' in first), `initial plan should succeed: ${'error' in first ? first.error : ''}`);
209
+ assert.ok(!('error' in first), `unexpected error: ${'error' in first ? first.error : ''}`);
211
210
 
212
- // Mark S01 as complete
213
- updateSliceStatus('M001', 'S01', 'complete');
211
+ // Mark S01 as complete (simulates work done in a worktree)
212
+ updateSliceStatus('M001', 'S01', 'complete', new Date().toISOString());
214
213
 
215
- // Second plan should fail — S01 is already complete
216
- const second = await handlePlanMilestone({
214
+ const s01Before = getSlice('M001', 'S01');
215
+ assert.equal(s01Before?.status, 'complete', 'S01 should be complete before re-plan');
216
+
217
+ // Re-plan the same milestone — S01 must stay "complete", S02 stays "pending"
218
+ const second = await handlePlanMilestone(validParams(), base);
219
+ assert.ok(!('error' in second), `unexpected error: ${'error' in second ? second.error : ''}`);
220
+
221
+ const s01After = getSlice('M001', 'S01');
222
+ assert.equal(s01After?.status, 'complete', 'S01 status must be preserved as complete after re-plan');
223
+
224
+ const s02After = getSlice('M001', 'S02');
225
+ assert.equal(s02After?.status, 'pending', 'S02 should remain pending');
226
+ } finally {
227
+ cleanup(base);
228
+ }
229
+ });
230
+
231
+ test('plan-milestone re-plan preserves completed status and updates slice fields (#2558)', async () => {
232
+ const base = makeTmpBase();
233
+ const dbPath = join(base, '.gsd', 'gsd.db');
234
+ openDatabase(dbPath);
235
+
236
+ try {
237
+ // Initial plan — both slices start as "pending"
238
+ const first = await handlePlanMilestone(validParams(), base);
239
+ assert.ok(!('error' in first), `unexpected error: ${'error' in first ? first.error : ''}`);
240
+
241
+ // Mark S01 as complete (simulates work done in worktree, then reconciled)
242
+ updateSliceStatus('M001', 'S01', 'complete', new Date().toISOString());
243
+ assert.equal(getSlice('M001', 'S01')?.status, 'complete');
244
+
245
+ // Re-plan with updated title for S01.
246
+ // The handler must:
247
+ // 1. NOT downgrade S01 from "complete" to "pending"
248
+ // 2. Update S01's non-status fields (title, risk, depends, demo)
249
+ // 3. Keep S02 as "pending"
250
+ const updatedParams = {
217
251
  ...validParams(),
218
- vision: 'Should not overwrite',
219
- }, base);
220
- assert.ok('error' in second, 'should refuse to re-plan when slices are completed');
221
- assert.match(second.error, /cannot re-plan/i);
222
- assert.match(second.error, /S01/);
252
+ slices: [
253
+ { ...validParams().slices[0], title: 'Updated S01 title', risk: 'high' },
254
+ validParams().slices[1],
255
+ ],
256
+ };
257
+ const second = await handlePlanMilestone(updatedParams, base);
258
+ assert.ok(!('error' in second), `unexpected error: ${'error' in second ? second.error : ''}`);
223
259
 
224
- // Verify the completed slice was not overwritten
225
- const slices = getMilestoneSlices('M001');
226
- const s01 = slices.find(s => s.id === 'S01');
227
- assert.equal(s01?.status, 'complete', 'S01 should still be complete');
260
+ const s01After = getSlice('M001', 'S01');
261
+ assert.equal(s01After?.status, 'complete', 'completed slice status must survive re-plan');
262
+ assert.equal(s01After?.title, 'Updated S01 title', 'title should update on re-plan');
263
+ assert.equal(s01After?.risk, 'high', 'risk should update on re-plan');
264
+
265
+ const s02After = getSlice('M001', 'S02');
266
+ assert.equal(s02After?.status, 'pending', 'pending slice stays pending');
267
+ } finally {
268
+ cleanup(base);
269
+ }
270
+ });
271
+
272
+ test('handlePlanMilestone promotes pre-existing queued milestone to active (#3022)', async () => {
273
+ const base = makeTmpBase();
274
+ const dbPath = join(base, '.gsd', 'gsd.db');
275
+ openDatabase(dbPath);
276
+
277
+ try {
278
+ // Simulate ensureMilestoneDbRow: pre-create row with status "queued"
279
+ // (this is what gsd_milestone_generate_id does)
280
+ insertMilestone({ id: 'M001', status: 'queued' });
281
+
282
+ const before = getMilestone('M001');
283
+ assert.equal(before?.status, 'queued', 'pre-condition: milestone should start as queued');
284
+
285
+ // Now plan the milestone — status should be promoted to "active"
286
+ const result = await handlePlanMilestone(validParams(), base);
287
+ assert.ok(!('error' in result), `unexpected error: ${'error' in result ? result.error : ''}`);
288
+
289
+ const after = getMilestone('M001');
290
+ assert.equal(after?.status, 'active', 'milestone status should be promoted from queued to active');
291
+ assert.equal(after?.title, 'DB-backed planning', 'milestone title should be set');
228
292
  } finally {
229
293
  cleanup(base);
230
294
  }