gsd-pi 2.77.0-dev.eaa4973bc → 2.78.0-dev.aeeb2ca00

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 (545) hide show
  1. package/README.md +53 -17
  2. package/dist/claude-cli-check.js +46 -10
  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 +72 -16
  13. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +481 -17
  14. package/dist/resources/extensions/github-sync/templates.js +103 -0
  15. package/dist/resources/extensions/google-search/index.js +3 -2
  16. package/dist/resources/extensions/gsd/auto/loop.js +124 -2
  17. package/dist/resources/extensions/gsd/auto/phases.js +57 -39
  18. package/dist/resources/extensions/gsd/auto/session.js +6 -2
  19. package/dist/resources/extensions/gsd/auto-dispatch.js +142 -29
  20. package/dist/resources/extensions/gsd/auto-model-selection.js +124 -4
  21. package/dist/resources/extensions/gsd/auto-post-unit.js +150 -64
  22. package/dist/resources/extensions/gsd/auto-prompts.js +372 -104
  23. package/dist/resources/extensions/gsd/auto-recovery.js +197 -48
  24. package/dist/resources/extensions/gsd/auto-start.js +107 -29
  25. package/dist/resources/extensions/gsd/auto-tool-tracking.js +47 -7
  26. package/dist/resources/extensions/gsd/auto-worktree.js +122 -26
  27. package/dist/resources/extensions/gsd/auto.js +76 -21
  28. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +19 -1
  29. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +209 -0
  30. package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +3 -6
  31. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +7 -3
  32. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +127 -9
  33. package/dist/resources/extensions/gsd/component-loader.js +447 -0
  34. package/dist/resources/extensions/gsd/component-types.js +69 -0
  35. package/dist/resources/extensions/gsd/context-store.js +23 -7
  36. package/dist/resources/extensions/gsd/detection.js +49 -1
  37. package/dist/resources/extensions/gsd/dispatch-guard.js +2 -17
  38. package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  39. package/dist/resources/extensions/gsd/forensics.js +106 -0
  40. package/dist/resources/extensions/gsd/gate-registry.js +2 -2
  41. package/dist/resources/extensions/gsd/git-constants.js +28 -1
  42. package/dist/resources/extensions/gsd/git-self-heal.js +27 -0
  43. package/dist/resources/extensions/gsd/git-service.js +126 -2
  44. package/dist/resources/extensions/gsd/gsd-db.js +6 -3
  45. package/dist/resources/extensions/gsd/guided-flow.js +39 -13
  46. package/dist/resources/extensions/gsd/memory-extractor.js +7 -1
  47. package/dist/resources/extensions/gsd/milestone-scope-classifier.js +299 -0
  48. package/dist/resources/extensions/gsd/milestone-summary-classifier.js +37 -0
  49. package/dist/resources/extensions/gsd/model-cost-table.js +3 -0
  50. package/dist/resources/extensions/gsd/model-router.js +6 -0
  51. package/dist/resources/extensions/gsd/native-git-bridge.js +34 -4
  52. package/dist/resources/extensions/gsd/preferences-validation.js +23 -0
  53. package/dist/resources/extensions/gsd/prompt-cache-optimizer.js +4 -0
  54. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
  55. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +23 -4
  56. package/dist/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  57. package/dist/resources/extensions/gsd/prompts/plan-slice.md +15 -2
  58. package/dist/resources/extensions/gsd/safety/git-checkpoint.js +11 -0
  59. package/dist/resources/extensions/gsd/service-tier.js +5 -2
  60. package/dist/resources/extensions/gsd/session-lock.js +19 -10
  61. package/dist/resources/extensions/gsd/skill-manifest.js +168 -0
  62. package/dist/resources/extensions/gsd/slice-cadence.js +238 -0
  63. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +278 -8
  64. package/dist/resources/extensions/gsd/state-transition-matrix.js +118 -0
  65. package/dist/resources/extensions/gsd/state.js +69 -58
  66. package/dist/resources/extensions/gsd/sync-lock.js +98 -42
  67. package/dist/resources/extensions/gsd/tools/validate-milestone.js +7 -2
  68. package/dist/resources/extensions/gsd/unit-context-composer.js +147 -0
  69. package/dist/resources/extensions/gsd/unit-context-manifest.js +370 -0
  70. package/dist/resources/extensions/gsd/uok/dispatch-envelope.js +33 -0
  71. package/dist/resources/extensions/gsd/uok/execution-graph.js +10 -0
  72. package/dist/resources/extensions/gsd/uok/gate-runner.js +53 -5
  73. package/dist/resources/extensions/gsd/uok/gitops.js +2 -1
  74. package/dist/resources/extensions/gsd/uok/loop-adapter.js +37 -10
  75. package/dist/resources/extensions/gsd/uok/parity-report.js +58 -0
  76. package/dist/resources/extensions/gsd/uok/plan-v2.js +10 -4
  77. package/dist/resources/extensions/gsd/uok/writer.js +82 -0
  78. package/dist/resources/extensions/gsd/workflow-mcp.js +6 -0
  79. package/dist/resources/extensions/gsd/worktree-manager.js +85 -8
  80. package/dist/resources/extensions/gsd/worktree-resolver.js +86 -7
  81. package/dist/resources/extensions/gsd/worktree-telemetry.js +198 -0
  82. package/dist/resources/extensions/mcp-client/index.js +3 -1
  83. package/dist/resources/extensions/ollama/index.js +5 -1
  84. package/dist/resources/extensions/remote-questions/manager.js +11 -5
  85. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  86. package/dist/web/standalone/.next/BUILD_ID +1 -1
  87. package/dist/web/standalone/.next/app-path-routes-manifest.json +11 -11
  88. package/dist/web/standalone/.next/build-manifest.json +2 -2
  89. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  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.html +1 -1
  99. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  100. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  102. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  103. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  104. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  105. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  106. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  107. package/dist/web/standalone/.next/server/app/index.html +1 -1
  108. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  109. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  110. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  112. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  113. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  114. package/dist/web/standalone/.next/server/app-paths-manifest.json +11 -11
  115. package/dist/web/standalone/.next/server/chunks/1926.js +1 -1
  116. package/dist/web/standalone/.next/server/chunks/6897.js +1 -1
  117. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  118. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  119. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  120. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  121. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  122. package/package.json +2 -3
  123. package/packages/daemon/package.json +2 -2
  124. package/packages/daemon/src/logger.ts +4 -3
  125. package/packages/mcp-server/dist/server.d.ts +24 -0
  126. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  127. package/packages/mcp-server/dist/server.js +88 -87
  128. package/packages/mcp-server/dist/server.js.map +1 -1
  129. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  130. package/packages/mcp-server/dist/workflow-tools.js +15 -6
  131. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  132. package/packages/mcp-server/package.json +2 -2
  133. package/packages/mcp-server/src/mcp-server.test.ts +25 -3
  134. package/packages/mcp-server/src/readers/graph.test.ts +87 -15
  135. package/packages/mcp-server/src/secure-env-collect.test.ts +232 -237
  136. package/packages/mcp-server/src/server.ts +131 -105
  137. package/packages/mcp-server/src/workflow-tools.test.ts +85 -0
  138. package/packages/mcp-server/src/workflow-tools.ts +19 -6
  139. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  140. package/packages/native/package.json +2 -2
  141. package/packages/native/src/__tests__/_test-coverage-guard.test.mjs +98 -0
  142. package/packages/native/src/__tests__/module-compat.test.mjs +59 -27
  143. package/packages/native/src/__tests__/ps.test.mjs +14 -8
  144. package/packages/native/src/__tests__/stream-process.test.mjs +23 -2
  145. package/packages/native/src/__tests__/truncate.test.mjs +17 -2
  146. package/packages/pi-agent-core/package.json +1 -1
  147. package/packages/pi-agent-core/src/agent-loop.test.ts +5 -15
  148. package/packages/pi-agent-core/src/agent.test.ts +96 -102
  149. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  150. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -1
  151. package/packages/pi-ai/dist/models/capability-patches.js +9 -2
  152. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -1
  153. package/packages/pi-ai/dist/models/generated/index.d.ts +34 -0
  154. package/packages/pi-ai/dist/models/generated/index.d.ts.map +1 -1
  155. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +17 -0
  156. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -1
  157. package/packages/pi-ai/dist/models/generated/openai-codex.js +17 -0
  158. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -1
  159. package/packages/pi-ai/dist/models/generated/openai.d.ts +17 -0
  160. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -1
  161. package/packages/pi-ai/dist/models/generated/openai.js +17 -0
  162. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -1
  163. package/packages/pi-ai/dist/models.generated.test.js +43 -70
  164. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  165. package/packages/pi-ai/dist/models.test.js +36 -11
  166. package/packages/pi-ai/dist/models.test.js.map +1 -1
  167. package/packages/pi-ai/package.json +1 -1
  168. package/packages/pi-ai/scripts/generate-models.ts +44 -0
  169. package/packages/pi-ai/src/models/capability-patches.ts +10 -2
  170. package/packages/pi-ai/src/models/generated/openai-codex.ts +17 -0
  171. package/packages/pi-ai/src/models/generated/openai.ts +17 -0
  172. package/packages/pi-ai/src/models.generated.test.ts +46 -73
  173. package/packages/pi-ai/src/models.test.ts +48 -11
  174. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  175. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +96 -32
  176. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
  177. package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js +75 -12
  178. package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js.map +1 -1
  179. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js +99 -31
  180. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js.map +1 -1
  181. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +5 -0
  182. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  183. package/packages/pi-coding-agent/dist/core/extensions/loader.js +61 -0
  184. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  185. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js +30 -4
  186. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js.map +1 -1
  187. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +17 -0
  188. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
  189. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js +76 -18
  190. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js.map +1 -1
  191. package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
  192. package/packages/pi-coding-agent/dist/core/retry-handler.js +2 -6
  193. package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
  194. package/packages/pi-coding-agent/dist/core/retry-handler.test.js +5 -1
  195. package/packages/pi-coding-agent/dist/core/retry-handler.test.js.map +1 -1
  196. package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts +18 -0
  197. package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts.map +1 -0
  198. package/packages/pi-coding-agent/dist/core/retryable-error-regex.js +18 -0
  199. package/packages/pi-coding-agent/dist/core/retryable-error-regex.js.map +1 -0
  200. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts +20 -0
  201. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  202. package/packages/pi-coding-agent/dist/core/system-prompt.js +16 -2
  203. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  204. package/packages/pi-coding-agent/dist/index.d.ts +1 -0
  205. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  206. package/packages/pi-coding-agent/dist/index.js +1 -0
  207. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  208. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +36 -5
  209. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  210. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +20 -13
  211. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -1
  212. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  213. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +30 -12
  214. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  215. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
  216. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +18 -3
  217. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  218. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +125 -0
  219. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  220. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts +2 -0
  221. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts.map +1 -1
  222. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.js.map +1 -1
  223. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +4 -0
  224. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  225. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +105 -13
  226. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  227. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts +2 -0
  228. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts.map +1 -0
  229. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js +130 -0
  230. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js.map +1 -0
  231. package/packages/pi-coding-agent/package.json +1 -1
  232. package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +113 -37
  233. package/packages/pi-coding-agent/src/core/agent-session-model-switch.test.ts +89 -17
  234. package/packages/pi-coding-agent/src/core/agent-session-tool-refresh.test.ts +112 -43
  235. package/packages/pi-coding-agent/src/core/extensions/loader.ts +58 -0
  236. package/packages/pi-coding-agent/src/core/lsp/lsp-integration.test.ts +35 -4
  237. package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +20 -0
  238. package/packages/pi-coding-agent/src/core/resource-loader-cache-reset.test.ts +93 -28
  239. package/packages/pi-coding-agent/src/core/retry-handler.test.ts +5 -1
  240. package/packages/pi-coding-agent/src/core/retry-handler.ts +2 -8
  241. package/packages/pi-coding-agent/src/core/retryable-error-regex.ts +18 -0
  242. package/packages/pi-coding-agent/src/core/system-prompt.ts +35 -1
  243. package/packages/pi-coding-agent/src/index.ts +1 -0
  244. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +49 -3
  245. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +26 -20
  246. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +48 -9
  247. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +146 -1
  248. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +20 -3
  249. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-state.ts +2 -0
  250. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +119 -13
  251. package/packages/pi-coding-agent/src/tests/system-prompt-skill-filter.test.ts +157 -0
  252. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  253. package/packages/pi-tui/dist/__tests__/autocomplete.test.js +18 -8
  254. package/packages/pi-tui/dist/__tests__/autocomplete.test.js.map +1 -1
  255. package/packages/pi-tui/dist/__tests__/overlay-layout.test.js +128 -17
  256. package/packages/pi-tui/dist/__tests__/overlay-layout.test.js.map +1 -1
  257. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js +37 -11
  258. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js.map +1 -1
  259. package/packages/pi-tui/dist/__tests__/tui.test.js +18 -30
  260. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  261. package/packages/pi-tui/dist/components/__tests__/input.test.js +10 -3
  262. package/packages/pi-tui/dist/components/__tests__/input.test.js.map +1 -1
  263. package/packages/pi-tui/dist/components/__tests__/loader.test.js +53 -9
  264. package/packages/pi-tui/dist/components/__tests__/loader.test.js.map +1 -1
  265. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js +6 -2
  266. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js.map +1 -1
  267. package/packages/pi-tui/dist/components/editor.d.ts +14 -0
  268. package/packages/pi-tui/dist/components/editor.d.ts.map +1 -1
  269. package/packages/pi-tui/dist/components/editor.js +19 -0
  270. package/packages/pi-tui/dist/components/editor.js.map +1 -1
  271. package/packages/pi-tui/dist/components/image.test.js +6 -5
  272. package/packages/pi-tui/dist/components/image.test.js.map +1 -1
  273. package/packages/pi-tui/dist/editor-component.d.ts +2 -0
  274. package/packages/pi-tui/dist/editor-component.d.ts.map +1 -1
  275. package/packages/pi-tui/dist/editor-component.js.map +1 -1
  276. package/packages/pi-tui/package.json +1 -1
  277. package/packages/pi-tui/src/__tests__/autocomplete.test.ts +24 -8
  278. package/packages/pi-tui/src/__tests__/overlay-layout.test.ts +140 -17
  279. package/packages/pi-tui/src/__tests__/stdin-buffer.test.ts +42 -11
  280. package/packages/pi-tui/src/__tests__/tui.test.ts +18 -37
  281. package/packages/pi-tui/src/components/__tests__/input.test.ts +19 -3
  282. package/packages/pi-tui/src/components/__tests__/loader.test.ts +112 -35
  283. package/packages/pi-tui/src/components/__tests__/markdown-maxlines.test.ts +9 -2
  284. package/packages/pi-tui/src/components/editor.ts +22 -0
  285. package/packages/pi-tui/src/components/image.test.ts +10 -5
  286. package/packages/pi-tui/src/editor-component.ts +3 -0
  287. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  288. package/packages/rpc-client/dist/rpc-client.test.js +101 -51
  289. package/packages/rpc-client/dist/rpc-client.test.js.map +1 -1
  290. package/packages/rpc-client/package.json +1 -1
  291. package/packages/rpc-client/src/rpc-client.test.ts +109 -52
  292. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  293. package/pkg/package.json +1 -1
  294. package/scripts/install.js +15 -1
  295. package/src/resources/extensions/browser-tools/capture.ts +12 -0
  296. package/src/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +8 -59
  297. package/src/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +36 -24
  298. package/src/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +69 -71
  299. package/src/resources/extensions/browser-tools/tools/forms.ts +5 -1
  300. package/src/resources/extensions/browser-tools/tools/intent.ts +5 -1
  301. package/src/resources/extensions/claude-code-cli/readiness.ts +75 -16
  302. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +518 -19
  303. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +919 -75
  304. package/src/resources/extensions/github-sync/templates.ts +151 -0
  305. package/src/resources/extensions/github-sync/tests/cli.test.ts +76 -7
  306. package/src/resources/extensions/github-sync/tests/templates.test.ts +92 -1
  307. package/src/resources/extensions/google-search/index.ts +3 -2
  308. package/src/resources/extensions/gsd/auto/loop.ts +142 -2
  309. package/src/resources/extensions/gsd/auto/phases.ts +62 -38
  310. package/src/resources/extensions/gsd/auto/session.ts +7 -2
  311. package/src/resources/extensions/gsd/auto-dispatch.ts +156 -29
  312. package/src/resources/extensions/gsd/auto-model-selection.ts +131 -4
  313. package/src/resources/extensions/gsd/auto-post-unit.ts +163 -73
  314. package/src/resources/extensions/gsd/auto-prompts.ts +385 -93
  315. package/src/resources/extensions/gsd/auto-recovery.ts +230 -51
  316. package/src/resources/extensions/gsd/auto-start.ts +127 -9
  317. package/src/resources/extensions/gsd/auto-tool-tracking.ts +51 -7
  318. package/src/resources/extensions/gsd/auto-worktree.ts +130 -26
  319. package/src/resources/extensions/gsd/auto.ts +90 -23
  320. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +20 -1
  321. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +221 -0
  322. package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +3 -7
  323. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +7 -3
  324. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +158 -9
  325. package/src/resources/extensions/gsd/component-loader.ts +598 -0
  326. package/src/resources/extensions/gsd/component-types.ts +362 -0
  327. package/src/resources/extensions/gsd/context-store.ts +25 -8
  328. package/src/resources/extensions/gsd/detection.ts +58 -1
  329. package/src/resources/extensions/gsd/dispatch-guard.ts +2 -20
  330. package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  331. package/src/resources/extensions/gsd/forensics.ts +118 -1
  332. package/src/resources/extensions/gsd/gate-registry.ts +2 -2
  333. package/src/resources/extensions/gsd/git-constants.ts +30 -1
  334. package/src/resources/extensions/gsd/git-self-heal.ts +31 -0
  335. package/src/resources/extensions/gsd/git-service.ts +149 -2
  336. package/src/resources/extensions/gsd/gsd-db.ts +6 -3
  337. package/src/resources/extensions/gsd/guided-flow.ts +57 -14
  338. package/src/resources/extensions/gsd/journal.ts +11 -1
  339. package/src/resources/extensions/gsd/memory-extractor.ts +11 -3
  340. package/src/resources/extensions/gsd/milestone-scope-classifier.ts +366 -0
  341. package/src/resources/extensions/gsd/milestone-summary-classifier.ts +42 -0
  342. package/src/resources/extensions/gsd/model-cost-table.ts +3 -0
  343. package/src/resources/extensions/gsd/model-router.ts +6 -0
  344. package/src/resources/extensions/gsd/native-git-bridge.ts +34 -4
  345. package/src/resources/extensions/gsd/preferences-validation.ts +21 -0
  346. package/src/resources/extensions/gsd/prompt-cache-optimizer.ts +4 -0
  347. package/src/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
  348. package/src/resources/extensions/gsd/prompts/discuss-headless.md +23 -4
  349. package/src/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  350. package/src/resources/extensions/gsd/prompts/plan-slice.md +15 -2
  351. package/src/resources/extensions/gsd/safety/git-checkpoint.ts +15 -0
  352. package/src/resources/extensions/gsd/service-tier.ts +5 -2
  353. package/src/resources/extensions/gsd/session-lock.ts +20 -10
  354. package/src/resources/extensions/gsd/skill-manifest.ts +175 -0
  355. package/src/resources/extensions/gsd/slice-cadence.ts +299 -0
  356. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +309 -8
  357. package/src/resources/extensions/gsd/state-transition-matrix.ts +152 -0
  358. package/src/resources/extensions/gsd/state.ts +76 -66
  359. package/src/resources/extensions/gsd/sync-lock.ts +97 -39
  360. package/src/resources/extensions/gsd/tests/artifact-retry-cap.test.ts +270 -0
  361. package/src/resources/extensions/gsd/tests/artifacts-table-preserved-on-cache-invalidate.test.ts +2 -1
  362. package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +341 -0
  363. package/src/resources/extensions/gsd/tests/auto-discuss-milestone-deadlock-4973.test.ts +264 -0
  364. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +133 -292
  365. package/src/resources/extensions/gsd/tests/auto-model-selection-tool-poisoning.test.ts +742 -0
  366. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +78 -0
  367. package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +61 -0
  368. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +93 -0
  369. package/src/resources/extensions/gsd/tests/auto-remediate-slice-status.test.ts +4 -1
  370. package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +8 -194
  371. package/src/resources/extensions/gsd/tests/auto-start-clean-runtime-db-gated.test.ts +3 -2
  372. package/src/resources/extensions/gsd/tests/auto-start-cold-db-bootstrap.test.ts +2 -2
  373. package/src/resources/extensions/gsd/tests/auto-start-needs-discussion.test.ts +15 -58
  374. package/src/resources/extensions/gsd/tests/auto-start-worktree-db-path.test.ts +2 -2
  375. package/src/resources/extensions/gsd/tests/auto-thinking-restore.test.ts +3 -2
  376. package/src/resources/extensions/gsd/tests/auto-warning-noise-regression.test.ts +3 -2
  377. package/src/resources/extensions/gsd/tests/bootstrap-derive-state-db-open.test.ts +2 -1
  378. package/src/resources/extensions/gsd/tests/cache-staleness-regression.test.ts +17 -21
  379. package/src/resources/extensions/gsd/tests/canonical-milestone-root.test.ts +108 -0
  380. package/src/resources/extensions/gsd/tests/complete-milestone-excerpt.test.ts +263 -0
  381. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +25 -0
  382. package/src/resources/extensions/gsd/tests/complete-slice-composer.test.ts +192 -0
  383. package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +2 -1
  384. package/src/resources/extensions/gsd/tests/complete-task.test.ts +16 -8
  385. package/src/resources/extensions/gsd/tests/component-loader.test.ts +589 -0
  386. package/src/resources/extensions/gsd/tests/component-types.test.ts +127 -0
  387. package/src/resources/extensions/gsd/tests/context-store.test.ts +79 -0
  388. package/src/resources/extensions/gsd/tests/copy-planning-artifacts-samepath.test.ts +2 -1
  389. package/src/resources/extensions/gsd/tests/crash-recovery.test.ts +50 -1
  390. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +159 -0
  391. package/src/resources/extensions/gsd/tests/db-access-guardrails.test.ts +1 -0
  392. package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -3
  393. package/src/resources/extensions/gsd/tests/derive-state-db-disk-reconcile.test.ts +40 -0
  394. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +91 -3
  395. package/src/resources/extensions/gsd/tests/derive-state.test.ts +4 -4
  396. package/src/resources/extensions/gsd/tests/discuss-slice-structured-questions.test.ts +2 -1
  397. package/src/resources/extensions/gsd/tests/discuss-tool-scope-leak.test.ts +2 -1
  398. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +5 -0
  399. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +25 -0
  400. package/src/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +14 -0
  401. package/src/resources/extensions/gsd/tests/dispatcher-stuck-planning.test.ts +3 -2
  402. package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +4 -3
  403. package/src/resources/extensions/gsd/tests/empty-content-abort-loop.test.ts +4 -3
  404. package/src/resources/extensions/gsd/tests/execution-entry-missing-context-4671.test.ts +173 -0
  405. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +139 -129
  406. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +8 -104
  407. package/src/resources/extensions/gsd/tests/gate-state-canonicalization.test.ts +102 -0
  408. package/src/resources/extensions/gsd/tests/gate-storage.test.ts +1 -1
  409. package/src/resources/extensions/gsd/tests/google-search-stub.test.ts +14 -4
  410. package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +117 -0
  411. package/src/resources/extensions/gsd/tests/hook-key-parsing.test.ts +4 -55
  412. package/src/resources/extensions/gsd/tests/integration/all-milestones-complete-merge.test.ts +7 -56
  413. package/src/resources/extensions/gsd/tests/integration/auto-recovery.test.ts +20 -0
  414. package/src/resources/extensions/gsd/tests/integration/doctor-proactive.test.ts +18 -2
  415. package/src/resources/extensions/gsd/tests/integration/queue-completed-milestone-perf.test.ts +10 -4
  416. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +144 -7
  417. package/src/resources/extensions/gsd/tests/integration/state-machine-live-validation.test.ts +4 -0
  418. package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +2 -16
  419. package/src/resources/extensions/gsd/tests/interactive-routing-bypass.test.ts +9 -3
  420. package/src/resources/extensions/gsd/tests/interrupted-session-ui.test.ts +6 -9
  421. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +64 -0
  422. package/src/resources/extensions/gsd/tests/knowledge.test.ts +93 -1
  423. package/src/resources/extensions/gsd/tests/mcp-client-security.test.ts +8 -37
  424. package/src/resources/extensions/gsd/tests/memory-extractor.test.ts +5 -15
  425. package/src/resources/extensions/gsd/tests/merge-conflict-stops-loop.test.ts +227 -55
  426. package/src/resources/extensions/gsd/tests/milestone-scope-classifier.test.ts +187 -0
  427. package/src/resources/extensions/gsd/tests/milestone-summary-classifier.test.ts +30 -0
  428. package/src/resources/extensions/gsd/tests/model-cost-table.test.ts +9 -1
  429. package/src/resources/extensions/gsd/tests/model-router.test.ts +1 -1
  430. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +6 -48
  431. package/src/resources/extensions/gsd/tests/notification-widget.test.ts +6 -3
  432. package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +59 -2
  433. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +273 -130
  434. package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +301 -0
  435. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +32 -1
  436. package/src/resources/extensions/gsd/tests/preferences-worktree-sync.test.ts +2 -1
  437. package/src/resources/extensions/gsd/tests/prompt-cache-optimizer.test.ts +12 -0
  438. package/src/resources/extensions/gsd/tests/prompt-step-ordering.test.ts +15 -4
  439. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +23 -24
  440. package/src/resources/extensions/gsd/tests/queue-auto-guard.test.ts +32 -0
  441. package/src/resources/extensions/gsd/tests/queue-draft-detection.test.ts +3 -2
  442. package/src/resources/extensions/gsd/tests/queued-discuss-fast-path.test.ts +4 -5
  443. package/src/resources/extensions/gsd/tests/ready-phrase-no-files-4573.test.ts +75 -2
  444. package/src/resources/extensions/gsd/tests/reassess-default-optin.test.ts +132 -0
  445. package/src/resources/extensions/gsd/tests/recovery-attempts-reset.test.ts +8 -40
  446. package/src/resources/extensions/gsd/tests/regex-hardening.test.ts +136 -256
  447. package/src/resources/extensions/gsd/tests/research-milestone-composer.test.ts +114 -0
  448. package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +6 -3
  449. package/src/resources/extensions/gsd/tests/run-uat-composer.test.ts +148 -0
  450. package/src/resources/extensions/gsd/tests/service-tier.test.ts +4 -0
  451. package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +29 -0
  452. package/src/resources/extensions/gsd/tests/sidecar-queue.test.ts +3 -2
  453. package/src/resources/extensions/gsd/tests/silent-catch-diagnostics.test.ts +55 -95
  454. package/src/resources/extensions/gsd/tests/single-writer-v3-tool-surface.test.ts +158 -0
  455. package/src/resources/extensions/gsd/tests/skill-activation.test.ts +120 -1
  456. package/src/resources/extensions/gsd/tests/skill-manifest.test.ts +112 -0
  457. package/src/resources/extensions/gsd/tests/slice-cadence.test.ts +242 -0
  458. package/src/resources/extensions/gsd/tests/slice-context-injection.test.ts +3 -2
  459. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +164 -1
  460. package/src/resources/extensions/gsd/tests/smart-entry-draft.test.ts +2 -1
  461. package/src/resources/extensions/gsd/tests/stale-dirlistcache-4648.test.ts +112 -0
  462. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +29 -5
  463. package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +44 -0
  464. package/src/resources/extensions/gsd/tests/stop-auto-race-null-unit.test.ts +3 -3
  465. package/src/resources/extensions/gsd/tests/structured-data-formatter.test.ts +11 -92
  466. package/src/resources/extensions/gsd/tests/subagent-model-dispatch.test.ts +7 -6
  467. package/src/resources/extensions/gsd/tests/survivor-branch-complete.test.ts +102 -101
  468. package/src/resources/extensions/gsd/tests/sync-lock.test.ts +31 -0
  469. package/src/resources/extensions/gsd/tests/sync-worktree-skip-current.test.ts +4 -3
  470. package/src/resources/extensions/gsd/tests/test-helpers.test.ts +98 -0
  471. package/src/resources/extensions/gsd/tests/test-helpers.ts +153 -0
  472. package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -1
  473. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +61 -1
  474. package/src/resources/extensions/gsd/tests/tool-naming.test.ts +8 -1
  475. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +355 -0
  476. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +258 -0
  477. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +51 -0
  478. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +16 -0
  479. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +75 -0
  480. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +49 -26
  481. package/src/resources/extensions/gsd/tests/uok-loop-adapter-writer.test.ts +65 -0
  482. package/src/resources/extensions/gsd/tests/uok-parity-report.test.ts +42 -0
  483. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +19 -2
  484. package/src/resources/extensions/gsd/tests/uok-writer.test.ts +75 -0
  485. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +12 -0
  486. package/src/resources/extensions/gsd/tests/verify-artifact-tightened.test.ts +144 -80
  487. package/src/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +20 -54
  488. package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +342 -277
  489. package/src/resources/extensions/gsd/tests/worker-model-override.test.ts +37 -29
  490. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +226 -266
  491. package/src/resources/extensions/gsd/tests/worktree-health-monorepo.test.ts +103 -67
  492. package/src/resources/extensions/gsd/tests/worktree-nested-git-safety.test.ts +92 -90
  493. package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +238 -59
  494. package/src/resources/extensions/gsd/tests/worktree-sync-overwrite-loop.test.ts +113 -161
  495. package/src/resources/extensions/gsd/tests/worktree-telemetry.test.ts +210 -0
  496. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +262 -0
  497. package/src/resources/extensions/gsd/tests/write-gate-predicates.test.ts +186 -0
  498. package/src/resources/extensions/gsd/tests/write-gate.test.ts +7 -5
  499. package/src/resources/extensions/gsd/tests/zombie-gsd-state.test.ts +80 -96
  500. package/src/resources/extensions/gsd/tools/validate-milestone.ts +8 -2
  501. package/src/resources/extensions/gsd/types.ts +3 -3
  502. package/src/resources/extensions/gsd/unit-context-composer.ts +218 -0
  503. package/src/resources/extensions/gsd/unit-context-manifest.ts +574 -0
  504. package/src/resources/extensions/gsd/uok/contracts.ts +65 -0
  505. package/src/resources/extensions/gsd/uok/dispatch-envelope.ts +56 -0
  506. package/src/resources/extensions/gsd/uok/execution-graph.ts +22 -0
  507. package/src/resources/extensions/gsd/uok/gate-runner.ts +65 -5
  508. package/src/resources/extensions/gsd/uok/gitops.ts +6 -1
  509. package/src/resources/extensions/gsd/uok/loop-adapter.ts +45 -10
  510. package/src/resources/extensions/gsd/uok/parity-report.ts +84 -0
  511. package/src/resources/extensions/gsd/uok/plan-v2.ts +13 -5
  512. package/src/resources/extensions/gsd/uok/writer.ts +113 -0
  513. package/src/resources/extensions/gsd/workflow-mcp.ts +6 -0
  514. package/src/resources/extensions/gsd/worktree-manager.ts +108 -7
  515. package/src/resources/extensions/gsd/worktree-resolver.ts +96 -9
  516. package/src/resources/extensions/gsd/worktree-telemetry.ts +322 -0
  517. package/src/resources/extensions/mcp-client/index.ts +3 -1
  518. package/src/resources/extensions/mcp-client/tests/server-name-spaces.test.ts +70 -36
  519. package/src/resources/extensions/ollama/index.ts +5 -1
  520. package/src/resources/extensions/ollama/ollama-auth-mode.test.ts +123 -15
  521. package/src/resources/extensions/ollama/ollama-status-indicator.test.ts +206 -19
  522. package/src/resources/extensions/remote-questions/manager.ts +36 -4
  523. package/src/resources/extensions/remote-questions/tests/command-polling.test.ts +200 -190
  524. package/src/resources/extensions/shared/tests/interview-preview.test.ts +11 -3
  525. package/src/resources/extensions/voice/tests/linux-ready.test.ts +129 -113
  526. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts +0 -2
  527. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts.map +0 -1
  528. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js +0 -289
  529. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js.map +0 -1
  530. package/packages/pi-ai/src/utils/oauth/oauth-providers.test.ts +0 -363
  531. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +0 -143
  532. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +0 -157
  533. package/src/resources/extensions/gsd/tests/dashboard-model-label-ordering.test.ts +0 -107
  534. package/src/resources/extensions/gsd/tests/find-missing-summaries-closed.test.ts +0 -48
  535. package/src/resources/extensions/gsd/tests/forensics-context-persist.test.ts +0 -159
  536. package/src/resources/extensions/gsd/tests/forensics-db-completion.test.ts +0 -96
  537. package/src/resources/extensions/gsd/tests/forensics-dedup.test.ts +0 -79
  538. package/src/resources/extensions/gsd/tests/forensics-hook-key-parse.test.ts +0 -74
  539. package/src/resources/extensions/gsd/tests/forensics-journal.test.ts +0 -162
  540. package/src/resources/extensions/gsd/tests/gitignore-bg-shell.test.ts +0 -38
  541. package/src/resources/extensions/gsd/tests/gsd-no-project-error.test.ts +0 -73
  542. package/src/resources/extensions/gsd/tests/idle-watchdog-stall-override.test.ts +0 -125
  543. package/src/resources/extensions/gsd/tests/import-done-milestones.test.ts +0 -42
  544. /package/dist/web/standalone/.next/static/{5wbu35_C2_MQ3Jj1lEVDx → cAJH99yNS1UPbeSEiNRrV}/_buildManifest.js +0 -0
  545. /package/dist/web/standalone/.next/static/{5wbu35_C2_MQ3Jj1lEVDx → cAJH99yNS1UPbeSEiNRrV}/_ssgManifest.js +0 -0
