gsd-pi 2.82.0 → 3.0.0-dev.2e8b124f7

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 (760) hide show
  1. package/README.md +125 -32
  2. package/dist/cli.js +20 -9
  3. package/dist/headless-ui.js +13 -6
  4. package/dist/headless.js +9 -2
  5. package/dist/resources/.managed-resources-content-hash +1 -1
  6. package/dist/resources/GSD-WORKFLOW.md +10 -1
  7. package/dist/resources/extensions/browser-tools/tools/screenshot.js +1 -0
  8. package/dist/resources/extensions/browser-tools/tools/zoom.js +1 -0
  9. package/dist/resources/extensions/claude-code-cli/partial-builder.js +2 -1
  10. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +44 -6
  11. package/dist/resources/extensions/cmux/index.js +5 -0
  12. package/dist/resources/extensions/gsd/auto/infra-errors.js +9 -3
  13. package/dist/resources/extensions/gsd/auto/loop.js +110 -37
  14. package/dist/resources/extensions/gsd/auto/orchestrator.js +124 -6
  15. package/dist/resources/extensions/gsd/auto/phases.js +113 -38
  16. package/dist/resources/extensions/gsd/auto/session.js +6 -0
  17. package/dist/resources/extensions/gsd/auto/unit-runner-events.js +7 -1
  18. package/dist/resources/extensions/gsd/auto/workflow-kernel.js +3 -0
  19. package/dist/resources/extensions/gsd/auto/workflow-memory-pressure.js +12 -0
  20. package/dist/resources/extensions/gsd/auto-dashboard.js +66 -1
  21. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +1 -0
  22. package/dist/resources/extensions/gsd/auto-dispatch.js +71 -20
  23. package/dist/resources/extensions/gsd/auto-model-selection.js +2 -0
  24. package/dist/resources/extensions/gsd/auto-post-unit.js +278 -137
  25. package/dist/resources/extensions/gsd/auto-prompts.js +47 -13
  26. package/dist/resources/extensions/gsd/auto-recovery.js +79 -14
  27. package/dist/resources/extensions/gsd/auto-start.js +90 -14
  28. package/dist/resources/extensions/gsd/auto-timers.js +11 -3
  29. package/dist/resources/extensions/gsd/auto-verification.js +102 -34
  30. package/dist/resources/extensions/gsd/auto-worktree.js +178 -11
  31. package/dist/resources/extensions/gsd/auto.js +225 -81
  32. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +31 -7
  33. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +10 -9
  34. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +10 -4
  35. package/dist/resources/extensions/gsd/bootstrap/subagent-input.js +21 -9
  36. package/dist/resources/extensions/gsd/bootstrap/system-context.js +55 -12
  37. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +24 -9
  38. package/dist/resources/extensions/gsd/clean-root-preflight.js +170 -8
  39. package/dist/resources/extensions/gsd/commands/catalog.js +10 -1
  40. package/dist/resources/extensions/gsd/commands/handlers/core.js +39 -1
  41. package/dist/resources/extensions/gsd/commands/handlers/ops.js +20 -0
  42. package/dist/resources/extensions/gsd/commands-bootstrap.js +5 -0
  43. package/dist/resources/extensions/gsd/commands-handlers.js +17 -2
  44. package/dist/resources/extensions/gsd/commands-mcp-status.js +9 -0
  45. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +10 -2
  46. package/dist/resources/extensions/gsd/commands-verdict.js +139 -0
  47. package/dist/resources/extensions/gsd/context-store.js +112 -0
  48. package/dist/resources/extensions/gsd/crash-recovery.js +52 -7
  49. package/dist/resources/extensions/gsd/db/milestone-leases.js +24 -0
  50. package/dist/resources/extensions/gsd/db/unit-dispatches.js +3 -2
  51. package/dist/resources/extensions/gsd/db-base-schema.js +2 -0
  52. package/dist/resources/extensions/gsd/db-migration-steps.js +4 -0
  53. package/dist/resources/extensions/gsd/db-task-slice-rows.js +2 -0
  54. package/dist/resources/extensions/gsd/db-writer.js +150 -84
  55. package/dist/resources/extensions/gsd/dispatch-guard.js +46 -2
  56. package/dist/resources/extensions/gsd/docs/preferences-reference.md +9 -1
  57. package/dist/resources/extensions/gsd/doctor-git-checks.js +87 -7
  58. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +28 -11
  59. package/dist/resources/extensions/gsd/doctor.js +2 -28
  60. package/dist/resources/extensions/gsd/export-html.js +27 -425
  61. package/dist/resources/extensions/gsd/forensics.js +10 -3
  62. package/dist/resources/extensions/gsd/git-service.js +138 -10
  63. package/dist/resources/extensions/gsd/gsd-db.js +76 -33
  64. package/dist/resources/extensions/gsd/guided-flow-queue.js +4 -3
  65. package/dist/resources/extensions/gsd/guided-flow.js +110 -117
  66. package/dist/resources/extensions/gsd/guided-unit-context.js +23 -0
  67. package/dist/resources/extensions/gsd/init-wizard.js +17 -2
  68. package/dist/resources/extensions/gsd/knowledge-backfill.js +144 -0
  69. package/dist/resources/extensions/gsd/knowledge-capture.js +136 -0
  70. package/dist/resources/extensions/gsd/knowledge-parser.js +154 -0
  71. package/dist/resources/extensions/gsd/knowledge-projection.js +210 -0
  72. package/dist/resources/extensions/gsd/markdown-renderer.js +20 -12
  73. package/dist/resources/extensions/gsd/mcp-filter.js +58 -0
  74. package/dist/resources/extensions/gsd/md-importer.js +1 -1
  75. package/dist/resources/extensions/gsd/memory-backfill.js +73 -17
  76. package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +222 -0
  77. package/dist/resources/extensions/gsd/migrate/command.js +5 -0
  78. package/dist/resources/extensions/gsd/migrate/parsers.js +10 -0
  79. package/dist/resources/extensions/gsd/migrate/preview.js +9 -0
  80. package/dist/resources/extensions/gsd/migrate/transformer.js +51 -4
  81. package/dist/resources/extensions/gsd/migrate/writer.js +11 -1
  82. package/dist/resources/extensions/gsd/migration-auto-check.js +12 -17
  83. package/dist/resources/extensions/gsd/milestone-actions.js +11 -4
  84. package/dist/resources/extensions/gsd/native-git-bridge.js +57 -14
  85. package/dist/resources/extensions/gsd/parallel-orchestrator.js +3 -0
  86. package/dist/resources/extensions/gsd/paths.js +4 -0
  87. package/dist/resources/extensions/gsd/pending-auto-start.js +52 -0
  88. package/dist/resources/extensions/gsd/planning-path-scope.js +9 -3
  89. package/dist/resources/extensions/gsd/post-execution-checks.js +73 -9
  90. package/dist/resources/extensions/gsd/pre-execution-checks.js +38 -11
  91. package/dist/resources/extensions/gsd/preferences-mcp.js +19 -0
  92. package/dist/resources/extensions/gsd/preferences-types.js +2 -0
  93. package/dist/resources/extensions/gsd/preferences-validation.js +138 -0
  94. package/dist/resources/extensions/gsd/preferences.js +2 -0
  95. package/dist/resources/extensions/gsd/prompt-loader.js +1 -1
  96. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  97. package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  98. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +8 -8
  99. package/dist/resources/extensions/gsd/prompts/discuss.md +9 -9
  100. package/dist/resources/extensions/gsd/prompts/forensics.md +3 -3
  101. package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +4 -4
  102. package/dist/resources/extensions/gsd/prompts/guided-discuss-requirements.md +3 -3
  103. package/dist/resources/extensions/gsd/prompts/plan-slice.md +4 -4
  104. package/dist/resources/extensions/gsd/prompts/queue.md +4 -4
  105. package/dist/resources/extensions/gsd/prompts/reactive-execute.md +1 -1
  106. package/dist/resources/extensions/gsd/prompts/refine-slice.md +2 -2
  107. package/dist/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
  108. package/dist/resources/extensions/gsd/prompts/system.md +2 -2
  109. package/dist/resources/extensions/gsd/provider-switch-observer.js +146 -0
  110. package/dist/resources/extensions/gsd/queue-reorder-ui.js +30 -13
  111. package/dist/resources/extensions/gsd/repository-registry.js +44 -0
  112. package/dist/resources/extensions/gsd/safety/evidence-collector.js +2 -0
  113. package/dist/resources/extensions/gsd/safety/evidence-cross-ref.js +42 -18
  114. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +59 -2
  115. package/dist/resources/extensions/gsd/smart-entry-routing.js +36 -0
  116. package/dist/resources/extensions/gsd/state-reconciliation/drift/merge-state.js +6 -1
  117. package/dist/resources/extensions/gsd/state-reconciliation/drift/project-md.js +9 -14
  118. package/dist/resources/extensions/gsd/state-reconciliation/drift/roadmap.js +19 -24
  119. package/dist/resources/extensions/gsd/state.js +14 -4
  120. package/dist/resources/extensions/gsd/status-guards.js +14 -2
  121. package/dist/resources/extensions/gsd/templates/knowledge.md +2 -2
  122. package/dist/resources/extensions/gsd/templates/plan.md +9 -5
  123. package/dist/resources/extensions/gsd/templates/task-plan.md +10 -2
  124. package/dist/resources/extensions/gsd/tools/complete-milestone.js +6 -8
  125. package/dist/resources/extensions/gsd/tools/complete-slice.js +6 -8
  126. package/dist/resources/extensions/gsd/tools/plan-milestone.js +7 -1
  127. package/dist/resources/extensions/gsd/tools/plan-slice.js +151 -15
  128. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +119 -0
  129. package/dist/resources/extensions/gsd/unit-context-composer.js +2 -0
  130. package/dist/resources/extensions/gsd/unit-context-manifest.js +69 -17
  131. package/dist/resources/extensions/gsd/validation.js +23 -1
  132. package/dist/resources/extensions/gsd/verification-gate.js +142 -7
  133. package/dist/resources/extensions/gsd/verification-verdict.js +26 -0
  134. package/dist/resources/extensions/gsd/workflow-manifest.js +2 -0
  135. package/dist/resources/extensions/gsd/workflow-mcp.js +17 -1
  136. package/dist/resources/extensions/gsd/workflow-projections.js +6 -8
  137. package/dist/resources/extensions/gsd/worktree-lifecycle.js +83 -19
  138. package/dist/resources/extensions/gsd/worktree-manager.js +11 -2
  139. package/dist/resources/extensions/gsd/worktree-safety.js +33 -1
  140. package/dist/resources/extensions/gsd/worktree-state-projection.js +31 -0
  141. package/dist/resources/extensions/shared/html-shell.js +388 -0
  142. package/dist/resources/extensions/shared/interview-ui.js +6 -4
  143. package/dist/resources/extensions/subagent/index.js +448 -78
  144. package/dist/resources/extensions/subagent/launch.js +77 -0
  145. package/dist/resources/extensions/subagent/run-store.js +148 -0
  146. package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
  147. package/dist/resources/extensions/visual-brief/artifact-policy.js +29 -0
  148. package/dist/resources/extensions/visual-brief/extension-manifest.json +8 -0
  149. package/dist/resources/extensions/visual-brief/index.js +5 -0
  150. package/dist/resources/extensions/visual-brief/page-contract.js +124 -0
  151. package/dist/resources/extensions/visual-brief/prompts.js +140 -0
  152. package/dist/resources/skills/forensics/SKILL.md +1 -1
  153. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  154. package/dist/web/standalone/.next/BUILD_ID +1 -1
  155. package/dist/web/standalone/.next/app-path-routes-manifest.json +9 -9
  156. package/dist/web/standalone/.next/build-manifest.json +4 -4
  157. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  158. package/dist/web/standalone/.next/react-loadable-manifest.json +5 -5
  159. package/dist/web/standalone/.next/required-server-files.json +4 -4
  160. package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
  161. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  162. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  163. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  164. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  165. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  166. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  167. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  168. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  169. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  170. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  171. package/dist/web/standalone/.next/server/app/_not-found/page.js.nft.json +1 -1
  172. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  173. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  174. package/dist/web/standalone/.next/server/app/_not-found.rsc +4 -7
  175. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +4 -7
  176. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  177. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +4 -5
  178. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  179. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  180. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -5
  181. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  182. package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
  183. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  184. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
  185. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  186. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
  187. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  188. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
  189. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  190. package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
  191. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  192. package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
  193. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  194. package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
  195. package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
  196. package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
  197. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  198. package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
  199. package/dist/web/standalone/.next/server/app/api/experimental/route.js +2 -2
  200. package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -1
  201. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  202. package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
  203. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  204. package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
  205. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  206. package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
  207. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  208. package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
  209. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  210. package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
  211. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  212. package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
  213. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  214. package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
  215. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  216. package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
  217. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  218. package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
  219. package/dist/web/standalone/.next/server/app/api/notifications/route.js +2 -2
  220. package/dist/web/standalone/.next/server/app/api/notifications/route_client-reference-manifest.js +1 -1
  221. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  222. package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
  223. package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
  224. package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
  225. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  226. package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  227. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  228. package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
  229. package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +2 -2
  230. package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
  231. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  232. package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
  233. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  234. package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
  235. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  236. package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
  237. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  238. package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
  239. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  240. package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
  241. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  242. package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
  243. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  244. package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
  245. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  246. package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
  247. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  248. package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
  249. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +1 -1
  250. package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
  251. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
  252. package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
  253. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
  254. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
  255. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +2 -2
  256. package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
  257. package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js +1 -1
  258. package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
  259. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  260. package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
  261. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  262. package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  263. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  264. package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
  265. package/dist/web/standalone/.next/server/app/index.html +1 -1
  266. package/dist/web/standalone/.next/server/app/index.rsc +5 -8
  267. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  268. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +5 -8
  269. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  270. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +4 -5
  271. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -5
  272. package/dist/web/standalone/.next/server/app/page.js +2 -2
  273. package/dist/web/standalone/.next/server/app/page.js.nft.json +1 -1
  274. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  275. package/dist/web/standalone/.next/server/app-paths-manifest.json +9 -9
  276. package/dist/web/standalone/.next/server/chunks/4266.js +2 -0
  277. package/dist/web/standalone/.next/server/chunks/63.js +3 -3
  278. package/dist/web/standalone/.next/server/chunks/6897.js +1 -1
  279. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  280. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  281. package/dist/web/standalone/.next/server/middleware.js +2 -2
  282. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  283. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  284. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  285. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  286. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  287. package/dist/web/standalone/.next/static/chunks/2973.33f26573894b6153.js +2 -0
  288. package/dist/web/standalone/.next/static/chunks/8359.65b24fac92188a6b.js +10 -0
  289. package/dist/web/standalone/.next/static/chunks/9441.ff70bb53f6835771.js +1 -0
  290. package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-2f24283c162b6ab3.js → page-f2a7482d42a5614b.js} +1 -1
  291. package/dist/web/standalone/.next/static/chunks/app/layout-8c10ec293ae0f1d5.js +1 -0
  292. package/dist/web/standalone/.next/static/chunks/app/page-752f1e2ebdaa3e45.js +1 -0
  293. package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +1 -0
  294. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +1 -0
  295. package/dist/web/standalone/.next/static/chunks/{webpack-de742b64187e13fe.js → webpack-855d616060cb6e59.js} +1 -1
  296. package/dist/web/standalone/.next/static/css/746ee28c929d1880.css +1 -0
  297. package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
  298. package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
  299. package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
  300. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
  301. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
  302. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
  303. package/dist/web/standalone/server.js +1 -1
  304. package/package.json +6 -5
  305. package/packages/contracts/dist/rpc.test.js +7 -0
  306. package/packages/contracts/dist/rpc.test.js.map +1 -1
  307. package/packages/contracts/dist/workflow.d.ts +21 -0
  308. package/packages/contracts/dist/workflow.d.ts.map +1 -1
  309. package/packages/contracts/dist/workflow.js +24 -0
  310. package/packages/contracts/dist/workflow.js.map +1 -1
  311. package/packages/contracts/src/rpc.test.ts +8 -0
  312. package/packages/contracts/src/workflow.ts +24 -0
  313. package/packages/daemon/package.json +2 -2
  314. package/packages/mcp-server/README.md +14 -3
  315. package/packages/mcp-server/dist/workflow-tools.d.ts +0 -3
  316. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  317. package/packages/mcp-server/dist/workflow-tools.js +80 -0
  318. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  319. package/packages/mcp-server/package.json +2 -2
  320. package/packages/mcp-server/src/workflow-tools-parity.test.ts +244 -0
  321. package/packages/mcp-server/src/workflow-tools.test.ts +23 -1
  322. package/packages/mcp-server/src/workflow-tools.ts +168 -0
  323. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  324. package/packages/native/package.json +1 -1
  325. package/packages/native/tsconfig.json +2 -1
  326. package/packages/native/tsconfig.tsbuildinfo +1 -1
  327. package/packages/pi-agent-core/package.json +1 -1
  328. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  329. package/packages/pi-ai/dist/index.d.ts +2 -2
  330. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  331. package/packages/pi-ai/dist/index.js +1 -1
  332. package/packages/pi-ai/dist/index.js.map +1 -1
  333. package/packages/pi-ai/dist/providers/google-gemini-cli.d.ts.map +1 -1
  334. package/packages/pi-ai/dist/providers/google-gemini-cli.js +5 -0
  335. package/packages/pi-ai/dist/providers/google-gemini-cli.js.map +1 -1
  336. package/packages/pi-ai/dist/providers/google-gemini-cli.test.d.ts +2 -0
  337. package/packages/pi-ai/dist/providers/google-gemini-cli.test.d.ts.map +1 -0
  338. package/packages/pi-ai/dist/providers/google-gemini-cli.test.js +41 -0
  339. package/packages/pi-ai/dist/providers/google-gemini-cli.test.js.map +1 -0
  340. package/packages/pi-ai/dist/providers/openai-codex-responses.d.ts.map +1 -1
  341. package/packages/pi-ai/dist/providers/openai-codex-responses.js +82 -1
  342. package/packages/pi-ai/dist/providers/openai-codex-responses.js.map +1 -1
  343. package/packages/pi-ai/dist/providers/openai-codex-responses.test.d.ts +2 -0
  344. package/packages/pi-ai/dist/providers/openai-codex-responses.test.d.ts.map +1 -0
  345. package/packages/pi-ai/dist/providers/openai-codex-responses.test.js +52 -0
  346. package/packages/pi-ai/dist/providers/openai-codex-responses.test.js.map +1 -0
  347. package/packages/pi-ai/dist/providers/simple-options.d.ts +2 -4
  348. package/packages/pi-ai/dist/providers/simple-options.d.ts.map +1 -1
  349. package/packages/pi-ai/dist/providers/simple-options.js +5 -6
  350. package/packages/pi-ai/dist/providers/simple-options.js.map +1 -1
  351. package/packages/pi-ai/dist/providers/simple-options.test.d.ts +2 -0
  352. package/packages/pi-ai/dist/providers/simple-options.test.d.ts.map +1 -0
  353. package/packages/pi-ai/dist/providers/simple-options.test.js +50 -0
  354. package/packages/pi-ai/dist/providers/simple-options.test.js.map +1 -0
  355. package/packages/pi-ai/dist/providers/transform-messages.d.ts +11 -0
  356. package/packages/pi-ai/dist/providers/transform-messages.d.ts.map +1 -1
  357. package/packages/pi-ai/dist/providers/transform-messages.js +20 -0
  358. package/packages/pi-ai/dist/providers/transform-messages.js.map +1 -1
  359. package/packages/pi-ai/package.json +1 -1
  360. package/packages/pi-ai/src/index.ts +7 -2
  361. package/packages/pi-ai/src/providers/google-gemini-cli.test.ts +49 -0
  362. package/packages/pi-ai/src/providers/google-gemini-cli.ts +7 -0
  363. package/packages/pi-ai/src/providers/openai-codex-responses.test.ts +63 -0
  364. package/packages/pi-ai/src/providers/openai-codex-responses.ts +91 -1
  365. package/packages/pi-ai/src/providers/simple-options.test.ts +60 -0
  366. package/packages/pi-ai/src/providers/simple-options.ts +5 -6
  367. package/packages/pi-ai/src/providers/transform-messages.ts +24 -0
  368. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  369. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts +2 -0
  370. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts.map +1 -0
  371. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js +66 -0
  372. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js.map +1 -0
  373. package/packages/pi-coding-agent/dist/core/agent-session.js +1 -1
  374. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  375. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +44 -3
  376. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  377. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts +6 -1
  378. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts.map +1 -1
  379. package/packages/pi-coding-agent/dist/core/compaction/compaction.js +7 -2
  380. package/packages/pi-coding-agent/dist/core/compaction/compaction.js.map +1 -1
  381. package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js +14 -1
  382. package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js.map +1 -1
  383. package/packages/pi-coding-agent/dist/core/sdk.js +1 -1
  384. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  385. package/packages/pi-coding-agent/dist/core/system-prompt.js +4 -4
  386. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  387. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +8 -2
  388. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  389. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
  390. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +24 -6
  391. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
  392. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +1 -1
  393. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  394. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  395. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +71 -97
  396. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  397. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +7 -7
  398. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  399. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +11 -0
  400. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  401. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +25 -1
  402. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -1
  403. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +2 -0
  404. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  405. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +24 -10
  406. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  407. package/packages/pi-coding-agent/dist/tests/system-prompt-file-safety.test.d.ts +2 -0
  408. package/packages/pi-coding-agent/dist/tests/system-prompt-file-safety.test.d.ts.map +1 -0
  409. package/packages/pi-coding-agent/dist/tests/system-prompt-file-safety.test.js +17 -0
  410. package/packages/pi-coding-agent/dist/tests/system-prompt-file-safety.test.js.map +1 -0
  411. package/packages/pi-coding-agent/package.json +1 -1
  412. package/packages/pi-coding-agent/src/core/agent-session-thinking-level.test.ts +79 -0
  413. package/packages/pi-coding-agent/src/core/agent-session.ts +1 -1
  414. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +53 -3
  415. package/packages/pi-coding-agent/src/core/compaction/compaction.test.ts +23 -1
  416. package/packages/pi-coding-agent/src/core/compaction/compaction.ts +7 -2
  417. package/packages/pi-coding-agent/src/core/sdk.ts +1 -1
  418. package/packages/pi-coding-agent/src/core/system-prompt.ts +4 -4
  419. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +17 -1
  420. package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +23 -7
  421. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +1 -1
  422. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +75 -102
  423. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +15 -1
  424. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +9 -9
  425. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +30 -1
  426. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +29 -10
  427. package/packages/pi-coding-agent/src/tests/system-prompt-file-safety.test.ts +22 -0
  428. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  429. package/packages/pi-tui/dist/__tests__/terminal.test.d.ts +2 -0
  430. package/packages/pi-tui/dist/__tests__/terminal.test.d.ts.map +1 -0
  431. package/packages/pi-tui/dist/__tests__/terminal.test.js +103 -0
  432. package/packages/pi-tui/dist/__tests__/terminal.test.js.map +1 -0
  433. package/packages/pi-tui/dist/__tests__/tui.test.js +45 -2
  434. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  435. package/packages/pi-tui/dist/terminal.d.ts +2 -0
  436. package/packages/pi-tui/dist/terminal.d.ts.map +1 -1
  437. package/packages/pi-tui/dist/terminal.js +12 -0
  438. package/packages/pi-tui/dist/terminal.js.map +1 -1
  439. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  440. package/packages/pi-tui/dist/tui.js +108 -24
  441. package/packages/pi-tui/dist/tui.js.map +1 -1
  442. package/packages/pi-tui/package.json +1 -1
  443. package/packages/pi-tui/src/__tests__/terminal.test.ts +121 -0
  444. package/packages/pi-tui/src/__tests__/tui.test.ts +59 -2
  445. package/packages/pi-tui/src/terminal.ts +11 -0
  446. package/packages/pi-tui/src/tui.ts +111 -24
  447. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  448. package/packages/rpc-client/package.json +1 -1
  449. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  450. package/pkg/package.json +1 -1
  451. package/src/resources/GSD-WORKFLOW.md +10 -1
  452. package/src/resources/extensions/browser-tools/tools/screenshot.ts +1 -0
  453. package/src/resources/extensions/browser-tools/tools/zoom.ts +1 -0
  454. package/src/resources/extensions/claude-code-cli/partial-builder.ts +2 -1
  455. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +52 -6
  456. package/src/resources/extensions/claude-code-cli/tests/partial-builder.test.ts +19 -2
  457. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +49 -2
  458. package/src/resources/extensions/cmux/index.ts +6 -0
  459. package/src/resources/extensions/gsd/auto/contracts.ts +62 -16
  460. package/src/resources/extensions/gsd/auto/infra-errors.ts +9 -3
  461. package/src/resources/extensions/gsd/auto/loop.ts +111 -38
  462. package/src/resources/extensions/gsd/auto/orchestrator.ts +129 -6
  463. package/src/resources/extensions/gsd/auto/phases.ts +141 -49
  464. package/src/resources/extensions/gsd/auto/session.ts +16 -0
  465. package/src/resources/extensions/gsd/auto/types.ts +3 -0
  466. package/src/resources/extensions/gsd/auto/unit-runner-events.ts +6 -2
  467. package/src/resources/extensions/gsd/auto/workflow-kernel.ts +5 -1
  468. package/src/resources/extensions/gsd/auto/workflow-memory-pressure.ts +13 -0
  469. package/src/resources/extensions/gsd/auto-dashboard.ts +72 -1
  470. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +1 -0
  471. package/src/resources/extensions/gsd/auto-dispatch.ts +74 -20
  472. package/src/resources/extensions/gsd/auto-model-selection.ts +2 -1
  473. package/src/resources/extensions/gsd/auto-post-unit.ts +312 -148
  474. package/src/resources/extensions/gsd/auto-prompts.ts +47 -16
  475. package/src/resources/extensions/gsd/auto-recovery.ts +83 -11
  476. package/src/resources/extensions/gsd/auto-start.ts +99 -12
  477. package/src/resources/extensions/gsd/auto-timers.ts +10 -3
  478. package/src/resources/extensions/gsd/auto-verification.ts +124 -42
  479. package/src/resources/extensions/gsd/auto-worktree.ts +195 -11
  480. package/src/resources/extensions/gsd/auto.ts +226 -70
  481. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +42 -7
  482. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +10 -9
  483. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +11 -4
  484. package/src/resources/extensions/gsd/bootstrap/subagent-input.ts +19 -7
  485. package/src/resources/extensions/gsd/bootstrap/system-context.ts +58 -15
  486. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +27 -10
  487. package/src/resources/extensions/gsd/clean-root-preflight.ts +174 -8
  488. package/src/resources/extensions/gsd/commands/catalog.ts +10 -1
  489. package/src/resources/extensions/gsd/commands/handlers/core.ts +42 -1
  490. package/src/resources/extensions/gsd/commands/handlers/ops.ts +21 -0
  491. package/src/resources/extensions/gsd/commands-bootstrap.ts +10 -0
  492. package/src/resources/extensions/gsd/commands-handlers.ts +21 -2
  493. package/src/resources/extensions/gsd/commands-mcp-status.ts +8 -0
  494. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +11 -3
  495. package/src/resources/extensions/gsd/commands-verdict.ts +202 -0
  496. package/src/resources/extensions/gsd/context-store.ts +120 -1
  497. package/src/resources/extensions/gsd/crash-recovery.ts +52 -6
  498. package/src/resources/extensions/gsd/db/milestone-leases.ts +26 -0
  499. package/src/resources/extensions/gsd/db/unit-dispatches.ts +4 -3
  500. package/src/resources/extensions/gsd/db-base-schema.ts +2 -0
  501. package/src/resources/extensions/gsd/db-migration-steps.ts +5 -0
  502. package/src/resources/extensions/gsd/db-task-slice-rows.ts +4 -0
  503. package/src/resources/extensions/gsd/db-writer.ts +167 -84
  504. package/src/resources/extensions/gsd/dispatch-guard.ts +60 -2
  505. package/src/resources/extensions/gsd/docs/preferences-reference.md +9 -1
  506. package/src/resources/extensions/gsd/doctor-git-checks.ts +89 -7
  507. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +25 -13
  508. package/src/resources/extensions/gsd/doctor-types.ts +3 -0
  509. package/src/resources/extensions/gsd/doctor.ts +2 -27
  510. package/src/resources/extensions/gsd/export-html.ts +27 -427
  511. package/src/resources/extensions/gsd/forensics.ts +9 -3
  512. package/src/resources/extensions/gsd/git-service.ts +166 -11
  513. package/src/resources/extensions/gsd/gsd-db.ts +80 -31
  514. package/src/resources/extensions/gsd/guided-flow-queue.ts +4 -3
  515. package/src/resources/extensions/gsd/guided-flow.ts +142 -134
  516. package/src/resources/extensions/gsd/guided-unit-context.ts +30 -0
  517. package/src/resources/extensions/gsd/init-wizard.ts +17 -2
  518. package/src/resources/extensions/gsd/journal.ts +8 -1
  519. package/src/resources/extensions/gsd/knowledge-backfill.ts +164 -0
  520. package/src/resources/extensions/gsd/knowledge-capture.ts +160 -0
  521. package/src/resources/extensions/gsd/knowledge-parser.ts +174 -0
  522. package/src/resources/extensions/gsd/knowledge-projection.ts +241 -0
  523. package/src/resources/extensions/gsd/markdown-renderer.ts +20 -12
  524. package/src/resources/extensions/gsd/mcp-filter.ts +80 -0
  525. package/src/resources/extensions/gsd/md-importer.ts +1 -1
  526. package/src/resources/extensions/gsd/memory-backfill.ts +89 -17
  527. package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +277 -0
  528. package/src/resources/extensions/gsd/migrate/command.ts +5 -0
  529. package/src/resources/extensions/gsd/migrate/parsers.ts +11 -0
  530. package/src/resources/extensions/gsd/migrate/preview.ts +10 -0
  531. package/src/resources/extensions/gsd/migrate/transformer.ts +58 -4
  532. package/src/resources/extensions/gsd/migrate/writer.ts +14 -1
  533. package/src/resources/extensions/gsd/migration-auto-check.ts +15 -23
  534. package/src/resources/extensions/gsd/milestone-actions.ts +10 -4
  535. package/src/resources/extensions/gsd/native-git-bridge.ts +63 -14
  536. package/src/resources/extensions/gsd/parallel-orchestrator.ts +3 -0
  537. package/src/resources/extensions/gsd/paths.ts +5 -0
  538. package/src/resources/extensions/gsd/pending-auto-start.ts +79 -0
  539. package/src/resources/extensions/gsd/planning-path-scope.ts +10 -2
  540. package/src/resources/extensions/gsd/post-execution-checks.ts +87 -12
  541. package/src/resources/extensions/gsd/pre-execution-checks.ts +49 -11
  542. package/src/resources/extensions/gsd/preferences-mcp.ts +27 -0
  543. package/src/resources/extensions/gsd/preferences-types.ts +33 -0
  544. package/src/resources/extensions/gsd/preferences-validation.ts +145 -0
  545. package/src/resources/extensions/gsd/preferences.ts +5 -0
  546. package/src/resources/extensions/gsd/prompt-loader.ts +1 -1
  547. package/src/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  548. package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  549. package/src/resources/extensions/gsd/prompts/discuss-headless.md +8 -8
  550. package/src/resources/extensions/gsd/prompts/discuss.md +9 -9
  551. package/src/resources/extensions/gsd/prompts/forensics.md +3 -3
  552. package/src/resources/extensions/gsd/prompts/guided-discuss-project.md +4 -4
  553. package/src/resources/extensions/gsd/prompts/guided-discuss-requirements.md +3 -3
  554. package/src/resources/extensions/gsd/prompts/plan-slice.md +4 -4
  555. package/src/resources/extensions/gsd/prompts/queue.md +4 -4
  556. package/src/resources/extensions/gsd/prompts/reactive-execute.md +1 -1
  557. package/src/resources/extensions/gsd/prompts/refine-slice.md +2 -2
  558. package/src/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
  559. package/src/resources/extensions/gsd/prompts/system.md +2 -2
  560. package/src/resources/extensions/gsd/provider-switch-observer.ts +185 -0
  561. package/src/resources/extensions/gsd/queue-reorder-ui.ts +31 -13
  562. package/src/resources/extensions/gsd/repository-registry.ts +77 -0
  563. package/src/resources/extensions/gsd/safety/evidence-collector.ts +2 -0
  564. package/src/resources/extensions/gsd/safety/evidence-cross-ref.ts +54 -19
  565. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +52 -1
  566. package/src/resources/extensions/gsd/smart-entry-routing.ts +77 -0
  567. package/src/resources/extensions/gsd/state-reconciliation/drift/merge-state.ts +8 -1
  568. package/src/resources/extensions/gsd/state-reconciliation/drift/project-md.ts +12 -15
  569. package/src/resources/extensions/gsd/state-reconciliation/drift/roadmap.ts +17 -25
  570. package/src/resources/extensions/gsd/state.ts +15 -4
  571. package/src/resources/extensions/gsd/status-guards.ts +16 -2
  572. package/src/resources/extensions/gsd/templates/knowledge.md +2 -2
  573. package/src/resources/extensions/gsd/templates/plan.md +9 -5
  574. package/src/resources/extensions/gsd/templates/task-plan.md +10 -2
  575. package/src/resources/extensions/gsd/tests/auto-abort-pause-regression.test.ts +10 -1
  576. package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +71 -0
  577. package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +116 -0
  578. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +375 -1
  579. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +598 -8
  580. package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +151 -12
  581. package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +53 -2
  582. package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +18 -6
  583. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +129 -6
  584. package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +12 -0
  585. package/src/resources/extensions/gsd/tests/auto-runtime-state.test.ts +4 -4
  586. package/src/resources/extensions/gsd/tests/auto-start-orphan-bootstrap.test.ts +1 -0
  587. package/src/resources/extensions/gsd/tests/auto-stop-notification.test.ts +20 -0
  588. package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +69 -1
  589. package/src/resources/extensions/gsd/tests/autocomplete-regressions-1675.test.ts +21 -0
  590. package/src/resources/extensions/gsd/tests/brief-command.test.ts +89 -0
  591. package/src/resources/extensions/gsd/tests/browser-tools-compatibility-declarations.test.ts +62 -0
  592. package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +87 -0
  593. package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +107 -2
  594. package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +32 -4
  595. package/src/resources/extensions/gsd/tests/closeout-git-deferral.test.ts +16 -0
  596. package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +378 -0
  597. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +4 -1
  598. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +5 -9
  599. package/src/resources/extensions/gsd/tests/complete-task.test.ts +3 -1
  600. package/src/resources/extensions/gsd/tests/context-store-decisions-from-memories.test.ts +312 -0
  601. package/src/resources/extensions/gsd/tests/crash-recovery-via-db.test.ts +104 -2
  602. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +2 -0
  603. package/src/resources/extensions/gsd/tests/db-authority-regression.test.ts +208 -0
  604. package/src/resources/extensions/gsd/tests/db-task-slice-rows.test.ts +1 -0
  605. package/src/resources/extensions/gsd/tests/db-writer.test.ts +13 -8
  606. package/src/resources/extensions/gsd/tests/decisions-projection-from-memories.test.ts +453 -0
  607. package/src/resources/extensions/gsd/tests/decisions-stop-table-writes.test.ts +348 -0
  608. package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +61 -2
  609. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +111 -1
  610. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +68 -1
  611. package/src/resources/extensions/gsd/tests/doctor-empty-worktree.test.ts +65 -0
  612. package/src/resources/extensions/gsd/tests/doctor-forensics-db-open-regression.test.ts +50 -0
  613. package/src/resources/extensions/gsd/tests/evidence-cross-ref.test.ts +97 -0
  614. package/src/resources/extensions/gsd/tests/export-html-enhancements.test.ts +8 -0
  615. package/src/resources/extensions/gsd/tests/freeform-decisions.test.ts +8 -4
  616. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +28 -0
  617. package/src/resources/extensions/gsd/tests/gsd-tools.test.ts +11 -7
  618. package/src/resources/extensions/gsd/tests/gsdroot-worktree-detection.test.ts +5 -2
  619. package/src/resources/extensions/gsd/tests/guided-discuss-project-prompt-rendering.test.ts +2 -0
  620. package/src/resources/extensions/gsd/tests/guided-dispatch-root.test.ts +106 -0
  621. package/src/resources/extensions/gsd/tests/guided-flow-session-isolation.test.ts +59 -11
  622. package/src/resources/extensions/gsd/tests/guided-flow.test.ts +21 -0
  623. package/src/resources/extensions/gsd/tests/guided-tool-contract.test.ts +65 -0
  624. package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +7 -7
  625. package/src/resources/extensions/gsd/tests/hook-model-resolution.test.ts +5 -0
  626. package/src/resources/extensions/gsd/tests/infra-error.test.ts +2 -2
  627. package/src/resources/extensions/gsd/tests/infra-errors-cooldown.test.ts +9 -0
  628. package/src/resources/extensions/gsd/tests/init-prefs-routing.test.ts +22 -0
  629. package/src/resources/extensions/gsd/tests/integration/doctor-git.test.ts +44 -0
  630. package/src/resources/extensions/gsd/tests/integration/doctor-runtime.test.ts +20 -0
  631. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +199 -2
  632. package/src/resources/extensions/gsd/tests/integration/integration-lifecycle.test.ts +13 -5
  633. package/src/resources/extensions/gsd/tests/integration/migrate-command.test.ts +48 -3
  634. package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +6 -1
  635. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +49 -3
  636. package/src/resources/extensions/gsd/tests/journal.test.ts +32 -0
  637. package/src/resources/extensions/gsd/tests/knowledge-backfill-projection.test.ts +323 -0
  638. package/src/resources/extensions/gsd/tests/knowledge-capture.test.ts +242 -0
  639. package/src/resources/extensions/gsd/tests/knowledge.test.ts +47 -2
  640. package/src/resources/extensions/gsd/tests/load-knowledge-block-rules-only.test.ts +209 -0
  641. package/src/resources/extensions/gsd/tests/mcp-filter.test.ts +287 -0
  642. package/src/resources/extensions/gsd/tests/mcp-status.test.ts +11 -0
  643. package/src/resources/extensions/gsd/tests/memory-consolidation-scanner.test.ts +316 -0
  644. package/src/resources/extensions/gsd/tests/merge-db-cycle.test.ts +179 -0
  645. package/src/resources/extensions/gsd/tests/migrate-transformer.test.ts +5 -1
  646. package/src/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +24 -1
  647. package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +6 -1
  648. package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +26 -18
  649. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +80 -2
  650. package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +121 -1
  651. package/src/resources/extensions/gsd/tests/parallel-orchestrator-zombie-cleanup.test.ts +55 -0
  652. package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +55 -1
  653. package/src/resources/extensions/gsd/tests/pending-autostart-scope.test.ts +29 -5
  654. package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +2 -1
  655. package/src/resources/extensions/gsd/tests/plan-milestone-sketch-render.test.ts +157 -0
  656. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +26 -0
  657. package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +2 -0
  658. package/src/resources/extensions/gsd/tests/plan-slice.test.ts +343 -3
  659. package/src/resources/extensions/gsd/tests/plan-task.test.ts +18 -1
  660. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  661. package/src/resources/extensions/gsd/tests/post-execution-checks.test.ts +105 -3
  662. package/src/resources/extensions/gsd/tests/post-unit-git-failure.test.ts +1 -1
  663. package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +84 -0
  664. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +59 -0
  665. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +54 -0
  666. package/src/resources/extensions/gsd/tests/preferences-mcp.test.ts +128 -0
  667. package/src/resources/extensions/gsd/tests/preferences.test.ts +75 -0
  668. package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +78 -0
  669. package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +8 -0
  670. package/src/resources/extensions/gsd/tests/prompt-loader.test.ts +23 -0
  671. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +37 -1
  672. package/src/resources/extensions/gsd/tests/provider-switch-observer.test.ts +252 -0
  673. package/src/resources/extensions/gsd/tests/quality-gates.test.ts +6 -0
  674. package/src/resources/extensions/gsd/tests/queue-reorder-ui.test.ts +54 -0
  675. package/src/resources/extensions/gsd/tests/remediation-completion-guard.test.ts +89 -2
  676. package/src/resources/extensions/gsd/tests/repository-registry.test.ts +52 -0
  677. package/src/resources/extensions/gsd/tests/run-uat-replay-cap.test.ts +2 -3
  678. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +16 -4
  679. package/src/resources/extensions/gsd/tests/session-switch-abort-misclassification.test.ts +21 -1
  680. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +72 -1
  681. package/src/resources/extensions/gsd/tests/smart-entry-routing.test.ts +113 -0
  682. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +86 -2
  683. package/src/resources/extensions/gsd/tests/state-corruption-2945.test.ts +6 -0
  684. package/src/resources/extensions/gsd/tests/state-reconciliation-drift.test.ts +119 -23
  685. package/src/resources/extensions/gsd/tests/status-guards.test.ts +17 -1
  686. package/src/resources/extensions/gsd/tests/stuck-state-via-db.test.ts +64 -1
  687. package/src/resources/extensions/gsd/tests/subagent-model-dispatch.test.ts +2 -1
  688. package/src/resources/extensions/gsd/tests/summary-render-parity.test.ts +7 -3
  689. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +1 -0
  690. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +131 -9
  691. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +111 -1
  692. package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +29 -2
  693. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +42 -1
  694. package/src/resources/extensions/gsd/tests/verification-gate.test.ts +173 -1
  695. package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +78 -0
  696. package/src/resources/extensions/gsd/tests/workflow-kernel.test.ts +7 -0
  697. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +19 -1
  698. package/src/resources/extensions/gsd/tests/workflow-memory-pressure.test.ts +21 -1
  699. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +1 -1
  700. package/src/resources/extensions/gsd/tests/worktree-git-pathspec.test.ts +39 -0
  701. package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +64 -12
  702. package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +57 -2
  703. package/src/resources/extensions/gsd/tests/worktree-preferences-sync.test.ts +25 -0
  704. package/src/resources/extensions/gsd/tests/worktree-safety.test.ts +46 -0
  705. package/src/resources/extensions/gsd/tests/worktree-sync-milestones.test.ts +95 -0
  706. package/src/resources/extensions/gsd/tests/worktree-write-gate.test.ts +27 -0
  707. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +59 -0
  708. package/src/resources/extensions/gsd/tools/complete-milestone.ts +8 -10
  709. package/src/resources/extensions/gsd/tools/complete-slice.ts +6 -8
  710. package/src/resources/extensions/gsd/tools/plan-milestone.ts +5 -1
  711. package/src/resources/extensions/gsd/tools/plan-slice.ts +172 -12
  712. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +135 -0
  713. package/src/resources/extensions/gsd/types.ts +1 -1
  714. package/src/resources/extensions/gsd/unit-context-composer.ts +3 -0
  715. package/src/resources/extensions/gsd/unit-context-manifest.ts +86 -19
  716. package/src/resources/extensions/gsd/validation.ts +23 -1
  717. package/src/resources/extensions/gsd/verification-gate.ts +170 -6
  718. package/src/resources/extensions/gsd/verification-verdict.ts +47 -0
  719. package/src/resources/extensions/gsd/workflow-logger.ts +4 -0
  720. package/src/resources/extensions/gsd/workflow-manifest.ts +2 -0
  721. package/src/resources/extensions/gsd/workflow-mcp.ts +18 -1
  722. package/src/resources/extensions/gsd/workflow-projections.ts +6 -8
  723. package/src/resources/extensions/gsd/worktree-lifecycle.ts +93 -20
  724. package/src/resources/extensions/gsd/worktree-manager.ts +14 -2
  725. package/src/resources/extensions/gsd/worktree-safety.ts +45 -6
  726. package/src/resources/extensions/gsd/worktree-state-projection.ts +43 -0
  727. package/src/resources/extensions/shared/html-shell.ts +412 -0
  728. package/src/resources/extensions/shared/interview-ui.ts +6 -4
  729. package/src/resources/extensions/shared/tests/interview-notes-loop.test.ts +15 -0
  730. package/src/resources/extensions/subagent/index.ts +567 -103
  731. package/src/resources/extensions/subagent/launch.ts +131 -0
  732. package/src/resources/extensions/subagent/run-store.ts +218 -0
  733. package/src/resources/extensions/subagent/tests/launch.test.ts +115 -0
  734. package/src/resources/extensions/subagent/tests/run-store.test.ts +111 -0
  735. package/src/resources/extensions/ttsr/ttsr-manager.ts +5 -1
  736. package/src/resources/extensions/visual-brief/artifact-policy.ts +41 -0
  737. package/src/resources/extensions/visual-brief/extension-manifest.json +8 -0
  738. package/src/resources/extensions/visual-brief/index.ts +8 -0
  739. package/src/resources/extensions/visual-brief/page-contract.ts +136 -0
  740. package/src/resources/extensions/visual-brief/prompts.ts +183 -0
  741. package/src/resources/extensions/visual-brief/tests/visual-brief.test.ts +212 -0
  742. package/src/resources/skills/forensics/SKILL.md +1 -1
  743. package/dist/web/standalone/.next/server/chunks/5822.js +0 -2
  744. package/dist/web/standalone/.next/static/chunks/2556.0527fea66e123b7f.js +0 -1
  745. package/dist/web/standalone/.next/static/chunks/8359.e059d86b255fce1c.js +0 -10
  746. package/dist/web/standalone/.next/static/chunks/9441.1081da1125d1764f.js +0 -1
  747. package/dist/web/standalone/.next/static/chunks/app/layout-9ecfd95f343793f0.js +0 -1
  748. package/dist/web/standalone/.next/static/chunks/app/page-200592a7f3baf579.js +0 -1
  749. package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +0 -1
  750. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +0 -1
  751. package/dist/web/standalone/.next/static/css/54ec2745c1da488b.css +0 -1
  752. package/dist/web/standalone/.next/static/css/de70bee13400563f.css +0 -1
  753. package/dist/web/standalone/.next/static/media/4cf2300e9c8272f7-s.p.woff2 +0 -0
  754. package/dist/web/standalone/.next/static/media/747892c23ea88013-s.woff2 +0 -0
  755. package/dist/web/standalone/.next/static/media/8d697b304b401681-s.woff2 +0 -0
  756. package/dist/web/standalone/.next/static/media/93f479601ee12b01-s.p.woff2 +0 -0
  757. package/dist/web/standalone/.next/static/media/9610d9e46709d722-s.woff2 +0 -0
  758. package/dist/web/standalone/.next/static/media/ba015fad6dcf6784-s.woff2 +0 -0
  759. /package/dist/web/standalone/.next/static/{S44UQTFCUdA44dkjfYt6S → zCegwxH2e6vLp1vEZLLuZ}/_buildManifest.js +0 -0
  760. /package/dist/web/standalone/.next/static/{S44UQTFCUdA44dkjfYt6S → zCegwxH2e6vLp1vEZLLuZ}/_ssgManifest.js +0 -0
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=system-prompt-file-safety.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"system-prompt-file-safety.test.d.ts","sourceRoot":"","sources":["../../src/tests/system-prompt-file-safety.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,17 @@
1
+ // GSD-2 pi-coding-agent system prompt file-safety regression tests.
2
+ import test from "node:test";
3
+ import assert from "node:assert/strict";
4
+ import { buildSystemPrompt } from "../core/system-prompt.js";
5
+ test("buildSystemPrompt: read and write tools require reading before overwrite", () => {
6
+ const prompt = buildSystemPrompt({ selectedTools: ["read", "write"] });
7
+ assert.match(prompt, /before editing or overwriting/i);
8
+ assert.match(prompt, /Before write creates or replaces a file, verify the target path/i);
9
+ assert.match(prompt, /if it exists, read it first/i);
10
+ assert.match(prompt, /Use write only for new files or complete rewrites after verifying the target path/i);
11
+ });
12
+ test("buildSystemPrompt: write-only tool guidance does not reference unavailable read tool", () => {
13
+ const prompt = buildSystemPrompt({ selectedTools: ["write"] });
14
+ assert.doesNotMatch(prompt, /read it first/i);
15
+ assert.match(prompt, /Use write only for new files or complete rewrites after verifying the target path/i);
16
+ });
17
+ //# sourceMappingURL=system-prompt-file-safety.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"system-prompt-file-safety.test.js","sourceRoot":"","sources":["../../src/tests/system-prompt-file-safety.test.ts"],"names":[],"mappings":"AAAA,oEAAoE;AAEpE,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,MAAM,MAAM,oBAAoB,CAAC;AAExC,OAAO,EAAE,iBAAiB,EAAE,MAAM,0BAA0B,CAAC;AAE7D,IAAI,CAAC,0EAA0E,EAAE,GAAG,EAAE;IACrF,MAAM,MAAM,GAAG,iBAAiB,CAAC,EAAE,aAAa,EAAE,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,CAAC,CAAC;IAEvE,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,gCAAgC,CAAC,CAAC;IACvD,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,kEAAkE,CAAC,CAAC;IACzF,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,8BAA8B,CAAC,CAAC;IACrD,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,oFAAoF,CAAC,CAAC;AAC5G,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,sFAAsF,EAAE,GAAG,EAAE;IACjG,MAAM,MAAM,GAAG,iBAAiB,CAAC,EAAE,aAAa,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;IAE/D,MAAM,CAAC,YAAY,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAAC;IAC9C,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,oFAAoF,CAAC,CAAC;AAC5G,CAAC,CAAC,CAAC","sourcesContent":["// GSD-2 pi-coding-agent system prompt file-safety regression tests.\n\nimport test from \"node:test\";\nimport assert from \"node:assert/strict\";\n\nimport { buildSystemPrompt } from \"../core/system-prompt.js\";\n\ntest(\"buildSystemPrompt: read and write tools require reading before overwrite\", () => {\n\tconst prompt = buildSystemPrompt({ selectedTools: [\"read\", \"write\"] });\n\n\tassert.match(prompt, /before editing or overwriting/i);\n\tassert.match(prompt, /Before write creates or replaces a file, verify the target path/i);\n\tassert.match(prompt, /if it exists, read it first/i);\n\tassert.match(prompt, /Use write only for new files or complete rewrites after verifying the target path/i);\n});\n\ntest(\"buildSystemPrompt: write-only tool guidance does not reference unavailable read tool\", () => {\n\tconst prompt = buildSystemPrompt({ selectedTools: [\"write\"] });\n\n\tassert.doesNotMatch(prompt, /read it first/i);\n\tassert.match(prompt, /Use write only for new files or complete rewrites after verifying the target path/i);\n});\n"]}
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@gsd/pi-coding-agent",
3
- "version": "2.82.0",
3
+ "version": "3.0.0",
4
4
  "description": "Coding agent CLI (vendored from pi-mono)",
5
5
  "type": "module",
6
6
  "gsd": {
@@ -0,0 +1,79 @@
1
+ // Regression test for #5102: disabling thinking on one reasoning-capable
2
+ // model must not silently persist "off" as the global default.
3
+
4
+ import assert from "node:assert/strict";
5
+ import { mkdtempSync, rmSync } from "node:fs";
6
+ import { tmpdir } from "node:os";
7
+ import { join } from "node:path";
8
+ import { afterEach, beforeEach, describe, it } from "node:test";
9
+
10
+ import { Agent } from "@gsd/pi-agent-core";
11
+ import { getModel } from "@gsd/pi-ai";
12
+ import { AgentSession } from "./agent-session.js";
13
+ import { AuthStorage } from "./auth-storage.js";
14
+ import { ModelRegistry } from "./model-registry.js";
15
+ import { DefaultResourceLoader } from "./resource-loader.js";
16
+ import { SessionManager } from "./session-manager.js";
17
+ import { SettingsManager } from "./settings-manager.js";
18
+
19
+ let testDir: string;
20
+
21
+ async function createSession(): Promise<{ session: AgentSession; settingsManager: SettingsManager }> {
22
+ const agentDir = join(testDir, "agent-home");
23
+ const authStorage = AuthStorage.inMemory({});
24
+ const modelRegistry = new ModelRegistry(authStorage, join(agentDir, "models.json"));
25
+ const settingsManager = SettingsManager.inMemory({ defaultThinkingLevel: "high" });
26
+ const resourceLoader = new DefaultResourceLoader({
27
+ cwd: testDir,
28
+ agentDir,
29
+ settingsManager,
30
+ noExtensions: true,
31
+ noPromptTemplates: true,
32
+ noThemes: true,
33
+ });
34
+ await resourceLoader.reload();
35
+
36
+ const session = new AgentSession({
37
+ agent: new Agent({
38
+ initialState: {
39
+ model: getModel("zai", "glm-5.1" as any),
40
+ thinkingLevel: "high",
41
+ },
42
+ }),
43
+ sessionManager: SessionManager.inMemory(testDir),
44
+ settingsManager,
45
+ cwd: testDir,
46
+ resourceLoader,
47
+ modelRegistry,
48
+ });
49
+
50
+ return { session, settingsManager };
51
+ }
52
+
53
+ describe("AgentSession thinking level persistence", () => {
54
+ beforeEach(() => {
55
+ testDir = mkdtempSync(join(tmpdir(), "agent-session-thinking-level-"));
56
+ });
57
+
58
+ afterEach(() => {
59
+ rmSync(testDir, { recursive: true, force: true });
60
+ });
61
+
62
+ it("does not persist off as the global default for reasoning-capable models", async () => {
63
+ const { session, settingsManager } = await createSession();
64
+
65
+ session.setThinkingLevel("off");
66
+
67
+ assert.equal(session.thinkingLevel, "off");
68
+ assert.equal(settingsManager.getDefaultThinkingLevel(), "high");
69
+ });
70
+
71
+ it("still persists non-off thinking levels as the global default", async () => {
72
+ const { session, settingsManager } = await createSession();
73
+
74
+ session.setThinkingLevel("low");
75
+
76
+ assert.equal(session.thinkingLevel, "low");
77
+ assert.equal(settingsManager.getDefaultThinkingLevel(), "low");
78
+ });
79
+ });
@@ -1945,7 +1945,7 @@ export class AgentSession {
1945
1945
 
1946
1946
  if (isChanging) {
1947
1947
  this.sessionManager.appendThinkingLevelChange(effectiveLevel);
1948
- if (this.supportsThinking() || effectiveLevel !== "off") {
1948
+ if (effectiveLevel !== "off") {
1949
1949
  this.settingsManager.setDefaultThinkingLevel(effectiveLevel);
1950
1950
  }
1951
1951
  this._emitSessionStateChanged("set_thinking_level");
@@ -1,3 +1,5 @@
1
+ // Project/App: GSD-2
2
+ // File Purpose: Regression tests for streamed interactive chat ordering.
1
3
  import assert from "node:assert/strict";
2
4
  import { test } from "node:test";
3
5
 
@@ -97,9 +99,7 @@ function createHost() {
97
99
  return host;
98
100
  }
99
101
 
100
- test("chat-controller renders content blocks in content[] index order (tool-first stream)", async () => {
101
- // ToolExecutionComponent uses the global theme singleton.
102
- // Install a minimal no-op theme implementation for this unit test.
102
+ function installTheme() {
103
103
  (globalThis as any)[Symbol.for("@gsd/pi-coding-agent:theme")] = {
104
104
  fg: (_key: string, text: string) => text,
105
105
  bg: (_key: string, text: string) => text,
@@ -107,6 +107,12 @@ test("chat-controller renders content blocks in content[] index order (tool-firs
107
107
  italic: (text: string) => text,
108
108
  truncate: (text: string) => text,
109
109
  };
110
+ }
111
+
112
+ test("chat-controller renders content blocks in content[] index order (tool-first stream)", async () => {
113
+ // ToolExecutionComponent uses the global theme singleton.
114
+ // Install a minimal no-op theme implementation for this unit test.
115
+ installTheme();
110
116
 
111
117
  const host = createHost();
112
118
  const toolId = "mcp-tool-1";
@@ -168,6 +174,50 @@ test("chat-controller renders content blocks in content[] index order (tool-firs
168
174
  assert.equal(host.chatContainer.children[1]?.constructor?.name, "AssistantMessageComponent");
169
175
  });
170
176
 
177
+ test("chat-controller skips empty GPT reasoning blocks before tool-only turns", async () => {
178
+ installTheme();
179
+
180
+ const host = createHost();
181
+ host.getMarkdownThemeWithSettings = () => ({});
182
+ const toolId = "gpt-tool-1";
183
+ const toolCall = {
184
+ type: "toolCall",
185
+ id: toolId,
186
+ name: "read",
187
+ arguments: { filePath: "todo.js" },
188
+ };
189
+ const content = [
190
+ { type: "thinking", thinking: "", thinkingSignature: "encrypted" },
191
+ toolCall,
192
+ ];
193
+
194
+ await handleAgentEvent(host, { type: "message_start", message: makeAssistant([]) } as any);
195
+
196
+ await handleAgentEvent(
197
+ host,
198
+ {
199
+ type: "message_update",
200
+ message: makeAssistant(content),
201
+ assistantMessageEvent: {
202
+ type: "toolcall_end",
203
+ contentIndex: 1,
204
+ toolCall: {
205
+ ...toolCall,
206
+ externalResult: {
207
+ content: [{ type: "text", text: "todo contents" }],
208
+ details: {},
209
+ isError: false,
210
+ },
211
+ },
212
+ partial: makeAssistant(content),
213
+ },
214
+ } as any,
215
+ );
216
+
217
+ assert.equal(host.chatContainer.children.length, 1, "empty reasoning should not create a blank assistant block");
218
+ assert.equal(host.chatContainer.children[0]?.constructor?.name, "ToolExecutionComponent");
219
+ });
220
+
171
221
  test("chat-controller renders serverToolUse before trailing text matching content[] index order", async () => {
172
222
  (globalThis as any)[Symbol.for("@gsd/pi-coding-agent:theme")] = {
173
223
  fg: (_key: string, text: string) => text,
@@ -9,7 +9,14 @@ import { describe, it, mock } from "node:test";
9
9
  import type { AgentMessage } from "@gsd/pi-agent-core";
10
10
  import type { Model, AssistantMessage } from "@gsd/pi-ai";
11
11
 
12
- import { generateSummary, estimateTokens, chunkMessages, isDegenerateSummary, CompactionProducedNoSummaryError } from "./compaction.js";
12
+ import {
13
+ generateSummary,
14
+ estimateTokens,
15
+ chunkMessages,
16
+ isDegenerateSummary,
17
+ CompactionProducedNoSummaryError,
18
+ calculateContextTokens,
19
+ } from "./compaction.js";
13
20
  import { estimateSerializedTokens } from "./utils.js";
14
21
 
15
22
  // ---------------------------------------------------------------------------
@@ -187,6 +194,21 @@ describe("chunkMessages", () => {
187
194
  });
188
195
  });
189
196
 
197
+ describe("calculateContextTokens", () => {
198
+ it("uses prompt-relevant usage only (input + cacheRead + cacheWrite)", () => {
199
+ const usage = {
200
+ input: 65,
201
+ output: 39_846,
202
+ cacheRead: 2_945_563,
203
+ cacheWrite: 243_452,
204
+ totalTokens: 3_228_926,
205
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 },
206
+ };
207
+
208
+ assert.equal(calculateContextTokens(usage), 3_189_080);
209
+ });
210
+ });
211
+
190
212
  // ---------------------------------------------------------------------------
191
213
  // generateSummary chunked fallback tests
192
214
  // ---------------------------------------------------------------------------
@@ -107,10 +107,15 @@ export const DEFAULT_COMPACTION_SETTINGS: CompactionSettings = {
107
107
 
108
108
  /**
109
109
  * Calculate total context tokens from usage.
110
- * Uses the native totalTokens field when available, falls back to computing from components.
110
+ * Uses prompt-relevant components only:
111
+ * - input: current user/tool payload sent to model
112
+ * - cacheRead/cacheWrite: context replay + newly cached context
113
+ *
114
+ * Excludes output because output tokens are not part of the current prompt size.
115
+ * Excludes totalTokens because some providers report cumulative loop/session totals.
111
116
  */
112
117
  export function calculateContextTokens(usage: Usage): number {
113
- return usage.totalTokens || usage.input + usage.output + usage.cacheRead + usage.cacheWrite;
118
+ return usage.input + usage.cacheRead + usage.cacheWrite;
114
119
  }
115
120
 
116
121
  /**
@@ -573,7 +573,7 @@ export async function createAgentSession(options: CreateAgentSessionOptions = {}
573
573
  const removed = modelRegistry.authStorage.removeLegacyOAuthCredential(resolvedProvider);
574
574
  if (removed) {
575
575
  console.warn(
576
- `[auth] Removed unsupported Anthropic OAuth credential from auth.json (#3952).`,
576
+ `[auth] Removed unsupported Anthropic OAuth credential from auth.json.`,
577
577
  );
578
578
  }
579
579
  if (isClaudeCodeBinaryInPath()) {
@@ -203,9 +203,9 @@ export function buildSystemPrompt(options: BuildSystemPromptOptions = {}): strin
203
203
  addGuideline("Prefer grep/find/ls tools over bash for file exploration (faster, respects .gitignore)");
204
204
  }
205
205
 
206
- // Read before edit guideline
207
- if (hasRead && hasEdit) {
208
- addGuideline("Use read to examine files before editing. You must use this tool instead of cat or sed.");
206
+ // Read before file mutation guideline
207
+ if (hasRead && (hasEdit || hasWrite)) {
208
+ addGuideline("Use read to examine relevant existing files before editing or overwriting. Before write creates or replaces a file, verify the target path; if it exists, read it first. Use read instead of cat or sed for file inspection.");
209
209
  }
210
210
 
211
211
  // Edit guideline
@@ -215,7 +215,7 @@ export function buildSystemPrompt(options: BuildSystemPromptOptions = {}): strin
215
215
 
216
216
  // Write guideline
217
217
  if (hasWrite) {
218
- addGuideline("Use write only for new files or complete rewrites");
218
+ addGuideline("Use write only for new files or complete rewrites after verifying the target path");
219
219
  }
220
220
 
221
221
  // LSP guideline
@@ -39,6 +39,7 @@ function renderToolCollapsed(
39
39
  details?: Record<string, unknown>;
40
40
  },
41
41
  toolDefinition?: { label?: string; renderCall?: (...args: any[]) => any; renderResult?: (...args: any[]) => any },
42
+ width = 120,
42
43
  ): string {
43
44
  const component = new ToolExecutionComponent(
44
45
  toolName,
@@ -48,7 +49,7 @@ function renderToolCollapsed(
48
49
  { requestRender() {} } as any,
49
50
  );
50
51
  if (result) component.updateResult(result);
51
- return stripAnsi(component.render(120).join("\n"));
52
+ return stripAnsi(component.render(width).join("\n"));
52
53
  }
53
54
 
54
55
  describe("ToolExecutionComponent", () => {
@@ -294,6 +295,21 @@ describe("ToolExecutionComponent", () => {
294
295
  assert.doesNotMatch(rendered, /\bok\b/);
295
296
  });
296
297
 
298
+ test("uses available row width for compact bash command text", () => {
299
+ const rendered = renderToolCollapsed(
300
+ "bash",
301
+ {
302
+ command:
303
+ 'grep -n "expanded\\|toolOutputExpanded\\|setExpanded\\|defaultExpanded" /tmp/project/src/tool-execution.ts',
304
+ },
305
+ { content: [{ type: "text", text: "ok" }], isError: false, details: { cwd: "/tmp/project" } },
306
+ undefined,
307
+ 200,
308
+ );
309
+
310
+ assert.match(rendered, /defaultExpanded/);
311
+ });
312
+
297
313
  test("keeps failed tools expanded and error visible", () => {
298
314
  const rendered = renderToolCollapsed(
299
315
  "edit",
@@ -20,6 +20,17 @@ function sanitizeStatusText(text: string): string {
20
20
  .trim();
21
21
  }
22
22
 
23
+ function truncateFooterPath(text: string, width: number): string {
24
+ if (visibleWidth(text) <= width) return text;
25
+ const tailMatch = text.match(/( \([^)]+\)(?: • .*)?)$/);
26
+ if (!tailMatch) return truncateToWidth(text, width, "...");
27
+ const tail = tailMatch[1];
28
+ const tailWidth = visibleWidth(tail);
29
+ if (tailWidth >= width - 4) return truncateToWidth(text, width, "...");
30
+ const head = text.slice(0, -tail.length);
31
+ return `${truncateToWidth(head, width - tailWidth, "...")}${tail}`;
32
+ }
33
+
23
34
  /**
24
35
  * Format token counts (similar to web-ui)
25
36
  */
@@ -222,10 +233,6 @@ export class FooterComponent implements Component {
222
233
  }
223
234
  }
224
235
 
225
- // Apply dim to the stats group before handing it to the shared footer strip.
226
- // statsLeft may contain color codes for context %, so keep coloring local to the group.
227
- const dimStatsLeft = theme.fg("dim", statsLeft);
228
-
229
236
  // Extension statuses right-aligned on the pwd line (sorted by key).
230
237
  // Keeps the footer compact by avoiding a dedicated row when the content
231
238
  // fits alongside pwd. Falls back to pwd-only if the combined line would
@@ -239,12 +246,21 @@ export class FooterComponent implements Component {
239
246
  .join(" ")
240
247
  : "";
241
248
 
249
+ const footerRight = [rightSide, extStatusText].filter(Boolean).join(" ");
250
+ const gsdSegment = theme.fg("accent", "● GSD");
251
+ const dimStatsLeft = theme.fg("dim", statsLeft);
252
+ const innerWidth = Math.max(1, width - 2);
253
+ const rightWidth = visibleWidth(footerRight);
254
+ const leftBudget = footerRight ? Math.max(1, innerWidth - rightWidth - 3) : innerWidth;
255
+ const sepWidth = visibleWidth(" │ ");
256
+ const pwdBudget = Math.max(1, leftBudget - visibleWidth(gsdSegment) - visibleWidth(dimStatsLeft) - sepWidth * 2);
257
+ const pwdSegment = theme.fg("dim", truncateFooterPath(pwd, pwdBudget));
258
+
242
259
  const leftSegments = [
243
- theme.fg("accent", "● GSD"),
244
- theme.fg("dim", pwd),
260
+ gsdSegment,
261
+ pwdSegment,
245
262
  dimStatsLeft,
246
263
  ];
247
- const footerRight = [rightSide, extStatusText].filter(Boolean).join(" ");
248
264
  return renderFooterStrip(leftSegments, footerRight, width);
249
265
  }
250
266
  }
@@ -117,7 +117,7 @@ function formatElapsed(ms: number): string {
117
117
  }
118
118
 
119
119
  function formatCommandPreview(command: string): string {
120
- return truncateToWidth(command.replace(/\s+/g, " ").trim(), 64, "");
120
+ return command.replace(/\s+/g, " ").trim();
121
121
  }
122
122
 
123
123
  function appendLineOrRange(displayPath: string | undefined, target: ToolTargetMetadata): string | undefined {
@@ -1,4 +1,5 @@
1
- // GSD-2 Interactive Chat Controller
1
+ // Project/App: GSD-2
2
+ // File Purpose: Interactive TUI chat stream controller.
2
3
  import { Loader, Markdown, Spacer, Text } from "@gsd/pi-tui";
3
4
 
4
5
  import type { InteractiveModeEvent, InteractiveModeStateHost } from "../interactive-mode-state.js";
@@ -31,18 +32,77 @@ type RenderedSegment =
31
32
  | { kind: "tool"; contentIndex: number; component: ToolExecutionComponent }
32
33
  | { kind: "tool-summary"; component: ToolPhaseSummaryComponent; phases: ToolExecutionPhase[] };
33
34
 
35
+ type DesiredSegment =
36
+ | { kind: "text-run"; startIndex: number; endIndex: number; contentType: "text" | "thinking" }
37
+ | { kind: "tool"; contentIndex: number; toolId: string };
38
+
34
39
  let renderedSegments: RenderedSegment[] = [];
35
40
  // When providers reuse one assistant lifecycle across internal sub-turns,
36
41
  // a content[] shrink resets renderedSegments. Keep the displaced segments so
37
42
  // claude-code MCP pruning can remove stale provisional text later.
38
43
  let orphanedSegments: RenderedSegment[] = [];
39
44
 
45
+ function getVisibleTextLikeBlockType(block: any): "text" | "thinking" | undefined {
46
+ if (block?.type === "text" && typeof block.text === "string" && block.text.trim().length > 0) return "text";
47
+ if (block?.type === "thinking" && typeof block.thinking === "string" && block.thinking.trim().length > 0) return "thinking";
48
+ return undefined;
49
+ }
50
+
51
+ function buildDesiredSegments(
52
+ blocks: Array<any>,
53
+ options: { shouldSkipTextBlock?: (block: any, index: number) => boolean } = {},
54
+ ): DesiredSegment[] {
55
+ const desired: DesiredSegment[] = [];
56
+ let runStart = -1;
57
+ let runEnd = -1;
58
+ let runType: "text" | "thinking" | undefined;
59
+ const closeRun = () => {
60
+ if (runStart !== -1 && runType) {
61
+ desired.push({ kind: "text-run", startIndex: runStart, endIndex: runEnd, contentType: runType });
62
+ runStart = -1;
63
+ runEnd = -1;
64
+ runType = undefined;
65
+ }
66
+ };
67
+
68
+ for (let i = 0; i < blocks.length; i++) {
69
+ const block = blocks[i];
70
+ const blockType = getVisibleTextLikeBlockType(block);
71
+ const isInvisibleTextLike = blockType === undefined && (block?.type === "text" || block?.type === "thinking");
72
+ const isTool = block?.type === "toolCall" || block?.type === "serverToolUse";
73
+
74
+ if (blockType) {
75
+ if (options.shouldSkipTextBlock?.(block, i)) {
76
+ closeRun();
77
+ continue;
78
+ }
79
+ if (runStart === -1) {
80
+ runStart = i;
81
+ runEnd = i;
82
+ runType = blockType;
83
+ } else if (runType !== blockType) {
84
+ closeRun();
85
+ runStart = i;
86
+ runEnd = i;
87
+ runType = blockType;
88
+ } else {
89
+ runEnd = i;
90
+ }
91
+ } else {
92
+ if (isInvisibleTextLike) continue;
93
+ closeRun();
94
+ if (isTool) {
95
+ desired.push({ kind: "tool", contentIndex: i, toolId: block.id });
96
+ }
97
+ }
98
+ }
99
+ closeRun();
100
+
101
+ return desired;
102
+ }
103
+
40
104
  function hasVisibleAssistantContent(message: { content: Array<any> }): boolean {
41
- return message.content.some(
42
- (c) =>
43
- (c.type === "text" && typeof c.text === "string" && c.text.trim().length > 0)
44
- || (c.type === "thinking" && typeof c.thinking === "string" && c.thinking.trim().length > 0),
45
- );
105
+ return message.content.some((c) => getVisibleTextLikeBlockType(c) !== undefined);
46
106
  }
47
107
 
48
108
  function hasAssistantToolBlocks(message: { content: Array<any> }): boolean {
@@ -470,59 +530,14 @@ export async function handleAgentEvent(host: InteractiveModeStateHost & {
470
530
  // Only prune provisional pre-tool prose after post-tool prose exists,
471
531
  // so MCP tool-only windows do not blank the assistant content.
472
532
  const shouldDropPreToolProse = isClaudeCodeProvider && hasMcpToolBlock && hasPostToolText;
473
- type DesiredSegment =
474
- | { kind: "text-run"; startIndex: number; endIndex: number; contentType: "text" | "thinking" }
475
- | { kind: "tool"; contentIndex: number; toolId: string };
476
- const desired: DesiredSegment[] = [];
477
- let runStart = -1;
478
- let runEnd = -1;
479
- let runType: "text" | "thinking" | undefined;
480
- const closeRun = () => {
481
- if (runStart !== -1 && runType) {
482
- desired.push({ kind: "text-run", startIndex: runStart, endIndex: runEnd, contentType: runType });
483
- runStart = -1;
484
- runEnd = -1;
485
- runType = undefined;
486
- }
487
- };
488
- for (let i = 0; i < blocks.length; i++) {
489
- const b = blocks[i];
490
- const blockType = b.type === "text" || b.type === "thinking" ? b.type : undefined;
491
- const isTextLike = blockType === "text" || blockType === "thinking";
492
- const isTool = b.type === "toolCall" || b.type === "serverToolUse";
493
- // For Claude Code MCP turns, prune only pre-tool prose, never thinking.
494
- const textValue = blockType === "text" && typeof b?.text === "string" ? b.text : "";
495
- const isLikelyQuestion = blockType === "text" && typeof textValue === "string" && /\?\s*$/.test(textValue.trim());
496
- const shouldSkipProse = shouldDropPreToolProse
497
- && firstToolIdx >= 0
498
- && i < firstToolIdx
499
- && blockType === "text"
500
- && !isLikelyQuestion;
501
- if (shouldSkipProse) {
502
- closeRun();
503
- continue;
504
- }
505
- if (isTextLike) {
506
- if (runStart === -1) {
507
- runStart = i;
508
- runEnd = i;
509
- runType = blockType;
510
- } else if (runType !== blockType) {
511
- closeRun();
512
- runStart = i;
513
- runEnd = i;
514
- runType = blockType;
515
- } else {
516
- runEnd = i;
517
- }
518
- } else {
519
- closeRun();
520
- if (isTool) {
521
- desired.push({ kind: "tool", contentIndex: i, toolId: b.id });
522
- }
523
- }
524
- }
525
- closeRun();
533
+ const desired = buildDesiredSegments(blocks, {
534
+ shouldSkipTextBlock: (block: any, index: number) => {
535
+ if (!shouldDropPreToolProse || firstToolIdx < 0 || index >= firstToolIdx) return false;
536
+ if (getVisibleTextLikeBlockType(block) !== "text") return false;
537
+ const textValue = typeof block?.text === "string" ? block.text : "";
538
+ return !/\?\s*$/.test(textValue.trim());
539
+ },
540
+ });
526
541
 
527
542
  // Claude Code MCP can emit provisional pre-tool prose that gets
528
543
  // superseded by post-tool output. Prune stale text-run segments so
@@ -742,49 +757,7 @@ export async function handleAgentEvent(host: InteractiveModeStateHost & {
742
757
  // ranges/components don't keep stale partial indices.
743
758
  if (renderedSegments.length > 0) {
744
759
  const finalBlocks = host.streamingMessage.content;
745
- type DesiredSegment =
746
- | { kind: "text-run"; startIndex: number; endIndex: number; contentType: "text" | "thinking" }
747
- | { kind: "tool"; contentIndex: number; toolId: string };
748
- const desired: DesiredSegment[] = [];
749
- let runStart = -1;
750
- let runEnd = -1;
751
- let runType: "text" | "thinking" | undefined;
752
- const closeRun = () => {
753
- if (runStart !== -1 && runType) {
754
- desired.push({ kind: "text-run", startIndex: runStart, endIndex: runEnd, contentType: runType });
755
- runStart = -1;
756
- runEnd = -1;
757
- runType = undefined;
758
- }
759
- };
760
-
761
- for (let i = 0; i < finalBlocks.length; i++) {
762
- const block = finalBlocks[i] as any;
763
- const blockType = block?.type === "text" || block?.type === "thinking" ? block.type : undefined;
764
- const isTextLike = blockType === "text" || blockType === "thinking";
765
- const isTool = block?.type === "toolCall" || block?.type === "serverToolUse";
766
-
767
- if (isTextLike) {
768
- if (runStart === -1) {
769
- runStart = i;
770
- runEnd = i;
771
- runType = blockType;
772
- } else if (runType !== blockType) {
773
- closeRun();
774
- runStart = i;
775
- runEnd = i;
776
- runType = blockType;
777
- } else {
778
- runEnd = i;
779
- }
780
- } else {
781
- closeRun();
782
- if (isTool) {
783
- desired.push({ kind: "tool", contentIndex: i, toolId: block.id });
784
- }
785
- }
786
- }
787
- closeRun();
760
+ const desired = buildDesiredSegments(finalBlocks);
788
761
 
789
762
  const toolComponentsById = new Map<string, ToolExecutionComponent>();
790
763
  for (const [toolId, component] of host.pendingTools.entries()) {
@@ -45,7 +45,7 @@ function createMockHost() {
45
45
  toolOutputExpanded: false,
46
46
  hideThinkingBlock: false,
47
47
  isBashMode: false,
48
- onInputCallback: undefined,
48
+ onInputCallback: undefined as ((text: string) => void) | undefined,
49
49
  isInitialized: true,
50
50
  loadingAnimation: undefined,
51
51
  pendingWorkingMessage: undefined,
@@ -146,6 +146,20 @@ describe("input-controller pending images", () => {
146
146
  assert.equal(host.pendingImages.length, 0);
147
147
  assert.equal(promptCalls.length, 0); // bash commands don't go through prompt
148
148
  });
149
+
150
+ it("preserves pending images when using onInputCallback path", async () => {
151
+ let callbackText = "";
152
+ host.onInputCallback = (text: string) => {
153
+ callbackText = text;
154
+ };
155
+ host.pendingImages.push({ ...TEST_IMAGE });
156
+
157
+ await host.defaultEditor.onSubmit!("[Image #1] describe");
158
+
159
+ assert.equal(callbackText, "[Image #1] describe");
160
+ assert.equal(host.pendingImages.length, 1);
161
+ assert.equal(promptCalls.length, 0);
162
+ });
149
163
  });
150
164
 
151
165
  type HostOptions = {