gsd-pi 2.78.0-dev.aeeb2ca00 → 2.78.1-dev.84a383f51

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 (383) hide show
  1. package/README.md +7 -7
  2. package/dist/claude-cli-check.js +64 -37
  3. package/dist/cli-policy.d.ts +13 -0
  4. package/dist/cli-policy.js +17 -0
  5. package/dist/cli.js +95 -55
  6. package/dist/headless-query.d.ts +22 -0
  7. package/dist/headless-query.js +24 -4
  8. package/dist/headless.d.ts +10 -0
  9. package/dist/headless.js +16 -1
  10. package/dist/loader.js +7 -10
  11. package/dist/onboarding.d.ts +10 -0
  12. package/dist/onboarding.js +2 -2
  13. package/dist/provider-migrations.d.ts +2 -2
  14. package/dist/provider-migrations.js +5 -2
  15. package/dist/resource-loader.d.ts +5 -2
  16. package/dist/resource-loader.js +28 -5
  17. package/dist/resources/.managed-resources-content-hash +1 -0
  18. package/dist/resources/extensions/claude-code-cli/readiness.js +77 -45
  19. package/dist/resources/extensions/gsd/auto/loop.js +23 -0
  20. package/dist/resources/extensions/gsd/auto/phases.js +2 -2
  21. package/dist/resources/extensions/gsd/auto/run-unit.js +3 -1
  22. package/dist/resources/extensions/gsd/auto/session.js +3 -0
  23. package/dist/resources/extensions/gsd/auto-recovery.js +43 -4
  24. package/dist/resources/extensions/gsd/auto-runtime-state.js +31 -0
  25. package/dist/resources/extensions/gsd/auto-start.js +1 -1
  26. package/dist/resources/extensions/gsd/auto-tool-tracking.js +2 -2
  27. package/dist/resources/extensions/gsd/auto-worktree.js +30 -0
  28. package/dist/resources/extensions/gsd/auto.js +14 -5
  29. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +14 -2
  30. package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +7 -5
  31. package/dist/resources/extensions/gsd/bootstrap/query-tools.js +2 -2
  32. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +5 -4
  33. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +94 -31
  34. package/dist/resources/extensions/gsd/bootstrap/register-shortcuts.js +11 -6
  35. package/dist/resources/extensions/gsd/bootstrap/system-context.js +34 -8
  36. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +38 -2
  37. package/dist/resources/extensions/gsd/commands/catalog.js +69 -5
  38. package/dist/resources/extensions/gsd/commands/handlers/core.js +22 -1
  39. package/dist/resources/extensions/gsd/commands-mcp-status.js +3 -1
  40. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +10 -1
  41. package/dist/resources/extensions/gsd/dashboard-overlay.js +1 -1
  42. package/dist/resources/extensions/gsd/docs/preferences-reference.md +4 -0
  43. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +39 -1
  44. package/dist/resources/extensions/gsd/error-classifier.js +1 -1
  45. package/dist/resources/extensions/gsd/forensics.js +2 -2
  46. package/dist/resources/extensions/gsd/git-service.js +12 -5
  47. package/dist/resources/extensions/gsd/gsd-db.js +11 -2
  48. package/dist/resources/extensions/gsd/guided-flow.js +23 -23
  49. package/dist/resources/extensions/gsd/memory-store.js +66 -31
  50. package/dist/resources/extensions/gsd/milestone-id-reservation.js +36 -0
  51. package/dist/resources/extensions/gsd/model-router.js +114 -9
  52. package/dist/resources/extensions/gsd/native-git-bridge.js +7 -1
  53. package/dist/resources/extensions/gsd/preferences-models.js +91 -15
  54. package/dist/resources/extensions/gsd/preferences-types.js +2 -0
  55. package/dist/resources/extensions/gsd/preferences-validation.js +32 -0
  56. package/dist/resources/extensions/gsd/preferences.js +5 -3
  57. package/dist/resources/extensions/gsd/prompt-loader.js +23 -12
  58. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +9 -3
  59. package/dist/resources/extensions/gsd/state.js +42 -0
  60. package/dist/resources/extensions/gsd/templates/PREFERENCES.md +1 -0
  61. package/dist/resources/extensions/gsd/tools/memory-tools.js +18 -1
  62. package/dist/resources/extensions/gsd/visualizer-overlay.js +1 -1
  63. package/dist/resources/extensions/gsd/watch/header-renderer.js +3 -1
  64. package/dist/resources/extensions/gsd/worktree-command.js +26 -46
  65. package/dist/resources/extensions/gsd/worktree-session-state.js +33 -0
  66. package/dist/resources/extensions/mcp-client/index.js +6 -3
  67. package/dist/resources/extensions/slash-commands/create-extension.js +36 -22
  68. package/dist/resources/skills/create-gsd-extension/SKILL.md +9 -5
  69. package/dist/resources/skills/create-gsd-extension/references/custom-commands.md +1 -1
  70. package/dist/resources/skills/create-gsd-extension/references/custom-rendering.md +5 -5
  71. package/dist/resources/skills/create-gsd-extension/references/custom-tools.md +4 -4
  72. package/dist/resources/skills/create-gsd-extension/references/custom-ui.md +6 -6
  73. package/dist/resources/skills/create-gsd-extension/references/events-reference.md +3 -3
  74. package/dist/resources/skills/create-gsd-extension/references/packaging-distribution.md +1 -1
  75. package/dist/resources/skills/create-gsd-extension/references/remote-execution-overrides.md +3 -3
  76. package/dist/resources/skills/create-gsd-extension/workflows/create-extension.md +32 -12
  77. package/dist/rtk-shared.d.ts +3 -0
  78. package/dist/rtk-shared.js +17 -0
  79. package/dist/rtk.d.ts +2 -5
  80. package/dist/rtk.js +3 -20
  81. package/dist/runtime-checks.d.ts +27 -0
  82. package/dist/runtime-checks.js +38 -0
  83. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  84. package/dist/web/standalone/.next/BUILD_ID +1 -1
  85. package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
  86. package/dist/web/standalone/.next/build-manifest.json +3 -3
  87. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  88. package/dist/web/standalone/.next/react-loadable-manifest.json +44 -4
  89. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  90. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  91. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  92. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  93. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  94. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  95. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  96. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  97. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  98. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  99. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  100. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  102. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  103. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  104. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  105. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  106. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  107. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  108. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  109. package/dist/web/standalone/.next/server/app/api/session/events/route.js +4 -2
  110. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  111. package/dist/web/standalone/.next/server/app/index.html +1 -1
  112. package/dist/web/standalone/.next/server/app/index.rsc +2 -2
  113. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  114. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +2 -2
  115. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  116. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  117. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  118. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  119. package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
  120. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  121. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  122. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  123. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  124. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  125. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  126. package/dist/web/standalone/.next/server/webpack-runtime.js +1 -1
  127. package/dist/web/standalone/.next/static/chunks/2556.0527fea66e123b7f.js +1 -0
  128. package/dist/web/standalone/.next/static/chunks/2824.08296bc2f9654698.js +1 -0
  129. package/dist/web/standalone/.next/static/chunks/3026.3af53b279375f082.js +1 -0
  130. package/dist/web/standalone/.next/static/chunks/315.6f68ae79b67d25cf.js +1 -0
  131. package/dist/web/standalone/.next/static/chunks/3497.4bfc60a3b3dea717.js +1 -0
  132. package/dist/web/standalone/.next/static/chunks/5516.4a07c872b5c3a663.js +1 -0
  133. package/dist/web/standalone/.next/static/chunks/8336.31b019697882acfb.js +10 -0
  134. package/dist/web/standalone/.next/static/chunks/8845.c9702695e8c5a9c5.js +2 -0
  135. package/dist/web/standalone/.next/static/chunks/9058.01ef3a463bda88f1.js +20 -0
  136. package/dist/web/standalone/.next/static/chunks/9441.1081da1125d1764f.js +1 -0
  137. package/dist/web/standalone/.next/static/chunks/app/{page-5b113fd32bc2a1c3.js → page-9bf2e0c50fb2ca05.js} +1 -1
  138. package/dist/web/standalone/.next/static/chunks/webpack-f9f0dc45e4f3ac10.js +1 -0
  139. package/dist/web/standalone/package.json +2 -1
  140. package/dist/worktree-status-banner.d.ts +1 -0
  141. package/dist/worktree-status-banner.js +132 -0
  142. package/package.json +1 -1
  143. package/packages/daemon/package.json +2 -2
  144. package/packages/mcp-server/dist/alias-telemetry.d.ts +8 -0
  145. package/packages/mcp-server/dist/alias-telemetry.d.ts.map +1 -0
  146. package/packages/mcp-server/dist/alias-telemetry.js +30 -0
  147. package/packages/mcp-server/dist/alias-telemetry.js.map +1 -0
  148. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  149. package/packages/mcp-server/dist/workflow-tools.js +74 -46
  150. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  151. package/packages/mcp-server/package.json +2 -2
  152. package/packages/mcp-server/src/alias-telemetry.test.ts +78 -0
  153. package/packages/mcp-server/src/alias-telemetry.ts +30 -0
  154. package/packages/mcp-server/src/workflow-tools.test.ts +26 -0
  155. package/packages/mcp-server/src/workflow-tools.ts +93 -58
  156. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  157. package/packages/native/package.json +1 -1
  158. package/packages/pi-agent-core/package.json +1 -1
  159. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  160. package/packages/pi-ai/dist/providers/anthropic-shared.cache-breakpoint.test.d.ts +2 -0
  161. package/packages/pi-ai/dist/providers/anthropic-shared.cache-breakpoint.test.d.ts.map +1 -0
  162. package/packages/pi-ai/dist/providers/anthropic-shared.cache-breakpoint.test.js +231 -0
  163. package/packages/pi-ai/dist/providers/anthropic-shared.cache-breakpoint.test.js.map +1 -0
  164. package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -1
  165. package/packages/pi-ai/dist/providers/anthropic-shared.js +48 -19
  166. package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -1
  167. package/packages/pi-ai/dist/types.d.ts +13 -0
  168. package/packages/pi-ai/dist/types.d.ts.map +1 -1
  169. package/packages/pi-ai/dist/types.js.map +1 -1
  170. package/packages/pi-ai/dist/utils/repair-tool-json.d.ts.map +1 -1
  171. package/packages/pi-ai/dist/utils/repair-tool-json.js +24 -3
  172. package/packages/pi-ai/dist/utils/repair-tool-json.js.map +1 -1
  173. package/packages/pi-ai/dist/utils/tests/repair-tool-json.test.js +26 -0
  174. package/packages/pi-ai/dist/utils/tests/repair-tool-json.test.js.map +1 -1
  175. package/packages/pi-ai/package.json +1 -1
  176. package/packages/pi-ai/src/providers/anthropic-shared.cache-breakpoint.test.ts +289 -0
  177. package/packages/pi-ai/src/providers/anthropic-shared.ts +52 -20
  178. package/packages/pi-ai/src/types.ts +13 -0
  179. package/packages/pi-ai/src/utils/repair-tool-json.ts +24 -3
  180. package/packages/pi-ai/src/utils/tests/repair-tool-json.test.ts +32 -0
  181. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  182. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  183. package/packages/pi-coding-agent/dist/core/agent-session.js +6 -0
  184. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  185. package/packages/pi-coding-agent/dist/core/messages.d.ts.map +1 -1
  186. package/packages/pi-coding-agent/dist/core/messages.js +4 -0
  187. package/packages/pi-coding-agent/dist/core/messages.js.map +1 -1
  188. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +19 -2
  189. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
  190. package/packages/pi-coding-agent/dist/core/model-registry.d.ts +10 -0
  191. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  192. package/packages/pi-coding-agent/dist/core/model-registry.js +18 -0
  193. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  194. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts +13 -0
  195. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  196. package/packages/pi-coding-agent/dist/core/system-prompt.js +20 -16
  197. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  198. package/packages/pi-coding-agent/dist/core/token-telemetry.d.ts +37 -0
  199. package/packages/pi-coding-agent/dist/core/token-telemetry.d.ts.map +1 -0
  200. package/packages/pi-coding-agent/dist/core/token-telemetry.js +49 -0
  201. package/packages/pi-coding-agent/dist/core/token-telemetry.js.map +1 -0
  202. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.d.ts +2 -0
  203. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.d.ts.map +1 -0
  204. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.js +133 -0
  205. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.js.map +1 -0
  206. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +1 -1
  207. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  208. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +14 -1
  209. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  210. package/packages/pi-coding-agent/dist/tests/system-prompt-cache-stability.test.d.ts +2 -0
  211. package/packages/pi-coding-agent/dist/tests/system-prompt-cache-stability.test.d.ts.map +1 -0
  212. package/packages/pi-coding-agent/dist/tests/system-prompt-cache-stability.test.js +78 -0
  213. package/packages/pi-coding-agent/dist/tests/system-prompt-cache-stability.test.js.map +1 -0
  214. package/packages/pi-coding-agent/dist/tests/token-telemetry.test.d.ts +2 -0
  215. package/packages/pi-coding-agent/dist/tests/token-telemetry.test.d.ts.map +1 -0
  216. package/packages/pi-coding-agent/dist/tests/token-telemetry.test.js +181 -0
  217. package/packages/pi-coding-agent/dist/tests/token-telemetry.test.js.map +1 -0
  218. package/packages/pi-coding-agent/package.json +1 -1
  219. package/packages/pi-coding-agent/src/core/agent-session.ts +7 -0
  220. package/packages/pi-coding-agent/src/core/messages.ts +4 -0
  221. package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +32 -2
  222. package/packages/pi-coding-agent/src/core/model-registry.ts +21 -0
  223. package/packages/pi-coding-agent/src/core/system-prompt.ts +33 -15
  224. package/packages/pi-coding-agent/src/core/token-telemetry.ts +77 -0
  225. package/packages/pi-coding-agent/src/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.ts +212 -0
  226. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +17 -1
  227. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +1 -1
  228. package/packages/pi-coding-agent/src/tests/system-prompt-cache-stability.test.ts +102 -0
  229. package/packages/pi-coding-agent/src/tests/token-telemetry.test.ts +200 -0
  230. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  231. package/packages/pi-tui/dist/__tests__/autocomplete.test.js +17 -3
  232. package/packages/pi-tui/dist/__tests__/autocomplete.test.js.map +1 -1
  233. package/packages/pi-tui/dist/components/__tests__/leak-fixes-runtime.test.d.ts +2 -0
  234. package/packages/pi-tui/dist/components/__tests__/leak-fixes-runtime.test.d.ts.map +1 -0
  235. package/packages/pi-tui/dist/components/__tests__/leak-fixes-runtime.test.js +161 -0
  236. package/packages/pi-tui/dist/components/__tests__/leak-fixes-runtime.test.js.map +1 -0
  237. package/packages/pi-tui/package.json +1 -1
  238. package/packages/pi-tui/src/__tests__/autocomplete.test.ts +20 -3
  239. package/packages/pi-tui/src/components/__tests__/leak-fixes-runtime.test.ts +219 -0
  240. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  241. package/packages/rpc-client/package.json +1 -1
  242. package/pkg/package.json +1 -1
  243. package/src/resources/extensions/claude-code-cli/readiness.ts +78 -46
  244. package/src/resources/extensions/gsd/auto/loop.ts +24 -2
  245. package/src/resources/extensions/gsd/auto/phases.ts +3 -3
  246. package/src/resources/extensions/gsd/auto/run-unit.ts +3 -1
  247. package/src/resources/extensions/gsd/auto/session.ts +3 -0
  248. package/src/resources/extensions/gsd/auto/types.ts +1 -0
  249. package/src/resources/extensions/gsd/auto-recovery.ts +46 -8
  250. package/src/resources/extensions/gsd/auto-runtime-state.ts +51 -0
  251. package/src/resources/extensions/gsd/auto-start.ts +1 -1
  252. package/src/resources/extensions/gsd/auto-tool-tracking.ts +2 -4
  253. package/src/resources/extensions/gsd/auto-worktree.ts +38 -0
  254. package/src/resources/extensions/gsd/auto.ts +14 -4
  255. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +15 -13
  256. package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +8 -7
  257. package/src/resources/extensions/gsd/bootstrap/query-tools.ts +2 -2
  258. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +10 -9
  259. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +102 -31
  260. package/src/resources/extensions/gsd/bootstrap/register-shortcuts.ts +12 -6
  261. package/src/resources/extensions/gsd/bootstrap/system-context.ts +39 -8
  262. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +39 -11
  263. package/src/resources/extensions/gsd/commands/catalog.ts +75 -5
  264. package/src/resources/extensions/gsd/commands/handlers/core.ts +22 -1
  265. package/src/resources/extensions/gsd/commands-mcp-status.ts +3 -1
  266. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +15 -1
  267. package/src/resources/extensions/gsd/dashboard-overlay.ts +1 -1
  268. package/src/resources/extensions/gsd/docs/preferences-reference.md +4 -0
  269. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +39 -1
  270. package/src/resources/extensions/gsd/doctor-types.ts +3 -1
  271. package/src/resources/extensions/gsd/error-classifier.ts +1 -1
  272. package/src/resources/extensions/gsd/forensics.ts +2 -2
  273. package/src/resources/extensions/gsd/git-service.ts +13 -5
  274. package/src/resources/extensions/gsd/gsd-db.ts +12 -2
  275. package/src/resources/extensions/gsd/guided-flow.ts +25 -25
  276. package/src/resources/extensions/gsd/memory-store.ts +81 -28
  277. package/src/resources/extensions/gsd/milestone-id-reservation.ts +47 -0
  278. package/src/resources/extensions/gsd/model-router.ts +172 -9
  279. package/src/resources/extensions/gsd/native-git-bridge.ts +7 -1
  280. package/src/resources/extensions/gsd/preferences-models.ts +101 -15
  281. package/src/resources/extensions/gsd/preferences-types.ts +6 -0
  282. package/src/resources/extensions/gsd/preferences-validation.ts +35 -0
  283. package/src/resources/extensions/gsd/preferences.ts +16 -2
  284. package/src/resources/extensions/gsd/prompt-loader.ts +26 -12
  285. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +9 -3
  286. package/src/resources/extensions/gsd/state.ts +42 -0
  287. package/src/resources/extensions/gsd/templates/PREFERENCES.md +1 -0
  288. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +178 -1
  289. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +58 -0
  290. package/src/resources/extensions/gsd/tests/auto-session-encapsulation.test.ts +9 -5
  291. package/src/resources/extensions/gsd/tests/auto-supervisor.test.mjs +21 -4
  292. package/src/resources/extensions/gsd/tests/bootstrap-derive-state-db-open.test.ts +1 -1
  293. package/src/resources/extensions/gsd/tests/budget-prediction.test.ts +138 -211
  294. package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +142 -59
  295. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +7 -4
  296. package/src/resources/extensions/gsd/tests/completed-at-reconcile.test.ts +89 -32
  297. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +41 -23
  298. package/src/resources/extensions/gsd/tests/db-path-worktree-symlink.test.ts +3 -43
  299. package/src/resources/extensions/gsd/tests/debug-logger.test.ts +5 -3
  300. package/src/resources/extensions/gsd/tests/deferred-milestone-dir-4996.test.ts +116 -0
  301. package/src/resources/extensions/gsd/tests/discuss-empty-db-fallback.test.ts +22 -87
  302. package/src/resources/extensions/gsd/tests/discuss-queued-milestones.test.ts +7 -118
  303. package/src/resources/extensions/gsd/tests/discuss-tool-scope-leak.test.ts +18 -60
  304. package/src/resources/extensions/gsd/tests/doctor-orphan-milestone-4996.test.ts +100 -0
  305. package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +14 -76
  306. package/src/resources/extensions/gsd/tests/ensure-preconditions-guard-4996.test.ts +93 -0
  307. package/src/resources/extensions/gsd/tests/false-degraded-mode-warning.test.ts +22 -83
  308. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +1 -63
  309. package/src/resources/extensions/gsd/tests/find-missing-summaries-closed-runtime.test.ts +47 -0
  310. package/src/resources/extensions/gsd/tests/forensics-stuck-loops.test.ts +26 -1
  311. package/src/resources/extensions/gsd/tests/gitignore-bg-shell-runtime.test.ts +63 -0
  312. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +30 -0
  313. package/src/resources/extensions/gsd/tests/gsd-no-project-error-runtime.test.ts +81 -0
  314. package/src/resources/extensions/gsd/tests/headless-answers.test.ts +14 -4
  315. package/src/resources/extensions/gsd/tests/health-widget.test.ts +22 -12
  316. package/src/resources/extensions/gsd/tests/help-menu-coverage.test.ts +57 -0
  317. package/src/resources/extensions/gsd/tests/import-done-milestones-runtime.test.ts +145 -0
  318. package/src/resources/extensions/gsd/tests/init-prefs-routing.test.ts +64 -1
  319. package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +22 -0
  320. package/src/resources/extensions/gsd/tests/integration/token-savings.test.ts +0 -23
  321. package/src/resources/extensions/gsd/tests/memory-store.test.ts +128 -0
  322. package/src/resources/extensions/gsd/tests/memory-tools.test.ts +33 -1
  323. package/src/resources/extensions/gsd/tests/merge-self-branch-guard.test.ts +124 -0
  324. package/src/resources/extensions/gsd/tests/milestone-id-gap-reuse-4996.test.ts +152 -0
  325. package/src/resources/extensions/gsd/tests/model-router.test.ts +169 -8
  326. package/src/resources/extensions/gsd/tests/native-git-infra-errors.test.ts +50 -0
  327. package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +8 -0
  328. package/src/resources/extensions/gsd/tests/parallel-crash-recovery.test.ts +32 -43
  329. package/src/resources/extensions/gsd/tests/phases-merge-error-stops-auto.test.ts +4 -10
  330. package/src/resources/extensions/gsd/tests/preferences.test.ts +127 -0
  331. package/src/resources/extensions/gsd/tests/prompt-step-ordering.test.ts +16 -0
  332. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +7 -0
  333. package/src/resources/extensions/gsd/tests/quick-turn-end-cleanup.test.ts +6 -6
  334. package/src/resources/extensions/gsd/tests/register-hooks-compaction-checkpoint.test.ts +93 -0
  335. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +168 -19
  336. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +7 -1
  337. package/src/resources/extensions/gsd/tests/smart-entry-complete.test.ts +23 -1
  338. package/src/resources/extensions/gsd/tests/system-context-message-routing.test.ts +101 -0
  339. package/src/resources/extensions/gsd/tests/token-profile.test.ts +51 -4
  340. package/src/resources/extensions/gsd/tests/turn-epoch.test.ts +7 -16
  341. package/src/resources/extensions/gsd/tests/unstructured-continue-context-injection.test.ts +5 -7
  342. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +15 -1
  343. package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +6 -6
  344. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +15 -0
  345. package/src/resources/extensions/gsd/tools/memory-tools.ts +17 -1
  346. package/src/resources/extensions/gsd/unit-context-manifest.ts +8 -8
  347. package/src/resources/extensions/gsd/visualizer-overlay.ts +1 -1
  348. package/src/resources/extensions/gsd/watch/header-renderer.ts +3 -1
  349. package/src/resources/extensions/gsd/workflow-logger.ts +1 -0
  350. package/src/resources/extensions/gsd/worktree-command.ts +31 -44
  351. package/src/resources/extensions/gsd/worktree-session-state.ts +35 -0
  352. package/src/resources/extensions/mcp-client/index.ts +6 -3
  353. package/src/resources/extensions/mcp-client/tests/global-config.test.ts +91 -0
  354. package/src/resources/extensions/slash-commands/create-extension.ts +38 -24
  355. package/src/resources/skills/create-gsd-extension/SKILL.md +9 -5
  356. package/src/resources/skills/create-gsd-extension/references/custom-commands.md +1 -1
  357. package/src/resources/skills/create-gsd-extension/references/custom-rendering.md +5 -5
  358. package/src/resources/skills/create-gsd-extension/references/custom-tools.md +4 -4
  359. package/src/resources/skills/create-gsd-extension/references/custom-ui.md +6 -6
  360. package/src/resources/skills/create-gsd-extension/references/events-reference.md +3 -3
  361. package/src/resources/skills/create-gsd-extension/references/packaging-distribution.md +1 -1
  362. package/src/resources/skills/create-gsd-extension/references/remote-execution-overrides.md +3 -3
  363. package/src/resources/skills/create-gsd-extension/templates/extension-skeleton.ts +2 -2
  364. package/src/resources/skills/create-gsd-extension/templates/stateful-tool-skeleton.ts +3 -3
  365. package/src/resources/skills/create-gsd-extension/templates/templates.test.ts +58 -0
  366. package/src/resources/skills/create-gsd-extension/workflows/create-extension.md +32 -12
  367. package/dist/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +0 -601
  368. package/dist/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +0 -651
  369. package/dist/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +0 -91
  370. package/dist/resources/extensions/gsd/tests/auto-supervisor.test.mjs +0 -53
  371. package/dist/resources/extensions/gsd/tests/dist-redirect.mjs +0 -112
  372. package/dist/resources/extensions/gsd/tests/resolve-ts-hooks.mjs +0 -23
  373. package/dist/resources/extensions/gsd/tests/resolve-ts.mjs +0 -5
  374. package/dist/resources/skills/github-workflows/references/gh/tests/__init__.py +0 -0
  375. package/dist/resources/skills/github-workflows/references/gh/tests/test_github_project_setup.py +0 -608
  376. package/dist/web/standalone/.next/static/chunks/2826.e9f5195e91f9cad2.js +0 -11
  377. package/dist/web/standalone/.next/static/chunks/3621.fc7480022c972438.js +0 -20
  378. package/dist/web/standalone/.next/static/chunks/webpack-2e68521d7c82f7c2.js +0 -1
  379. package/src/resources/extensions/gsd/tests/copy-planning-artifacts-samepath.test.ts +0 -22
  380. package/src/resources/extensions/gsd/tests/discuss-slice-structured-questions.test.ts +0 -47
  381. package/src/resources/extensions/gsd/tests/empty-content-abort-loop.test.ts +0 -75
  382. /package/dist/web/standalone/.next/static/{cAJH99yNS1UPbeSEiNRrV → UF5VF4F1tB0miEtJS7LyX}/_buildManifest.js +0 -0
  383. /package/dist/web/standalone/.next/static/{cAJH99yNS1UPbeSEiNRrV → UF5VF4F1tB0miEtJS7LyX}/_ssgManifest.js +0 -0