@@ -1,281 +1,161 @@
1
- // Regex-hardening tests for S02/T02 — proves all 12 regex/parser sites
2
- // accept both M001 (classic) and M001-abc123 (unique) milestone ID formats.
3
- //
4
- // Sections:
5
- // (a) Directory scanning regex findMilestoneIds pattern
6
- // (b) Title-strip regex milestone title cleanup
7
- // (c) SLICE_BRANCH_RE branch name parsing (with/without worktree prefix)
8
- // (d) Milestone detection regex hasExistingMilestones pattern
9
- // (e) MILESTONE_CONTEXT_REcontext write-gate filename match
10
- // (f) Prompt dispatch regexes — executeMatch and resumeMatch capture
11
- // (g) milestoneIdSort — mixed-format ordering
12
- // (h) extractMilestoneSeq numeric extraction from both formats
13
-
14
- import { test } from 'node:test';
1
+ /**
2
+ * regex-hardening.test.ts verifies production regexes accept both the
3
+ * legacy (M001) and unique (M001-abc123) milestone ID formats.
4
+ *
5
+ * The previous version of this file advertised 12 parser sites but
6
+ * only 3 tested imports (SLICE_BRANCH_RE, MILESTONE_ID_RE helpers).
7
+ * The remaining 9 sections (a, b, d, e, f) declared local `const
8
+ * *_RE = ...` copies of production regexes and asserted against the
9
+ * copies a bug in the real regex would not fail those tests. See
10
+ * #4835.
11
+ *
12
+ * This rewrite imports every production pattern it exercises. Four
13
+ * call sites whose regexes are inline at the use site (state.ts:313
14
+ * title-strip, workspace-index.ts:80 title extraction, worktree-
15
+ * command.ts hasExistingMilestones, and the prompt dispatch regexes
16
+ * in index.ts) are intentionally NOT reimplemented here — they should
17
+ * be covered by behaviour tests of their parent functions, not by
18
+ * regex-copy assertions. A follow-up issue tracks extracting those
19
+ * regexes to a shared patterns module so they can be tested directly.
20
+ */
21
+
22
+ import test from "node:test";
23
+ import assert from "node:assert/strict";
15
24
 
