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,145 @@
1
+ /**
2
+ * Runtime regression — milestones with all-done roadmap slices import as
3
+ * `complete` (#3699 / #3390 / #3379), follow-up #4902.
4
+ *
5
+ * The deleted `import-done-milestones.test.ts` was a source-grep check
6
+ * for the literal `roadmap.slices.every(s => s.done)`. This rewrite
7
+ * exercises `migrateHierarchyToDb()` against a fixture roadmap whose
8
+ * slices are all `[x]` and asserts the milestone row's `status` is
9
+ * `complete` — the actual behaviour the every() check exists to
10
+ * produce.
11
+ */
12
+
13
+ import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from 'node:fs';
14
+ import { join } from 'node:path';
15
+ import { tmpdir } from 'node:os';
16
+
17
+ import {
18
+ openDatabase,
19
+ closeDatabase,
20
+ getAllMilestones,
21
+ } from '../gsd-db.ts';
22
+ import { migrateHierarchyToDb } from '../md-importer.ts';
23
+ import { describe, test } from 'node:test';
24
+ import assert from 'node:assert/strict';
25
+
26
+ function createFixtureBase(): string {
27
+ const base = mkdtempSync(join(tmpdir(), 'gsd-import-done-'));
28
+ mkdirSync(join(base, '.gsd', 'milestones'), { recursive: true });
29
+ return base;
30
+ }
31
+
32
+ function writeFile(base: string, relativePath: string, content: string): void {
33
+ const full = join(base, '.gsd', relativePath);
34
+ mkdirSync(join(full, '..'), { recursive: true });
35
+ writeFileSync(full, content);
36
+ }
37
+
38
+ function cleanup(base: string): void {
39
+ rmSync(base, { recursive: true, force: true });
40
+ }
41
+
42
+ const ROADMAP_ALL_DONE = `# M001: Finished Milestone
43
+
44
+ **Vision:** Done work.
45
+
46
+ ## Slices
47
+
48
+ - [x] **S01: First Slice** \`risk:low\` \`depends:[]\`
49
+ > After this: Done.
50
+
51
+ - [x] **S02: Second Slice** \`risk:medium\` \`depends:[S01]\`
52
+ > After this: Also done.
53
+ `;
54
+
55
+ const ROADMAP_PARTIAL = `# M002: In-Progress Milestone
56
+
57
+ **Vision:** Mid-flight.
58
+
59
+ ## Slices
60
+
61
+ - [x] **S01: Done Slice** \`risk:low\` \`depends:[]\`
62
+ > After this: Done.
63
+
64
+ - [ ] **S02: Pending Slice** \`risk:medium\` \`depends:[S01]\`
65
+ > After this: TBD.
66
+ `;
67
+
68
+ const ROADMAP_EMPTY = `# M003: Empty Milestone
69
+
70
+ **Vision:** No slices yet.
71
+
72
+ ## Slices
73
+
74
+ `;
75
+
76
+ describe('migrateHierarchyToDb: all-done milestones import as complete (#4902)', () => {
77
+ test('milestone with all [x] slices and no SUMMARY imports as complete', () => {
78
+ const base = createFixtureBase();
79
+ try {
80
+ writeFile(base, 'milestones/M001/M001-ROADMAP.md', ROADMAP_ALL_DONE);
81
+ // No SUMMARY.md — the all-done roadmap check is the authoritative signal.
82
+
83
+ openDatabase(':memory:');
84
+ migrateHierarchyToDb(base);
85
+
86
+ const milestones = getAllMilestones();
87
+ const m001 = milestones.find((m) => m.id === 'M001');
88
+ assert.ok(m001, 'M001 should be imported');
89
+ assert.equal(
90
+ m001!.status,
91
+ 'complete',
92
+ 'milestone with all-done slices must import as complete',
93
+ );
94
+ } finally {
95
+ closeDatabase();
96
+ cleanup(base);
97
+ }
98
+ });
99
+
100
+ test('milestone with one pending slice imports as active (negative case)', () => {
101
+ const base = createFixtureBase();
102
+ try {
103
+ writeFile(base, 'milestones/M002/M002-ROADMAP.md', ROADMAP_PARTIAL);
104
+
105
+ openDatabase(':memory:');
106
+ migrateHierarchyToDb(base);
107
+
108
+ const milestones = getAllMilestones();
109
+ const m002 = milestones.find((m) => m.id === 'M002');
110
+ assert.ok(m002, 'M002 should be imported');
111
+ assert.equal(
112
+ m002!.status,
113
+ 'active',
114
+ 'milestone with at least one pending slice must NOT be marked complete',
115
+ );
116
+ } finally {
117
+ closeDatabase();
118
+ cleanup(base);
119
+ }
120
+ });
121
+
122
+ test('milestone with empty slice list does not import as complete', () => {
123
+ // Guards the `roadmap.slices.length > 0` precondition: an empty roadmap
124
+ // must not be misread as "everything is done" (vacuous truth bug).
125
+ const base = createFixtureBase();
126
+ try {
127
+ writeFile(base, 'milestones/M003/M003-ROADMAP.md', ROADMAP_EMPTY);
128
+
129
+ openDatabase(':memory:');
130
+ migrateHierarchyToDb(base);
131
+
132
+ const milestones = getAllMilestones();
133
+ const m003 = milestones.find((m) => m.id === 'M003');
134
+ assert.ok(m003, 'M003 should be imported');
135
+ assert.notEqual(
136
+ m003!.status,
137
+ 'complete',
138
+ 'empty roadmap (no slices) must not import as complete',
139
+ );
140
+ } finally {
141
+ closeDatabase();
142
+ cleanup(base);
143
+ }
144
+ });
145
+ });
@@ -12,7 +12,7 @@ import { tmpdir } from "node:os";
12
12
  import { join } from "node:path";