@@ -0,0 +1,152 @@
1
+ // GSD Extension — Regression test for #4996: ghost milestone ID reuse
2
+ // Verifies that isReusableGhostMilestone correctly identifies reclaim-safe stub dirs,
3
+ // and that nextMilestoneIdReserved (guided-flow) prefers the lowest reusable ghost
4
+ // over max+1. Also covers the race-window regression: a queued DB row must NOT be reused.
5
+
6
+ import { describe, it, beforeEach, afterEach } from "node:test";
7
+ import assert from "node:assert/strict";
8
+ import { mkdtempSync, mkdirSync, writeFileSync, rmSync } from "node:fs";
9
+ import { join } from "node:path";
10
+ import { tmpdir } from "node:os";
11
+
12
+ import { isReusableGhostMilestone } from "../state.ts";
13
+ import { nextMilestoneIdReserved } from "../milestone-id-reservation.ts";
14
+ import {
15
+ openDatabase,
16
+ closeDatabase,
17
+ insertMilestone,
18
+ } from "../gsd-db.ts";
19
+ import { clearReservedMilestoneIds, findMilestoneIds } from "../milestone-ids.ts";
20
+ import { invalidateAllCaches } from "../cache.ts";
21
+
22
+ function makeBase(prefix = "gsd-gap-4996-"): string {
23
+ const base = mkdtempSync(join(tmpdir(), prefix));
24
+ mkdirSync(join(base, ".gsd", "milestones"), { recursive: true });
25
+ return base;
26
+ }
27
+
28
+ function stubDir(base: string, mid: string): void {
29
+ // Create an empty stub — the phantom pattern
30
+ mkdirSync(join(base, ".gsd", "milestones", mid, "slices"), { recursive: true });
31
+ }
32
+
33
+ function populateDir(base: string, mid: string): void {
34
+ mkdirSync(join(base, ".gsd", "milestones", mid), { recursive: true });
35
+ writeFileSync(join(base, ".gsd", "milestones", mid, `${mid}-CONTEXT.md`), `# ${mid} Context\n`);
36
+ }
37
+
38
+ describe("isReusableGhostMilestone (#4996)", () => {
39
+ let base: string;
40
+
41
+ afterEach(() => {
42
+ try { closeDatabase(); } catch { /* ignore */ }
43
+ try { invalidateAllCaches(); } catch { /* ignore */ }
44
+ try { rmSync(base, { recursive: true, force: true }); } catch { /* ignore */ }
45
+ });
46
+
47
+ it("(a) fails closed when the DB is unavailable", () => {
48
+ base = makeBase();
49
+ stubDir(base, "M003");
50
+ assert.equal(isReusableGhostMilestone(base, "M003"), false, "closed DB should block reusable-ghost claims");
51
+ });
52
+
53
+ it("(b) empty stub dir with an open DB and no DB row is reusable", () => {
54
+ base = makeBase();
55
+ openDatabase(join(base, ".gsd", "gsd.db"));
56
+ stubDir(base, "M003");
57
+ assert.ok(isReusableGhostMilestone(base, "M003"), "empty stub with no DB row should be reusable");
58
+ });
59
+
60
+ it("(c) queued DB row with no content must NOT be reusable (race window regression)", () => {
61
+ base = makeBase();
62
+ stubDir(base, "M003");
63
+ const dbPath = join(base, ".gsd", "gsd.db");
64
+ openDatabase(dbPath);
65
+ insertMilestone({ id: "M003", status: "queued" });
66
+ // Even though no content files exist, the queued DB row means an in-flight discuss
67
+ // is reserving this ID — it must not be reclaimed.
68
+ assert.ok(!isReusableGhostMilestone(base, "M003"), "queued DB row must block reuse");
69
+ });
70
+
71
+ it("(d) populated milestone dir is not reusable", () => {
72
+ base = makeBase();
73
+ openDatabase(join(base, ".gsd", "gsd.db"));
74
+ populateDir(base, "M001");
75
+ assert.ok(!isReusableGhostMilestone(base, "M001"), "populated dir must not be reusable");
76
+ });
77
+
78
+ it("(e) stub dir with worktree is not reusable (legitimate in-flight)", () => {
79
+ base = makeBase();
80
+ openDatabase(join(base, ".gsd", "gsd.db"));
81
+ stubDir(base, "M003");
82
+ // Simulate an existing worktree
83
+ mkdirSync(join(base, ".gsd", "worktrees", "M003"), { recursive: true });
84
+ assert.ok(!isReusableGhostMilestone(base, "M003"), "dir with worktree must not be reusable");
85
+ });
86
+
87
+ it("(f) active DB row makes dir not reusable", () => {
88
+ base = makeBase();
89
+ stubDir(base, "M003");
90
+ const dbPath = join(base, ".gsd", "gsd.db");
91
+ openDatabase(dbPath);
92
+ insertMilestone({ id: "M003", status: "active" });
93
+ assert.ok(!isReusableGhostMilestone(base, "M003"), "active DB row must block reuse");
94
+ });
95
+ });
96
+
97
+ describe("primary regression: M003/M004 stubs returned as next ID (#4996)", () => {
98
+ let base: string;
99
+
100
+ beforeEach(() => {
101
+ clearReservedMilestoneIds();
102
+ });
103
+
104
+ afterEach(() => {
105
+ try { closeDatabase(); } catch { /* ignore */ }
106
+ try { invalidateAllCaches(); } catch { /* ignore */ }
107
+ try { clearReservedMilestoneIds(); } catch { /* ignore */ }
108
+ try { rmSync(base, { recursive: true, force: true }); } catch { /* ignore */ }
109
+ });
110
+
111
+ it("M001/M002 populated + M003/M004 stubs → isReusableGhostMilestone returns true for M003 and M004", () => {
112
+ base = makeBase();
113
+ openDatabase(join(base, ".gsd", "gsd.db"));
114
+ populateDir(base, "M001");
115
+ populateDir(base, "M002");
116
+ stubDir(base, "M003");
117
+ stubDir(base, "M004");
118
+
119
+ assert.ok(isReusableGhostMilestone(base, "M003"), "M003 should be identified as reusable ghost");
120
+ assert.ok(isReusableGhostMilestone(base, "M004"), "M004 should be identified as reusable ghost");
121
+ assert.ok(!isReusableGhostMilestone(base, "M001"), "M001 should not be reusable");
122
+ assert.ok(!isReusableGhostMilestone(base, "M002"), "M002 should not be reusable");
123
+
124
+ const nextId = nextMilestoneIdReserved(findMilestoneIds(base), false, base);
125
+ assert.equal(nextId, "M003", "ID reservation should select the lowest reusable ghost");
126
+ });
127
+
128
+ it("when all dirs are populated, no ghost exists and the function returns false for all", () => {
129
+ base = makeBase();
130
+ openDatabase(join(base, ".gsd", "gsd.db"));
131
+ populateDir(base, "M001");
132
+ populateDir(base, "M002");
133
+
134
+ assert.ok(!isReusableGhostMilestone(base, "M001"), "M001 is populated, not reusable");
135
+ assert.ok(!isReusableGhostMilestone(base, "M002"), "M002 is populated, not reusable");
136
+
137
+ const nextId = nextMilestoneIdReserved(findMilestoneIds(base), false, base);
138
+ assert.equal(nextId, "M003", "ID reservation should fall back to max+1 when no ghost is reusable");
139
+ });
140
+
141
+ it("does not return an already-reserved reusable ghost twice", () => {
142
+ base = makeBase();
143
+ openDatabase(join(base, ".gsd", "gsd.db"));
144
+ stubDir(base, "M001");
145
+
146
+ const firstId = nextMilestoneIdReserved(findMilestoneIds(base), false, base);
147
+ const secondId = nextMilestoneIdReserved(findMilestoneIds(base), false, base);
148
+
149
+ assert.equal(firstId, "M001", "first reservation should reuse the ghost");
150
+ assert.equal(secondId, "M002", "second reservation must skip the already-reserved ghost");
151
+ });
152
+ });
@@ -5,6 +5,7 @@ import {
5
5
  resolveModelForComplexity,
6
6
  escalateTier,
7
7
  defaultRoutingConfig,
8
+ resolveModelForTier,
8
9
  scoreModel,
9
10
  computeTaskRequirements,
10
11
  scoreEligibleModels,
@@ -248,6 +249,172 @@ test("#2192: known model is still downgraded normally", () => {
248
249
  assert.notEqual(result.modelId, "claude-opus-4-6");
249
250
  });
250
251
 
252
+ // ─── Cross-provider fallback ──────────────────────────────────────────────────
253
+
254
+ test("uses cross-provider equivalent when configured primary is unavailable", () => {
255
+ const config = { ...defaultRoutingConfig(), enabled: true };
256
+ // Profile default says claude-opus-4-6 for planning, but user is on GPT only
257
+ const result = resolveModelForComplexity(
258
+ makeClassification("heavy"),
259
+ { primary: "claude-opus-4-6", fallbacks: [] },
260
+ config,
261
+ ["gpt-4o", "gpt-4o-mini", "o1"],
262
+ );
263
+ // o1 is the heavy-tier GPT model — should be selected as cross-provider equivalent
264
+ assert.equal(result.modelId, "o1");
265
+ assert.equal(result.wasDowngraded, false);
266
+ assert.match(result.reason, /cross-provider/);
267
+ });
268
+
269
+ test("cross-provider: selects standard-tier equivalent when primary unavailable", () => {
270
+ const config = { ...defaultRoutingConfig(), enabled: true };
271
+ // Planning configured with Opus, but only GPT standard models available
272
+ const result = resolveModelForComplexity(
273
+ makeClassification("heavy"),
274
+ { primary: "claude-opus-4-6", fallbacks: [] },
275
+ config,
276
+ ["gpt-4o", "gpt-4o-mini"],
277
+ );
278
+ // gpt-4o is standard tier, not heavy — no heavy-tier model available
279
+ // Should fall back to gpt-4o (best available)
280
+ assert.ok(result.modelId === "gpt-4o" || result.modelId === "claude-opus-4-6");
281
+ assert.equal(result.wasDowngraded, false);
282
+ });
283
+
284
+ test("cross-provider: configured primary available by bare ID wins over equivalent", () => {
285
+ const config = { ...defaultRoutingConfig(), enabled: true };
286
+ // Provider-prefixed ID — bare match should find it
287
+ const result = resolveModelForComplexity(
288
+ makeClassification("heavy"),
289
+ { primary: "claude-opus-4-6", fallbacks: [] },
290
+ config,
291
+ ["anthropic/claude-opus-4-6", "o1"],
292
+ );
293
+ assert.equal(result.modelId, "claude-opus-4-6");
294
+ assert.equal(result.wasDowngraded, false);
295
+ });
296
+
297
+ // ─── resolveModelForTier (provider-agnostic tier resolution) ────────────────
298
+
299
+ test("resolveModelForTier: returns canonical Anthropic model when no available models", () => {
300
+ assert.equal(resolveModelForTier("heavy", []), "claude-opus-4-6");
301
+ assert.equal(resolveModelForTier("standard", []), "claude-sonnet-4-6");
302
+ assert.equal(resolveModelForTier("light", []), "claude-haiku-4-5");
303
+ });
304
+
305
+ test("resolveModelForTier: returns canonical model when it is available", () => {
306
+ assert.equal(
307
+ resolveModelForTier("heavy", ["claude-opus-4-6", "claude-sonnet-4-6"]),
308
+ "claude-opus-4-6",
309
+ );
310
+ });
311
+
312
+ test("resolveModelForTier: does not prefer canonical over cheaper same-tier model", () => {
313
+ const result = resolveModelForTier("light", ["claude-haiku-4-5", "gpt-4o-mini"]);
314
+ assert.equal(result, "gpt-4o-mini");
315
+ });
316
+
317
+ test("resolveModelForTier: honors configured tier_models pins", () => {
318
+ const config: DynamicRoutingConfig = {
319
+ ...defaultRoutingConfig(),
320
+ tier_models: { light: "claude-haiku-4-5" },
321
+ };
322
+ const result = resolveModelForTier("light", ["claude-haiku-4-5", "gpt-4o-mini"], config);
323
+ assert.equal(result, "claude-haiku-4-5");
324
+ });
325
+
326
+ test("resolveModelForTier: picks cross-provider equivalent when Anthropic unavailable", () => {
327
+ // Only OpenAI models available
328
+ const result = resolveModelForTier("heavy", ["gpt-4o", "gpt-4o-mini", "o1"]);
329
+ // o1 is the heavy-tier model in the OpenAI lineup
330
+ assert.equal(result, "o1");
331
+ });
332
+
333
+ test("resolveModelForTier: picks standard-tier cross-provider model", () => {
334
+ const result = resolveModelForTier("standard", ["gpt-4o", "gpt-4o-mini"]);
335
+ assert.equal(result, "gpt-4o");
336
+ });
337
+
338
+ test("resolveModelForTier: picks light-tier cross-provider model", () => {
339
+ const result = resolveModelForTier("light", ["gpt-4o", "gpt-4o-mini"]);
340
+ assert.equal(result, "gpt-4o-mini");
341
+ });
342
+
343
+ test("resolveModelForTier: falls back to canonical when no tier match available", () => {
344
+ // Only unknown models available — getModelTier classifies unknowns as
345
+ // "standard", so a request for "heavy" finds no match and the canonical
346
+ // Anthropic ID is returned as a documented fallback.
347
+ const result = resolveModelForTier("heavy", ["some-custom-model"]);
348
+ assert.equal(result, "claude-opus-4-6");
349
+ });
350
+
351
+ test("resolveModelForTier: handles provider-prefixed available models", () => {
352
+ const result = resolveModelForTier("heavy", ["anthropic/claude-opus-4-6"]);
353
+ assert.equal(result, "claude-opus-4-6");
354
+ });
355
+
356
+ test("resolveModelForTier: picks Gemini models when only Google available", () => {
357
+ const result = resolveModelForTier("light", ["gemini-2.5-pro", "gemini-2.0-flash"]);
358
+ assert.equal(result, "gemini-2.0-flash");
359
+ });
360
+
361
+ // ─── Behavioral: profile defaults are provider-agnostic at runtime ──────────
362
+
363
+ test("resolveProfileDefaults: balanced with only OpenAI models returns OpenAI IDs", async () => {
364
+ const { resolveProfileDefaults } = await import("../preferences-models.js");
365
+ const defaults = resolveProfileDefaults("balanced", ["gpt-4o", "gpt-4o-mini"]);
366
+ assert.ok(defaults.models, "balanced should populate models");
367
+ // All slots must resolve to an available OpenAI ID — not a claude- canonical.
368
+ for (const [phase, modelId] of Object.entries(defaults.models!)) {
369
+ assert.ok(typeof modelId === "string" && modelId.length > 0, `${phase} should resolve to a model ID`);
370
+ assert.ok(
371
+ !String(modelId).startsWith("claude-"),
372
+ `${phase} resolved to ${modelId} but no claude-* model is available — should be OpenAI`,
373
+ );
374
+ }
375
+ });
376
+
377
+ test("resolveProfileDefaults: budget with only OpenAI models picks gpt-4o-mini for light slots", async () => {
378
+ const { resolveProfileDefaults } = await import("../preferences-models.js");
379
+ const defaults = resolveProfileDefaults("budget", ["gpt-4o", "gpt-4o-mini"]);
380
+ // light-tier slots in budget: research, execution_simple, completion, subagent
381
+ assert.equal(defaults.models?.research, "gpt-4o-mini");
382
+ assert.equal(defaults.models?.execution_simple, "gpt-4o-mini");
383
+ assert.equal(defaults.models?.completion, "gpt-4o-mini");
384
+ assert.equal(defaults.models?.subagent, "gpt-4o-mini");
385
+ // standard-tier slots: planning, execution
386
+ assert.equal(defaults.models?.planning, "gpt-4o");
387
+ assert.equal(defaults.models?.execution, "gpt-4o");
388
+ });
389
+
390
+ test("resolveProfileDefaults: honors dynamic routing tier_models pins", async () => {
391
+ const { resolveProfileDefaults } = await import("../preferences-models.js");
392
+ const defaults = resolveProfileDefaults(
393
+ "budget",
394
+ ["claude-haiku-4-5", "gpt-4o-mini", "gpt-4o"],
395
+ { ...defaultRoutingConfig(), tier_models: { light: "claude-haiku-4-5" } },
396
+ );
397
+ assert.equal(defaults.models?.research, "claude-haiku-4-5");
398
+ assert.equal(defaults.models?.execution_simple, "claude-haiku-4-5");
399
+ assert.equal(defaults.models?.completion, "claude-haiku-4-5");
400
+ assert.equal(defaults.models?.subagent, "claude-haiku-4-5");
401
+ });
402
+
403
+ test("resolveProfileDefaults: empty availableModelIds falls back to canonical Anthropic IDs", async () => {
404
+ const { resolveProfileDefaults } = await import("../preferences-models.js");
405
+ const defaults = resolveProfileDefaults("balanced", []);
406
+ // Documented fallback only — when registry is unavailable at bootstrap.
407
+ const planningModel = defaults.models?.planning;
408
+ assert.ok(typeof planningModel === "string" && planningModel.startsWith("claude-"));
409
+ });
410
+
411
+ test("resolveProfileDefaults: burn-max omits models so user choice is preserved", async () => {
412
+ const { resolveProfileDefaults } = await import("../preferences-models.js");
413
+ const defaults = resolveProfileDefaults("burn-max", ["gpt-4o"]);
414
+ assert.equal(defaults.models, undefined, "burn-max must not write model defaults");
415
+ assert.equal(defaults.dynamic_routing?.enabled, false);
416
+ });
417
+
251
418
  // ─── Capability Scoring (ADR-004 Phase 2) ───────────────────────────────────
252
419
 
253
420
  test("defaultRoutingConfig includes capability_routing: true", () => {
@@ -279,14 +446,8 @@ test("scoreModel computes weighted average of capability × requirement", () =>
279
446
  assert.ok(Math.abs(score - 88.21) < 0.1, `score ${score} should be ~88.21`);
280
447
  });
281
448
 
282
- test("scoreModel returns 50 for empty requirements", () => {
283
- const caps: ModelCapabilities = {
284
- coding: 90, debugging: 80, research: 70,
285
- reasoning: 85, speed: 50, longContext: 60, instruction: 75,
286
- };
287
- const score = scoreModel(caps, {});
288
- assert.equal(score, 50);
289
- });
449
+ // (Removed duplicate "scoreModel returns 50 for empty requirements" the
450
+ // `describe("scoreModel")` block below has the same scenario.)
290
451
 
291
452
  test("computeTaskRequirements returns base vector for known unit type", () => {
292
453
  const reqs = computeTaskRequirements("execute-task");
@@ -0,0 +1,50 @@
1
+ import test from "node:test";
2
+ import assert from "node:assert/strict";
3
+ import { chmodSync, mkdirSync, mkdtempSync, rmSync, writeFileSync } from "node:fs";
4
+ import { tmpdir } from "node:os";
5
+ import { join } from "node:path";
6
+
7
+ import { git } from "./test-utils.ts";
8
+ import { GSD_GIT_ERROR } from "../errors.js";
9
+
10
+ test("nativeAddAllWithExclusions preserves infrastructure failures from git add", async () => {
11
+ const base = mkdtempSync(join(tmpdir(), "gsd-native-git-infra-"));
12
+ const repo = join(base, "repo");
13
+ const bin = join(base, "bin");
14
+ mkdirSync(repo);
15
+ mkdirSync(bin);
16
+
17
+ const fakeGit = join(bin, "git");
18
+ writeFileSync(
19
+ fakeGit,
20
+ "#!/bin/sh\n" +
21
+ "echo 'fatal: ENFILE: file table overflow' >&2\n" +
22
+ "exit 1\n",
23
+ "utf-8",
24
+ );
25
+ chmodSync(fakeGit, 0o755);
26
+
27
+ const originalPath = process.env.PATH ?? "";
28
+ try {
29
+ git(repo, "init");
30
+ git(repo, "config", "user.email", "test@example.com");
31
+ git(repo, "config", "user.name", "Test User");
32
+ writeFileSync(join(repo, "README.md"), "# Test\n", "utf-8");
33
+
34
+ process.env.PATH = `${bin}:${originalPath}`;
35
+ const { nativeAddAllWithExclusions } = await import("../native-git-bridge.ts");
36
+
37
+ assert.throws(
38
+ () => nativeAddAllWithExclusions(repo, [".gsd/activity/"]),
39
+ (err) => {
40
+ const shaped = err as { code?: string; stderr?: string; message?: string };
41
+ assert.notEqual(shaped.code, GSD_GIT_ERROR);
42
+ assert.match(`${shaped.stderr ?? ""}${shaped.message ?? ""}`, /ENFILE/);
43
+ return true;
44
+ },
45
+ );
46
+ } finally {
47
+ process.env.PATH = originalPath;
48
+ rmSync(base, { recursive: true, force: true });
49
+ }
50
+ });
@@ -101,6 +101,14 @@ describe("auditOrphanedMilestoneBranches", () => {
101
101
  result.warnings.some(w => w.includes("NOT merged")),
102
102
  "should warn about unmerged branch",
103
103
  );
104
+ assert.ok(
105
+ result.warnings.some(w => w.includes("/gsd doctor fix")),
106
+ `warning should suggest the real remediation command; got: ${JSON.stringify(result.warnings)}`,
107
+ );
108
+ assert.ok(
109
+ result.warnings.every(w => !w.includes("/gsd health --fix")),
110
+ `warning must not suggest the removed health --fix command; got: ${JSON.stringify(result.warnings)}`,
111
+ );
104
112
 
105
113
  // Branch should still exist (data safety)
106
114
  const branches = run("git branch --list milestone/M001", dir);
@@ -19,13 +19,15 @@ import { join } from "node:path";
19
19
  import { tmpdir } from "node:os";
20
20
 
21
21
  import {
22
- persistState,
23
22
  restoreState,
24
23
  resetOrchestrator,
25
- getOrchestratorState,
26
24
  type PersistedState,
27
25
  } from "../parallel-orchestrator.ts";
28
- import { writeSessionStatus, readAllSessionStatuses, removeSessionStatus } from "../session-status-io.ts";
26
+ import {
27
+ writeSessionStatus,
28
+ readAllSessionStatuses,
29
+ cleanupStaleSessions,
30
+ } from "../session-status-io.ts";
29
31
  // ─── Helpers ──────────────────────────────────────────────────────────────────
30
32
 
31
33
  function makeTempDir(): string {
@@ -57,17 +59,20 @@ function makePersistedState(overrides: Partial<PersistedState> = {}): PersistedS
57
59
 
58
60
 
59
61
  describe('parallel-crash-recovery', () => {
60
- test('Test 1: persistState writes valid JSON', () => {
62
+ test('Test 1: orchestrator.json round-trips through restoreState (preserves worker fields)', () => {
61
63
  const basePath = makeTempDir();
62
64
  try {
63
- // We can't call persistState directly without internal state set up,
64
- // so we test the round-trip by writing a state file and reading it back
65
+ // Write a full state file to disk and then exercise the real production
66
+ // restoreState() reader against it. This verifies the persisted file
67
+ // schema (the contract between persistState's writer and the reader)
68
+ // — earlier this test inlined a test-only writer and re-parsed JSON,
69
+ // bypassing production code entirely.
65
70
  const state = makePersistedState({
66
71
  workers: [
67
72
  {
68
73
  milestoneId: "M001",
69
74
  title: "M001",
70
- pid: process.pid,
75
+ pid: process.pid, // alive — survives restoreState's PID filter
71
76
  worktreePath: "/tmp/wt-M001",
72
77
  startedAt: Date.now(),
73
78
  state: "running",
@@ -78,13 +83,13 @@ test('Test 1: persistState writes valid JSON', () => {
78
83
  });
79
84
  writeStateFile(basePath, state);
80
85
 
81
- const raw = readFileSync(stateFilePath(basePath), "utf-8");
82
- const parsed = JSON.parse(raw) as PersistedState;
83
- assert.deepStrictEqual(parsed.active, true, "persistState: active field preserved");
84
- assert.deepStrictEqual(parsed.workers.length, 1, "persistState: worker count preserved");
85
- assert.deepStrictEqual(parsed.workers[0].milestoneId, "M001", "persistState: milestoneId preserved");
86
- assert.deepStrictEqual(parsed.workers[0].cost, 0.15, "persistState: cost preserved");
87
- assert.deepStrictEqual(parsed.totalCost, 0.15, "persistState: totalCost preserved");
86
+ const restored = restoreState(basePath);
87
+ assert.ok(restored !== null, "restoreState: returns state for live worker");
88
+ assert.deepStrictEqual(restored!.active, true, "active field preserved through round-trip");
89
+ assert.deepStrictEqual(restored!.workers.length, 1, "worker count preserved");
90
+ assert.deepStrictEqual(restored!.workers[0].milestoneId, "M001", "milestoneId preserved");
91
+ assert.deepStrictEqual(restored!.workers[0].cost, 0.15, "cost preserved");
92
+ assert.deepStrictEqual(restored!.totalCost, 0.15, "totalCost preserved");
88
93
  } finally {
89
94
  rmSync(basePath, { recursive: true, force: true });
90
95
  }
@@ -201,11 +206,12 @@ test('Test 5: restoreState skips stopped/error workers even with alive PIDs', ()
201
206
  }
202
207
  });
203
208
 
204
- test('Test 6: orphan detection finds stale sessions', () => {
209
+ test('Test 6: cleanupStaleSessions removes dead-PID sessions and keeps live ones', () => {
205
210
  const basePath = makeTempDir();
206
211
  try {
207
- // Write a session status with a dead PID
208
212
  mkdirSync(join(basePath, ".gsd", "parallel"), { recursive: true });
213
+
214
+ // Dead PID
209
215
  writeSessionStatus(basePath, {
210
216
  milestoneId: "M001",
211
217
  pid: 99999999,
@@ -218,7 +224,7 @@ test('Test 6: orphan detection finds stale sessions', () => {
218
224
  worktreePath: "/tmp/wt-M001",
219
225
  });
220
226
 
221
- // Write a session status with alive PID
227
+ // Live PID (this process)
222
228
  writeSessionStatus(basePath, {
223
229
  milestoneId: "M002",
224
230
  pid: process.pid,
@@ -231,37 +237,20 @@ test('Test 6: orphan detection finds stale sessions', () => {
231
237
  worktreePath: "/tmp/wt-M002",
232
238
  });
233
239
 
234
- // Read all sessions — both should exist initially
235
240
  const before = readAllSessionStatuses(basePath);
236
- assert.deepStrictEqual(before.length, 2, "orphan: both sessions exist before detection");
241
+ assert.deepStrictEqual(before.length, 2, "both sessions exist before cleanup");
237
242
 
238
- // Now simulate orphan detection logic (same as prepareParallelStart)
239
- const sessions = readAllSessionStatuses(basePath);
240
- const orphans: Array<{ milestoneId: string; pid: number; alive: boolean }> = [];
241
- for (const session of sessions) {
242
- let alive: boolean;
243
- try {
244
- process.kill(session.pid, 0);
245
- alive = true;
246
- } catch {
247
- alive = false;
248
- }
249
- orphans.push({ milestoneId: session.milestoneId, pid: session.pid, alive });
250
- if (!alive) {
251
- removeSessionStatus(basePath, session.milestoneId);
252
- }
253
- }
243
+ // Drive the real production cleanup function. Earlier this test
244
+ // re-implemented the cleanup loop inline (process.kill + remove*) and
245
+ // never exercised cleanupStaleSessions itself so changes to the
246
+ // production sweep would not have been caught.
247
+ const removed = cleanupStaleSessions(basePath);
254
248
 
255
- assert.ok(orphans.length === 2, "orphan: detected both sessions");
256
- const deadOrphan = orphans.find(o => o.milestoneId === "M001");
257
- assert.ok(deadOrphan !== undefined && !deadOrphan.alive, "orphan: M001 detected as dead");
258
- const aliveOrphan = orphans.find(o => o.milestoneId === "M002");
259
- assert.ok(aliveOrphan !== undefined && aliveOrphan.alive, "orphan: M002 detected as alive");
249
+ assert.deepStrictEqual(removed, ["M001"], "dead-PID session id is reported as removed");
260
250
 
261
- // Dead session should be cleaned up
262
251
  const after = readAllSessionStatuses(basePath);
263
- assert.deepStrictEqual(after.length, 1, "orphan: dead session cleaned up");
264
- assert.deepStrictEqual(after[0].milestoneId, "M002", "orphan: alive session remains");
252
+ assert.deepStrictEqual(after.length, 1, "dead session cleaned up");
253
+ assert.deepStrictEqual(after[0].milestoneId, "M002", "alive session remains");
265
254
  } finally {
266
255
  rmSync(basePath, { recursive: true, force: true });
267
256
  }
@@ -18,20 +18,14 @@ const phasesSrc = readFileSync(phasesPath, "utf-8");
18
18
 
19
19
  console.log("\n=== #2766: Non-MergeConflictError stops auto mode ===");
20
20
 
21
- // ── Test 1: phases.ts calls logError for non-conflict merge errors ──────
22
-
23
- assertTrue(
24
- phasesPath.length > 0 && phasesPath.endsWith("phases.ts"),
25
- "phases.ts file exists and is readable",
26
- );
21
+ // ── Test 1: every mergeAndExit call site has a catch (mergeErr) block ──
27
22
 
28
23
  // Count all mergeAndExit catch blocks by finding "} catch (mergeErr)" patterns
29
- const mergeErrCatches = [...phasesPath.matchAll(/\} catch \(mergeErr\)/g)];
30
- // Use the source itself for matching
24
+ const mergeAndExitCallCount = [...phasesSrc.matchAll(/\.mergeAndExit\(/g)].length;
31
25
  const mergeErrCatchCount = [...phasesSrc.matchAll(/\} catch \(mergeErr\)/g)].length;
32
26
  assertTrue(
33
- mergeErrCatchCount >= 3,
34
- `all mergeAndExit call sites have catch (mergeErr) blocks (found ${mergeErrCatchCount}, expected >= 3)`,
27
+ mergeErrCatchCount === mergeAndExitCallCount && mergeAndExitCallCount > 0,
28
+ `every mergeAndExit call site has a catch (mergeErr) block (calls=${mergeAndExitCallCount}, catches=${mergeErrCatchCount})`,
35
29
  );
36
30
 
37
31
  // ── Test 2: Every mergeErr catch block handles non-MergeConflictError ───