16
25
  import {
17
26
  MILESTONE_ID_RE,
18
27
  extractMilestoneSeq,
19
28
  milestoneIdSort,
20
- } from '../guided-flow.ts';
21
-
22
- import { SLICE_BRANCH_RE } from '../worktree.ts';
23
- import { createTestContext } from './test-helpers.ts';
24
-
25
-
26
- const { assertEq, assertTrue, report } = createTestContext();
27
- // ─── Tests ─────────────────────────────────────────────────────────────────
28
-
29
- async function main(): Promise<void> {
30
- console.log('regex-hardening tests');
31
-
32
- // (a) Directory scanning regex — used in state.ts, workspace-index.ts, files.ts
33
- // Pattern: /^(M\d+(?:-[a-z0-9]{6})?)/
34
- {
35
- console.log(' (a) Directory scanning regex');
36
- const DIR_SCAN_RE = /^(M\d+(?:-[a-z0-9]{6})?)/;
37
-
38
- // Classic format matches
39
- assertTrue(DIR_SCAN_RE.test('M001'), 'dir scan matches M001');
40
- assertTrue(DIR_SCAN_RE.test('M042'), 'dir scan matches M042');
41
- assertTrue(DIR_SCAN_RE.test('M999'), 'dir scan matches M999');
42
- assertEq(('M001' as string).match(DIR_SCAN_RE)?.[1], 'M001', 'captures M001');
43
-
44
- // Unique format matches
45
- assertTrue(DIR_SCAN_RE.test('M001-abc123'), 'dir scan matches M001-abc123');
46
- assertTrue(DIR_SCAN_RE.test('M042-z9a8b7'), 'dir scan matches M042-z9a8b7');
47
- assertEq(('M001-abc123' as string).match(DIR_SCAN_RE)?.[1], 'M001-abc123', 'captures M001-abc123 from dir name');
48
-
49
- // Rejects
50
- assertTrue(!DIR_SCAN_RE.test('S01'), 'dir scan rejects S01');
51
- assertTrue(!DIR_SCAN_RE.test('X001'), 'dir scan rejects X001');
52
- assertTrue(!DIR_SCAN_RE.test('.DS_Store'), 'dir scan rejects .DS_Store');
53
- assertTrue(!DIR_SCAN_RE.test('notes'), 'dir scan rejects notes');
54
- }
55
-
56
- // (b) Title-strip regex — used in state.ts, workspace-index.ts
57
- // Pattern: /^M\d+(?:-[a-z0-9]{6})?[^:]*:\s*/
58
- {
59
- console.log(' (b) Title-strip regex');
60
- const TITLE_STRIP_RE = /^M\d+(?:-[a-z0-9]{6})?[^:]*:\s*/;
61
-
62
- // Classic format strip
63
- assertEq('M001: Title'.replace(TITLE_STRIP_RE, ''), 'Title', 'strips M001: Title → Title');
64
- assertEq('M042: Payment Integration'.replace(TITLE_STRIP_RE, ''), 'Payment Integration', 'strips M042: Payment Integration');
29
+ } from "../guided-flow.ts";
30
+ import { SLICE_BRANCH_RE } from "../worktree.ts";
31
+ import { MILESTONE_CONTEXT_RE } from "../bootstrap/write-gate.ts";
65
32
 
66
- // Unique format strip
67
- assertEq('M001-abc123: Title'.replace(TITLE_STRIP_RE, ''), 'Title', 'strips M001-abc123: Title → Title');
68
- assertEq('M042-z9a8b7: Dashboard'.replace(TITLE_STRIP_RE, ''), 'Dashboard', 'strips M042-z9a8b7: Dashboard');
33
+ // ─── MILESTONE_ID_RE ──────────────────────────────────────────────────────
69
34
 
70
- // Em dash in title — current format (M001: Title) correctly preserves em dash in title body
71
- assertEq(
72
- 'M001: Foundation — Build Core'.replace(TITLE_STRIP_RE, ''),
73
- 'Foundation — Build Core',
74
- 'strips M001: prefix and preserves em dash in title body',
75
- );
76
- assertEq(
77
- 'M001-abc123: Foundation — Build Core'.replace(TITLE_STRIP_RE, ''),
78
- 'Foundation — Build Core',
79
- 'strips M001-abc123: prefix and preserves em dash in title body (unique format)',
80
- );
81
-
82
- // Edge case: dash-style separator (M001 — Title: Subtitle preserves colon in body)
83
- assertEq(
84
- 'M001 — Unique Milestone IDs: Foo'.replace(TITLE_STRIP_RE, ''),
85
- 'Foo',
86
- 'strips M001 — Unique Milestone IDs: Foo → Foo (first colon consumed)',
87
- );
88
-
89
- // Edge case: colon inside title body preserved
90
- assertEq(
91
- 'M001: Note: important'.replace(TITLE_STRIP_RE, ''),
92
- 'Note: important',
93
- 'preserves colons in title body',
94
- );
95
-
96
- // No match — leaves non-milestone strings alone
97
- assertEq('S01: Slice Title'.replace(TITLE_STRIP_RE, ''), 'S01: Slice Title', 'does not strip S01 prefix');
98
- }
99
-
100
- // (c) SLICE_BRANCH_RE — from worktree.ts
101
- // Pattern: /^gsd\/(?:([a-zA-Z0-9_-]+)\/)?(M\d+(?:-[a-z0-9]{6})?)\/(S\d+)$/
102
- {
103
- console.log(' (c) SLICE_BRANCH_RE');
104
-
105
- // Classic format — no worktree prefix
106
- {
107
- const m = 'gsd/M001/S01'.match(SLICE_BRANCH_RE);
108
- assertTrue(m !== null, 'matches gsd/M001/S01');
109
- assertEq(m?.[1], undefined, 'no worktree prefix for gsd/M001/S01');
110
- assertEq(m?.[2], 'M001', 'captures M001');
111
- assertEq(m?.[3], 'S01', 'captures S01');
112
- }
113
-
114
- // Unique format — no worktree prefix
115
- {
116
- const m = 'gsd/M001-abc123/S01'.match(SLICE_BRANCH_RE);
117
- assertTrue(m !== null, 'matches gsd/M001-abc123/S01');
118
- assertEq(m?.[1], undefined, 'no worktree prefix for unique format');
119
- assertEq(m?.[2], 'M001-abc123', 'captures M001-abc123');
120
- assertEq(m?.[3], 'S01', 'captures S01');
121
- }
122
-
123
- // Classic format — with worktree prefix
124
- {
125
- const m = 'gsd/worktree/M001/S01'.match(SLICE_BRANCH_RE);
126
- assertTrue(m !== null, 'matches gsd/worktree/M001/S01');
127
- assertEq(m?.[1], 'worktree', 'captures worktree prefix');
128
- assertEq(m?.[2], 'M001', 'captures M001 with worktree');
129
- assertEq(m?.[3], 'S01', 'captures S01 with worktree');
130
- }
131
-
132
- // Unique format — with worktree prefix
133
- {
134
- const m = 'gsd/worktree/M001-abc123/S01'.match(SLICE_BRANCH_RE);
135
- assertTrue(m !== null, 'matches gsd/worktree/M001-abc123/S01');
136
- assertEq(m?.[1], 'worktree', 'captures worktree prefix with unique format');
137
- assertEq(m?.[2], 'M001-abc123', 'captures M001-abc123 with worktree');
138
- assertEq(m?.[3], 'S01', 'captures S01 with worktree and unique format');
139
- }
140
-
141
- // Rejects
142
- assertTrue(!SLICE_BRANCH_RE.test('gsd/S01'), 'rejects gsd/S01 (no milestone)');
143
- assertTrue(!SLICE_BRANCH_RE.test('main'), 'rejects main');
144
- assertTrue(!SLICE_BRANCH_RE.test('gsd/M001'), 'rejects gsd/M001 (no slice)');
145
- assertTrue(!SLICE_BRANCH_RE.test('feature/M001/S01'), 'rejects feature/ prefix');
146
- }
147
-
148
- // (d) Milestone detection regex — used in worktree-command.ts (hasExistingMilestones)
149
- // Pattern: /^M\d+(?:-[a-z0-9]{6})?/
150
- {
151
- console.log(' (d) Milestone detection regex');
152
- const MILESTONE_DETECT_RE = /^M\d+(?:-[a-z0-9]{6})?/;
35
+ test("MILESTONE_ID_RE accepts classic M001 format", () => {
36
+ assert.ok(MILESTONE_ID_RE.test("M001"));
37
+ assert.ok(MILESTONE_ID_RE.test("M042"));
38
+ assert.ok(MILESTONE_ID_RE.test("M999"));
39
+ });
153
40
 
154
- // Classic format matches
155
- assertTrue(MILESTONE_DETECT_RE.test('M001'), 'detect matches M001');
156
- assertTrue(MILESTONE_DETECT_RE.test('M042'), 'detect matches M042');
41
+ test("MILESTONE_ID_RE accepts unique M001-abc123 format", () => {
42
+ assert.ok(MILESTONE_ID_RE.test("M001-abc123"));
43
+ assert.ok(MILESTONE_ID_RE.test("M042-z9a8b7"));
44
+ });
157
45
 
158
- // Unique format matches
159
- assertTrue(MILESTONE_DETECT_RE.test('M001-abc123'), 'detect matches M001-abc123');
160
- assertTrue(MILESTONE_DETECT_RE.test('M042-z9a8b7'), 'detect matches M042-z9a8b7');
46
+ test("MILESTONE_ID_RE rejects non-milestone strings", () => {
47
+ assert.ok(!MILESTONE_ID_RE.test("S01"));
48
+ assert.ok(!MILESTONE_ID_RE.test("X001"));
49
+ assert.ok(!MILESTONE_ID_RE.test("notes"));
50
+ assert.ok(!MILESTONE_ID_RE.test(".DS_Store"));
51
+ assert.ok(!MILESTONE_ID_RE.test(""));
52
+ // Must be a bare id — not a prefix match.
53
+ assert.ok(!MILESTONE_ID_RE.test("M001-ABCDEF"), "uppercase suffix rejected");
54
+ assert.ok(!MILESTONE_ID_RE.test("M001 "), "trailing space rejected");
55
+ });
161
56
 
162
- // Rejects
163
- assertTrue(!MILESTONE_DETECT_RE.test('S01'), 'detect rejects S01');
164
- assertTrue(!MILESTONE_DETECT_RE.test('notes'), 'detect rejects notes');
165
- assertTrue(!MILESTONE_DETECT_RE.test('.DS_Store'), 'detect rejects .DS_Store');
57
+ // ─── SLICE_BRANCH_RE ──────────────────────────────────────────────────────
58
+
59
+ test("SLICE_BRANCH_RE captures milestone + slice without worktree prefix", () => {
60
+ for (const { input, expectMid } of [
61
+ { input: "gsd/M001/S01", expectMid: "M001" },
62
+ { input: "gsd/M001-abc123/S01", expectMid: "M001-abc123" },
63
+ ]) {
64
+ const m = input.match(SLICE_BRANCH_RE);
65
+ assert.ok(m, `should match ${input}`);
66
+ assert.equal(m?.[1], undefined, "no worktree prefix");
67
+ assert.equal(m?.[2], expectMid);
68
+ assert.equal(m?.[3], "S01");
166
69
  }
70
+ });
167
71
 
168
- // (e) MILESTONE_CONTEXT_RE used in index.ts (write-gate)
169
- // Pattern: /M\d+(?:-[a-z0-9]{6})?-CONTEXT\.md$/
170
- {
171
- console.log(' (e) MILESTONE_CONTEXT_RE');
172
- const CONTEXT_RE = /M\d+(?:-[a-z0-9]{6})?-CONTEXT\.md$/;
173
-
174
- // Classic format matches
175
- assertTrue(CONTEXT_RE.test('M001-CONTEXT.md'), 'context matches M001-CONTEXT.md');
176
- assertTrue(CONTEXT_RE.test('.gsd/milestones/M001/M001-CONTEXT.md'), 'context matches full path classic format');
177
-
178
- // Unique format matches
179
- assertTrue(CONTEXT_RE.test('M001-abc123-CONTEXT.md'), 'context matches M001-abc123-CONTEXT.md');
180
- assertTrue(CONTEXT_RE.test('.gsd/milestones/M001-abc123/M001-abc123-CONTEXT.md'), 'context matches full path unique format');
181
-
182
- // Rejects
183
- assertTrue(!CONTEXT_RE.test('M001-ROADMAP.md'), 'context rejects M001-ROADMAP.md');
184
- assertTrue(!CONTEXT_RE.test('M001-SUMMARY.md'), 'context rejects M001-SUMMARY.md');
185
- assertTrue(!CONTEXT_RE.test('CONTEXT.md'), 'context rejects bare CONTEXT.md');
72
+ test("SLICE_BRANCH_RE captures worktree prefix when present", () => {
73
+ for (const { input, expectMid } of [
74
+ { input: "gsd/worktree/M001/S01", expectMid: "M001" },
75
+ { input: "gsd/worktree/M001-abc123/S01", expectMid: "M001-abc123" },
76
+ ]) {
77
+ const m = input.match(SLICE_BRANCH_RE);
78
+ assert.ok(m, `should match ${input}`);
79
+ assert.equal(m?.[1], "worktree");
80
+ assert.equal(m?.[2], expectMid);
81
+ assert.equal(m?.[3], "S01");
186
82
  }
83
+ });
187
84
 
188
- // (f) Prompt dispatch regexes used in index.ts (executeMatch, resumeMatch)
189
- {
190
- console.log(' (f) Prompt dispatch regexes');
191
- const EXECUTE_RE = /Execute the next task:\s+(T\d+)\s+\("([^"]+)"\)\s+in slice\s+(S\d+)\s+of milestone\s+(M\d+(?:-[a-z0-9]{6})?)/i;
192
- const RESUME_RE = /Resume interrupted work\.[\s\S]*?slice\s+(S\d+)\s+of milestone\s+(M\d+(?:-[a-z0-9]{6})?)/i;
193
-
194
- // Execute — classic format
195
- {
196
- const prompt = 'Execute the next task: T01 ("Write tests") in slice S01 of milestone M001';
197
- const m = prompt.match(EXECUTE_RE);
198
- assertTrue(m !== null, 'execute matches classic format');
199
- assertEq(m?.[1], 'T01', 'execute captures T01');
200
- assertEq(m?.[3], 'S01', 'execute captures S01');
201
- assertEq(m?.[4], 'M001', 'execute captures M001');
202
- }
203
-
204
- // Execute — unique format
205
- {
206
- const prompt = 'Execute the next task: T02 ("Build feature") in slice S03 of milestone M001-abc123';
207
- const m = prompt.match(EXECUTE_RE);
208
- assertTrue(m !== null, 'execute matches unique format');
209
- assertEq(m?.[1], 'T02', 'execute captures T02 (unique format)');
210
- assertEq(m?.[3], 'S03', 'execute captures S03 (unique format)');
211
- assertEq(m?.[4], 'M001-abc123', 'execute captures M001-abc123');
212
- }
213
-
214
- // Resume — classic format
215
- {
216
- const prompt = 'Resume interrupted work.\nContinuing slice S02 of milestone M001';
217
- const m = prompt.match(RESUME_RE);
218
- assertTrue(m !== null, 'resume matches classic format');
219
- assertEq(m?.[1], 'S02', 'resume captures S02');
220
- assertEq(m?.[2], 'M001', 'resume captures M001');
221
- }
222
-
223
- // Resume — unique format
224
- {
225
- const prompt = 'Resume interrupted work.\nContinuing slice S01 of milestone M042-z9a8b7';
226
- const m = prompt.match(RESUME_RE);
227
- assertTrue(m !== null, 'resume matches unique format');
228
- assertEq(m?.[1], 'S01', 'resume captures S01 (unique format)');
229
- assertEq(m?.[2], 'M042-z9a8b7', 'resume captures M042-z9a8b7');
230
- }
231
- }
85
+ test("SLICE_BRANCH_RE rejects malformed inputs", () => {
86
+ assert.ok(!SLICE_BRANCH_RE.test("gsd/S01"), "no milestone");
87
+ assert.ok(!SLICE_BRANCH_RE.test("main"), "non-gsd branch");
88
+ assert.ok(!SLICE_BRANCH_RE.test("gsd/M001"), "no slice");
89
+ assert.ok(!SLICE_BRANCH_RE.test("feature/M001/S01"), "wrong prefix");
90
+ });
232
91
 
233
- // (g) milestoneIdSort — mixed-format ordering
234
- {
235
- console.log(' (g) milestoneIdSort');
236
- const mixed = ['M002-abc123', 'M001', 'M001-xyz789'];
237
- const sorted = [...mixed].sort(milestoneIdSort);
238
- assertEq(sorted, ['M001', 'M001-xyz789', 'M002-abc123'], 'sorts mixed IDs by sequence number');
92
+ // ─── MILESTONE_CONTEXT_RE ────────────────────────────────────────────────
93
+
94
+ test("MILESTONE_CONTEXT_RE matches legacy and unique CONTEXT.md names", () => {
95
+ assert.ok(MILESTONE_CONTEXT_RE.test("M001-CONTEXT.md"));
96
+ assert.ok(MILESTONE_CONTEXT_RE.test("M001-abc123-CONTEXT.md"));
97
+ assert.ok(
98
+ MILESTONE_CONTEXT_RE.test(".gsd/milestones/M001/M001-CONTEXT.md"),
99
+ "full path legacy format",
100
+ );
101
+ assert.ok(
102
+ MILESTONE_CONTEXT_RE.test(".gsd/milestones/M001-abc123/M001-abc123-CONTEXT.md"),
103
+ "full path unique format",
104
+ );
105
+ });
239
106
 
240
- // Stable within same seq preserves insertion order
241
- const sameSorted = ['M001-abc123', 'M001'].sort(milestoneIdSort);
242
- assertEq(sameSorted[0], 'M001-abc123', 'same seq preserves order (first)');
243
- assertEq(sameSorted[1], 'M001', 'same seq preserves order (second)');
107
+ test("MILESTONE_CONTEXT_RE rejects non-CONTEXT artifact names", () => {
108
+ assert.ok(!MILESTONE_CONTEXT_RE.test("M001-ROADMAP.md"));
109
+ assert.ok(!MILESTONE_CONTEXT_RE.test("M001-SUMMARY.md"));
110
+ assert.ok(!MILESTONE_CONTEXT_RE.test("CONTEXT.md"), "bare name without milestone prefix");
111
+ });
244
112
 
245
- // Classic format only
246
- const oldOnly = ['M003', 'M001', 'M002'];
247
- assertEq([...oldOnly].sort(milestoneIdSort), ['M001', 'M002', 'M003'], 'sorts classic-format IDs');
113
+ // ─── extractMilestoneSeq ──────────────────────────────────────────────────
248
114
 
249
- // Unique format only
250
- const newOnly = ['M003-abc123', 'M001-def456', 'M002-ghi789'];
251
- assertEq([...newOnly].sort(milestoneIdSort), ['M001-def456', 'M002-ghi789', 'M003-abc123'], 'sorts unique-format IDs');
252
- }
115
+ test("extractMilestoneSeq returns numeric sequence for both formats", () => {
116
+ assert.equal(extractMilestoneSeq("M001"), 1);
117
+ assert.equal(extractMilestoneSeq("M042"), 42);
118
+ assert.equal(extractMilestoneSeq("M999"), 999);
119
+ assert.equal(extractMilestoneSeq("M001-abc123"), 1);
120
+ assert.equal(extractMilestoneSeq("M042-z9a8b7"), 42);
121
+ assert.equal(extractMilestoneSeq("M100-xyz789"), 100);
122
+ });
253
123
 
254
- // (h) extractMilestoneSeq numeric extraction from both formats
255
- {
256
- console.log(' (h) extractMilestoneSeq');
124
+ test("extractMilestoneSeq returns 0 (not NaN) for invalid inputs", () => {
125
+ assert.equal(extractMilestoneSeq(""), 0);
126
+ assert.equal(extractMilestoneSeq("notes"), 0);
127
+ assert.equal(extractMilestoneSeq("S01"), 0);
128
+ // Specific regression: the parseInt(slice(1)) implementation returned
129
+ // NaN on inputs like "M001-abc123" because parseInt stopped at the
130
+ // dash but then the rest of the logic treated the result as a number.
131
+ // Current impl returns a real number.
132
+ assert.ok(!Number.isNaN(extractMilestoneSeq("M001-abc123")));
133
+ });
257
134
 
258
- // Classic format
259
- assertEq(extractMilestoneSeq('M001'), 1, 'M001 → 1');
260
- assertEq(extractMilestoneSeq('M042'), 42, 'M042 → 42');
261
- assertEq(extractMilestoneSeq('M999'), 999, 'M999 → 999');
135
+ // ─── milestoneIdSort ──────────────────────────────────────────────────────
262
136
 
263
- // Unique format confirms dispatch-guard refactor correctness
264
- assertEq(extractMilestoneSeq('M001-abc123'), 1, 'M001-abc123 → 1');
265
- assertEq(extractMilestoneSeq('M042-z9a8b7'), 42, 'M042-z9a8b7 → 42');
266
- assertEq(extractMilestoneSeq('M100-xyz789'), 100, 'M100-xyz789 → 100');
137
+ test("milestoneIdSort orders by numeric sequence across both formats", () => {
138
+ const mixed = ["M002-abc123", "M001", "M001-xyz789"];
139
+ assert.deepEqual(
140
+ [...mixed].sort(milestoneIdSort),
141
+ ["M001", "M001-xyz789", "M002-abc123"],
142
+ "mixed formats sort by seq number",
143
+ );
267
144
 
268
- // Invalid 0 (not NaN — the old parseInt(slice(1)) bug)
269
- assertEq(extractMilestoneSeq(''), 0, 'empty → 0');
270
- assertEq(extractMilestoneSeq('notes'), 0, 'notes → 0');
271
- assertEq(extractMilestoneSeq('S01'), 0, 'S01 → 0');
272
- assertTrue(!Number.isNaN(extractMilestoneSeq('M001-abc123')), 'unique format does not return NaN');
273
- assertTrue(!Number.isNaN(extractMilestoneSeq('M001-ABCDEF')), 'invalid format does not return NaN');
274
- }
145
+ const legacy = ["M003", "M001", "M002"];
146
+ assert.deepEqual([...legacy].sort(milestoneIdSort), ["M001", "M002", "M003"]);
275
147
 
276
- report();
277
- }
148
+ const unique = ["M003-abc123", "M001-def456", "M002-ghi789"];
149
+ assert.deepEqual(
150
+ [...unique].sort(milestoneIdSort),
151
+ ["M001-def456", "M002-ghi789", "M003-abc123"],
152
+ );
153
+ });
278
154
 