13
13
 
14
14
  import { mapInitPrefsToWizardShape } from "../init-wizard.ts";
15
- import { writePreferencesFile } from "../commands-prefs-wizard.ts";
15
+ import { handlePrefsWizard, writePreferencesFile } from "../commands-prefs-wizard.ts";
16
16
 
17
17
  test("mapInitPrefsToWizardShape — full roundtrip with all fields", () => {
18
18
  const out = mapInitPrefsToWizardShape({
@@ -123,6 +123,69 @@ test("writePreferencesFile — falls back to default body for new files", async
123
123
  }
124
124
  });
125
125
 
126
+ test("handlePrefsWizard — Advanced config writes min_request_interval_ms", async () => {
127
+ const tmp = mkdtempSync(join(tmpdir(), "gsd-init-prefs-routing-"));
128
+ const path = join(tmp, "PREFERENCES.md");
129
+
130
+ try {
131
+ const selectResponses = [
132
+ "Advanced",
133
+ "(keep current)",
134
+ "(keep current)",
135
+ "(keep current)",
136
+ "(keep current)",
137
+ "(keep current)",
138
+ "(keep current)",
139
+ "(keep current)",
140
+ "── Save & Exit ──",
141
+ ];
142
+ const inputResponses = ["250"];
143
+ const ctx = {
144
+ ui: {
145
+ notify: () => {},
146
+ select: async (_label: string, options: string[]) => {
147
+ const response = selectResponses.shift();
148
+ if (response === undefined) {
149
+ throw new Error(
150
+ `Unexpected extra select prompt in handlePrefsWizard flow: selectResponses queue exhausted for "${_label}" ` +
151
+ "(expected no additional select prompts)",
152
+ );
153
+ }
154
+ if (response === "Advanced") {
155
+ const advancedOption = options.find((option) => option.startsWith("Advanced"));
156
+ if (!advancedOption) {
157
+ throw new Error(`Expected an "Advanced" option in "${_label}" menu`);
158
+ }
159
+ return advancedOption;
160
+ }
161
+ return response;
162
+ },
163
+ input: async () => {
164
+ const response = inputResponses.shift();
165
+ if (response === undefined) {
166
+ throw new Error(
167
+ "Unexpected extra input prompt in handlePrefsWizard flow: inputResponses queue exhausted " +
168
+ "(expected no additional input prompts)",
169
+ );
170
+ }
171
+ return response;
172
+ },
173
+ },
174
+ waitForIdle: async () => {},
175
+ reload: async () => {},
176
+ };
177
+
178
+ await handlePrefsWizard(ctx as any, "project", {}, { pathOverride: path });
179
+
180
+ assert.equal(selectResponses.length, 0, "Expected all queued selectResponses to be consumed");
181
+ assert.equal(inputResponses.length, 0, "Expected all queued inputResponses to be consumed");
182
+ const content = readFileSync(path, "utf-8");
183
+ assert.match(content, /^min_request_interval_ms:\s*250$/m);
184
+ } finally {
185
+ rmSync(tmp, { recursive: true, force: true });
186
+ }
187
+ });
188
+
126
189
  // ─── Regression tests from #4457 codex adversarial review ──────────────────
127
190
 
128
191
  test("init — Step 9b shape: 'not_yet' option is recognized as defer (#4457 review)", async () => {
@@ -345,4 +345,26 @@ describe("auto-worktree lifecycle", () => {
345
345
  teardownAutoWorktree(tempDir, "M003");
346
346
  }
347
347
  });
348
+
349
+ test("#2482: throws GSDError when repo has no commits", () => {
350
+ // Create a bare git init with no commits — HEAD is invalid
351
+ tempDir = realpathSync(mkdtempSync(join(tmpdir(), "auto-wt-empty-")));
352
+ run("git init", tempDir);
353
+ run("git config user.email test@test.com", tempDir);
354
+ run("git config user.name Test", tempDir);
355
+
356
+ assert.throws(
357
+ () => createAutoWorktree(tempDir, "M001"),
358
+ (err: unknown) => {
359
+ assert.ok(err instanceof Error, "should throw an Error");
360
+ assert.ok("code" in err, "should have a code property (GSDError)");
361
+ assert.strictEqual((err as { code: string }).code, "GSD_GIT_ERROR");
362
+ assert.ok(
363
+ err.message.includes("repository has no commits yet"),
364
+ `message should mention no commits, got: ${err.message}`,
365
+ );
366
+ return true;
367
+ },
368
+ );
369
+ });
348
370
  });
@@ -338,27 +338,4 @@ console.log('\n=== token-savings: quality — correct scoping, no cross-contamin
338
338
  rmSync(base, { recursive: true, force: true });
339
339
  }
340
340
 
341
- // ═══════════════════════════════════════════════════════════════════════════
342
- // Test: Fixture data realism — sufficient volume and distribution
343
- // ═══════════════════════════════════════════════════════════════════════════
344
-
345
- console.log('\n=== token-savings: fixture data realism ===');
346
- {
347
- // Verify fixture generators produce sufficient volume
348
- assert.ok(DECISIONS_COUNT >= 20, `decisions count ≥ 20 (actual: ${DECISIONS_COUNT})`);
349
- assert.ok(REQUIREMENTS_COUNT >= 20, `requirements count ≥ 20 (actual: ${REQUIREMENTS_COUNT})`);
350
- assert.ok(MILESTONES.length >= 3, `milestones ≥ 3 (actual: ${MILESTONES.length})`);
351
- assert.ok(SLICE_ASSIGNMENTS.length >= 5, `slice assignments ≥ 5 (actual: ${SLICE_ASSIGNMENTS.length})`);
352
-
353
- // Verify markdown content is substantial
354
- assert.ok(decisionsMarkdown.length > 1000, `decisions markdown > 1000 chars (actual: ${decisionsMarkdown.length})`);
355
- assert.ok(requirementsMarkdown.length > 1000, `requirements markdown > 1000 chars (actual: ${requirementsMarkdown.length})`);
356
-
357
- // Verify content structure
358
- assert.match(decisionsMarkdown, /\| D001 \|/, 'decisions markdown has D001');
359
- assert.match(decisionsMarkdown, /\| D024 \|/, 'decisions markdown has D024');
360
- assert.match(requirementsMarkdown, /### R001/, 'requirements markdown has R001');
361
- assert.match(requirementsMarkdown, /### R021/, 'requirements markdown has R021');
362
- }
363
-
364
341
  // ─── Report ────────────────────────────────────────────────────────────────
@@ -4,6 +4,11 @@ import {
4
4
  isDbAvailable,
5
5
  _getAdapter,
6
6
  } from '../gsd-db.ts';
7
+ import {
8
+ _resetLogs,
9
+ peekLogs,
10
+ setStderrLoggingEnabled,
11
+ } from '../workflow-logger.ts';
7
12
  import {
8
13
  getActiveMemories,
9
14
  getActiveMemoriesRanked,
@@ -329,3 +334,126 @@ test('memory-store: schema includes memories table', () => {
329
334
 
330
335
  closeDatabase();
331
336
  });
337
+
338
+ // ═══════════════════════════════════════════════════════════════════════════
339
+ // regression #4967 — createMemory must not silently swallow SQL errors
340
+ // ═══════════════════════════════════════════════════════════════════════════
341
+
342
+ test('memory-store: createMemory throws on memory-table SQL errors (regression #4967)', () => {
343
+ openDatabase(':memory:');
344
+
345
+ const adapter = _getAdapter()!;
346
+ // Drop FTS + dependents first to satisfy SQLite's trigger ordering, then
347
+ // the base memories table. IF EXISTS makes setup robust against schema
348
+ // versions that may not have created every dependent (e.g. embeddings).
349
+ adapter.prepare('DROP TABLE IF EXISTS memory_embeddings').run();
350
+ adapter.prepare('DROP TABLE IF EXISTS memories_fts').run();
351
+ adapter.prepare('DROP TABLE IF EXISTS memories').run();
352
+
353
+ // Pre-fix behaviour: returns null and the caller has no idea why.
354
+ // Post-fix behaviour: throws so the caller can surface the real SQL message.
355
+ assert.throws(
356
+ () => createMemory({ category: 'gotcha', content: 'broken store' }),
357
+ /memories|no such table/i,
358
+ 'createMemory must surface SQL errors instead of returning null',
359
+ );
360
+
361
+ closeDatabase();
362
+ });
363
+
364
+ test('memory-store: VACUUM retry rolls back partial memory and logs recovery', () => {
365
+ openDatabase(':memory:');
366
+
367
+ const adapter = _getAdapter()!;
368
+ const originalPrepareMethod = adapter.prepare;
369
+ const originalPrepare = adapter.prepare.bind(adapter);
370
+ const previousStderrLogging = setStderrLoggingEnabled(false);
371
+ const streamAny = process.stderr as unknown as {
372
+ write: (chunk: string | Uint8Array, ...rest: unknown[]) => boolean;
373
+ };
374
+ const originalStderrWrite = streamAny.write.bind(streamAny);
375
+ let selectFailures = 0;
376
+ let vacuumRuns = 0;
377
+ _resetLogs();
378
+
379
+ adapter.prepare = ((sql: string) => {
380
+ if (sql === 'SELECT seq FROM memories WHERE id = :id' && selectFailures === 0) {
381
+ const stmt = originalPrepare(sql);
382
+ return {
383
+ run: (...params: unknown[]) => stmt.run(...params),
384
+ get: (..._params: unknown[]) => {
385
+ selectFailures++;
386
+ throw new Error('database disk image is malformed');
387
+ },
388
+ all: (...params: unknown[]) => stmt.all(...params),
389
+ };
390
+ }
391
+
392
+ if (sql === 'VACUUM') {
393
+ const stmt = originalPrepare(sql);
394
+ return {
395
+ run: (...params: unknown[]) => {
396
+ vacuumRuns++;
397
+ return stmt.run(...params);
398
+ },
399
+ get: (...params: unknown[]) => stmt.get(...params),
400
+ all: (...params: unknown[]) => stmt.all(...params),
401
+ };
402
+ }
403
+
404
+ return originalPrepare(sql);
405
+ }) as typeof adapter.prepare;
406
+ streamAny.write = (): boolean => true;
407
+
408
+ try {
409
+ const id = createMemory({ category: 'gotcha', content: 'recover without duplicate' });
410
+ assert.equal(id, 'MEM001', 'retry should create a single first memory');
411
+
412
+ const rows = adapter.prepare('SELECT id FROM memories ORDER BY seq').all();
413
+ assert.deepStrictEqual(
414
+ rows.map((row) => row['id']),
415
+ ['MEM001'],
416
+ 'failed first attempt should not leave a live _TMP_ memory behind',
417
+ );
418
+ assert.equal(selectFailures, 1, 'test should simulate one malformed SELECT after INSERT');
419
+ assert.equal(vacuumRuns, 1, 'malformed recovery should run VACUUM once');
420
+ assert.ok(
421
+ peekLogs().some((entry) =>
422
+ entry.component === 'memory-store' &&
423
+ entry.message === 'recovered malformed memory store via VACUUM'
424
+ ),
425
+ 'successful VACUUM recovery should be emitted to the workflow logger',
426
+ );
427
+ } finally {
428
+ adapter.prepare = originalPrepareMethod;
429
+ streamAny.write = originalStderrWrite;
430
+ setStderrLoggingEnabled(previousStderrLogging);
431
+ _resetLogs();
432
+ closeDatabase();
433
+ }
434
+ });
435
+
436
+ test('memory-store: applyMemoryActions stays non-fatal when memory store is broken (regression #4967)', () => {
437
+ openDatabase(':memory:');
438
+
439
+ const adapter = _getAdapter()!;
440
+ // Drop FTS + dependents first to satisfy SQLite's trigger ordering, then
441
+ // the base memories table. IF EXISTS makes setup robust against schema
442
+ // versions that may not have created every dependent (e.g. embeddings).
443
+ adapter.prepare('DROP TABLE IF EXISTS memory_embeddings').run();
444
+ adapter.prepare('DROP TABLE IF EXISTS memories_fts').run();
445
+ adapter.prepare('DROP TABLE IF EXISTS memories').run();
446
+
447
+ // applyMemoryActions wraps createMemory in a transaction with an outer
448
+ // catch. Even with createMemory now throwing, applyMemoryActions must not
449
+ // crash the auto-mode flow that calls it (memory extraction is best-effort).
450
+ const actions: MemoryAction[] = [
451
+ { action: 'CREATE', category: 'gotcha', content: 'inside-transaction call' },
452
+ ];
453
+ assert.doesNotThrow(
454
+ () => applyMemoryActions(actions),
455
+ 'applyMemoryActions must absorb thrown errors so callers continue',
456
+ );
457
+
458
+ closeDatabase();
459
+ });
@@ -1,7 +1,7 @@
1
1
  import { test } from 'node:test';
2
2
  import assert from 'node:assert/strict';
3
3
 
4
- import { closeDatabase, openDatabase } from '../gsd-db.ts';
4
+ import { _getAdapter, closeDatabase, openDatabase } from '../gsd-db.ts';
5
5
  import { createMemory, supersedeMemory } from '../memory-store.ts';
6
6
  import {
7
7
  executeGsdGraph,
@@ -293,3 +293,35 @@ test('memory-tools: gsd_graph errors when DB is closed', () => {
293
293
  assert.ok(result.isError);
294
294
  assert.equal(result.details.error, 'db_unavailable');
295
295
  });
296
+
297
+ // ═══════════════════════════════════════════════════════════════════════════
298
+ // regression #4967 — capture_thought must surface real SQL errors
299
+ // ═══════════════════════════════════════════════════════════════════════════
300
+
301
+ test('memory-tools: capture_thought surfaces underlying SQL error (regression #4967)', () => {
302
+ openDatabase(':memory:');
303
+
304
+ // Simulate the real-world failure mode where the memories table is gone.
305
+ // The original bug was a "database disk image is malformed" on the memory
306
+ // store; dropping the table produces a deterministic statement-time SQL
307
+ // error of the same shape — a thrown sqlite error during INSERT.
308
+ const adapter = _getAdapter()!;
309
+ adapter.prepare('DROP TABLE IF EXISTS memory_embeddings').run();
310
+ adapter.prepare('DROP TABLE IF EXISTS memories_fts').run();
311
+ adapter.prepare('DROP TABLE IF EXISTS memories').run();
312
+
313
+ const result = executeMemoryCapture({
314
+ category: 'gotcha',
315
+ content: 'should reveal the real reason',
316
+ });
317
+
318
+ assert.ok(result.isError, 'broken store should produce an error result');
319
+ assert.equal(result.details.operation, 'memory_capture');
320
+
321
+ const err = String(result.details.error ?? '');
322
+ assert.notEqual(err, 'create_failed', 'must not collapse to opaque create_failed');
323
+ assert.ok(err.length > 0, 'error detail must be populated');
324
+ assert.match(err, /memories|no such table/i, 'error must reference the underlying SQL fault');
325
+
326
+ closeDatabase();
327
+ });
@@ -0,0 +1,124 @@
1
+ // gsd-2 / merge-self-branch-guard.test.ts — regression for #5024
2
+ //
3
+ // mergeMilestoneToMain() must fail closed when the resolved integration
4
+ // branch is the same ref as the milestone branch. Stale or corrupt
5
+ // integration metadata (e.g. integrationBranch recorded as "milestone/<MID>")
6
+ // would otherwise let the squash merge resolve to a self-merge: the post-
7
+ // merge no-op safety check (#1792) compares main vs milestone and finds an
8
+ // empty diff (because they're the same ref), so the helper returns success
9
+ // for work that never landed on a distinct integration branch.
10
+
11
+ import test from "node:test";
12
+ import assert from "node:assert/strict";
13
+ import {
14
+ mkdtempSync,
15
+ mkdirSync,
16
+ rmSync,
17
+ existsSync,
18
+ realpathSync,
19
+ writeFileSync,
20
+ } from "node:fs";
21
+ import { join } from "node:path";
22
+ import { tmpdir } from "node:os";
23
+ import { execFileSync } from "node:child_process";
24
+
25
+ import { mergeMilestoneToMain } from "../auto-worktree.ts";
26
+ import { _resetServiceCache } from "../worktree.ts";
27
+ import { _clearGsdRootCache } from "../paths.ts";
28
+
29
+ function git(args: string[], cwd: string): string {
30
+ return execFileSync("git", args, { cwd, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8" }).trim();
31
+ }
32
+
33
+ function createTempRepo(): string {
34
+ const dir = realpathSync(mkdtempSync(join(tmpdir(), "merge-self-guard-")));
35
+ git(["init"], dir);
36
+ git(["config", "user.email", "test@test.com"], dir);
37
+ git(["config", "user.name", "Test"], dir);
38
+ writeFileSync(join(dir, "README.md"), "# test\n");
39
+ git(["add", "."], dir);
40
+ git(["commit", "-m", "init"], dir);
41
+ git(["branch", "-M", "main"], dir);
42
+ return dir;
43
+ }
44
+
45
+ function assertSelfMergeRefIsRejected(recordedIntegrationBranch: string): void {
46
+ const savedCwd = process.cwd();
47
+ let tempDir = "";
48
+
49
+ // Isolate from user's global preferences so prefs.main_branch can't
50
+ // override the corrupt-metadata path under test.
51
+ const originalHome = process.env.HOME;
52
+ const fakeHome = realpathSync(mkdtempSync(join(tmpdir(), "gsd-fake-home-")));
53
+ process.env.HOME = fakeHome;
54
+ _clearGsdRootCache();
55
+ _resetServiceCache();
56
+
57
+ try {
58
+ tempDir = createTempRepo();
59
+
60
+ // Plant corrupt integration metadata: integrationBranch points at the
61
+ // milestone branch itself. Commit it so mergeMilestoneToMain's
62
+ // autoCommitDirtyState pre-step has nothing to capture and the
63
+ // postcondition (no new commits) cleanly reflects the guard.
64
+ const msDir = join(tempDir, ".gsd", "milestones", "M001");
65
+ mkdirSync(msDir, { recursive: true });
66
+ writeFileSync(
67
+ join(msDir, "M001-META.json"),
68
+ JSON.stringify({ integrationBranch: recordedIntegrationBranch }),
69
+ );
70
+ git(["add", "."], tempDir);
71
+ git(["commit", "-m", "chore: plant corrupt M001 meta"], tempDir);
72
+
73
+ // Create the milestone branch ref so any pre-guard branch operations
74
+ // wouldn't fail for unrelated reasons.
75
+ git(["branch", "milestone/M001"], tempDir);
76
+
77
+ const mainHeadBefore = git(["rev-parse", "main"], tempDir);
78
+ const milestoneHeadBefore = git(["rev-parse", "milestone/M001"], tempDir);
79
+
80
+ process.chdir(tempDir);
81
+
82
+ assert.throws(
83
+ () => mergeMilestoneToMain(tempDir, "M001", ""),
84
+ (err: unknown) => {
85
+ assert.ok(err instanceof Error, "expected an Error to be thrown");
86
+ assert.match(
87
+ err.message,
88
+ /self-merge|same ref/i,
89
+ "error message should explain the self-merge refusal",
90
+ );
91
+ return true;
92
+ },
93
+ );
94
+
95
+ // Postcondition: neither branch should have been advanced by a merge
96
+ // commit. The guard fires before checkout/merge, so both refs must be
97
+ // unchanged from their pre-call state.
98
+ const mainHeadAfter = git(["rev-parse", "main"], tempDir);
99
+ const milestoneHeadAfter = git(["rev-parse", "milestone/M001"], tempDir);
100
+ assert.equal(mainHeadAfter, mainHeadBefore, "main must not have advanced");
101
+ assert.equal(
102
+ milestoneHeadAfter,
103
+ milestoneHeadBefore,
104
+ "milestone branch must not have advanced",
105
+ );
106
+ } finally {
107
+ process.chdir(savedCwd);
108
+ process.env.HOME = originalHome;
109
+ _clearGsdRootCache();
110
+ _resetServiceCache();
111
+ if (tempDir && existsSync(tempDir)) {
112
+ rmSync(tempDir, { recursive: true, force: true });
113
+ }
114
+ rmSync(fakeHome, { recursive: true, force: true });
115
+ }
116
+ }
117
+
118
+ test("mergeMilestoneToMain refuses exact milestone branch self-merge metadata (#5024)", () => {
119
+ assertSelfMergeRefIsRejected("milestone/M001");
120
+ });
121
+
122
+ test("mergeMilestoneToMain refuses refs/heads milestone branch self-merge metadata (#5024)", () => {
123
+ assertSelfMergeRefIsRejected("refs/heads/milestone/M001");
124
+ });