gsd-pi 2.77.0-dev.58d3d4d6c → 2.77.0-dev.cfd69e714

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 (429) hide show
  1. package/README.md +1 -1
  2. package/dist/claude-cli-check.js +5 -1
  3. package/dist/headless.js +49 -4
  4. package/dist/resource-loader.d.ts +40 -0
  5. package/dist/resource-loader.js +32 -13
  6. package/dist/resources/extensions/browser-tools/capture.js +9 -0
  7. package/dist/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +8 -59
  8. package/dist/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +36 -24
  9. package/dist/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +69 -71
  10. package/dist/resources/extensions/browser-tools/tools/forms.js +5 -1
  11. package/dist/resources/extensions/browser-tools/tools/intent.js +5 -1
  12. package/dist/resources/extensions/claude-code-cli/readiness.js +5 -1
  13. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +481 -17
  14. package/dist/resources/extensions/gsd/auto/loop.js +43 -0
  15. package/dist/resources/extensions/gsd/auto/phases.js +15 -21
  16. package/dist/resources/extensions/gsd/auto/session.js +0 -2
  17. package/dist/resources/extensions/gsd/auto-dispatch.js +102 -24
  18. package/dist/resources/extensions/gsd/auto-model-selection.js +124 -4
  19. package/dist/resources/extensions/gsd/auto-post-unit.js +71 -64
  20. package/dist/resources/extensions/gsd/auto-prompts.js +329 -102
  21. package/dist/resources/extensions/gsd/auto-recovery.js +195 -23
  22. package/dist/resources/extensions/gsd/auto-start.js +34 -24
  23. package/dist/resources/extensions/gsd/auto-tool-tracking.js +47 -7
  24. package/dist/resources/extensions/gsd/auto-worktree.js +122 -26
  25. package/dist/resources/extensions/gsd/auto.js +31 -20
  26. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +9 -1
  27. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +209 -0
  28. package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +3 -6
  29. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +7 -3
  30. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +127 -9
  31. package/dist/resources/extensions/gsd/component-loader.js +447 -0
  32. package/dist/resources/extensions/gsd/component-types.js +69 -0
  33. package/dist/resources/extensions/gsd/detection.js +49 -1
  34. package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  35. package/dist/resources/extensions/gsd/gate-registry.js +2 -2
  36. package/dist/resources/extensions/gsd/git-constants.js +28 -1
  37. package/dist/resources/extensions/gsd/git-self-heal.js +27 -0
  38. package/dist/resources/extensions/gsd/git-service.js +126 -2
  39. package/dist/resources/extensions/gsd/gsd-db.js +6 -3
  40. package/dist/resources/extensions/gsd/guided-flow.js +17 -5
  41. package/dist/resources/extensions/gsd/memory-extractor.js +7 -1
  42. package/dist/resources/extensions/gsd/milestone-scope-classifier.js +299 -0
  43. package/dist/resources/extensions/gsd/model-cost-table.js +3 -0
  44. package/dist/resources/extensions/gsd/model-router.js +6 -0
  45. package/dist/resources/extensions/gsd/native-git-bridge.js +34 -4
  46. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
  47. package/dist/resources/extensions/gsd/prompts/plan-slice.md +15 -2
  48. package/dist/resources/extensions/gsd/safety/git-checkpoint.js +11 -0
  49. package/dist/resources/extensions/gsd/service-tier.js +5 -2
  50. package/dist/resources/extensions/gsd/session-lock.js +19 -10
  51. package/dist/resources/extensions/gsd/skill-manifest.js +168 -0
  52. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +278 -8
  53. package/dist/resources/extensions/gsd/state.js +44 -33
  54. package/dist/resources/extensions/gsd/sync-lock.js +98 -42
  55. package/dist/resources/extensions/gsd/unit-context-composer.js +147 -0
  56. package/dist/resources/extensions/gsd/unit-context-manifest.js +370 -0
  57. package/dist/resources/extensions/gsd/uok/gate-runner.js +53 -5
  58. package/dist/resources/extensions/gsd/workflow-mcp.js +6 -0
  59. package/dist/resources/extensions/gsd/worktree-manager.js +34 -8
  60. package/dist/resources/extensions/mcp-client/index.js +3 -1
  61. package/dist/resources/extensions/ollama/index.js +5 -1
  62. package/dist/resources/extensions/remote-questions/manager.js +11 -5
  63. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  64. package/dist/web/standalone/.next/BUILD_ID +1 -1
  65. package/dist/web/standalone/.next/app-path-routes-manifest.json +5 -5
  66. package/dist/web/standalone/.next/build-manifest.json +2 -2
  67. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  68. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  69. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  70. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  71. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  72. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  73. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  74. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  75. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  76. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  77. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  78. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  79. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  80. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  81. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  82. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  83. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  85. package/dist/web/standalone/.next/server/app/index.html +1 -1
  86. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  87. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  88. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  89. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  90. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  91. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  92. package/dist/web/standalone/.next/server/app-paths-manifest.json +5 -5
  93. package/dist/web/standalone/.next/server/chunks/1926.js +1 -1
  94. package/dist/web/standalone/.next/server/chunks/6897.js +1 -1
  95. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  96. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  97. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  98. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  99. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  100. package/package.json +2 -3
  101. package/packages/daemon/src/logger.ts +4 -3
  102. package/packages/mcp-server/dist/server.d.ts +24 -0
  103. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  104. package/packages/mcp-server/dist/server.js +88 -87
  105. package/packages/mcp-server/dist/server.js.map +1 -1
  106. package/packages/mcp-server/src/mcp-server.test.ts +25 -3
  107. package/packages/mcp-server/src/readers/graph.test.ts +87 -15
  108. package/packages/mcp-server/src/secure-env-collect.test.ts +232 -237
  109. package/packages/mcp-server/src/server.ts +131 -105
  110. package/packages/mcp-server/src/workflow-tools.test.ts +80 -39
  111. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  112. package/packages/native/package.json +1 -1
  113. package/packages/native/src/__tests__/_test-coverage-guard.test.mjs +98 -0
  114. package/packages/native/src/__tests__/module-compat.test.mjs +59 -27
  115. package/packages/native/src/__tests__/ps.test.mjs +14 -8
  116. package/packages/native/src/__tests__/stream-process.test.mjs +23 -2
  117. package/packages/native/src/__tests__/truncate.test.mjs +17 -2
  118. package/packages/pi-agent-core/src/agent-loop.test.ts +5 -15
  119. package/packages/pi-agent-core/src/agent.test.ts +96 -102
  120. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  121. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -1
  122. package/packages/pi-ai/dist/models/capability-patches.js +9 -2
  123. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -1
  124. package/packages/pi-ai/dist/models/generated/index.d.ts +34 -0
  125. package/packages/pi-ai/dist/models/generated/index.d.ts.map +1 -1
  126. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +17 -0
  127. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -1
  128. package/packages/pi-ai/dist/models/generated/openai-codex.js +17 -0
  129. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -1
  130. package/packages/pi-ai/dist/models/generated/openai.d.ts +17 -0
  131. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -1
  132. package/packages/pi-ai/dist/models/generated/openai.js +17 -0
  133. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -1
  134. package/packages/pi-ai/dist/models.generated.test.js +43 -70
  135. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  136. package/packages/pi-ai/dist/models.test.js +36 -11
  137. package/packages/pi-ai/dist/models.test.js.map +1 -1
  138. package/packages/pi-ai/scripts/generate-models.ts +44 -0
  139. package/packages/pi-ai/src/models/capability-patches.ts +10 -2
  140. package/packages/pi-ai/src/models/generated/openai-codex.ts +17 -0
  141. package/packages/pi-ai/src/models/generated/openai.ts +17 -0
  142. package/packages/pi-ai/src/models.generated.test.ts +46 -73
  143. package/packages/pi-ai/src/models.test.ts +48 -11
  144. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  145. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +96 -32
  146. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
  147. package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js +75 -12
  148. package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js.map +1 -1
  149. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js +99 -31
  150. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js.map +1 -1
  151. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +5 -0
  152. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  153. package/packages/pi-coding-agent/dist/core/extensions/loader.js +61 -0
  154. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  155. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js +30 -4
  156. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js.map +1 -1
  157. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +17 -0
  158. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
  159. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js +76 -18
  160. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js.map +1 -1
  161. package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
  162. package/packages/pi-coding-agent/dist/core/retry-handler.js +2 -6
  163. package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
  164. package/packages/pi-coding-agent/dist/core/retry-handler.test.js +5 -1
  165. package/packages/pi-coding-agent/dist/core/retry-handler.test.js.map +1 -1
  166. package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts +18 -0
  167. package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts.map +1 -0
  168. package/packages/pi-coding-agent/dist/core/retryable-error-regex.js +18 -0
  169. package/packages/pi-coding-agent/dist/core/retryable-error-regex.js.map +1 -0
  170. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts +20 -0
  171. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  172. package/packages/pi-coding-agent/dist/core/system-prompt.js +16 -2
  173. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  174. package/packages/pi-coding-agent/dist/index.d.ts +1 -0
  175. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  176. package/packages/pi-coding-agent/dist/index.js +1 -0
  177. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  178. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +20 -13
  179. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -1
  180. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
  181. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +18 -3
  182. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  183. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +125 -0
  184. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  185. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts +2 -0
  186. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts.map +1 -1
  187. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.js.map +1 -1
  188. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +4 -0
  189. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  190. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +105 -13
  191. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  192. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts +2 -0
  193. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts.map +1 -0
  194. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js +130 -0
  195. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js.map +1 -0
  196. package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +113 -37
  197. package/packages/pi-coding-agent/src/core/agent-session-model-switch.test.ts +89 -17
  198. package/packages/pi-coding-agent/src/core/agent-session-tool-refresh.test.ts +112 -43
  199. package/packages/pi-coding-agent/src/core/extensions/loader.ts +58 -0
  200. package/packages/pi-coding-agent/src/core/lsp/lsp-integration.test.ts +35 -4
  201. package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +20 -0
  202. package/packages/pi-coding-agent/src/core/resource-loader-cache-reset.test.ts +93 -28
  203. package/packages/pi-coding-agent/src/core/retry-handler.test.ts +5 -1
  204. package/packages/pi-coding-agent/src/core/retry-handler.ts +2 -8
  205. package/packages/pi-coding-agent/src/core/retryable-error-regex.ts +18 -0
  206. package/packages/pi-coding-agent/src/core/system-prompt.ts +35 -1
  207. package/packages/pi-coding-agent/src/index.ts +1 -0
  208. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +26 -20
  209. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +146 -1
  210. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +20 -3
  211. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-state.ts +2 -0
  212. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +119 -13
  213. package/packages/pi-coding-agent/src/tests/system-prompt-skill-filter.test.ts +157 -0
  214. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  215. package/packages/pi-tui/dist/__tests__/autocomplete.test.js +18 -8
  216. package/packages/pi-tui/dist/__tests__/autocomplete.test.js.map +1 -1
  217. package/packages/pi-tui/dist/__tests__/overlay-layout.test.js +128 -17
  218. package/packages/pi-tui/dist/__tests__/overlay-layout.test.js.map +1 -1
  219. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js +36 -12
  220. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js.map +1 -1
  221. package/packages/pi-tui/dist/__tests__/tui.test.js +18 -30
  222. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  223. package/packages/pi-tui/dist/components/__tests__/input.test.js +10 -3
  224. package/packages/pi-tui/dist/components/__tests__/input.test.js.map +1 -1
  225. package/packages/pi-tui/dist/components/__tests__/loader.test.js +53 -9
  226. package/packages/pi-tui/dist/components/__tests__/loader.test.js.map +1 -1
  227. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js +6 -2
  228. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js.map +1 -1
  229. package/packages/pi-tui/dist/components/editor.d.ts +14 -0
  230. package/packages/pi-tui/dist/components/editor.d.ts.map +1 -1
  231. package/packages/pi-tui/dist/components/editor.js +19 -0
  232. package/packages/pi-tui/dist/components/editor.js.map +1 -1
  233. package/packages/pi-tui/dist/components/image.test.js +6 -5
  234. package/packages/pi-tui/dist/components/image.test.js.map +1 -1
  235. package/packages/pi-tui/dist/editor-component.d.ts +2 -0
  236. package/packages/pi-tui/dist/editor-component.d.ts.map +1 -1
  237. package/packages/pi-tui/dist/editor-component.js.map +1 -1
  238. package/packages/pi-tui/src/__tests__/autocomplete.test.ts +24 -8
  239. package/packages/pi-tui/src/__tests__/overlay-layout.test.ts +140 -17
  240. package/packages/pi-tui/src/__tests__/stdin-buffer.test.ts +41 -12
  241. package/packages/pi-tui/src/__tests__/tui.test.ts +18 -37
  242. package/packages/pi-tui/src/components/__tests__/input.test.ts +19 -3
  243. package/packages/pi-tui/src/components/__tests__/loader.test.ts +112 -35
  244. package/packages/pi-tui/src/components/__tests__/markdown-maxlines.test.ts +9 -2
  245. package/packages/pi-tui/src/components/editor.ts +22 -0
  246. package/packages/pi-tui/src/components/image.test.ts +10 -5
  247. package/packages/pi-tui/src/editor-component.ts +3 -0
  248. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  249. package/packages/rpc-client/dist/rpc-client.test.js +101 -51
  250. package/packages/rpc-client/dist/rpc-client.test.js.map +1 -1
  251. package/packages/rpc-client/src/rpc-client.test.ts +109 -52
  252. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  253. package/scripts/install.js +15 -1
  254. package/src/resources/extensions/browser-tools/capture.ts +12 -0
  255. package/src/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +8 -59
  256. package/src/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +36 -24
  257. package/src/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +69 -71
  258. package/src/resources/extensions/browser-tools/tools/forms.ts +5 -1
  259. package/src/resources/extensions/browser-tools/tools/intent.ts +5 -1
  260. package/src/resources/extensions/claude-code-cli/readiness.ts +5 -1
  261. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +518 -19
  262. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +919 -75
  263. package/src/resources/extensions/github-sync/tests/cli.test.ts +76 -7
  264. package/src/resources/extensions/github-sync/tests/templates.test.ts +33 -1
  265. package/src/resources/extensions/gsd/auto/loop.ts +47 -0
  266. package/src/resources/extensions/gsd/auto/phases.ts +16 -20
  267. package/src/resources/extensions/gsd/auto/session.ts +0 -2
  268. package/src/resources/extensions/gsd/auto-dispatch.ts +113 -24
  269. package/src/resources/extensions/gsd/auto-model-selection.ts +131 -4
  270. package/src/resources/extensions/gsd/auto-post-unit.ts +82 -73
  271. package/src/resources/extensions/gsd/auto-prompts.ts +330 -90
  272. package/src/resources/extensions/gsd/auto-recovery.ts +225 -24
  273. package/src/resources/extensions/gsd/auto-start.ts +54 -6
  274. package/src/resources/extensions/gsd/auto-tool-tracking.ts +51 -7
  275. package/src/resources/extensions/gsd/auto-worktree.ts +130 -26
  276. package/src/resources/extensions/gsd/auto.ts +43 -22
  277. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +9 -1
  278. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +221 -0
  279. package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +3 -7
  280. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +7 -3
  281. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +158 -9
  282. package/src/resources/extensions/gsd/component-loader.ts +598 -0
  283. package/src/resources/extensions/gsd/component-types.ts +362 -0
  284. package/src/resources/extensions/gsd/detection.ts +58 -1
  285. package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  286. package/src/resources/extensions/gsd/gate-registry.ts +2 -2
  287. package/src/resources/extensions/gsd/git-constants.ts +30 -1
  288. package/src/resources/extensions/gsd/git-self-heal.ts +31 -0
  289. package/src/resources/extensions/gsd/git-service.ts +133 -2
  290. package/src/resources/extensions/gsd/gsd-db.ts +6 -3
  291. package/src/resources/extensions/gsd/guided-flow.ts +20 -5
  292. package/src/resources/extensions/gsd/memory-extractor.ts +11 -3
  293. package/src/resources/extensions/gsd/milestone-scope-classifier.ts +366 -0
  294. package/src/resources/extensions/gsd/model-cost-table.ts +3 -0
  295. package/src/resources/extensions/gsd/model-router.ts +6 -0
  296. package/src/resources/extensions/gsd/native-git-bridge.ts +34 -4
  297. package/src/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
  298. package/src/resources/extensions/gsd/prompts/plan-slice.md +15 -2
  299. package/src/resources/extensions/gsd/safety/git-checkpoint.ts +15 -0
  300. package/src/resources/extensions/gsd/service-tier.ts +5 -2
  301. package/src/resources/extensions/gsd/session-lock.ts +20 -10
  302. package/src/resources/extensions/gsd/skill-manifest.ts +175 -0
  303. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +309 -8
  304. package/src/resources/extensions/gsd/state.ts +49 -44
  305. package/src/resources/extensions/gsd/sync-lock.ts +97 -39
  306. package/src/resources/extensions/gsd/tests/artifact-retry-cap.test.ts +270 -0
  307. package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +341 -0
  308. package/src/resources/extensions/gsd/tests/auto-discuss-milestone-deadlock-4973.test.ts +264 -0
  309. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +94 -289
  310. package/src/resources/extensions/gsd/tests/auto-model-selection-tool-poisoning.test.ts +742 -0
  311. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +78 -0
  312. package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +61 -0
  313. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +93 -0
  314. package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +8 -197
  315. package/src/resources/extensions/gsd/tests/auto-start-needs-discussion.test.ts +15 -58
  316. package/src/resources/extensions/gsd/tests/cache-staleness-regression.test.ts +17 -21
  317. package/src/resources/extensions/gsd/tests/complete-milestone-excerpt.test.ts +263 -0
  318. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +25 -0
  319. package/src/resources/extensions/gsd/tests/complete-slice-composer.test.ts +192 -0
  320. package/src/resources/extensions/gsd/tests/complete-task.test.ts +16 -8
  321. package/src/resources/extensions/gsd/tests/component-loader.test.ts +589 -0
  322. package/src/resources/extensions/gsd/tests/component-types.test.ts +127 -0
  323. package/src/resources/extensions/gsd/tests/crash-recovery.test.ts +50 -1
  324. package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -3
  325. package/src/resources/extensions/gsd/tests/derive-state-db-disk-reconcile.test.ts +40 -0
  326. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +91 -3
  327. package/src/resources/extensions/gsd/tests/derive-state.test.ts +4 -3
  328. package/src/resources/extensions/gsd/tests/dispatcher-stuck-planning.test.ts +3 -2
  329. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +139 -129
  330. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +9 -105
  331. package/src/resources/extensions/gsd/tests/gate-state-canonicalization.test.ts +102 -0
  332. package/src/resources/extensions/gsd/tests/gate-storage.test.ts +1 -1
  333. package/src/resources/extensions/gsd/tests/hook-key-parsing.test.ts +4 -55
  334. package/src/resources/extensions/gsd/tests/integration/all-milestones-complete-merge.test.ts +7 -57
  335. package/src/resources/extensions/gsd/tests/integration/auto-recovery.test.ts +20 -0
  336. package/src/resources/extensions/gsd/tests/integration/doctor-proactive.test.ts +18 -2
  337. package/src/resources/extensions/gsd/tests/integration/queue-completed-milestone-perf.test.ts +10 -4
  338. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +144 -7
  339. package/src/resources/extensions/gsd/tests/integration/state-machine-live-validation.test.ts +4 -0
  340. package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +2 -16
  341. package/src/resources/extensions/gsd/tests/interrupted-session-ui.test.ts +6 -9
  342. package/src/resources/extensions/gsd/tests/mcp-client-security.test.ts +8 -37
  343. package/src/resources/extensions/gsd/tests/memory-extractor.test.ts +5 -15
  344. package/src/resources/extensions/gsd/tests/merge-conflict-stops-loop.test.ts +227 -62
  345. package/src/resources/extensions/gsd/tests/milestone-scope-classifier.test.ts +187 -0
  346. package/src/resources/extensions/gsd/tests/model-cost-table.test.ts +9 -1
  347. package/src/resources/extensions/gsd/tests/model-router.test.ts +1 -1
  348. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +6 -49
  349. package/src/resources/extensions/gsd/tests/notification-widget.test.ts +6 -3
  350. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +273 -133
  351. package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +301 -0
  352. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +32 -1
  353. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +23 -24
  354. package/src/resources/extensions/gsd/tests/queue-auto-guard.test.ts +32 -0
  355. package/src/resources/extensions/gsd/tests/ready-phrase-no-files-4573.test.ts +75 -2
  356. package/src/resources/extensions/gsd/tests/reassess-default-optin.test.ts +132 -0
  357. package/src/resources/extensions/gsd/tests/recovery-attempts-reset.test.ts +8 -40
  358. package/src/resources/extensions/gsd/tests/regex-hardening.test.ts +136 -256
  359. package/src/resources/extensions/gsd/tests/research-milestone-composer.test.ts +114 -0
  360. package/src/resources/extensions/gsd/tests/run-uat-composer.test.ts +148 -0
  361. package/src/resources/extensions/gsd/tests/service-tier.test.ts +4 -0
  362. package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +29 -0
  363. package/src/resources/extensions/gsd/tests/silent-catch-diagnostics.test.ts +55 -95
  364. package/src/resources/extensions/gsd/tests/single-writer-v3-tool-surface.test.ts +158 -0
  365. package/src/resources/extensions/gsd/tests/skill-activation.test.ts +120 -1
  366. package/src/resources/extensions/gsd/tests/skill-manifest.test.ts +112 -0
  367. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +164 -1
  368. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +5 -5
  369. package/src/resources/extensions/gsd/tests/structured-data-formatter.test.ts +11 -92
  370. package/src/resources/extensions/gsd/tests/survivor-branch-complete.test.ts +102 -101
  371. package/src/resources/extensions/gsd/tests/sync-lock.test.ts +31 -0
  372. package/src/resources/extensions/gsd/tests/test-helpers.test.ts +12 -61
  373. package/src/resources/extensions/gsd/tests/test-helpers.ts +21 -8
  374. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +61 -1
  375. package/src/resources/extensions/gsd/tests/tool-naming.test.ts +8 -1
  376. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +355 -0
  377. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +258 -0
  378. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +75 -0
  379. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +49 -26
  380. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +1 -0
  381. package/src/resources/extensions/gsd/tests/verify-artifact-tightened.test.ts +144 -81
  382. package/src/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +20 -54
  383. package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +342 -277
  384. package/src/resources/extensions/gsd/tests/worker-model-override.test.ts +37 -29
  385. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +226 -266
  386. package/src/resources/extensions/gsd/tests/worktree-health-monorepo.test.ts +103 -67
  387. package/src/resources/extensions/gsd/tests/worktree-nested-git-safety.test.ts +92 -90
  388. package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +238 -59
  389. package/src/resources/extensions/gsd/tests/worktree-sync-overwrite-loop.test.ts +113 -161
  390. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +262 -0
  391. package/src/resources/extensions/gsd/tests/write-gate-predicates.test.ts +186 -0
  392. package/src/resources/extensions/gsd/tests/write-gate.test.ts +7 -5
  393. package/src/resources/extensions/gsd/tests/zombie-gsd-state.test.ts +80 -96
  394. package/src/resources/extensions/gsd/types.ts +3 -3
  395. package/src/resources/extensions/gsd/unit-context-composer.ts +218 -0
  396. package/src/resources/extensions/gsd/unit-context-manifest.ts +574 -0
  397. package/src/resources/extensions/gsd/uok/gate-runner.ts +65 -5
  398. package/src/resources/extensions/gsd/workflow-mcp.ts +6 -0
  399. package/src/resources/extensions/gsd/worktree-manager.ts +55 -7
  400. package/src/resources/extensions/mcp-client/index.ts +3 -1
  401. package/src/resources/extensions/mcp-client/tests/server-name-spaces.test.ts +70 -36
  402. package/src/resources/extensions/ollama/index.ts +5 -1
  403. package/src/resources/extensions/ollama/ollama-auth-mode.test.ts +123 -15
  404. package/src/resources/extensions/ollama/ollama-status-indicator.test.ts +206 -19
  405. package/src/resources/extensions/remote-questions/manager.ts +36 -4
  406. package/src/resources/extensions/remote-questions/tests/command-polling.test.ts +200 -190
  407. package/src/resources/extensions/shared/tests/interview-preview.test.ts +11 -3
  408. package/src/resources/extensions/voice/tests/linux-ready.test.ts +129 -113
  409. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts +0 -2
  410. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts.map +0 -1
  411. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js +0 -289
  412. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js.map +0 -1
  413. package/packages/pi-ai/src/utils/oauth/oauth-providers.test.ts +0 -363
  414. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +0 -144
  415. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +0 -157
  416. package/src/resources/extensions/gsd/tests/dashboard-model-label-ordering.test.ts +0 -107
  417. package/src/resources/extensions/gsd/tests/find-missing-summaries-closed.test.ts +0 -48
  418. package/src/resources/extensions/gsd/tests/forensics-context-persist.test.ts +0 -159
  419. package/src/resources/extensions/gsd/tests/forensics-db-completion.test.ts +0 -96
  420. package/src/resources/extensions/gsd/tests/forensics-dedup.test.ts +0 -79
  421. package/src/resources/extensions/gsd/tests/forensics-hook-key-parse.test.ts +0 -75
  422. package/src/resources/extensions/gsd/tests/forensics-journal.test.ts +0 -162
  423. package/src/resources/extensions/gsd/tests/forensics-worktree-telemetry.test.ts +0 -145
  424. package/src/resources/extensions/gsd/tests/gitignore-bg-shell.test.ts +0 -38
  425. package/src/resources/extensions/gsd/tests/gsd-no-project-error.test.ts +0 -73
  426. package/src/resources/extensions/gsd/tests/idle-watchdog-stall-override.test.ts +0 -130
  427. package/src/resources/extensions/gsd/tests/import-done-milestones.test.ts +0 -43
  428. /package/dist/web/standalone/.next/static/{Cev5xrAYA3ZGTRLyjR2fX → SvCJDZPQW104bR1KnBQg1}/_buildManifest.js +0 -0
  429. /package/dist/web/standalone/.next/static/{Cev5xrAYA3ZGTRLyjR2fX → SvCJDZPQW104bR1KnBQg1}/_ssgManifest.js +0 -0