279
- test('regex-hardening: all 12 sites accept both formats', async () => {
280
- await main();
155
+ test("milestoneIdSort preserves input order for same-sequence ids", () => {
156
+ // sort is stable per ECMAScript 2019+ when the comparator returns 0.
157
+ const sameSeq = ["M001-abc123", "M001"];
158
+ const sorted = [...sameSeq].sort(milestoneIdSort);
159
+ assert.equal(sorted[0], "M001-abc123");
160
+ assert.equal(sorted[1], "M001");
281
161
  });
@@ -0,0 +1,114 @@
1
+ // GSD-2 — #4782 phase 3 batch 2: research-milestone 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 { buildResearchMilestonePrompt } from "../auto-prompts.ts";
10
+ import { invalidateAllCaches } from "../cache.ts";
11
+ import {
12
+ openDatabase,
13
+ closeDatabase,
14
+ insertMilestone,
15
+ upsertMilestonePlanning,
16
+ insertArtifact,
17
+ } from "../gsd-db.ts";
18
+
19
+ function makeBase(): string {
20
+ const base = mkdtempSync(join(tmpdir(), "gsd-research-ms-composer-"));
21
+ mkdirSync(join(base, ".gsd", "milestones", "M001"), { recursive: true });
22
+ return base;
23
+ }
24
+
25
+ function cleanup(base: string): void {
26
+ try { closeDatabase(); } catch { /* noop */ }
27
+ invalidateAllCaches();
28
+ rmSync(base, { recursive: true, force: true });
29
+ }
30
+
31
+ function seed(base: string, mid: string): void {
32
+ openDatabase(join(base, ".gsd", "gsd.db"));
33
+ insertMilestone({ id: mid, title: "Research Test", status: "active", depends_on: [] });
34
+ upsertMilestonePlanning(mid, {
35
+ title: "Research Test",
36
+ status: "active",
37
+ vision: "Research composer migration",
38
+ successCriteria: ["Prompt compiles"],
39
+ keyRisks: [],
40
+ proofStrategy: [],
41
+ verificationContract: "",
42
+ verificationIntegration: "",
43
+ verificationOperational: "",
44
+ verificationUat: "",
45
+ definitionOfDone: [],
46
+ requirementCoverage: "",
47
+ boundaryMapMarkdown: "",
48
+ });
49
+ }
50
+
51
+ test("#4782 phase 3: buildResearchMilestonePrompt emits milestone-context then research template via composer", async (t) => {
52
+ const base = makeBase();
53
+ t.after(() => cleanup(base));
54
+ invalidateAllCaches();
55
+
56
+ seed(base, "M001");
57
+
58
+ writeFileSync(
59
+ join(base, ".gsd", "milestones", "M001", "M001-CONTEXT.md"),
60
+ "# M001 Context\n\nA research test milestone.\n",
61
+ );
62
+
63
+ const prompt = await buildResearchMilestonePrompt("M001", "Research Test", base);
64
+
65
+ // Context wrapper present
66
+ assert.match(prompt, /## Inlined Context \(preloaded — do not re-read these files\)/);
67
+
68
+ // Milestone context inlined first (manifest order)
69
+ assert.match(prompt, /### Milestone Context/);
70
+ assert.match(prompt, /A research test milestone/);
71
+
72
+ // Research template inlined as the templates artifact
73
+ assert.match(prompt, /### Output Template: Research/);
74
+
75
+ // Ordering: milestone-context precedes the research template
76
+ const contextIdx = prompt.indexOf("### Milestone Context");
77
+ const researchIdx = prompt.indexOf("### Output Template: Research");
78
+ assert.ok(contextIdx > -1 && researchIdx > contextIdx,
79
+ `milestone-context (${contextIdx}) must precede research template (${researchIdx})`);
80
+ });
81
+
82
+ test("#4782 phase 3: buildResearchMilestonePrompt preserves manifest order across optional artifacts (#4925 review)", async (t) => {
83
+ const base = makeBase();
84
+ t.after(() => cleanup(base));
85
+ invalidateAllCaches();
86
+
87
+ seed(base, "M001");
88
+ writeFileSync(join(base, ".gsd", "milestones", "M001", "M001-CONTEXT.md"), "# M001 Context\n");
89
+ // Seed PROJECT.md into the artifacts table so inlineProjectFromDb resolves
90
+ // to a non-null body. Lets us verify the project block sits between
91
+ // milestone-context and the templates block per manifest order.
92
+ insertArtifact({
93
+ path: "PROJECT.md",
94
+ artifact_type: "project",
95
+ milestone_id: null,
96
+ slice_id: null,
97
+ task_id: null,
98
+ full_content: "# Project\n\nResearch composer fixture project.\n",
99
+ });
100
+
101
+ const prompt = await buildResearchMilestonePrompt("M001", "Research Test", base);
102
+
103
+ // Manifest-declared order: milestone-context, project, requirements, decisions, templates.
104
+ const contextIdx = prompt.indexOf("### Milestone Context");
105
+ const projectIdx = prompt.indexOf("### Project");
106
+ const researchIdx = prompt.indexOf("### Output Template: Research");
107
+ assert.ok(contextIdx > -1, "milestone-context block missing");
108
+ assert.ok(projectIdx > -1, "project block missing — seed should have populated it");
109
+ assert.ok(researchIdx > -1, "research template block missing");
110
+ assert.ok(
111
+ contextIdx < projectIdx && projectIdx < researchIdx,
112
+ `manifest order violated: milestone-context (${contextIdx}) < project (${projectIdx}) < research-template (${researchIdx})`,
113
+ );
114
+ });
@@ -13,6 +13,7 @@ import { describe, it } from 'node:test'
13
13
  import assert from 'node:assert/strict'
14
14
  import { readFileSync } from 'node:fs'
15
15
  import { resolve } from 'node:path'
16
+ import { extractSourceRegion } from "./test-helpers.ts";
16
17
 
17
18
  const src = readFileSync(
18
19
  resolve(process.cwd(), 'src', 'resources', 'extensions', 'gsd', 'guided-flow.ts'),
@@ -37,7 +38,7 @@ describe('restore tools after discuss flow scoping (#3628)', () => {
37
38
  assert.ok(discussCheck !== -1)
38
39
 
39
40
  // Look for savedTools assignment within the discuss block
40
- const blockAfter = src.slice(discussCheck, discussCheck + 500)
41
+ const blockAfter = extractSourceRegion(src, 'if (unitType?.startsWith("discuss-"))')
41
42
  assert.ok(
42
43
  blockAfter.includes('savedTools = currentTools'),
43
44
  'savedTools must be assigned from currentTools inside the discuss block',
@@ -55,8 +56,10 @@ describe('restore tools after discuss flow scoping (#3628)', () => {
55
56
  const sendMsg = src.indexOf('triggerTurn: true', savedToolsAssign)
56
57
  assert.ok(sendMsg !== -1, 'discuss-flow sendMessage with triggerTurn must exist after savedTools capture')
57
58
 
58
- // After sendMessage, savedTools should be restored via setActiveTools
59
- const afterSend = src.slice(sendMsg, sendMsg + 500)
59
+ // After sendMessage, savedTools should be restored via setActiveTools.
60
+ // Use fromIdx to anchor at the discuss-flow sendMessage, not the first
61
+ // triggerTurn: true occurrence in the file.
62
+ const afterSend = extractSourceRegion(src, 'triggerTurn: true', { fromIdx: savedToolsAssign })
60
63
  assert.ok(
61
64
  afterSend.includes('if (savedTools)'),
62
65
  'savedTools restoration guard must exist after sendMessage',