gsd-pi 2.76.0-dev.fe143342a → 2.77.0-dev.1d17f366c

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 (597) hide show
  1. package/README.md +17 -35
  2. package/dist/claude-cli-check.js +32 -3
  3. package/dist/cli-web-branch.d.ts +1 -0
  4. package/dist/cli-web-branch.js +3 -0
  5. package/dist/cli.js +38 -2
  6. package/dist/extension-discovery.d.ts +6 -0
  7. package/dist/extension-discovery.js +37 -0
  8. package/dist/extension-registry.d.ts +3 -0
  9. package/dist/extension-sort.d.ts +18 -0
  10. package/dist/extension-sort.js +114 -0
  11. package/dist/extension-validator.d.ts +47 -0
  12. package/dist/extension-validator.js +127 -0
  13. package/dist/loader.js +35 -7
  14. package/dist/onboarding.js +45 -0
  15. package/dist/provider-migrations.d.ts +18 -0
  16. package/dist/provider-migrations.js +14 -0
  17. package/dist/resources/extensions/claude-code-cli/readiness.js +4 -3
  18. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +78 -59
  19. package/dist/resources/extensions/cmux/index.js +20 -0
  20. package/dist/resources/extensions/github-sync/templates.js +103 -0
  21. package/dist/resources/extensions/google-search/extension-manifest.json +5 -4
  22. package/dist/resources/extensions/google-search/index.js +3 -375
  23. package/dist/resources/extensions/gsd/abandon-detect.js +44 -0
  24. package/dist/resources/extensions/gsd/auto/loop.js +90 -2
  25. package/dist/resources/extensions/gsd/auto/phases.js +95 -21
  26. package/dist/resources/extensions/gsd/auto/resolve.js +24 -0
  27. package/dist/resources/extensions/gsd/auto/run-unit.js +48 -4
  28. package/dist/resources/extensions/gsd/auto/session.js +18 -1
  29. package/dist/resources/extensions/gsd/auto/turn-epoch.js +95 -0
  30. package/dist/resources/extensions/gsd/auto-dispatch.js +115 -17
  31. package/dist/resources/extensions/gsd/auto-loop.js +1 -1
  32. package/dist/resources/extensions/gsd/auto-model-selection.js +1 -1
  33. package/dist/resources/extensions/gsd/auto-post-unit.js +90 -2
  34. package/dist/resources/extensions/gsd/auto-prompts.js +14 -0
  35. package/dist/resources/extensions/gsd/auto-recovery.js +46 -1
  36. package/dist/resources/extensions/gsd/auto-start.js +45 -39
  37. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +11 -5
  38. package/dist/resources/extensions/gsd/auto-unit-closeout.js +11 -2
  39. package/dist/resources/extensions/gsd/auto-worktree.js +109 -61
  40. package/dist/resources/extensions/gsd/auto.js +97 -31
  41. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +27 -1
  42. package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +4 -2
  43. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +11 -0
  44. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +17 -6
  45. package/dist/resources/extensions/gsd/bootstrap/system-context.js +11 -6
  46. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +34 -2
  47. package/dist/resources/extensions/gsd/clean-root-preflight.js +93 -0
  48. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +31 -4
  49. package/dist/resources/extensions/gsd/commands-cmux.js +9 -6
  50. package/dist/resources/extensions/gsd/commands-extensions.js +634 -43
  51. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +968 -23
  52. package/dist/resources/extensions/gsd/dispatch-guard.js +29 -3
  53. package/dist/resources/extensions/gsd/file-lock.js +49 -9
  54. package/dist/resources/extensions/gsd/git-service.js +1 -0
  55. package/dist/resources/extensions/gsd/gitignore.js +2 -0
  56. package/dist/resources/extensions/gsd/gsd-db.js +90 -30
  57. package/dist/resources/extensions/gsd/guided-flow-queue.js +4 -1
  58. package/dist/resources/extensions/gsd/guided-flow.js +212 -9
  59. package/dist/resources/extensions/gsd/health-widget.js +4 -1
  60. package/dist/resources/extensions/gsd/journal.js +17 -2
  61. package/dist/resources/extensions/gsd/key-manager.js +22 -0
  62. package/dist/resources/extensions/gsd/milestone-actions.js +15 -0
  63. package/dist/resources/extensions/gsd/milestone-summary-classifier.js +37 -0
  64. package/dist/resources/extensions/gsd/model-router.js +36 -3
  65. package/dist/resources/extensions/gsd/notifications.js +30 -16
  66. package/dist/resources/extensions/gsd/pre-execution-checks.js +31 -6
  67. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +29 -2
  68. package/dist/resources/extensions/gsd/prompts/discuss.md +29 -2
  69. package/dist/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  70. package/dist/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
  71. package/dist/resources/extensions/gsd/prompts/system.md +1 -0
  72. package/dist/resources/extensions/gsd/reports.js +5 -4
  73. package/dist/resources/extensions/gsd/safety/evidence-collector.js +96 -0
  74. package/dist/resources/extensions/gsd/safety/file-change-validator.js +12 -4
  75. package/dist/resources/extensions/gsd/safety/safety-harness.js +5 -1
  76. package/dist/resources/extensions/gsd/state-transition-matrix.js +118 -0
  77. package/dist/resources/extensions/gsd/state.js +25 -25
  78. package/dist/resources/extensions/gsd/token-counter.js +22 -5
  79. package/dist/resources/extensions/gsd/tools/complete-milestone.js +16 -10
  80. package/dist/resources/extensions/gsd/tools/complete-slice.js +21 -0
  81. package/dist/resources/extensions/gsd/tools/complete-task.js +31 -0
  82. package/dist/resources/extensions/gsd/uok/audit.js +18 -2
  83. package/dist/resources/extensions/gsd/uok/dispatch-envelope.js +33 -0
  84. package/dist/resources/extensions/gsd/uok/execution-graph.js +10 -0
  85. package/dist/resources/extensions/gsd/uok/gitops.js +2 -1
  86. package/dist/resources/extensions/gsd/uok/loop-adapter.js +37 -10
  87. package/dist/resources/extensions/gsd/uok/parity-report.js +58 -0
  88. package/dist/resources/extensions/gsd/uok/plan-v2.js +30 -7
  89. package/dist/resources/extensions/gsd/uok/writer.js +82 -0
  90. package/dist/resources/extensions/gsd/workflow-logger.js +10 -2
  91. package/dist/resources/extensions/gsd/worktree-manager.js +1 -0
  92. package/dist/resources/extensions/gsd/worktree-resolver.js +50 -10
  93. package/dist/resources/extensions/mcp-client/auth.js +10 -1
  94. package/dist/resources/extensions/mcp-client/index.js +118 -9
  95. package/dist/resources/extensions/shared/cmux-events.js +12 -0
  96. package/dist/resources/extensions/shared/rtk-session-stats.js +1 -2
  97. package/dist/resources/skills/create-skill/SKILL.md +2 -2
  98. package/dist/resources/skills/create-skill/references/gsd-skill-ecosystem.md +4 -4
  99. package/dist/resources/skills/create-skill/workflows/audit-skill.md +4 -4
  100. package/dist/resources/skills/create-skill/workflows/create-new-skill.md +5 -5
  101. package/dist/resources/skills/verify-before-complete/SKILL.md +2 -1
  102. package/dist/resources/skills/write-docs/SKILL.md +2 -1
  103. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  104. package/dist/web/standalone/.next/BUILD_ID +1 -1
  105. package/dist/web/standalone/.next/app-path-routes-manifest.json +9 -9
  106. package/dist/web/standalone/.next/build-manifest.json +3 -3
  107. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  108. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  109. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  110. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  112. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  113. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  114. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  115. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  116. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  117. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  118. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  119. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  120. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  121. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  122. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  123. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  124. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  125. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  126. package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
  127. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  128. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
  129. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  130. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
  131. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +1 -1
  132. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
  133. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  134. package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
  135. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  136. package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
  137. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  138. package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
  139. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  140. package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
  141. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  142. package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
  143. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  144. package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
  145. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  146. package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
  147. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  148. package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
  149. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  150. package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
  151. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  152. package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
  153. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  154. package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
  155. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  156. package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
  157. package/dist/web/standalone/.next/server/app/api/notifications/route.js +1 -1
  158. package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
  159. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  160. package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
  161. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  162. package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
  163. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  164. package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
  165. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  166. package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
  167. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  168. package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
  169. package/dist/web/standalone/.next/server/app/api/session/events/route.js +1 -1
  170. package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
  171. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  172. package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
  173. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  174. package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
  175. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  176. package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
  177. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  178. package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
  179. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  180. package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
  181. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
  182. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
  183. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
  184. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
  185. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  186. package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
  187. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  188. package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
  189. package/dist/web/standalone/.next/server/app/index.html +1 -1
  190. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  191. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  192. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  193. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  194. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  195. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  196. package/dist/web/standalone/.next/server/app-paths-manifest.json +9 -9
  197. package/dist/web/standalone/.next/server/chunks/1926.js +1 -0
  198. package/dist/web/standalone/.next/server/chunks/6897.js +2 -2
  199. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  200. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  201. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  202. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  203. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  204. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  205. package/dist/web/standalone/.next/static/chunks/2826.e9f5195e91f9cad2.js +11 -0
  206. package/dist/web/standalone/.next/static/chunks/{webpack-5fc74f13a25fa1bb.js → webpack-2e68521d7c82f7c2.js} +1 -1
  207. package/dist/welcome-screen.js +6 -1
  208. package/dist/wizard.js +2 -0
  209. package/package.json +16 -14
  210. package/packages/daemon/package.json +2 -2
  211. package/packages/mcp-server/README.md +3 -3
  212. package/packages/mcp-server/dist/env-writer.d.ts +1 -0
  213. package/packages/mcp-server/dist/env-writer.d.ts.map +1 -1
  214. package/packages/mcp-server/dist/env-writer.js +74 -6
  215. package/packages/mcp-server/dist/env-writer.js.map +1 -1
  216. package/packages/mcp-server/dist/remote-questions.d.ts +45 -0
  217. package/packages/mcp-server/dist/remote-questions.d.ts.map +1 -0
  218. package/packages/mcp-server/dist/remote-questions.js +732 -0
  219. package/packages/mcp-server/dist/remote-questions.js.map +1 -0
  220. package/packages/mcp-server/dist/server.d.ts +7 -0
  221. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  222. package/packages/mcp-server/dist/server.js +95 -10
  223. package/packages/mcp-server/dist/server.js.map +1 -1
  224. package/packages/mcp-server/dist/session-manager.d.ts +14 -0
  225. package/packages/mcp-server/dist/session-manager.d.ts.map +1 -1
  226. package/packages/mcp-server/dist/session-manager.js +49 -1
  227. package/packages/mcp-server/dist/session-manager.js.map +1 -1
  228. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  229. package/packages/mcp-server/dist/workflow-tools.js +15 -6
  230. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  231. package/packages/mcp-server/package.json +9 -3
  232. package/packages/mcp-server/src/env-writer.test.ts +79 -1
  233. package/packages/mcp-server/src/env-writer.ts +76 -6
  234. package/packages/mcp-server/src/mcp-server.test.ts +67 -0
  235. package/packages/mcp-server/src/readers/readers.test.ts +5 -1
  236. package/packages/mcp-server/src/remote-questions.test.ts +294 -0
  237. package/packages/mcp-server/src/remote-questions.ts +916 -0
  238. package/packages/mcp-server/src/server.ts +118 -16
  239. package/packages/mcp-server/src/session-manager.ts +43 -1
  240. package/packages/mcp-server/src/workflow-tools.test.ts +44 -0
  241. package/packages/mcp-server/src/workflow-tools.ts +19 -6
  242. package/packages/mcp-server/tsconfig.test.json +19 -0
  243. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  244. package/packages/native/package.json +6 -1
  245. package/packages/native/src/__tests__/clipboard.test.mjs +69 -23
  246. package/packages/native/tsconfig.tsbuildinfo +1 -1
  247. package/packages/pi-agent-core/package.json +6 -1
  248. package/packages/pi-agent-core/src/agent-loop.test.ts +220 -15
  249. package/packages/pi-ai/dist/models/custom.d.ts +38 -0
  250. package/packages/pi-ai/dist/models/custom.d.ts.map +1 -1
  251. package/packages/pi-ai/dist/models/custom.js +41 -0
  252. package/packages/pi-ai/dist/models/custom.js.map +1 -1
  253. package/packages/pi-ai/dist/providers/anthropic-auth.test.js +1 -1
  254. package/packages/pi-ai/dist/providers/anthropic-auth.test.js.map +1 -1
  255. package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -1
  256. package/packages/pi-ai/dist/providers/anthropic-shared.js +27 -4
  257. package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -1
  258. package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
  259. package/packages/pi-ai/dist/providers/anthropic.js +8 -3
  260. package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
  261. package/packages/pi-ai/dist/providers/minimax-tool-name.test.d.ts +2 -0
  262. package/packages/pi-ai/dist/providers/minimax-tool-name.test.d.ts.map +1 -0
  263. package/packages/pi-ai/dist/providers/minimax-tool-name.test.js +80 -0
  264. package/packages/pi-ai/dist/providers/minimax-tool-name.test.js.map +1 -0
  265. package/packages/pi-ai/dist/providers/simple-options.d.ts +10 -0
  266. package/packages/pi-ai/dist/providers/simple-options.d.ts.map +1 -1
  267. package/packages/pi-ai/dist/providers/simple-options.js +16 -1
  268. package/packages/pi-ai/dist/providers/simple-options.js.map +1 -1
  269. package/packages/pi-ai/package.json +6 -1
  270. package/packages/pi-ai/src/models/custom.ts +42 -0
  271. package/packages/pi-ai/src/providers/anthropic-auth.test.ts +1 -1
  272. package/packages/pi-ai/src/providers/anthropic-shared.ts +26 -5
  273. package/packages/pi-ai/src/providers/anthropic.ts +9 -3
  274. package/packages/pi-ai/src/providers/minimax-tool-name.test.ts +98 -0
  275. package/packages/pi-ai/src/providers/simple-options.ts +17 -1
  276. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  277. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +3 -2
  278. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
  279. package/packages/pi-coding-agent/dist/core/agent-session.d.ts +2 -0
  280. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  281. package/packages/pi-coding-agent/dist/core/agent-session.js +7 -0
  282. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  283. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts +25 -0
  284. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts.map +1 -1
  285. package/packages/pi-coding-agent/dist/core/compaction/compaction.js +105 -6
  286. package/packages/pi-coding-agent/dist/core/compaction/compaction.js.map +1 -1
  287. package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js +230 -28
  288. package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js.map +1 -1
  289. package/packages/pi-coding-agent/dist/core/compaction/utils.d.ts +30 -2
  290. package/packages/pi-coding-agent/dist/core/compaction/utils.d.ts.map +1 -1
  291. package/packages/pi-coding-agent/dist/core/compaction/utils.js +113 -12
  292. package/packages/pi-coding-agent/dist/core/compaction/utils.js.map +1 -1
  293. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.d.ts +1 -0
  294. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.d.ts.map +1 -1
  295. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.js +29 -18
  296. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.js.map +1 -1
  297. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.d.ts +2 -0
  298. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.d.ts.map +1 -0
  299. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.js +130 -0
  300. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.js.map +1 -0
  301. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js +56 -1
  302. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js.map +1 -1
  303. package/packages/pi-coding-agent/dist/core/discovery-cache.test.js +8 -15
  304. package/packages/pi-coding-agent/dist/core/discovery-cache.test.js.map +1 -1
  305. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.d.ts +25 -0
  306. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.d.ts.map +1 -0
  307. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.js +109 -0
  308. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.js.map +1 -0
  309. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.d.ts +67 -0
  310. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.d.ts.map +1 -0
  311. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.js +167 -0
  312. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.js.map +1 -0
  313. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +3 -2
  314. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  315. package/packages/pi-coding-agent/dist/core/extensions/loader.js +24 -8
  316. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  317. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +2 -0
  318. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
  319. package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
  320. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +14 -0
  321. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
  322. package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
  323. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js +11 -0
  324. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js.map +1 -1
  325. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +2 -2
  326. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
  327. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.d.ts +2 -0
  328. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.d.ts.map +1 -0
  329. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.js +203 -0
  330. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.js.map +1 -0
  331. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  332. package/packages/pi-coding-agent/dist/core/model-registry.js +14 -0
  333. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  334. package/packages/pi-coding-agent/dist/core/resource-loader.js +1 -1
  335. package/packages/pi-coding-agent/dist/core/resource-loader.js.map +1 -1
  336. package/packages/pi-coding-agent/dist/core/sdk.d.ts +1 -0
  337. package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
  338. package/packages/pi-coding-agent/dist/core/sdk.js +4 -1
  339. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  340. package/packages/pi-coding-agent/dist/core/sdk.test.js +19 -1
  341. package/packages/pi-coding-agent/dist/core/sdk.test.js.map +1 -1
  342. package/packages/pi-coding-agent/dist/core/session-manager.js +1 -1
  343. package/packages/pi-coding-agent/dist/core/session-manager.js.map +1 -1
  344. package/packages/pi-coding-agent/dist/core/session-manager.test.js +21 -1
  345. package/packages/pi-coding-agent/dist/core/session-manager.test.js.map +1 -1
  346. package/packages/pi-coding-agent/dist/core/system-prompt.js +3 -3
  347. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  348. package/packages/pi-coding-agent/dist/core/tools/path-utils.test.js +2 -1
  349. package/packages/pi-coding-agent/dist/core/tools/path-utils.test.js.map +1 -1
  350. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/provider-display-name.test.js +15 -6
  351. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/provider-display-name.test.js.map +1 -1
  352. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
  353. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +14 -5
  354. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
  355. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts +7 -1
  356. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts.map +1 -1
  357. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js +31 -9
  358. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js.map +1 -1
  359. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +14 -0
  360. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  361. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  362. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +13 -1
  363. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  364. package/packages/pi-coding-agent/package.json +6 -1
  365. package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +3 -2
  366. package/packages/pi-coding-agent/src/core/agent-session.ts +11 -0
  367. package/packages/pi-coding-agent/src/core/compaction/compaction.test.ts +368 -28
  368. package/packages/pi-coding-agent/src/core/compaction/compaction.ts +122 -6
  369. package/packages/pi-coding-agent/src/core/compaction/utils.ts +111 -13
  370. package/packages/pi-coding-agent/src/core/compaction-orchestrator.test.ts +154 -0
  371. package/packages/pi-coding-agent/src/core/compaction-orchestrator.ts +32 -18
  372. package/packages/pi-coding-agent/src/core/compaction-utils.test.ts +68 -1
  373. package/packages/pi-coding-agent/src/core/discovery-cache.test.ts +9 -18
  374. package/packages/pi-coding-agent/src/core/extensions/extension-discovery.ts +119 -0
  375. package/packages/pi-coding-agent/src/core/extensions/extension-registry.ts +222 -0
  376. package/packages/pi-coding-agent/src/core/extensions/loader.ts +24 -11
  377. package/packages/pi-coding-agent/src/core/extensions/runner.ts +2 -0
  378. package/packages/pi-coding-agent/src/core/extensions/types.ts +15 -0
  379. package/packages/pi-coding-agent/src/core/lsp/lsp-integration.test.ts +13 -0
  380. package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +2 -2
  381. package/packages/pi-coding-agent/src/core/model-registry-custom-caps.test.ts +245 -0
  382. package/packages/pi-coding-agent/src/core/model-registry.ts +16 -0
  383. package/packages/pi-coding-agent/src/core/resource-loader.ts +1 -1
  384. package/packages/pi-coding-agent/src/core/sdk.test.ts +25 -1
  385. package/packages/pi-coding-agent/src/core/sdk.ts +10 -3
  386. package/packages/pi-coding-agent/src/core/session-manager.test.ts +30 -1
  387. package/packages/pi-coding-agent/src/core/session-manager.ts +1 -1
  388. package/packages/pi-coding-agent/src/core/system-prompt.ts +3 -3
  389. package/packages/pi-coding-agent/src/core/tools/path-utils.test.ts +2 -1
  390. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/provider-display-name.test.ts +17 -7
  391. package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +14 -5
  392. package/packages/pi-coding-agent/src/modes/interactive/components/model-selector.ts +45 -11
  393. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +14 -0
  394. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +13 -1
  395. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  396. package/packages/pi-tui/dist/__tests__/autocomplete.test.js +12 -5
  397. package/packages/pi-tui/dist/__tests__/autocomplete.test.js.map +1 -1
  398. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js +21 -0
  399. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js.map +1 -1
  400. package/packages/pi-tui/dist/stdin-buffer.d.ts +7 -0
  401. package/packages/pi-tui/dist/stdin-buffer.d.ts.map +1 -1
  402. package/packages/pi-tui/dist/stdin-buffer.js +20 -0
  403. package/packages/pi-tui/dist/stdin-buffer.js.map +1 -1
  404. package/packages/pi-tui/package.json +6 -1
  405. package/packages/pi-tui/src/__tests__/autocomplete.test.ts +20 -5
  406. package/packages/pi-tui/src/__tests__/stdin-buffer.test.ts +27 -0
  407. package/packages/pi-tui/src/stdin-buffer.ts +26 -0
  408. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  409. package/packages/rpc-client/package.json +6 -1
  410. package/pkg/package.json +1 -1
  411. package/scripts/install.js +512 -0
  412. package/scripts/lib/workspace-manifest.cjs +86 -0
  413. package/scripts/link-workspace-packages.cjs +5 -17
  414. package/scripts/postinstall.js +9 -178
  415. package/src/resources/extensions/claude-code-cli/readiness.ts +4 -3
  416. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +91 -63
  417. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +114 -12
  418. package/src/resources/extensions/cmux/index.ts +35 -10
  419. package/src/resources/extensions/github-sync/templates.ts +151 -0
  420. package/src/resources/extensions/github-sync/tests/templates.test.ts +59 -0
  421. package/src/resources/extensions/google-search/extension-manifest.json +5 -4
  422. package/src/resources/extensions/google-search/index.ts +9 -470
  423. package/src/resources/extensions/gsd/abandon-detect.ts +62 -0
  424. package/src/resources/extensions/gsd/auto/loop-deps.ts +14 -1
  425. package/src/resources/extensions/gsd/auto/loop.ts +104 -2
  426. package/src/resources/extensions/gsd/auto/phases.ts +123 -21
  427. package/src/resources/extensions/gsd/auto/resolve.ts +29 -0
  428. package/src/resources/extensions/gsd/auto/run-unit.ts +56 -4
  429. package/src/resources/extensions/gsd/auto/session.ts +28 -1
  430. package/src/resources/extensions/gsd/auto/turn-epoch.ts +108 -0
  431. package/src/resources/extensions/gsd/auto/types.ts +1 -1
  432. package/src/resources/extensions/gsd/auto-dispatch.ts +117 -16
  433. package/src/resources/extensions/gsd/auto-loop.ts +1 -1
  434. package/src/resources/extensions/gsd/auto-model-selection.ts +1 -1
  435. package/src/resources/extensions/gsd/auto-post-unit.ts +92 -3
  436. package/src/resources/extensions/gsd/auto-prompts.ts +28 -1
  437. package/src/resources/extensions/gsd/auto-recovery.ts +40 -1
  438. package/src/resources/extensions/gsd/auto-start.ts +48 -52
  439. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +12 -5
  440. package/src/resources/extensions/gsd/auto-unit-closeout.ts +14 -3
  441. package/src/resources/extensions/gsd/auto-worktree.ts +122 -68
  442. package/src/resources/extensions/gsd/auto.ts +105 -35
  443. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +34 -1
  444. package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +6 -2
  445. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +11 -0
  446. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +18 -6
  447. package/src/resources/extensions/gsd/bootstrap/system-context.ts +13 -9
  448. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +35 -2
  449. package/src/resources/extensions/gsd/clean-root-preflight.ts +111 -0
  450. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +27 -8
  451. package/src/resources/extensions/gsd/commands-cmux.ts +10 -6
  452. package/src/resources/extensions/gsd/commands-extensions.ts +747 -41
  453. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +898 -32
  454. package/src/resources/extensions/gsd/dispatch-guard.ts +26 -2
  455. package/src/resources/extensions/gsd/file-lock.ts +84 -11
  456. package/src/resources/extensions/gsd/git-service.ts +1 -0
  457. package/src/resources/extensions/gsd/gitignore.ts +2 -1
  458. package/src/resources/extensions/gsd/gsd-db.ts +92 -32
  459. package/src/resources/extensions/gsd/guided-flow-queue.ts +4 -1
  460. package/src/resources/extensions/gsd/guided-flow.ts +259 -10
  461. package/src/resources/extensions/gsd/health-widget.ts +3 -1
  462. package/src/resources/extensions/gsd/journal.ts +29 -3
  463. package/src/resources/extensions/gsd/key-manager.ts +22 -0
  464. package/src/resources/extensions/gsd/milestone-actions.ts +18 -0
  465. package/src/resources/extensions/gsd/milestone-summary-classifier.ts +42 -0
  466. package/src/resources/extensions/gsd/model-router.ts +42 -1
  467. package/src/resources/extensions/gsd/notifications.ts +27 -15
  468. package/src/resources/extensions/gsd/pre-execution-checks.ts +33 -7
  469. package/src/resources/extensions/gsd/preferences-types.ts +8 -0
  470. package/src/resources/extensions/gsd/prompts/discuss-headless.md +29 -2
  471. package/src/resources/extensions/gsd/prompts/discuss.md +29 -2
  472. package/src/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  473. package/src/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
  474. package/src/resources/extensions/gsd/prompts/system.md +1 -0
  475. package/src/resources/extensions/gsd/reports.ts +5 -4
  476. package/src/resources/extensions/gsd/safety/evidence-collector.ts +119 -0
  477. package/src/resources/extensions/gsd/safety/file-change-validator.ts +16 -3
  478. package/src/resources/extensions/gsd/safety/safety-harness.ts +9 -0
  479. package/src/resources/extensions/gsd/state-transition-matrix.ts +152 -0
  480. package/src/resources/extensions/gsd/state.ts +35 -30
  481. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +238 -4
  482. package/src/resources/extensions/gsd/tests/auto-mode-guards.test.ts +79 -0
  483. package/src/resources/extensions/gsd/tests/auto-paused-session-validation.test.ts +12 -0
  484. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +122 -0
  485. package/src/resources/extensions/gsd/tests/auto-start-bootstrap-await-3420.test.ts +141 -0
  486. package/src/resources/extensions/gsd/tests/auto-start-clean-runtime-db-gated.test.ts +63 -0
  487. package/src/resources/extensions/gsd/tests/auto-wrapup-inflight-guard.test.ts +23 -0
  488. package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +186 -0
  489. package/src/resources/extensions/gsd/tests/cmux.test.ts +5 -9
  490. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +15 -0
  491. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +61 -1
  492. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  493. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  494. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +161 -0
  495. package/src/resources/extensions/gsd/tests/db-access-guardrails.test.ts +1 -0
  496. package/src/resources/extensions/gsd/tests/derive-state.test.ts +1 -2
  497. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +14 -9
  498. package/src/resources/extensions/gsd/tests/dispatch-guard-summary-db-mismatch.test.ts +77 -0
  499. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +25 -0
  500. package/src/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +14 -0
  501. package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +31 -0
  502. package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +1 -1
  503. package/src/resources/extensions/gsd/tests/ensure-db-open.test.ts +1 -1
  504. package/src/resources/extensions/gsd/tests/escalation.test.ts +1 -1
  505. package/src/resources/extensions/gsd/tests/exec-history.test.ts +113 -0
  506. package/src/resources/extensions/gsd/tests/execution-entry-missing-context-4671.test.ts +173 -0
  507. package/src/resources/extensions/gsd/tests/file-change-validator.test.ts +38 -0
  508. package/src/resources/extensions/gsd/tests/file-lock.test.ts +86 -12
  509. package/src/resources/extensions/gsd/tests/google-search-stub.test.ts +131 -0
  510. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +296 -1
  511. package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +117 -0
  512. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +30 -0
  513. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +4 -2
  514. package/src/resources/extensions/gsd/tests/integration/gitignore-tracked-gsd.test.ts +1 -0
  515. package/src/resources/extensions/gsd/tests/integration/idle-recovery.test.ts +30 -0
  516. package/src/resources/extensions/gsd/tests/integration/worktree-e2e.test.ts +11 -0
  517. package/src/resources/extensions/gsd/tests/issue-4540-regressions.test.ts +288 -0
  518. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +66 -0
  519. package/src/resources/extensions/gsd/tests/key-manager.test.ts +2 -0
  520. package/src/resources/extensions/gsd/tests/mcp-client-security.test.ts +76 -0
  521. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -1
  522. package/src/resources/extensions/gsd/tests/memory-pressure-stuck-state.test.ts +12 -0
  523. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -2
  524. package/src/resources/extensions/gsd/tests/milestone-status-authoritative.test.ts +3 -3
  525. package/src/resources/extensions/gsd/tests/milestone-summary-classifier.test.ts +30 -0
  526. package/src/resources/extensions/gsd/tests/milestone-transition-state-rebuild.test.ts +4 -2
  527. package/src/resources/extensions/gsd/tests/parallel-commit-scope.test.ts +5 -0
  528. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +19 -0
  529. package/src/resources/extensions/gsd/tests/parallel-skill-prompt-integration.test.ts +150 -0
  530. package/src/resources/extensions/gsd/tests/plan-gate-failed-doctor-heal-hint.test.ts +37 -0
  531. package/src/resources/extensions/gsd/tests/pre-exec-gate-loop.test.ts +272 -0
  532. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +337 -0
  533. package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +44 -0
  534. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +39 -25
  535. package/src/resources/extensions/gsd/tests/queue-auto-guard.test.ts +181 -0
  536. package/src/resources/extensions/gsd/tests/quick-auto-guard.test.ts +13 -7
  537. package/src/resources/extensions/gsd/tests/ready-phrase-no-files-4573.test.ts +388 -0
  538. package/src/resources/extensions/gsd/tests/require-slice-discussion-dispatch.test.ts +170 -0
  539. package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +9 -3
  540. package/src/resources/extensions/gsd/tests/resume-dispatch-worktree.test.ts +230 -0
  541. package/src/resources/extensions/gsd/tests/rewrite-docs-abandon-detect.test.ts +195 -0
  542. package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +205 -0
  543. package/src/resources/extensions/gsd/tests/schema-v21-sequence.test.ts +413 -0
  544. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +32 -40
  545. package/src/resources/extensions/gsd/tests/stale-dirlistcache-4648.test.ts +112 -0
  546. package/src/resources/extensions/gsd/tests/stash-queued-context-files.test.ts +56 -0
  547. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +24 -0
  548. package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +44 -0
  549. package/src/resources/extensions/gsd/tests/stuck-detection-coverage.test.ts +2 -2
  550. package/src/resources/extensions/gsd/tests/token-counter.test.ts +105 -1
  551. package/src/resources/extensions/gsd/tests/tool-compatibility.test.ts +107 -0
  552. package/src/resources/extensions/gsd/tests/triage-resolution.test.ts +50 -2
  553. package/src/resources/extensions/gsd/tests/turn-epoch.test.ts +162 -0
  554. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +51 -0
  555. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +16 -0
  556. package/src/resources/extensions/gsd/tests/uok-loop-adapter-writer.test.ts +65 -0
  557. package/src/resources/extensions/gsd/tests/uok-parity-report.test.ts +42 -0
  558. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +42 -2
  559. package/src/resources/extensions/gsd/tests/uok-writer.test.ts +75 -0
  560. package/src/resources/extensions/gsd/tests/validate-extension-package.test.ts +168 -0
  561. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +138 -5
  562. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +25 -2
  563. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +65 -2
  564. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +35 -0
  565. package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +6 -1
  566. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +78 -5
  567. package/src/resources/extensions/gsd/tests/write-gate.test.ts +64 -0
  568. package/src/resources/extensions/gsd/token-counter.ts +22 -5
  569. package/src/resources/extensions/gsd/tools/complete-milestone.ts +15 -9
  570. package/src/resources/extensions/gsd/tools/complete-slice.ts +38 -0
  571. package/src/resources/extensions/gsd/tools/complete-task.ts +49 -0
  572. package/src/resources/extensions/gsd/uok/audit.ts +20 -2
  573. package/src/resources/extensions/gsd/uok/contracts.ts +65 -0
  574. package/src/resources/extensions/gsd/uok/dispatch-envelope.ts +56 -0
  575. package/src/resources/extensions/gsd/uok/execution-graph.ts +22 -0
  576. package/src/resources/extensions/gsd/uok/gitops.ts +6 -1
  577. package/src/resources/extensions/gsd/uok/loop-adapter.ts +45 -10
  578. package/src/resources/extensions/gsd/uok/parity-report.ts +84 -0
  579. package/src/resources/extensions/gsd/uok/plan-v2.ts +39 -8
  580. package/src/resources/extensions/gsd/uok/writer.ts +113 -0
  581. package/src/resources/extensions/gsd/workflow-logger.ts +23 -3
  582. package/src/resources/extensions/gsd/worktree-manager.ts +1 -0
  583. package/src/resources/extensions/gsd/worktree-resolver.ts +54 -9
  584. package/src/resources/extensions/mcp-client/auth.ts +12 -1
  585. package/src/resources/extensions/mcp-client/index.ts +129 -10
  586. package/src/resources/extensions/shared/cmux-events.ts +59 -0
  587. package/src/resources/extensions/shared/rtk-session-stats.ts +1 -2
  588. package/src/resources/skills/create-skill/SKILL.md +2 -2
  589. package/src/resources/skills/create-skill/references/gsd-skill-ecosystem.md +4 -4
  590. package/src/resources/skills/create-skill/workflows/audit-skill.md +4 -4
  591. package/src/resources/skills/create-skill/workflows/create-new-skill.md +5 -5
  592. package/src/resources/skills/verify-before-complete/SKILL.md +2 -1
  593. package/src/resources/skills/write-docs/SKILL.md +2 -1
  594. package/dist/web/standalone/.next/server/chunks/7461.js +0 -1
  595. package/dist/web/standalone/.next/static/chunks/2826.e59e8578e2e28639.js +0 -9
  596. /package/dist/web/standalone/.next/static/{n21VtX2hZlkpdEUO_nU4z → vidAVJkURvTJ0_V2-64ro}/_buildManifest.js +0 -0
  597. /package/dist/web/standalone/.next/static/{n21VtX2hZlkpdEUO_nU4z → vidAVJkURvTJ0_V2-64ro}/_ssgManifest.js +0 -0