@@ -69,11 +69,11 @@ describe("cache-staleness-regression", () => {
69
69
  ].join('\n');
70
70
  writeMilestoneFile(base, 'M001', 'ROADMAP', roadmap);
71
71
 
72
- // Step 3: WITHOUT invalidation, the old state might be cached
73
- // The state cache has a 100ms TTL, so wait just past it
74
- await new Promise(r => setTimeout(r, 150));
75
-
76
- // Step 4: Invalidate and re-derive — should see the new roadmap
72
+ // Step 3: Explicit invalidation — this is the #1240 fix path. We
73
+ // do NOT rely on the 100ms TTL here; the production code calls
74
+ // invalidateAllCaches() / invalidateStateCache() immediately after
75
+ // writing planning files, so the next deriveState() must see the
76
+ // new roadmap without any wall-clock wait.
77
77
  invalidateAllCaches();
78
78
  invalidateStateCache();
79
79
  const state2 = await deriveState(base);
@@ -100,10 +100,8 @@ describe("cache-staleness-regression", () => {
100
100
  // Simulate: discussion completes, CONTEXT.md is written
101
101
  writeMilestoneFile(base, 'M001', 'CONTEXT', '# M001: Test\n\nFull context after discussion.\n');
102
102
 
103
- // Wait past TTL
104
- await new Promise(r => setTimeout(r, 150));
105
-
106
- // Without invalidation, we'd still see 'needs-discussion'
103
+ // Explicit invalidation is the production fix path for #1249 —
104
+ // no wall-clock wait needed.
107
105
  invalidateAllCaches();
108
106
  invalidateStateCache();
109
107
  const state2 = await deriveState(base);
@@ -116,7 +114,7 @@ describe("cache-staleness-regression", () => {
116
114
  }
117
115
  });
118
116
 
119
- test("state cache TTL: fresh reads after 100ms", async () => {
117
+ test("state cache TTL: within window returns cached; past window re-derives", async () => {
120
118
  const base = createBase();
121
119
  try {
122
120
  writeMilestoneFile(base, 'M001', 'CONTEXT', '# M001\n\nDesc.\n');
@@ -126,7 +124,7 @@ describe("cache-staleness-regression", () => {
126
124
  const state1 = await deriveState(base);
127
125
  assert.strictEqual(state1.phase, 'pre-planning', 'initial: pre-planning');
128
126
 
129
- // Write roadmap immediately
127
+ // Write roadmap immediately — no invalidation
130
128
  writeMilestoneFile(base, 'M001', 'ROADMAP', [
131
129
  '# M001: Test',
132
130
  '',
@@ -136,18 +134,19 @@ describe("cache-staleness-regression", () => {
136
134
  '',
137
135
  ].join('\n'));
138
136
 
139
- // Immediately after writing (within 100ms TTL), the cache might be stale
137
+ // Within the TTL window, deriveState() must return the cached
138
+ // pre-planning state — this is the "cached" half of the TTL
139
+ // contract and the reason invalidateStateCache() exists.
140
140
  const state2 = await deriveState(base);
141
- // This MAY still show pre-planning if within TTL that's expected behavior
141
+ assert.strictEqual(state2.phase, 'pre-planning', 'within TTL: cached pre-planning is returned');
142
142
 
143
- // Wait past TTL
143
+ // Past the TTL + explicit parse-cache flush, the fresh derive must
144
+ // see the new roadmap. invalidateAllCaches() is required because
145
+ // the file-parse cache is independent of the state TTL.
144
146
  await new Promise(r => setTimeout(r, 150));
145
-
146
- // ALSO invalidate parse cache (not just state cache)
147
147
  invalidateAllCaches();
148
- invalidateStateCache();
149
148
  const state3 = await deriveState(base);
150
- assert.strictEqual(state3.phase, 'planning', 'after TTL expiry + invalidation → planning');
149
+ assert.strictEqual(state3.phase, 'planning', 'past TTL: re-derive sees new roadmap');
151
150
  } finally {
152
151
  cleanup(base);
153
152
  }
@@ -194,7 +193,6 @@ describe("cache-staleness-regression", () => {
194
193
  '- [ ] **T02: Second Task** `est:1h`',
195
194
  ].join('\n'));
196
195
 
197
- await new Promise(r => setTimeout(r, 150));
198
196
  invalidateAllCaches();
199
197
  invalidateStateCache();
200
198
  const state2 = await deriveState(base);
@@ -242,7 +240,6 @@ describe("cache-staleness-regression", () => {
242
240
  '- [x] **T01: Task** `est:1h`',
243
241
  ].join('\n'));
244
242
 
245
- await new Promise(r => setTimeout(r, 150));
246
243
  invalidateAllCaches();
247
244
  invalidateStateCache();
248
245
  const state2 = await deriveState(base);
@@ -282,7 +279,6 @@ describe("cache-staleness-regression", () => {
282
279
  '',
283
280
  ].join('\n'));
284
281
 
285
- await new Promise(r => setTimeout(r, 150));
286
282
  invalidateAllCaches();
287
283
  invalidateStateCache();
288
284
  const state2 = await deriveState(base);
@@ -0,0 +1,263 @@
1
+ // GSD-2 — #4780: slice-summary excerpts replace full inlining in
2
+ // buildCompleteMilestonePrompt. Verify (a) the excerpt helper emits
3
+ // frontmatter fields + section heads + on-demand path, (b) the closer
4
+ // prompt lists all slice SUMMARY paths under "On-demand Slice Summaries",
5
+ // (c) regression on prompt size.
6
+
7
+ import test from "node:test";
8
+ import assert from "node:assert/strict";
9
+ import { mkdtempSync, mkdirSync, writeFileSync, rmSync } from "node:fs";
10
+ import { join } from "node:path";
11
+ import { tmpdir } from "node:os";
12
+
13
+ import { buildSliceSummaryExcerpt, buildCompleteMilestonePrompt } from "../auto-prompts.ts";
14
+ import { invalidateAllCaches } from "../cache.ts";
15
+
16
+ // ─── Fixture helpers ──────────────────────────────────────────────────────
17
+
18
+ function createBase(): string {
19
+ const base = mkdtempSync(join(tmpdir(), "gsd-cm-excerpt-"));
20
+ mkdirSync(join(base, ".gsd", "milestones", "M001", "slices", "S01", "tasks"), { recursive: true });
21
+ mkdirSync(join(base, ".gsd", "milestones", "M001", "slices", "S02", "tasks"), { recursive: true });
22
+ return base;
23
+ }
24
+
25
+ function cleanup(base: string): void {
26
+ rmSync(base, { recursive: true, force: true });
27
+ }
28
+
29
+ function writeRoadmap(base: string, content: string): void {
30
+ writeFileSync(join(base, ".gsd", "milestones", "M001", "M001-ROADMAP.md"), content);
31
+ }
32
+
33
+ function writeSummary(base: string, sid: string, content: string): void {
34
+ writeFileSync(
35
+ join(base, ".gsd", "milestones", "M001", "slices", sid, `${sid}-SUMMARY.md`),
36
+ content,
37
+ );
38
+ }
39
+
40
+ // A summary with enough body narrative that full inlining would balloon the
41
+ // prompt. The excerpt should keep frontmatter + sections but drop the
42
+ // "What Happened" narrative.
43
+ function makeFatSummary(sid: string): string {
44
+ const narrativePara =
45
+ "The team discovered several subtle integration issues, traced them to the cache layer, and produced a patch set that threads the cache key through every call site. ".repeat(20);
46
+ return [
47
+ "---",
48
+ `id: ${sid}`,
49
+ "parent: M001",
50
+ "milestone: M001",
51
+ "provides:",
52
+ " - compact slice-summary excerpts",
53
+ " - on-demand read path registry",
54
+ "affects:",
55
+ " - complete-milestone prompt builder",
56
+ "key_decisions:",
57
+ " - use parseSummary for frontmatter extraction",
58
+ " - fall back to full inline when frontmatter fails",
59
+ "patterns_established:",
60
+ " - excerpt-first inlining for closer units",
61
+ "key_files:",
62
+ " - src/resources/extensions/gsd/auto-prompts.ts",
63
+ "duration: 1h",
64
+ "verification_result: passed",
65
+ "completed_at: 2026-04-24",
66
+ "blocker_discovered: false",
67
+ "---",
68
+ "",
69
+ `# ${sid}: Slice summary`,
70
+ "**Short one-liner for the slice**",
71
+ "",
72
+ "## What Happened",
73
+ "",
74
+ narrativePara,
75
+ "",
76
+ "## Deviations",
77
+ "",
78
+ "Extended the excerpt helper scope at review time.",
79
+ "",
80
+ "## Known Limitations",
81
+ "",
82
+ "Does not yet cover validate-milestone — follow-up.",
83
+ "",
84
+ "## Follow-ups",
85
+ "",
86
+ "- Wire the same excerpt into buildValidateMilestonePrompt",
87
+ ].join("\n");
88
+ }
89
+
90
+ function makeRoadmap(): string {
91
+ return [
92
+ "# M001 Roadmap",
93
+ "## Slices",
94
+ "- [x] **S01: Excerpt helper** `risk:medium` `depends:[]`",
95
+ "- [x] **S02: Closer wiring** `risk:low` `depends:[S01]`",
96
+ ].join("\n");
97
+ }
98
+
99
+ // ─── buildSliceSummaryExcerpt unit tests ──────────────────────────────────
100
+
101
+ test("#4780 excerpt: emits compact block with frontmatter fields + section heads", async (t) => {
102
+ const base = createBase();
103
+ t.after(() => cleanup(base));
104
+ invalidateAllCaches();
105
+
106
+ const absPath = join(base, ".gsd", "milestones", "M001", "slices", "S01", "S01-SUMMARY.md");
107
+ const relPath = ".gsd/milestones/M001/slices/S01/S01-SUMMARY.md";
108
+ writeSummary(base, "S01", makeFatSummary("S01"));
109
+
110
+ const out = await buildSliceSummaryExcerpt(absPath, relPath, "S01");
111
+
112
+ // Compact header with source path for on-demand Read
113
+ assert.match(out, /### S01 Summary \(excerpt\)/);
114
+ assert.match(out, /Source: `\.gsd\/milestones\/M001\/slices\/S01\/S01-SUMMARY\.md`/);
115
+
116
+ // Frontmatter fields surfaced
117
+ assert.match(out, /\*\*Title:\*\* S01: Slice summary/);
118
+ assert.match(out, /\*\*One-liner:\*\*/);
119
+ assert.match(out, /\*\*Verification:\*\* `passed`/);
120
+ assert.match(out, /\*\*Blockers:\*\* none/);
121
+ assert.match(out, /\*\*Provides:\*\* compact slice-summary excerpts;/);
122
+ assert.match(out, /\*\*Key decisions:\*\* use parseSummary/);
123
+ assert.match(out, /\*\*Patterns established:\*\* excerpt-first inlining/);
124
+
125
+ // Section heads included (body-section markdown), not whole sections inlined
126
+ assert.match(out, /#### Deviations/);
127
+ assert.match(out, /#### Known limitations/);
128
+ assert.match(out, /#### Follow-ups/);
129
+
130
+ // On-demand instruction present
131
+ assert.match(out, /On-demand.*read.*for the full "What Happened"/);
132
+
133
+ // Bulk narrative is NOT inlined — excerpt is meaningfully shorter than full
134
+ // A 20x-repeated paragraph produces ~2.5KB; excerpt should come in well under.
135
+ const fullSize = makeFatSummary("S01").length;
136
+ assert.ok(
137
+ out.length < fullSize * 0.6,
138
+ `excerpt length ${out.length} should be < 60% of full summary length ${fullSize}`,
139
+ );
140
+ });
141
+
142
+ test("#4780 excerpt: blocker_discovered=true surfaces prominent marker", async (t) => {
143
+ const base = createBase();
144
+ t.after(() => cleanup(base));
145
+ invalidateAllCaches();
146
+
147
+ const content = [
148
+ "---",
149
+ "id: S01",
150
+ "parent: M001",
151
+ "milestone: M001",
152
+ "blocker_discovered: true",
153
+ "---",
154
+ "# S01",
155
+ "**One-liner**",
156
+ "",
157
+ "## What Happened",
158
+ "content",
159
+ ].join("\n");
160
+ const absPath = join(base, ".gsd", "milestones", "M001", "slices", "S01", "S01-SUMMARY.md");
161
+ writeSummary(base, "S01", content);
162
+
163
+ const out = await buildSliceSummaryExcerpt(absPath, "rel", "S01");
164
+ assert.match(out, /Blockers:\*\* ⚠️ blocker recorded/);
165
+ });
166
+
167
+ test("#4780 excerpt: fall back to full inline when frontmatter is unrecognizable", async (t) => {
168
+ const base = createBase();
169
+ t.after(() => cleanup(base));
170
+ invalidateAllCaches();
171
+
172
+ // No frontmatter, no id — parser returns empty id, triggering fallback
173
+ const garbage = "# S99\n\nJust a wall of text with no frontmatter at all.\n";
174
+ const absPath = join(base, ".gsd", "milestones", "M001", "slices", "S01", "S99-SUMMARY.md");
175
+ writeFileSync(absPath, garbage);
176
+
177
+ const out = await buildSliceSummaryExcerpt(absPath, "rel/path.md", "S99");
178
+ // Full content preserved (no excerpt wrapper), no data-loss
179
+ assert.match(out, /Just a wall of text/);
180
+ assert.match(out, /### S99 Summary/);
181
+ });
182
+
183
+ test("#4780 excerpt: missing file reports not-found fallback", async (t) => {
184
+ const base = createBase();
185
+ t.after(() => cleanup(base));
186
+
187
+ const out = await buildSliceSummaryExcerpt(null, "rel/missing.md", "S42");
188
+ assert.match(out, /### S42 Summary \(excerpt\)/);
189
+ assert.match(out, /not found — file does not exist yet/);
190
+ });
191
+
192
+ test("#4780 excerpt: section bodies are capped (coderabbit review)", async (t) => {
193
+ const base = createBase();
194
+ t.after(() => cleanup(base));
195
+ invalidateAllCaches();
196
+
197
+ // Long Follow-ups section (~4.8KB) would balloon the excerpt without
198
+ // the cap — regression coverage for the coderabbit finding on #4908.
199
+ const longFollowUps = "A verbose follow-up bullet that keeps restating the same point. ".repeat(60);
200
+ const content = [
201
+ "---",
202
+ "id: S01",
203
+ "parent: M001",
204
+ "milestone: M001",
205
+ "---",
206
+ "# S01: Test",
207
+ "**One-liner**",
208
+ "",
209
+ "## Follow-ups",
210
+ longFollowUps,
211
+ ].join("\n");
212
+ const absPath = join(base, ".gsd", "milestones", "M001", "slices", "S01", "S01-SUMMARY.md");
213
+ writeSummary(base, "S01", content);
214
+
215
+ const out = await buildSliceSummaryExcerpt(absPath, "rel/path.md", "S01");
216
+
217
+ assert.match(out, /\(truncated — see full `rel\/path\.md`\)/);
218
+ assert.ok(
219
+ out.length < 2000,
220
+ `excerpt length ${out.length} should be well under 2KB when one section hits the cap`,
221
+ );
222
+ });
223
+
224
+ // ─── buildCompleteMilestonePrompt integration test ─────────────────────────
225
+
226
+ test("#4780 closer prompt: uses excerpts + lists on-demand slice SUMMARY paths", async (t) => {
227
+ const base = createBase();
228
+ t.after(() => cleanup(base));
229
+ invalidateAllCaches();
230
+
231
+ writeRoadmap(base, makeRoadmap());
232
+ writeSummary(base, "S01", makeFatSummary("S01"));
233
+ writeSummary(base, "S02", makeFatSummary("S02"));
234
+
235
+ const prompt = await buildCompleteMilestonePrompt("M001", "Test Milestone", base);
236
+
237
+ // Excerpt markers present for each slice
238
+ assert.match(prompt, /### S01 Summary \(excerpt\)/);
239
+ assert.match(prompt, /### S02 Summary \(excerpt\)/);
240
+
241
+ // On-demand path section exists with both slice paths
242
+ assert.match(prompt, /### On-demand Slice Summaries/);
243
+ assert.match(prompt, /S01-SUMMARY\.md/);
244
+ assert.match(prompt, /S02-SUMMARY\.md/);
245
+
246
+ // Fat narrative (the 20x-repeated paragraph) is NOT inlined
247
+ assert.ok(
248
+ !prompt.includes("threads the cache key through every call site."),
249
+ "closer prompt must not inline full 'What Happened' narrative after #4780",
250
+ );
251
+
252
+ // Prompt size is bounded — the two fat summaries' narratives alone would
253
+ // have exceeded ~4KB each. Post-fix closer prompt should be meaningfully
254
+ // smaller than their combined raw size.
255
+ const rawSize = makeFatSummary("S01").length + makeFatSummary("S02").length;
256
+ // Prompt includes roadmap, templates, and other inlines, so it may still
257
+ // be sizable — the guard is specifically that the fat narrative is gone.
258
+ // Use a soft bound: prompt - overhead should be less than 2x one summary.
259
+ assert.ok(
260
+ prompt.length < rawSize + 20_000,
261
+ `closer prompt length ${prompt.length} should be < raw summary size ${rawSize} + 20KB headroom`,
262
+ );
263
+ });
@@ -153,6 +153,31 @@ describe("complete-milestone", () => {
153
153
  );
154
154
  });
155
155
 
156
+ test("prompt does not hard-fail main self-diff as missing implementation (#4699)", () => {
157
+ const prompt = loadPromptFromWorktree("complete-milestone", {
158
+ workingDirectory: "/tmp/test-project",
159
+ milestoneId: "M001",
160
+ milestoneTitle: "Main Retry Test",
161
+ roadmapPath: ".gsd/milestones/M001/M001-ROADMAP.md",
162
+ inlinedContext: "context",
163
+ });
164
+
165
+ assert.ok(
166
+ !prompt.includes("git diff --stat HEAD $(git merge-base HEAD main) -- ':!.gsd/'"),
167
+ "prompt must not require the known self-diff command from #4699",
168
+ );
169
+ assert.match(
170
+ prompt,
171
+ /self-diff/i,
172
+ "prompt should explicitly guard retries where HEAD and the integration branch are the same commit",
173
+ );
174
+ assert.match(
175
+ prompt,
176
+ /GSD-(?:Task|Unit)/,
177
+ "prompt should direct main-branch retries toward milestone-scoped GSD commit evidence",
178
+ );
179
+ });
180
+
156
181
  test("handleCompleteMilestone rejects when verificationPassed is false", async () => {
157
182
  const { handleCompleteMilestone } = await import("../tools/complete-milestone.ts");
158
183
  const base = createFixtureBase();
@@ -0,0 +1,192 @@
1
+ // GSD-2 — #4782 phase 3 batch 3: complete-slice migrated through composer.
2
+
3
+ import test from "node:test";
4
+ import assert from "node:assert/strict";
5
+ import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
6
+ import { join } from "node:path";
7
+ import { tmpdir } from "node:os";
8
+
9
+ import { buildCompleteSlicePrompt } from "../auto-prompts.ts";
10
+ import { invalidateAllCaches } from "../cache.ts";
11
+ import {
12
+ openDatabase,
13
+ closeDatabase,
14
+ insertMilestone,
15
+ upsertMilestonePlanning,
16
+ insertSlice,
17
+ insertTask,
18
+ } from "../gsd-db.ts";
19
+
20
+ function makeBase(): string {
21
+ const base = mkdtempSync(join(tmpdir(), "gsd-completeslice-composer-"));
22
+ mkdirSync(join(base, ".gsd", "milestones", "M001", "slices", "S01", "tasks"), { recursive: true });
23
+ return base;
24
+ }
25
+
26
+ function cleanup(base: string): void {
27
+ try { closeDatabase(); } catch { /* noop */ }
28
+ invalidateAllCaches();
29
+ rmSync(base, { recursive: true, force: true });
30
+ }
31
+
32
+ function seed(base: string, mid: string): void {
33
+ openDatabase(join(base, ".gsd", "gsd.db"));
34
+ insertMilestone({ id: mid, title: "Composer Test", status: "active", depends_on: [] });
35
+ upsertMilestonePlanning(mid, {
36
+ title: "Composer Test",
37
+ status: "active",
38
+ vision: "Validate complete-slice migration",
39
+ successCriteria: ["Prompt compiles"],
40
+ keyRisks: [],
41
+ proofStrategy: [],
42
+ verificationContract: "",
43
+ verificationIntegration: "",
44
+ verificationOperational: "",
45
+ verificationUat: "",
46
+ definitionOfDone: [],
47
+ requirementCoverage: "",
48
+ boundaryMapMarkdown: "",
49
+ });
50
+ insertSlice({
51
+ id: "S01",
52
+ milestoneId: mid,
53
+ title: "First",
54
+ status: "complete",
55
+ risk: "low",
56
+ depends: [],
57
+ demo: "",
58
+ sequence: 1,
59
+ });
60
+ insertTask({
61
+ id: "T01",
62
+ sliceId: "S01",
63
+ milestoneId: mid,
64
+ title: "Task one",
65
+ status: "complete",
66
+ });
67
+ }
68
+
69
+ function writeArtifacts(base: string): void {
70
+ writeFileSync(
71
+ join(base, ".gsd", "milestones", "M001", "M001-ROADMAP.md"),
72
+ "# M001 Roadmap\n## Slices\n- [x] **S01: First** `risk:low` `depends:[]`\n",
73
+ );
74
+ writeFileSync(
75
+ join(base, ".gsd", "milestones", "M001", "slices", "S01", "S01-PLAN.md"),
76
+ "# S01 Plan\n\nSlice plan body.\n",
77
+ );
78
+ writeFileSync(
79
+ join(base, ".gsd", "milestones", "M001", "slices", "S01", "tasks", "T01-SUMMARY.md"),
80
+ "---\nid: T01\n---\n# T01 Summary\n\nTask one did the thing.\n",
81
+ );
82
+ }
83
+
84
+ test("#4782 phase 3: buildCompleteSlicePrompt composes roadmap → plan → task summaries → templates in declared order", async (t) => {
85
+ const base = makeBase();
86
+ t.after(() => cleanup(base));
87
+ invalidateAllCaches();
88
+
89
+ seed(base, "M001");
90
+ writeArtifacts(base);
91
+
92
+ const prompt = await buildCompleteSlicePrompt("M001", "Composer Test", "S01", "First", base);
93
+
94
+ // Context wrapper present
95
+ assert.match(prompt, /## Inlined Context \(preloaded — do not re-read these files\)/);
96
+
97
+ // Manifest-declared artifacts present
98
+ assert.match(prompt, /### Milestone Roadmap/);
99
+ assert.match(prompt, /### Slice Plan/);
100
+ assert.match(prompt, /### Task Summary: T01/);
101
+ assert.match(prompt, /### Output Template: Slice Summary/);
102
+
103
+ // Ordering: roadmap → slice plan → task summaries → slice summary template
104
+ const roadmapIdx = prompt.indexOf("### Milestone Roadmap");
105
+ const planIdx = prompt.indexOf("### Slice Plan");
106
+ const taskSummaryIdx = prompt.indexOf("### Task Summary: T01");
107
+ const sliceSummaryTemplateIdx = prompt.indexOf("### Output Template: Slice Summary");
108
+
109
+ assert.ok(roadmapIdx > -1 && planIdx > roadmapIdx, "roadmap precedes slice plan");
110
+ assert.ok(planIdx > -1 && taskSummaryIdx > planIdx, "slice plan precedes task summaries");
111
+ assert.ok(
112
+ taskSummaryIdx > -1 && sliceSummaryTemplateIdx > taskSummaryIdx,
113
+ "task summaries precede slice-summary template",
114
+ );
115
+
116
+ // Task body inlined
117
+ assert.match(prompt, /Task one did the thing/);
118
+ });
119
+
120
+ test("#4782 phase 3: buildCompleteSlicePrompt handles missing task summaries gracefully", async (t) => {
121
+ const base = makeBase();
122
+ t.after(() => cleanup(base));
123
+ invalidateAllCaches();
124
+
125
+ seed(base, "M001");
126
+ // Write roadmap + plan but no task summaries
127
+ writeFileSync(
128
+ join(base, ".gsd", "milestones", "M001", "M001-ROADMAP.md"),
129
+ "# M001 Roadmap\n## Slices\n- [x] **S01: First** `risk:low` `depends:[]`\n",
130
+ );
131
+ writeFileSync(
132
+ join(base, ".gsd", "milestones", "M001", "slices", "S01", "S01-PLAN.md"),
133
+ "# S01 Plan\n",
134
+ );
135
+
136
+ const prompt = await buildCompleteSlicePrompt("M001", "Composer Test", "S01", "First", base);
137
+
138
+ // Still succeeds — prior-task-summaries resolver returns null when dir is empty
139
+ assert.match(prompt, /### Milestone Roadmap/);
140
+ assert.match(prompt, /### Slice Plan/);
141
+ // No task summary blocks — they'd have a "### Task Summary:" prefix
142
+ assert.ok(!prompt.includes("### Task Summary:"));
143
+ // Roadmap still precedes slice plan despite the missing block
144
+ const roadmapIdx = prompt.indexOf("### Milestone Roadmap");
145
+ const planIdx = prompt.indexOf("### Slice Plan");
146
+ assert.ok(roadmapIdx > -1 && planIdx > roadmapIdx);
147
+ });
148
+
149
+ test("#4925 review: KNOWLEDGE splices BEFORE templates when no task summaries exist", async (t) => {
150
+ // Regression for the bug fixed in fcf3bfbe: the templates fallback was
151
+ // searching for "### Slice Summary" but inlineTemplate emits
152
+ // "### Output Template: Slice Summary", so templatesIdx stayed -1 and
153
+ // knowledge ended up appended after templates instead of before them.
154
+ const base = makeBase();
155
+ t.after(() => cleanup(base));
156
+ invalidateAllCaches();
157
+
158
+ seed(base, "M001");
159
+ // Roadmap + plan only — no T*-SUMMARY.md, so taskIdx must be -1 and the
160
+ // splice falls through to the templates anchor.
161
+ writeFileSync(
162
+ join(base, ".gsd", "milestones", "M001", "M001-ROADMAP.md"),
163
+ "# M001 Roadmap\n## Slices\n- [x] **S01: First** `risk:low` `depends:[]`\n",
164
+ );
165
+ writeFileSync(
166
+ join(base, ".gsd", "milestones", "M001", "slices", "S01", "S01-PLAN.md"),
167
+ "# S01 Plan\n",
168
+ );
169
+ // KNOWLEDGE.md with an H3 section whose header matches the slice title
170
+ // keyword "first" — queryKnowledge will return the section, so
171
+ // inlineKnowledgeBudgeted produces a non-null block and the splice
172
+ // path executes.
173
+ mkdirSync(join(base, ".gsd"), { recursive: true });
174
+ writeFileSync(
175
+ join(base, ".gsd", "KNOWLEDGE.md"),
176
+ "## Topics\n\n### First-slice notes\n\nNotes that should be inlined.\n",
177
+ );
178
+
179
+ const prompt = await buildCompleteSlicePrompt("M001", "Composer Test", "S01", "First", base);
180
+
181
+ // Sanity: the splice path actually fired.
182
+ assert.ok(!prompt.includes("### Task Summary:"), "fixture must have no task summaries to exercise the fallback");
183
+ const knowledgeIdx = prompt.indexOf("### Project Knowledge (scoped)");
184
+ const templatesIdx = prompt.indexOf("### Output Template: Slice Summary");
185
+ assert.ok(knowledgeIdx > -1, "knowledge block missing — fixture failed to populate KNOWLEDGE.md scope");
186
+ assert.ok(templatesIdx > -1, "templates block missing");
187
+ // The bug: knowledge appeared AFTER templates. The fix: knowledge before templates.
188
+ assert.ok(
189
+ knowledgeIdx < templatesIdx,
190
+ `knowledge (${knowledgeIdx}) must splice before templates (${templatesIdx}) when no task summaries exist`,
191
+ );
192
+ });
@@ -14,6 +14,7 @@ import {
14
14
  getTask,
15
15
  getSliceTasks,
16
16
  insertVerificationEvidence,
17
+ SCHEMA_VERSION,
17
18
  } from '../gsd-db.ts';
18
19
  import { handleCompleteTask } from '../tools/complete-task.ts';
19
20
 
@@ -99,19 +100,22 @@ function makeValidParams() {
99
100
  }
100
101
 
101
102
  // ═══════════════════════════════════════════════════════════════════════════
102
- // complete-task: Schema v5 migration
103
+ // complete-task: Fresh DB is migrated to the current schema version
103
104
  // ═══════════════════════════════════════════════════════════════════════════
104
105
 
105
- console.log('\n=== complete-task: schema v5 migration ===');
106
+ console.log('\n=== complete-task: fresh DB migrates to current schema version ===');
106
107
  {
107
108
  const dbPath = tempDbPath();
108
109
  openDatabase(dbPath);
109
110
 
110
111
  const adapter = _getAdapter()!;
111
112
 
112
- // Verify schema version is current (v22 — quality_gates DDL fix)
113
+ // Verify schema version matches the current source-of-truth constant.
114
+ // Asserting against SCHEMA_VERSION (not a hardcoded number) keeps this
115
+ // green across migration bumps while still catching a
116
+ // "fresh-DB-was-not-migrated" regression.
113
117
  const versionRow = adapter.prepare('SELECT MAX(version) as v FROM schema_version').get();
114
- assertEq(versionRow?.['v'], 22, 'schema version should be 22');
118
+ assertEq(versionRow?.['v'], SCHEMA_VERSION, 'fresh DB should be migrated to current SCHEMA_VERSION');
115
119
 
116
120
  // Verify all 4 new tables exist
117
121
  const tables = adapter.prepare(
@@ -399,9 +403,12 @@ console.log('\n=== complete-task: handler idempotency ===');
399
403
  const r1 = await handleCompleteTask(params, basePath);
400
404
  assertTrue(!('error' in r1), 'first call should succeed');
401
405
 
402
- // Verify only 1 task row
406
+ // Verify complete-task did not duplicate T01. State reconciliation may import
407
+ // the remaining plan task from disk so the DB stays aligned with S01-PLAN.md.
403
408
  const tasks = getSliceTasks('M001', 'S01');
404
- assertEq(tasks.length, 1, 'should have exactly 1 task row after first call');
409
+ assertEq(tasks.length, 2, 'should have T01 plus reconciled T02 after first call');
410
+ assertEq(tasks.filter(t => t.id === 'T01').length, 1, 'should have exactly one T01 row after first call');
411
+ assertEq(tasks.find(t => t.id === 'T02')?.status, 'pending', 'T02 should be reconciled as pending');
405
412
 
406
413
  // Second call with same params — state machine guard rejects (task is already complete)
407
414
  const r2 = await handleCompleteTask(params, basePath);
@@ -410,9 +417,10 @@ console.log('\n=== complete-task: handler idempotency ===');
410
417
  assertMatch(r2.error, /already complete/, 'error should mention already complete');
411
418
  }
412
419
 
413
- // Still only 1 task row (no duplication from rejected second call)
420
+ // Still no duplicate rows from the rejected second call.
414
421
  const tasksAfter = getSliceTasks('M001', 'S01');
415
- assertEq(tasksAfter.length, 1, 'should still have exactly 1 task row after rejected second call');
422
+ assertEq(tasksAfter.length, 2, 'should still have T01 plus reconciled T02 after rejected second call');
423
+ assertEq(tasksAfter.filter(t => t.id === 'T01').length, 1, 'should still have exactly one T01 row');
416
424
 
417
425
  cleanupDir(basePath);
418
426
  cleanup(dbPath);