@@ -0,0 +1,230 @@
1
+ /**
2
+ * Tests for fix of #3723: auto-mode resume/crash-recovery dispatches
3
+ * from project root instead of milestone worktree.
4
+ *
5
+ * During resume, the paused-session metadata may record `worktreePath` that
6
+ * was active when the session paused. The resume path must use that path (or
7
+ * derive the worktree path via filesystem lookup) to set the dispatch context
8
+ * (`s.basePath`), rather than defaulting to the project root.
9
+ *
10
+ * The fix adds an early worktree-path resolution step in the paused-session
11
+ * resume block of auto.ts — immediately after `s.basePath = base` — so that
12
+ * the correct dispatch directory is used before the dispatch loop runs.
13
+ */
14
+
15
+ import test from "node:test";
16
+ import assert from "node:assert/strict";
17
+ import { readFileSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
18
+ import { join } from "node:path";
19
+ import { tmpdir } from "node:os";
20
+ import { randomUUID } from "node:crypto";
21
+ import { fileURLToPath } from "node:url";
22
+ import { dirname } from "node:path";
23
+
24
+ const __dirname = dirname(fileURLToPath(import.meta.url));
25
+ const autoSrc = readFileSync(join(__dirname, "..", "auto.ts"), "utf-8");
26
+
27
+ // ── Source-structure tests ────────────────────────────────────────────────────
28
+
29
+ /**
30
+ * Extract the paused-session resume block from auto.ts.
31
+ *
32
+ * The block we care about is the `if (s.paused) { ... }` section inside
33
+ * startAuto, which contains `s.basePath = base` (line ~1473) followed by the
34
+ * `enterMilestone` call.
35
+ *
36
+ * We find it by locating the `s.basePath = base` assignment that appears
37
+ * WITHIN the s.paused branch (there's only one: all other basePath assignments
38
+ * use originalBasePath or a different value). We extract from that assignment
39
+ * up to just before `enterMilestone(`.
40
+ */
41
+ function getBasepathToEnterMilestoneSegment(): string {
42
+ // Find `s.basePath = base;` in the s.paused branch
43
+ // This assignment appears uniquely inside the resume block
44
+ const assignPattern = "s.basePath = base;";
45
+ const assignIdx = autoSrc.indexOf(assignPattern);
46
+ assert.ok(
47
+ assignIdx > -1,
48
+ `auto.ts must contain '${assignPattern}' in the resume block`,
49
+ );
50
+
51
+ // Find the next enterMilestone call after this assignment
52
+ const enterMilestoneIdx = autoSrc.indexOf("enterMilestone(", assignIdx);
53
+ assert.ok(
54
+ enterMilestoneIdx > assignIdx,
55
+ "auto.ts must call enterMilestone after the s.basePath = base assignment",
56
+ );
57
+
58
+ // Return the code between the assignment and enterMilestone
59
+ return autoSrc.slice(assignIdx, enterMilestoneIdx);
60
+ }
61
+
62
+ test("auto.ts resume block resolves paused-session worktreePath and applies it to s.basePath before entering worktree (fixes #3723)", () => {
63
+ // The segment between `s.basePath = base` and `enterMilestone(` must
64
+ // contain logic that resolves the paused-session worktreePath and applies
65
+ // it to s.basePath when the worktree exists on disk.
66
+ //
67
+ // The fix reads the worktree path from freshStartAssessment.pausedSession
68
+ // (since `meta` is out of scope at this point in startAuto) and assigns
69
+ // it to s.basePath, guarded by existsSync.
70
+ //
71
+ // Without this fix, the dispatch loop runs from `base` (project root)
72
+ // instead of the worktree, causing split-brain execution (#3723).
73
+ const segment = getBasepathToEnterMilestoneSegment();
74
+
75
+ // The fix uses freshStartAssessment.pausedSession?.worktreePath via a
76
+ // local variable (resumeWorktreePath) and assigns to s.basePath.
77
+ const hasWorktreePathResolution =
78
+ segment.includes("worktreePath") &&
79
+ segment.includes("existsSync") &&
80
+ segment.includes("s.basePath =");
81
+
82
+ assert.ok(
83
+ hasWorktreePathResolution,
84
+ "auto.ts must resolve the paused-session worktreePath, check existsSync, and assign " +
85
+ "s.basePath before enterMilestone — crash-recovery currently dispatches from project root " +
86
+ "instead of milestone worktree (issue #3723). The fix belongs between `s.basePath = base` " +
87
+ "and the enterMilestone call.",
88
+ );
89
+ });
90
+
91
+ test("auto.ts worktreePath assignment in resume block guards against non-existent path (fixes #3723)", () => {
92
+ // The assignment to s.basePath from the paused-session worktreePath must
93
+ // be guarded by existsSync to avoid setting an invalid basePath if the
94
+ // worktree was cleaned up between pause and resume.
95
+ const segment = getBasepathToEnterMilestoneSegment();
96
+
97
+ // Must have existsSync guard AND a worktreePath reference AND s.basePath assignment
98
+ const hasGuardedAssignment =
99
+ segment.includes("existsSync") &&
100
+ segment.includes("worktreePath") &&
101
+ segment.includes("s.basePath =");
102
+
103
+ assert.ok(
104
+ hasGuardedAssignment,
105
+ "auto.ts must guard the s.basePath = worktreePath assignment with existsSync (fixes #3723)",
106
+ );
107
+ });
108
+
109
+ // ── Functional tests ──────────────────────────────────────────────────────────
110
+
111
+ function makeTmpBase(): string {
112
+ const base = join(tmpdir(), `gsd-resume-wt-${randomUUID()}`);
113
+ mkdirSync(join(base, ".gsd", "runtime"), { recursive: true });
114
+ return base;
115
+ }
116
+
117
+ function cleanup(base: string): void {
118
+ try { rmSync(base, { recursive: true, force: true }); } catch { /* */ }
119
+ }
120
+
121
+ function writePausedSession(
122
+ base: string,
123
+ milestoneId: string,
124
+ worktreePath: string | null,
125
+ ): void {
126
+ writeFileSync(
127
+ join(base, ".gsd", "runtime", "paused-session.json"),
128
+ JSON.stringify({
129
+ milestoneId,
130
+ originalBasePath: base,
131
+ stepMode: false,
132
+ worktreePath,
133
+ pausedAt: new Date().toISOString(),
134
+ }, null, 2),
135
+ "utf-8",
136
+ );
137
+ }
138
+
139
+ function makeWorktreePath(base: string, milestoneId: string): string {
140
+ return join(base, ".gsd", "worktrees", milestoneId);
141
+ }
142
+
143
+ function setupWorktreeOnDisk(wt: string): void {
144
+ mkdirSync(wt, { recursive: true });
145
+ // Simulate a git worktree: .git file with gitdir pointer
146
+ writeFileSync(
147
+ join(wt, ".git"),
148
+ "gitdir: /project/.git/worktrees/M001-test\n",
149
+ "utf-8",
150
+ );
151
+ }
152
+
153
+ function writeRoadmap(base: string, milestoneId = "M001-test"): void {
154
+ const milestoneDir = join(base, ".gsd", "milestones", milestoneId);
155
+ mkdirSync(join(milestoneDir, "slices", "S01", "tasks"), { recursive: true });
156
+ writeFileSync(
157
+ join(milestoneDir, `${milestoneId}-ROADMAP.md`),
158
+ [
159
+ `# ${milestoneId}: Test Milestone`,
160
+ "",
161
+ "## Slices",
162
+ "",
163
+ "- [ ] **S01: Test slice** `risk:low`",
164
+ " After this: Demo",
165
+ ].join("\n"),
166
+ "utf-8",
167
+ );
168
+ }
169
+
170
+ test("readPausedSessionMetadata round-trips worktreePath from paused-session.json", () => {
171
+ // Verify that the paused-session metadata correctly stores and reads back
172
+ // the worktreePath field — this is what the resume path in auto.ts uses
173
+ // to determine the dispatch basePath (#3723).
174
+ //
175
+ // Implemented inline to avoid slow import chain from interrupted-session.ts.
176
+ const base = makeTmpBase();
177
+ const wt = makeWorktreePath(base, "M001-test");
178
+ try {
179
+ setupWorktreeOnDisk(wt);
180
+ writePausedSession(base, "M001-test", wt);
181
+
182
+ // Simulate readPausedSessionMetadata without importing the full module
183
+ const pausedPath = join(base, ".gsd", "runtime", "paused-session.json");
184
+ const meta = JSON.parse(readFileSync(pausedPath, "utf-8"));
185
+
186
+ assert.ok(meta, "paused-session metadata must be readable");
187
+ assert.equal(meta.milestoneId, "M001-test");
188
+ assert.equal(meta.worktreePath, wt, "worktreePath must round-trip through paused-session.json");
189
+ } finally {
190
+ cleanup(base);
191
+ }
192
+ });
193
+
194
+ test("auto.ts resume block uses worktreePath from freshStartAssessment.pausedSession as dispatch basePath when worktree exists (#3723)", () => {
195
+ // End-to-end structural verification: the auto.ts source must contain code
196
+ // that reads the worktreePath from freshStartAssessment.pausedSession AND
197
+ // applies it to s.basePath (guarded by existsSync) between the
198
+ // `s.basePath = base` assignment and `enterMilestone`.
199
+ //
200
+ // This is the core of the #3723 fix. Without this check, a session that
201
+ // paused while operating inside a worktree will resume dispatching from the
202
+ // project root, not the worktree — causing split-brain execution where some
203
+ // operations target the worktree and others target the project root.
204
+ const segment = getBasepathToEnterMilestoneSegment();
205
+
206
+ // Must reference freshStartAssessment.pausedSession?.worktreePath or
207
+ // equivalent local variable, AND contain an existsSync guard
208
+ const hasAssessmentWorktreePath =
209
+ segment.includes("freshStartAssessment.pausedSession") &&
210
+ segment.includes("worktreePath");
211
+
212
+ assert.ok(
213
+ hasAssessmentWorktreePath,
214
+ "auto.ts must read worktreePath from freshStartAssessment.pausedSession between " +
215
+ "s.basePath=base and enterMilestone",
216
+ );
217
+ assert.ok(
218
+ segment.includes("existsSync"),
219
+ "auto.ts must guard worktreePath usage with existsSync",
220
+ );
221
+ // The actual basePath re-assignment must be present in the segment
222
+ // (note: there are multiple `s.basePath =` assignments — we need one after
223
+ // the initial `s.basePath = base` assignment within this segment)
224
+ const worktreeAssignIdx = segment.lastIndexOf("s.basePath =");
225
+ const baseAssignIdx = segment.indexOf("s.basePath = base");
226
+ assert.ok(
227
+ worktreeAssignIdx > baseAssignIdx,
228
+ "auto.ts must assign s.basePath to the worktree path after `s.basePath = base` in the resume block",
229
+ );
230
+ });
@@ -0,0 +1,195 @@
1
+ /**
2
+ * Regression tests for #3490: post-unit handler for rewrite-docs must
3
+ * detect abandon/descope overrides that target the current milestone and
4
+ * park it — without false-positive parking on unrelated scope-change
5
+ * overrides that merely contain an abandon-family verb.
6
+ *
7
+ * Exercises detectAbandonMilestone() directly — a pure function over
8
+ * Override objects, no I/O and no production-code import chain.
9
+ * parkMilestone() end-to-end behavior is covered by the existing
10
+ * park-milestone.test.ts / park-edge-cases.test.ts / park-db-sync.test.ts
11
+ * suites; here we only validate the decision layer that feeds it.
12
+ */
13
+ import { test, describe } from "node:test";
14
+ import assert from "node:assert/strict";
15
+ import { detectAbandonMilestone } from "../abandon-detect.ts";
16
+ import type { Override } from "../files.ts";
17
+
18
+ function mkOverride(change: string): Override {
19
+ return {
20
+ timestamp: "2026-04-21T00:00:00.000Z",
21
+ change,
22
+ scope: "active",
23
+ appliedAt: "M001/S01/T01",
24
+ };
25
+ }
26
+
27
+ describe("detectAbandonMilestone (#3490)", () => {
28
+ test("parks on explicit abandon-milestone phrasing", () => {
29
+ const d = detectAbandonMilestone(
30
+ [mkOverride("abandon this milestone, it's no longer relevant")],
31
+ "M001",
32
+ );
33
+ assert.strictEqual(d.shouldPark, true);
34
+ assert.strictEqual(d.matched.length, 1);
35
+ assert.match(d.reason, /abandon this milestone/);
36
+ });
37
+
38
+ test("parks on descope-milestone phrasing", () => {
39
+ const d = detectAbandonMilestone(
40
+ [mkOverride("descope the entire milestone")],
41
+ "M001",
42
+ );
43
+ assert.strictEqual(d.shouldPark, true);
44
+ });
45
+
46
+ test("parks when override references milestone by ID instead of the word", () => {
47
+ const d = detectAbandonMilestone(
48
+ [mkOverride("shelve M003 for now, coming back next quarter")],
49
+ "M003",
50
+ );
51
+ assert.strictEqual(d.shouldPark, true);
52
+ assert.strictEqual(d.matched[0], "shelve M003 for now, coming back next quarter");
53
+ });
54
+
55
+ test("parks on past-tense / gerund verb forms (UK spelling)", () => {
56
+ const d = detectAbandonMilestone(
57
+ [mkOverride("this milestone was cancelled by product")],
58
+ "M001",
59
+ );
60
+ assert.strictEqual(d.shouldPark, true);
61
+ });
62
+
63
+ test("parks on US spelling 'canceled' (single-l)", () => {
64
+ const d = detectAbandonMilestone(
65
+ [mkOverride("the milestone was canceled by the PM")],
66
+ "M001",
67
+ );
68
+ assert.strictEqual(d.shouldPark, true);
69
+ });
70
+
71
+ test("parks on hyphenated 'de-scope' variant", () => {
72
+ const d = detectAbandonMilestone(
73
+ [mkOverride("de-scope this milestone — moved to v2")],
74
+ "M001",
75
+ );
76
+ assert.strictEqual(d.shouldPark, true);
77
+ });
78
+
79
+ test("parks on space-separated 'de scope' variant", () => {
80
+ const d = detectAbandonMilestone(
81
+ [mkOverride("de scope the milestone entirely")],
82
+ "M001",
83
+ );
84
+ assert.strictEqual(d.shouldPark, true);
85
+ });
86
+
87
+ test("parks on 'de-scoped' past-tense hyphen variant", () => {
88
+ const d = detectAbandonMilestone(
89
+ [mkOverride("M003 was de-scoped last week")],
90
+ "M003",
91
+ );
92
+ assert.strictEqual(d.shouldPark, true);
93
+ });
94
+
95
+ // ─── False-positive guards ────────────────────────────────────────────
96
+
97
+ test("does NOT park on 'cancel the standup reminder' (no milestone ref)", () => {
98
+ const d = detectAbandonMilestone(
99
+ [mkOverride("cancel the daily standup reminder")],
100
+ "M001",
101
+ );
102
+ assert.strictEqual(d.shouldPark, false);
103
+ assert.deepStrictEqual(d.matched, []);
104
+ });
105
+
106
+ test("does NOT park on 'drop the dependency on X' (no milestone ref)", () => {
107
+ const d = detectAbandonMilestone(
108
+ [mkOverride("drop the dependency on X and use Y instead")],
109
+ "M001",
110
+ );
111
+ assert.strictEqual(d.shouldPark, false);
112
+ });
113
+
114
+ test("does NOT park on 'scrap the v1 design' (no milestone ref)", () => {
115
+ const d = detectAbandonMilestone(
116
+ [mkOverride("scrap the v1 design for the landing page")],
117
+ "M001",
118
+ );
119
+ assert.strictEqual(d.shouldPark, false);
120
+ });
121
+
122
+ test("does NOT park on override with milestone word but no abandon verb", () => {
123
+ const d = detectAbandonMilestone(
124
+ [mkOverride("change the milestone title to something clearer")],
125
+ "M001",
126
+ );
127
+ assert.strictEqual(d.shouldPark, false);
128
+ });
129
+
130
+ test("does NOT park when a different milestone ID is referenced", () => {
131
+ // The abandon verb is present and an MID is present, but the MID is
132
+ // not the current milestone. We require either the literal word
133
+ // "milestone" OR the current MID — a reference to a different MID
134
+ // (M007) with neither the word "milestone" nor the current ID (M001)
135
+ // should not trigger.
136
+ const d = detectAbandonMilestone(
137
+ [mkOverride("drop M007")],
138
+ "M001",
139
+ );
140
+ assert.strictEqual(d.shouldPark, false);
141
+ });
142
+
143
+ // ─── Edge cases ───────────────────────────────────────────────────────
144
+
145
+ test("empty overrides list returns no-park", () => {
146
+ const d = detectAbandonMilestone([], "M001");
147
+ assert.strictEqual(d.shouldPark, false);
148
+ assert.deepStrictEqual(d.matched, []);
149
+ });
150
+
151
+ test("null/undefined currentMilestoneId returns no-park even with matching text", () => {
152
+ const d1 = detectAbandonMilestone(
153
+ [mkOverride("abandon this milestone")],
154
+ null,
155
+ );
156
+ assert.strictEqual(d1.shouldPark, false);
157
+
158
+ const d2 = detectAbandonMilestone(
159
+ [mkOverride("abandon this milestone")],
160
+ undefined,
161
+ );
162
+ assert.strictEqual(d2.shouldPark, false);
163
+ });
164
+
165
+ test("multiple abandon overrides are concatenated in reason", () => {
166
+ const d = detectAbandonMilestone(
167
+ [
168
+ mkOverride("abandon this milestone"),
169
+ mkOverride("cancel the standup"), // filtered out (no ref)
170
+ mkOverride("descope the milestone entirely"),
171
+ ],
172
+ "M001",
173
+ );
174
+ assert.strictEqual(d.shouldPark, true);
175
+ assert.strictEqual(d.matched.length, 2, "only the two milestone-scoped overrides match");
176
+ assert.match(d.reason, /abandon this milestone/);
177
+ assert.match(d.reason, /descope the milestone entirely/);
178
+ assert.doesNotMatch(d.reason, /cancel the standup/);
179
+ });
180
+
181
+ // ─── Regex-injection guard on milestone ID ────────────────────────────
182
+
183
+ test("milestone ID with regex metacharacters is escaped, not interpreted", () => {
184
+ // A pathological MID should not break the matcher — the function
185
+ // escapes regex metacharacters before building the pattern.
186
+ const d = detectAbandonMilestone(
187
+ [mkOverride("abandon M.01 the milestone")],
188
+ "M.01",
189
+ );
190
+ // 'milestone' is present, so this parks regardless of escaping,
191
+ // but the test confirms the RegExp construction does not throw.
192
+ assert.strictEqual(d.shouldPark, true);
193
+ });
194
+ });
195
+
@@ -0,0 +1,205 @@
1
+ /**
2
+ * Regression tests for three false-positive sources in the safety harness.
3
+ * Issue #4385
4
+ *
5
+ * Bug 1: Hardcoded BASH_READ_ONLY_RE — new legitimate commands blocked
6
+ * Bug 2: Non-persisted evidence — session restart causes false positive on resume
7
+ * Bug 3: git diff HEAD~1 scope check — fails on initial commits / shallow clones
8
+ */
9
+
10
+ import test from "node:test";
11
+ import assert from "node:assert/strict";
12
+ import { execFileSync } from "node:child_process";
13
+ import { mkdtempSync, mkdirSync, writeFileSync, rmSync } from "node:fs";
14
+ import { tmpdir } from "node:os";
15
+ import { join } from "node:path";
16
+
17
+ import { shouldBlockQueueExecution } from "../bootstrap/write-gate.ts";
18
+ import {
19
+ resetEvidence,
20
+ recordToolCall,
21
+ getEvidence,
22
+ saveEvidenceToDisk,
23
+ loadEvidenceFromDisk,
24
+ } from "../safety/evidence-collector.ts";
25
+ import { validateFileChanges } from "../safety/file-change-validator.ts";
26
+
27
+ // ─── Bug 1: Hardcoded Bash allowlist ────────────────────────────────────────
28
+
29
+ test("safety-harness-bug1: npm commands are not blocked during queue mode", () => {
30
+ const r = shouldBlockQueueExecution("bash", "npm run test", true);
31
+ assert.strictEqual(r.block, false, "npm run test must be read-only-safe");
32
+ });
33
+
34
+ test("safety-harness-bug1: npx commands are not blocked during queue mode", () => {
35
+ const r = shouldBlockQueueExecution("bash", "npx tsc --noEmit", true);
36
+ assert.strictEqual(r.block, false, "npx tsc --noEmit must pass");
37
+ });
38
+
39
+ test("safety-harness-bug1: tsx commands are not blocked during queue mode", () => {
40
+ const r = shouldBlockQueueExecution("bash", "tsx src/index.ts", true);
41
+ assert.strictEqual(
42
+ r.block,
43
+ false,
44
+ "tsx (TypeScript runner — read-only investigative) must pass",
45
+ );
46
+ });
47
+
48
+ test("safety-harness-bug1: node --print commands are not blocked during queue mode", () => {
49
+ const r = shouldBlockQueueExecution("bash", "node --print 'process.version'", true);
50
+ assert.strictEqual(r.block, false, "node --print must pass");
51
+ });
52
+
53
+ test("safety-harness-bug1: python read-only invocations are not blocked during queue mode", () => {
54
+ const r = shouldBlockQueueExecution("bash", "python -c 'import sys; print(sys.version)'", true);
55
+ assert.strictEqual(r.block, false, "python -c read-only must pass");
56
+ });
57
+
58
+ test("safety-harness-bug1: jq read-only command is not blocked during queue mode", () => {
59
+ const r = shouldBlockQueueExecution("bash", "jq '.version' package.json", true);
60
+ assert.strictEqual(r.block, false, "jq (read-only JSON query) must pass");
61
+ });
62
+
63
+ test("safety-harness-bug1: destructive commands are still blocked during queue mode", () => {
64
+ const r = shouldBlockQueueExecution("bash", "rm -rf dist/", true);
65
+ assert.strictEqual(r.block, true, "rm -rf must still be blocked");
66
+ });
67
+
68
+ // ─── Bug 2: Non-persisted evidence ──────────────────────────────────────────
69
+
70
+ test("safety-harness-bug2: evidence survives save/load round-trip (simulates session restart)", (t) => {
71
+ const base = mkdtempSync(join(tmpdir(), "gsd-evidence-persist-"));
72
+ t.after(() => rmSync(base, { recursive: true, force: true }));
73
+
74
+ resetEvidence();
75
+
76
+ // Simulate bash tool calls during unit execution
77
+ recordToolCall("tc-001", "Bash", { command: "npm run test:unit" });
78
+ recordToolCall("tc-002", "Bash", { command: "npx tsc --noEmit" });
79
+ recordToolCall("tc-003", "Write", { file_path: "src/foo.ts" });
80
+
81
+ const before = getEvidence();
82
+ assert.equal(before.length, 3, "three entries before save");
83
+
84
+ // Persist to disk
85
+ saveEvidenceToDisk(base, "M001", "S001", "T001");
86
+
87
+ // Simulate session restart: module-level array reset
88
+ resetEvidence();
89
+ assert.equal(getEvidence().length, 0, "in-memory cleared after reset");
90
+
91
+ // Resume: load from disk
92
+ loadEvidenceFromDisk(base, "M001", "S001", "T001");
93
+
94
+ const after = getEvidence();
95
+ assert.equal(after.length, 3, "evidence restored from disk after simulated restart");
96
+
97
+ const bashEntries = after.filter((e) => e.kind === "bash");
98
+ assert.equal(bashEntries.length, 2, "both bash entries restored");
99
+
100
+ const writeEntries = after.filter((e) => e.kind === "write");
101
+ assert.equal(writeEntries.length, 1, "write entry restored");
102
+ });
103
+
104
+ test("safety-harness-bug2: loadEvidenceFromDisk returns empty array when no file exists (fresh unit)", (t) => {
105
+ const base = mkdtempSync(join(tmpdir(), "gsd-evidence-nopersist-"));
106
+ t.after(() => rmSync(base, { recursive: true, force: true }));
107
+
108
+ resetEvidence();
109
+ loadEvidenceFromDisk(base, "M001", "S001", "T001");
110
+ assert.equal(getEvidence().length, 0, "no evidence on fresh unit is correct — not a false positive");
111
+ });
112
+
113
+ // ─── Bug 3: git diff HEAD~1 scope check ─────────────────────────────────────
114
+
115
+ test("safety-harness-bug3: validateFileChanges works on initial commit (no HEAD~1)", (t) => {
116
+ const base = mkdtempSync(join(tmpdir(), "gsd-initial-commit-"));
117
+ t.after(() => rmSync(base, { recursive: true, force: true }));
118
+
119
+ execFileSync("git", ["init"], { cwd: base });
120
+ execFileSync("git", ["config", "user.email", "test@example.com"], { cwd: base });
121
+ execFileSync("git", ["config", "user.name", "Test User"], { cwd: base });
122
+
123
+ writeFileSync(join(base, "index.ts"), "export const x = 1;\n");
124
+ execFileSync("git", ["add", "."], { cwd: base });
125
+ execFileSync("git", ["commit", "-m", "initial"], { cwd: base });
126
+
127
+ // On initial commit, HEAD~1 does not exist — must not throw or produce wrong results
128
+ const audit = validateFileChanges(base, ["index.ts"], []);
129
+
130
+ assert.ok(audit !== null, "audit must be produced for initial commit");
131
+ assert.deepEqual(audit!.unexpectedFiles, [], "no unexpected files on initial commit");
132
+ assert.deepEqual(audit!.missingFiles, [], "no missing files on initial commit");
133
+ });
134
+
135
+ test("safety-harness-bug3: validateFileChanges works on shallow clone (shallow repo without full history)", (t) => {
136
+ // Simulate shallow clone: create a repo, then clone it with depth=1
137
+ const origin = mkdtempSync(join(tmpdir(), "gsd-origin-"));
138
+ const shallow = mkdtempSync(join(tmpdir(), "gsd-shallow-"));
139
+ t.after(() => {
140
+ rmSync(origin, { recursive: true, force: true });
141
+ rmSync(shallow, { recursive: true, force: true });
142
+ });
143
+
144
+ // Set up origin with multiple commits
145
+ execFileSync("git", ["init"], { cwd: origin });
146
+ execFileSync("git", ["config", "user.email", "test@example.com"], { cwd: origin });
147
+ execFileSync("git", ["config", "user.name", "Test User"], { cwd: origin });
148
+ writeFileSync(join(origin, "a.ts"), "export const a = 1;\n");
149
+ execFileSync("git", ["add", "."], { cwd: origin });
150
+ execFileSync("git", ["commit", "-m", "first"], { cwd: origin });
151
+ writeFileSync(join(origin, "b.ts"), "export const b = 2;\n");
152
+ execFileSync("git", ["add", "."], { cwd: origin });
153
+ execFileSync("git", ["commit", "-m", "second"], { cwd: origin });
154
+
155
+ // Shallow clone with depth=1 — HEAD~1 will not exist
156
+ execFileSync("git", ["clone", "--depth=1", `file://${origin}`, shallow], {
157
+ stdio: ["ignore", "pipe", "pipe"],
158
+ });
159
+
160
+ // Verify the shallow clone has no parent (HEAD~1 unavailable)
161
+ let hasParent = true;
162
+ try {
163
+ execFileSync("git", ["rev-parse", "HEAD~1"], {
164
+ cwd: shallow,
165
+ stdio: ["ignore", "pipe", "pipe"],
166
+ });
167
+ } catch {
168
+ hasParent = false;
169
+ }
170
+ assert.equal(hasParent, false, "shallow clone should not have HEAD~1");
171
+
172
+ // validateFileChanges must not throw or give wrong results
173
+ const audit = validateFileChanges(shallow, ["b.ts"], []);
174
+ assert.ok(audit !== null, "audit must be produced even in shallow clone");
175
+ });
176
+
177
+ test("safety-harness-bug3: validateFileChanges works on merge commit", (t) => {
178
+ const base = mkdtempSync(join(tmpdir(), "gsd-merge-commit-"));
179
+ t.after(() => rmSync(base, { recursive: true, force: true }));
180
+
181
+ execFileSync("git", ["init", "-b", "main"], { cwd: base });
182
+ execFileSync("git", ["config", "user.email", "test@example.com"], { cwd: base });
183
+ execFileSync("git", ["config", "user.name", "Test User"], { cwd: base });
184
+
185
+ // Main branch: initial commit
186
+ writeFileSync(join(base, "main.ts"), "export const m = 1;\n");
187
+ execFileSync("git", ["add", "."], { cwd: base });
188
+ execFileSync("git", ["commit", "-m", "initial"], { cwd: base });
189
+
190
+ // Feature branch
191
+ execFileSync("git", ["checkout", "-b", "feature"], { cwd: base });
192
+ writeFileSync(join(base, "feature.ts"), "export const f = 2;\n");
193
+ execFileSync("git", ["add", "."], { cwd: base });
194
+ execFileSync("git", ["commit", "-m", "feature work"], { cwd: base });
195
+
196
+ // Merge back to main
197
+ execFileSync("git", ["checkout", "main"], { cwd: base });
198
+ execFileSync("git", ["merge", "--no-ff", "feature", "-m", "Merge feature"], { cwd: base });
199
+
200
+ // HEAD is now a merge commit with two parents — git diff HEAD~1 gives wrong scope
201
+ const audit = validateFileChanges(base, ["feature.ts"], []);
202
+
203
+ // Must produce a valid result without throwing
204
+ assert.ok(audit !== null, "audit must be produced for merge commit repo");
205
+ });