gsd-pi 2.73.1 → 2.74.0-dev.b741afb

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 (378) hide show
  1. package/dist/cli-web-branch.d.ts +4 -3
  2. package/dist/cli-web-branch.js +10 -7
  3. package/dist/cli.js +184 -206
  4. package/dist/headless-query.js +4 -1
  5. package/dist/help-text.js +23 -0
  6. package/dist/logo.d.ts +1 -1
  7. package/dist/logo.js +1 -1
  8. package/dist/onboarding.js +59 -53
  9. package/dist/resource-loader.js +2 -2
  10. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +68 -4
  11. package/dist/resources/extensions/gsd/auto/detect-stuck.js +11 -4
  12. package/dist/resources/extensions/gsd/auto/phases.js +60 -10
  13. package/dist/resources/extensions/gsd/auto-dispatch.js +11 -3
  14. package/dist/resources/extensions/gsd/auto-model-selection.js +54 -11
  15. package/dist/resources/extensions/gsd/auto-post-unit.js +93 -57
  16. package/dist/resources/extensions/gsd/auto-prompts.js +12 -0
  17. package/dist/resources/extensions/gsd/auto-start.js +23 -6
  18. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +13 -0
  19. package/dist/resources/extensions/gsd/auto-verification.js +88 -3
  20. package/dist/resources/extensions/gsd/auto.js +37 -10
  21. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +21 -8
  22. package/dist/resources/extensions/gsd/commands/catalog.js +26 -1
  23. package/dist/resources/extensions/gsd/commands/handlers/ops.js +20 -0
  24. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +68 -9
  25. package/dist/resources/extensions/gsd/commands-add-tests.js +111 -0
  26. package/dist/resources/extensions/gsd/commands-backlog.js +140 -0
  27. package/dist/resources/extensions/gsd/commands-do.js +79 -0
  28. package/dist/resources/extensions/gsd/commands-handlers.js +8 -2
  29. package/dist/resources/extensions/gsd/commands-maintenance.js +6 -6
  30. package/dist/resources/extensions/gsd/commands-pr-branch.js +180 -0
  31. package/dist/resources/extensions/gsd/commands-session-report.js +82 -0
  32. package/dist/resources/extensions/gsd/commands-ship.js +187 -0
  33. package/dist/resources/extensions/gsd/db-writer.js +3 -5
  34. package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  35. package/dist/resources/extensions/gsd/graph-context.js +66 -0
  36. package/dist/resources/extensions/gsd/gsd-db.js +321 -0
  37. package/dist/resources/extensions/gsd/index.js +15 -2
  38. package/dist/resources/extensions/gsd/md-importer.js +3 -4
  39. package/dist/resources/extensions/gsd/memory-store.js +19 -51
  40. package/dist/resources/extensions/gsd/milestone-validation-gates.js +13 -12
  41. package/dist/resources/extensions/gsd/native-git-bridge.js +7 -4
  42. package/dist/resources/extensions/gsd/notification-widget.js +2 -2
  43. package/dist/resources/extensions/gsd/preferences-models.js +43 -0
  44. package/dist/resources/extensions/gsd/preferences-types.js +1 -0
  45. package/dist/resources/extensions/gsd/preferences-validation.js +22 -0
  46. package/dist/resources/extensions/gsd/prompts/add-tests.md +35 -0
  47. package/dist/resources/extensions/gsd/state.js +66 -15
  48. package/dist/resources/extensions/gsd/tools/complete-slice.js +15 -0
  49. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +3 -14
  50. package/dist/resources/extensions/gsd/triage-resolution.js +2 -5
  51. package/dist/resources/extensions/gsd/workflow-manifest.js +8 -69
  52. package/dist/resources/extensions/gsd/workflow-migration.js +21 -22
  53. package/dist/resources/extensions/gsd/workflow-projections.js +4 -1
  54. package/dist/resources/extensions/gsd/workflow-reconcile.js +14 -11
  55. package/dist/tsconfig.extensions.tsbuildinfo +1 -0
  56. package/dist/update-check.d.ts +1 -0
  57. package/dist/update-check.js +13 -5
  58. package/dist/update-cmd.js +4 -3
  59. package/dist/web/standalone/.next/BUILD_ID +1 -1
  60. package/dist/web/standalone/.next/app-path-routes-manifest.json +9 -9
  61. package/dist/web/standalone/.next/build-manifest.json +3 -3
  62. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  63. package/dist/web/standalone/.next/required-server-files.json +3 -3
  64. package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
  65. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  66. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  67. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  68. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  69. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  70. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  71. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  72. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  73. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  74. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  75. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  76. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  77. package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
  78. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
  79. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  80. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +3 -3
  81. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  82. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  83. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  85. package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
  86. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  87. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
  88. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  89. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
  90. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  91. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
  92. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  93. package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
  94. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  95. package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
  96. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  97. package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
  98. package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
  99. package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
  100. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  101. package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
  102. package/dist/web/standalone/.next/server/app/api/experimental/route.js +2 -2
  103. package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -1
  104. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  105. package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
  106. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  107. package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
  108. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  109. package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
  110. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  111. package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
  112. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  113. package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
  114. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  115. package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
  116. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  117. package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
  118. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  119. package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
  120. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  121. package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
  122. package/dist/web/standalone/.next/server/app/api/notifications/route.js +2 -2
  123. package/dist/web/standalone/.next/server/app/api/notifications/route_client-reference-manifest.js +1 -1
  124. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  125. package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
  126. package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
  127. package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
  128. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  129. package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  130. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  131. package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
  132. package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +2 -2
  133. package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
  134. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  135. package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
  136. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  137. package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
  138. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  139. package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
  140. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  141. package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
  142. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  143. package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
  144. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  145. package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
  146. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  147. package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
  148. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  149. package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
  150. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  151. package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
  152. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +2 -2
  153. package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
  154. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
  155. package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
  156. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
  157. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
  158. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +4 -4
  159. package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
  160. package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js +1 -1
  161. package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
  162. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  163. package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
  164. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  165. package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  166. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  167. package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
  168. package/dist/web/standalone/.next/server/app/index.html +1 -1
  169. package/dist/web/standalone/.next/server/app/index.rsc +4 -4
  170. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  171. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
  172. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  173. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +3 -3
  174. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  175. package/dist/web/standalone/.next/server/app/page.js +2 -2
  176. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  177. package/dist/web/standalone/.next/server/app-paths-manifest.json +9 -9
  178. package/dist/web/standalone/.next/server/chunks/63.js +3 -3
  179. package/dist/web/standalone/.next/server/chunks/6897.js +1 -1
  180. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  181. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  182. package/dist/web/standalone/.next/server/middleware.js +2 -2
  183. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  184. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  185. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  186. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  187. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  188. package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-2f24283c162b6ab3.js → page-f2a7482d42a5614b.js} +1 -1
  189. package/dist/web/standalone/.next/static/chunks/app/{layout-9ecfd95f343793f0.js → layout-a16c7a7ecdf0c2cf.js} +1 -1
  190. package/dist/web/standalone/.next/static/chunks/app/page-f1e30ab6bb269149.js +1 -0
  191. package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +1 -0
  192. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +1 -0
  193. package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
  194. package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
  195. package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
  196. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
  197. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
  198. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
  199. package/dist/web/standalone/server.js +1 -1
  200. package/package.json +3 -3
  201. package/packages/daemon/package.json +2 -2
  202. package/packages/mcp-server/dist/index.d.ts +3 -0
  203. package/packages/mcp-server/dist/index.d.ts.map +1 -1
  204. package/packages/mcp-server/dist/index.js +3 -0
  205. package/packages/mcp-server/dist/index.js.map +1 -1
  206. package/packages/mcp-server/dist/readers/graph.d.ts +87 -0
  207. package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -0
  208. package/packages/mcp-server/dist/readers/graph.js +548 -0
  209. package/packages/mcp-server/dist/readers/graph.js.map +1 -0
  210. package/packages/mcp-server/dist/readers/index.d.ts +2 -0
  211. package/packages/mcp-server/dist/readers/index.d.ts.map +1 -1
  212. package/packages/mcp-server/dist/readers/index.js +1 -0
  213. package/packages/mcp-server/dist/readers/index.js.map +1 -1
  214. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  215. package/packages/mcp-server/dist/server.js +65 -0
  216. package/packages/mcp-server/dist/server.js.map +1 -1
  217. package/packages/mcp-server/package.json +2 -2
  218. package/packages/mcp-server/src/index.ts +15 -0
  219. package/packages/mcp-server/src/readers/graph.test.ts +426 -0
  220. package/packages/mcp-server/src/readers/graph.ts +708 -0
  221. package/packages/mcp-server/src/readers/index.ts +12 -0
  222. package/packages/mcp-server/src/server.ts +83 -0
  223. package/packages/mcp-server/tsconfig.json +1 -0
  224. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -0
  225. package/packages/native/package.json +2 -2
  226. package/packages/native/tsconfig.tsbuildinfo +1 -0
  227. package/packages/pi-agent-core/package.json +1 -1
  228. package/packages/pi-agent-core/tsconfig.json +1 -0
  229. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -0
  230. package/packages/pi-ai/dist/index.d.ts +1 -0
  231. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  232. package/packages/pi-ai/dist/index.js +1 -0
  233. package/packages/pi-ai/dist/index.js.map +1 -1
  234. package/packages/pi-ai/dist/utils/overflow.d.ts.map +1 -1
  235. package/packages/pi-ai/dist/utils/overflow.js +12 -0
  236. package/packages/pi-ai/dist/utils/overflow.js.map +1 -1
  237. package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts +2 -0
  238. package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts.map +1 -0
  239. package/packages/pi-ai/dist/utils/tests/overflow.test.js +50 -0
  240. package/packages/pi-ai/dist/utils/tests/overflow.test.js.map +1 -0
  241. package/packages/pi-ai/package.json +1 -1
  242. package/packages/pi-ai/src/index.ts +4 -0
  243. package/packages/pi-ai/src/utils/overflow.ts +14 -1
  244. package/packages/pi-ai/src/utils/tests/overflow.test.ts +58 -0
  245. package/packages/pi-ai/tsconfig.json +1 -0
  246. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -0
  247. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +313 -8
  248. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  249. package/packages/pi-coding-agent/dist/core/compaction/utils.js +5 -5
  250. package/packages/pi-coding-agent/dist/core/compaction/utils.js.map +1 -1
  251. package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts +2 -0
  252. package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts.map +1 -0
  253. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js +45 -0
  254. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js.map +1 -0
  255. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +12 -2
  256. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  257. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +61 -28
  258. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
  259. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts +2 -1
  260. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -1
  261. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js +9 -3
  262. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js.map +1 -1
  263. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts +2 -0
  264. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts.map +1 -0
  265. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +52 -0
  266. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -0
  267. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  268. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +94 -16
  269. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  270. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  271. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +11 -3
  272. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  273. package/packages/pi-coding-agent/package.json +1 -1
  274. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +355 -8
  275. package/packages/pi-coding-agent/src/core/compaction/utils.ts +5 -5
  276. package/packages/pi-coding-agent/src/core/compaction-utils.test.ts +50 -0
  277. package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +74 -32
  278. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +73 -0
  279. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.ts +9 -3
  280. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +113 -21
  281. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +11 -3
  282. package/packages/pi-coding-agent/tsconfig.json +1 -0
  283. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -0
  284. package/packages/pi-tui/dist/__tests__/tui.test.js +60 -1
  285. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  286. package/packages/pi-tui/dist/tui.d.ts +8 -0
  287. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  288. package/packages/pi-tui/dist/tui.js +32 -3
  289. package/packages/pi-tui/dist/tui.js.map +1 -1
  290. package/packages/pi-tui/package.json +1 -1
  291. package/packages/pi-tui/src/__tests__/tui.test.ts +76 -1
  292. package/packages/pi-tui/src/tui.ts +31 -3
  293. package/packages/pi-tui/tsconfig.json +1 -0
  294. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -0
  295. package/packages/rpc-client/package.json +1 -1
  296. package/packages/rpc-client/tsconfig.json +1 -0
  297. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -0
  298. package/pkg/package.json +1 -1
  299. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +107 -5
  300. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +111 -2
  301. package/src/resources/extensions/gsd/auto/detect-stuck.ts +12 -4
  302. package/src/resources/extensions/gsd/auto/loop-deps.ts +6 -0
  303. package/src/resources/extensions/gsd/auto/phases.ts +90 -10
  304. package/src/resources/extensions/gsd/auto-dispatch.ts +10 -4
  305. package/src/resources/extensions/gsd/auto-model-selection.ts +85 -11
  306. package/src/resources/extensions/gsd/auto-post-unit.ts +107 -58
  307. package/src/resources/extensions/gsd/auto-prompts.ts +13 -0
  308. package/src/resources/extensions/gsd/auto-start.ts +30 -6
  309. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +17 -0
  310. package/src/resources/extensions/gsd/auto-verification.ts +98 -3
  311. package/src/resources/extensions/gsd/auto.ts +38 -14
  312. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +24 -8
  313. package/src/resources/extensions/gsd/commands/catalog.ts +26 -1
  314. package/src/resources/extensions/gsd/commands/handlers/ops.ts +20 -0
  315. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +74 -9
  316. package/src/resources/extensions/gsd/commands-add-tests.ts +137 -0
  317. package/src/resources/extensions/gsd/commands-backlog.ts +182 -0
  318. package/src/resources/extensions/gsd/commands-do.ts +109 -0
  319. package/src/resources/extensions/gsd/commands-handlers.ts +8 -2
  320. package/src/resources/extensions/gsd/commands-maintenance.ts +6 -6
  321. package/src/resources/extensions/gsd/commands-pr-branch.ts +234 -0
  322. package/src/resources/extensions/gsd/commands-session-report.ts +101 -0
  323. package/src/resources/extensions/gsd/commands-ship.ts +219 -0
  324. package/src/resources/extensions/gsd/db-writer.ts +3 -5
  325. package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  326. package/src/resources/extensions/gsd/graph-context.ts +85 -0
  327. package/src/resources/extensions/gsd/gsd-db.ts +467 -0
  328. package/src/resources/extensions/gsd/index.ts +18 -2
  329. package/src/resources/extensions/gsd/md-importer.ts +3 -5
  330. package/src/resources/extensions/gsd/memory-store.ts +31 -62
  331. package/src/resources/extensions/gsd/milestone-validation-gates.ts +13 -14
  332. package/src/resources/extensions/gsd/native-git-bridge.ts +11 -12
  333. package/src/resources/extensions/gsd/notification-widget.ts +2 -2
  334. package/src/resources/extensions/gsd/preferences-models.ts +41 -0
  335. package/src/resources/extensions/gsd/preferences-types.ts +12 -0
  336. package/src/resources/extensions/gsd/preferences-validation.ts +23 -0
  337. package/src/resources/extensions/gsd/prompts/add-tests.md +35 -0
  338. package/src/resources/extensions/gsd/state.ts +80 -17
  339. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +2 -2
  340. package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +53 -0
  341. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +51 -2
  342. package/src/resources/extensions/gsd/tests/commands-backlog.test.ts +158 -0
  343. package/src/resources/extensions/gsd/tests/commands-do.test.ts +127 -0
  344. package/src/resources/extensions/gsd/tests/commands-pr-branch.test.ts +68 -0
  345. package/src/resources/extensions/gsd/tests/commands-session-report.test.ts +82 -0
  346. package/src/resources/extensions/gsd/tests/commands-ship.test.ts +71 -0
  347. package/src/resources/extensions/gsd/tests/commands-workflow-custom.test.ts +14 -0
  348. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +142 -0
  349. package/src/resources/extensions/gsd/tests/completed-at-reconcile.test.ts +42 -0
  350. package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -2
  351. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +3 -2
  352. package/src/resources/extensions/gsd/tests/derive-state-helpers.test.ts +68 -8
  353. package/src/resources/extensions/gsd/tests/derive-state.test.ts +3 -3
  354. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +154 -0
  355. package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +137 -1
  356. package/src/resources/extensions/gsd/tests/graph-context.test.ts +337 -0
  357. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +4 -2
  358. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +68 -1
  359. package/src/resources/extensions/gsd/tests/model-isolation.test.ts +91 -2
  360. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +140 -0
  361. package/src/resources/extensions/gsd/tests/preferences.test.ts +47 -0
  362. package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +180 -0
  363. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +5 -7
  364. package/src/resources/extensions/gsd/tests/token-profile.test.ts +1 -1
  365. package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +179 -0
  366. package/src/resources/extensions/gsd/tests/workflow-logger-wiring.test.ts +223 -0
  367. package/src/resources/extensions/gsd/tools/complete-slice.ts +19 -0
  368. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +3 -11
  369. package/src/resources/extensions/gsd/triage-resolution.ts +2 -7
  370. package/src/resources/extensions/gsd/workflow-manifest.ts +9 -104
  371. package/src/resources/extensions/gsd/workflow-migration.ts +21 -29
  372. package/src/resources/extensions/gsd/workflow-projections.ts +8 -1
  373. package/src/resources/extensions/gsd/workflow-reconcile.ts +15 -15
  374. package/dist/web/standalone/.next/static/chunks/app/page-7115e62689b5fd84.js +0 -1
  375. package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +0 -1
  376. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +0 -1
  377. /package/dist/web/standalone/.next/static/{Qr27MOHx0lxRGnJvlhxxu → XnHY5eXUsTCFmNodWHetD}/_buildManifest.js +0 -0
  378. /package/dist/web/standalone/.next/static/{Qr27MOHx0lxRGnJvlhxxu → XnHY5eXUsTCFmNodWHetD}/_ssgManifest.js +0 -0
@@ -6,6 +6,8 @@
6
6
  * and dynamic routing configuration.
7
7
  */
8
8
  import { existsSync, readFileSync, writeFileSync } from "node:fs";
9
+ import { homedir } from "node:os";
10
+ import { join } from "node:path";
9
11
  import { defaultRoutingConfig } from "./model-router.js";
10
12
  import { loadEffectiveGSDPreferences, getGlobalGSDPreferencesPath } from "./preferences.js";
11
13
  /**
@@ -161,6 +163,47 @@ export function resolveDefaultSessionModel(sessionProvider) {
161
163
  }
162
164
  return undefined;
163
165
  }
166
+ /**
167
+ * Returns true if `provider` is defined as a custom provider in the user's
168
+ * `~/.gsd/agent/models.json` (Ollama, vLLM, LM Studio, OpenAI-compatible
169
+ * proxies, etc.).
170
+ *
171
+ * Used by auto-mode bootstrap to decide whether the session model
172
+ * (set via `/gsd model`) should override `PREFERENCES.md`. Custom providers
173
+ * are never reachable from `PREFERENCES.md` (which only knows built-in
174
+ * providers), so when the user has explicitly selected one, it must take
175
+ * priority — otherwise auto-mode tries to start the built-in provider from
176
+ * PREFERENCES.md and fails with "Not logged in · Please run /login" (#4122).
177
+ *
178
+ * Reads models.json directly with a lightweight JSON parse to avoid
179
+ * pulling in the full model-registry at this call site. Falls back to
180
+ * `~/.pi/agent/models.json` for parity with `resolveModelsJsonPath()`.
181
+ * Any read or parse error yields `false` (treat as not-custom) so a
182
+ * malformed models.json never breaks the session bootstrap.
183
+ */
184
+ export function isCustomProvider(provider) {
185
+ if (!provider)
186
+ return false;
187
+ const candidates = [
188
+ join(homedir(), ".gsd", "agent", "models.json"),
189
+ join(homedir(), ".pi", "agent", "models.json"),
190
+ ];
191
+ for (const path of candidates) {
192
+ if (!existsSync(path))
193
+ continue;
194
+ try {
195
+ const raw = readFileSync(path, "utf-8");
196
+ const parsed = JSON.parse(raw);
197
+ if (parsed?.providers && Object.prototype.hasOwnProperty.call(parsed.providers, provider)) {
198
+ return true;
199
+ }
200
+ }
201
+ catch {
202
+ // Ignore — malformed models.json must not break bootstrap.
203
+ }
204
+ }
205
+ return false;
206
+ }
164
207
  /**
165
208
  * Determines the next fallback model to try when the current model fails.
166
209
  * If the current model is not in the configured list, returns the primary model.
@@ -83,6 +83,7 @@ export const KNOWN_PREFERENCE_KEYS = new Set([
83
83
  "discuss_preparation",
84
84
  "discuss_web_research",
85
85
  "discuss_depth",
86
+ "flat_rate_providers",
86
87
  ]);
87
88
  /** Canonical list of all dispatch unit types. */
88
89
  export const KNOWN_UNIT_TYPES = [
@@ -155,6 +155,28 @@ export function validatePreferences(preferences) {
155
155
  errors.push(`search_provider must be one of: brave, tavily, ollama, native, auto`);
156
156
  }
157
157
  }
158
+ // ─── Flat-rate Providers ────────────────────────────────────────────
159
+ // User-declared flat-rate providers for dynamic routing suppression.
160
+ // Built-in providers (github-copilot, copilot, claude-code) and any
161
+ // externalCli provider are already auto-detected; this list layers on
162
+ // top for private subscription proxies and custom CLI wrappers.
163
+ if (preferences.flat_rate_providers !== undefined) {
164
+ if (Array.isArray(preferences.flat_rate_providers)) {
165
+ const allStrings = preferences.flat_rate_providers.every((item) => typeof item === "string");
166
+ if (allStrings) {
167
+ // Strip empty/whitespace-only entries to avoid false matches.
168
+ validated.flat_rate_providers = preferences.flat_rate_providers
169
+ .map((s) => s.trim())
170
+ .filter((s) => s.length > 0);
171
+ }
172
+ else {
173
+ errors.push("flat_rate_providers must be an array of strings");
174
+ }
175
+ }
176
+ else {
177
+ errors.push("flat_rate_providers must be an array of strings");
178
+ }
179
+ }
158
180
  // ─── Phase Skip Preferences ─────────────────────────────────────────
159
181
  if (preferences.phases !== undefined) {
160
182
  if (typeof preferences.phases === "object" && preferences.phases !== null) {
@@ -0,0 +1,35 @@
1
+ You are generating tests for recently completed GSD work.
2
+
3
+ ## Slice: {{sliceId}} — {{sliceTitle}}
4
+
5
+ ### Summary
6
+
7
+ {{sliceSummary}}
8
+
9
+ ### Existing Test Patterns
10
+
11
+ {{existingTestPatterns}}
12
+
13
+ ## Working Directory
14
+
15
+ `{{workingDirectory}}`
16
+
17
+ ## Instructions
18
+
19
+ 1. Read the slice summary above to understand what was built
20
+ 2. Identify the source files that were created or modified for this slice
21
+ 3. Read the implementation code to understand behavior, edge cases, and error paths
22
+ 4. Write comprehensive tests following the project's existing test patterns and framework
23
+ 5. Run the tests to verify they pass
24
+ 6. Fix any failures
25
+
26
+ ### Rules
27
+
28
+ - Follow the project's existing test patterns (framework, assertions, file structure)
29
+ - Test behavior, not implementation details
30
+ - Cover: happy path, edge cases, error conditions, boundary values
31
+ - Do NOT modify implementation files — only create or update test files
32
+ - Name test files consistently with the project's conventions
33
+ - Keep tests focused and readable
34
+
35
+ {{skillActivation}}
@@ -271,7 +271,11 @@ function reconcileDiskToDb(basePath) {
271
271
  try {
272
272
  roadmapContent = readFileSync(roadmapPath, "utf-8");
273
273
  }
274
- catch {
274
+ catch (err) {
275
+ logWarning("state", "reconcileDiskToDb: roadmap read failed, skipping milestone", {
276
+ mid,
277
+ error: err.message,
278
+ });
275
279
  continue;
276
280
  }
277
281
  const parsed = parseRoadmap(roadmapContent);
@@ -312,6 +316,10 @@ function reconcileDiskToDb(basePath) {
312
316
  function buildCompletenessSet(basePath, milestones) {
313
317
  const completeMilestoneIds = new Set();
314
318
  const parkedMilestoneIds = new Set();
319
+ // DB-authoritative: a milestone is only "complete" when its DB row says so.
320
+ // SUMMARY-file presence is NOT a completion signal here — an orphan SUMMARY
321
+ // (crashed complete-milestone turn, partial merge, manual edit) must not
322
+ // flip derived state to complete and cascade into a false auto-merge (#4179).
315
323
  for (const m of milestones) {
316
324
  const parkedFile = resolveMilestoneFile(basePath, m.id, "PARKED");
317
325
  if (parkedFile || m.status === 'parked') {
@@ -322,11 +330,6 @@ function buildCompletenessSet(basePath, milestones) {
322
330
  completeMilestoneIds.add(m.id);
323
331
  continue;
324
332
  }
325
- const summaryFile = resolveMilestoneFile(basePath, m.id, "SUMMARY");
326
- if (summaryFile) {
327
- completeMilestoneIds.add(m.id);
328
- continue;
329
- }
330
333
  }
331
334
  return { completeMilestoneIds, parkedMilestoneIds };
332
335
  }
@@ -347,17 +350,22 @@ async function buildRegistryAndFindActive(basePath, milestones, completeMileston
347
350
  if (isGhostMilestone(basePath, m.id))
348
351
  continue;
349
352
  }
350
- const summaryFile = resolveMilestoneFile(basePath, m.id, "SUMMARY");
351
- if (completeMilestoneIds.has(m.id) || (summaryFile !== null)) {
353
+ // DB-authoritative completeness (#4179): only trust completeMilestoneIds,
354
+ // which is itself derived from DB status. SUMMARY-file presence alone must
355
+ // not imply completion. The summary file may still be consulted below as a
356
+ // title source for legitimately-complete milestones whose DB row has no title.
357
+ if (completeMilestoneIds.has(m.id)) {
352
358
  let title = stripMilestonePrefix(m.title) || m.id;
353
- if (summaryFile && !m.title) {
354
- const summaryContent = await loadFile(summaryFile);
355
- if (summaryContent) {
356
- title = parseSummary(summaryContent).title || m.id;
359
+ if (!m.title) {
360
+ const summaryFile = resolveMilestoneFile(basePath, m.id, "SUMMARY");
361
+ if (summaryFile) {
362
+ const summaryContent = await loadFile(summaryFile);
363
+ if (summaryContent) {
364
+ title = parseSummary(summaryContent).title || m.id;
365
+ }
357
366
  }
358
367
  }
359
368
  registry.push({ id: m.id, title, status: 'complete' });
360
- completeMilestoneIds.add(m.id);
361
369
  continue;
362
370
  }
363
371
  const allSlicesDone = slices.length > 0 && slices.every(s => isStatusDone(s.status));
@@ -391,7 +399,14 @@ async function buildRegistryAndFindActive(basePath, milestones, completeMileston
391
399
  const validationFile = resolveMilestoneFile(basePath, m.id, "VALIDATION");
392
400
  const validationContent = validationFile ? await loadFile(validationFile) : null;
393
401
  const validationTerminal = validationContent ? isValidationTerminal(validationContent) : false;
394
- if (!validationTerminal || (validationTerminal && !summaryFile)) {
402
+ // DB-authoritative (#4179): completeness is already decided by
403
+ // completeMilestoneIds above. If we reached this branch, the DB says
404
+ // the milestone is NOT complete — so any SUMMARY file on disk is an
405
+ // orphan (crashed complete-milestone, partial merge, manual edit) and
406
+ // must not short-circuit this path. When validation is terminal, fall
407
+ // through to the default active-push below so `complete-milestone` can
408
+ // re-run idempotently.
409
+ if (!validationTerminal) {
395
410
  activeMilestone = { id: m.id, title };
396
411
  activeMilestoneSlices = slices;
397
412
  activeMilestoneFound = true;
@@ -516,6 +531,9 @@ function resolveSliceDependencies(activeMilestoneSlices) {
516
531
  return { activeSlice: null, activeSliceRow: null };
517
532
  }
518
533
  }
534
+ // First pass: find a slice with ALL dependencies satisfied (strict)
535
+ let bestFallback = null;
536
+ let bestFallbackSatisfied = -1;
519
537
  for (const s of activeMilestoneSlices) {
520
538
  if (isStatusDone(s.status))
521
539
  continue;
@@ -524,6 +542,23 @@ function resolveSliceDependencies(activeMilestoneSlices) {
524
542
  if (s.depends.every(dep => doneSliceIds.has(dep))) {
525
543
  return { activeSlice: { id: s.id, title: s.title }, activeSliceRow: s };
526
544
  }
545
+ // Track the slice with the most satisfied dependencies as fallback
546
+ const satisfied = s.depends.filter(dep => doneSliceIds.has(dep)).length;
547
+ if (satisfied > bestFallbackSatisfied || (satisfied === bestFallbackSatisfied && !bestFallback)) {
548
+ bestFallback = s;
549
+ bestFallbackSatisfied = satisfied;
550
+ }
551
+ }
552
+ // Fallback: if no slice has all deps met but there ARE incomplete non-deferred
553
+ // slices, pick the one with the most deps satisfied. This prevents hard-blocking
554
+ // when dependency metadata is stale (e.g. after reassessment added/removed slices)
555
+ // or when deps reference slices from previous milestones.
556
+ if (bestFallback) {
557
+ const unmet = bestFallback.depends.filter(dep => !doneSliceIds.has(dep));
558
+ logWarning("state", `No slice has all deps satisfied — falling back to ${bestFallback.id} ` +
559
+ `(${bestFallbackSatisfied}/${bestFallback.depends.length} deps met, ` +
560
+ `unmet: ${unmet.join(", ")})`, { mid: activeMilestoneSlices[0]?.milestone_id, sid: bestFallback.id });
561
+ return { activeSlice: { id: bestFallback.id, title: bestFallback.title }, activeSliceRow: bestFallback };
527
562
  }
528
563
  return { activeSlice: null, activeSliceRow: null };
529
564
  }
@@ -567,7 +602,7 @@ async function reconcileSliceTasks(basePath, milestoneId, sliceId, planFile) {
567
602
  const summaryPath = resolveTaskFile(basePath, milestoneId, sliceId, t.id, "SUMMARY");
568
603
  if (summaryPath && existsSync(summaryPath)) {
569
604
  try {
570
- updateTaskStatus(milestoneId, sliceId, t.id, "complete");
605
+ updateTaskStatus(milestoneId, sliceId, t.id, "complete", new Date().toISOString());
571
606
  logWarning("reconcile", `task ${milestoneId}/${sliceId}/${t.id} status reconciled from "${t.status}" to "complete" (#2514)`, { mid: milestoneId, sid: sliceId, tid: t.id });
572
607
  reconciled = true;
573
608
  }
@@ -1269,6 +1304,8 @@ export async function _deriveStateImpl(basePath) {
1269
1304
  }
1270
1305
  }
1271
1306
  else {
1307
+ let bestFallbackLegacy = null;
1308
+ let bestFallbackLegacySatisfied = -1;
1272
1309
  for (const s of activeRoadmap.slices) {
1273
1310
  if (s.done)
1274
1311
  continue;
@@ -1276,6 +1313,20 @@ export async function _deriveStateImpl(basePath) {
1276
1313
  activeSlice = { id: s.id, title: s.title };
1277
1314
  break;
1278
1315
  }
1316
+ // Track best fallback
1317
+ const satisfied = s.depends.filter(dep => doneSliceIds.has(dep)).length;
1318
+ if (satisfied > bestFallbackLegacySatisfied) {
1319
+ bestFallbackLegacy = s;
1320
+ bestFallbackLegacySatisfied = satisfied;
1321
+ }
1322
+ }
1323
+ // Fallback: if no slice has all deps met, pick the one with the most deps satisfied
1324
+ if (!activeSlice && bestFallbackLegacy) {
1325
+ const unmet = bestFallbackLegacy.depends.filter(dep => !doneSliceIds.has(dep));
1326
+ logWarning("state", `No slice has all deps satisfied — falling back to ${bestFallbackLegacy.id} ` +
1327
+ `(${bestFallbackLegacySatisfied}/${bestFallbackLegacy.depends.length} deps met, ` +
1328
+ `unmet: ${unmet.join(", ")})`);
1329
+ activeSlice = { id: bestFallbackLegacy.id, title: bestFallbackLegacy.title };
1279
1330
  }
1280
1331
  }
1281
1332
  if (!activeSlice) {
@@ -354,6 +354,21 @@ export async function handleCompleteSlice(params, basePath) {
354
354
  catch (eventErr) {
355
355
  logError("tool", `complete-slice event log FAILED — completion invisible to reconciliation`, { error: eventErr.message });
356
356
  }
357
+ // Fire-and-forget graph rebuild — must NOT await, must NOT crash slice completion.
358
+ // Dynamic import of the package name (not a relative path) so it resolves
359
+ // correctly via package.json#exports in both development and production.
360
+ // eslint-disable-next-line @typescript-eslint/no-floating-promises
361
+ (async () => {
362
+ try {
363
+ const graphMod = await import("@gsd-build/mcp-server");
364
+ const g = await graphMod.buildGraph(basePath);
365
+ await graphMod.writeGraph(graphMod.resolveGsdRoot(basePath), g);
366
+ }
367
+ catch (graphErr) {
368
+ // Graph rebuild is best-effort — log at warning level but never propagate
369
+ logWarning("tool", `complete-slice graph rebuild failed (non-fatal): ${graphErr.message ?? String(graphErr)}`);
370
+ }
371
+ })();
357
372
  return {
358
373
  sliceId: params.sliceId,
359
374
  milestoneId: params.milestoneId,
@@ -1,7 +1,7 @@
1
1
  import { ensureDbOpen } from "../bootstrap/dynamic-tools.js";
2
2
  import { sanitizeCompleteMilestoneParams } from "../bootstrap/sanitize-complete-milestone.js";
3
3
  import { loadWriteGateSnapshot, shouldBlockContextArtifactSaveInSnapshot } from "../bootstrap/write-gate.js";
4
- import { getMilestone, getSliceStatusSummary, getSliceTaskCounts, _getAdapter, saveGateResult, } from "../gsd-db.js";
4
+ import { getMilestone, getSliceStatusSummary, getSliceTaskCounts, readTransaction, saveGateResult, } from "../gsd-db.js";
5
5
  import { GATE_REGISTRY } from "../gate-registry.js";
6
6
  import { saveArtifactToDb } from "../db-writer.js";
7
7
  import { handleCompleteMilestone } from "./complete-milestone.js";
@@ -493,12 +493,9 @@ export async function executeMilestoneStatus(params, basePath = process.cwd()) {
493
493
  isError: true,
494
494
  };
495
495
  }
496
- const adapter = _getAdapter();
497
- adapter.exec("BEGIN");
498
- try {
496
+ return readTransaction(() => {
499
497
  const milestone = getMilestone(params.milestoneId);
500
498
  if (!milestone) {
501
- adapter.exec("COMMIT");
502
499
  return {
503
500
  content: [{ type: "text", text: `Milestone ${params.milestoneId} not found in database.` }],
504
501
  details: { operation: "milestone_status", milestoneId: params.milestoneId, found: false },
@@ -510,7 +507,6 @@ export async function executeMilestoneStatus(params, basePath = process.cwd()) {
510
507
  status: s.status,
511
508
  taskCounts: getSliceTaskCounts(params.milestoneId, s.id),
512
509
  }));
513
- adapter.exec("COMMIT");
514
510
  const result = {
515
511
  milestoneId: milestone.id,
516
512
  title: milestone.title,
@@ -524,14 +520,7 @@ export async function executeMilestoneStatus(params, basePath = process.cwd()) {
524
520
  content: [{ type: "text", text: JSON.stringify(result, null, 2) }],
525
521
  details: { operation: "milestone_status", milestoneId: milestone.id, sliceCount: slices.length },
526
522
  };
527
- }
528
- catch (txErr) {
529
- try {
530
- adapter.exec("ROLLBACK");
531
- }
532
- catch { /* swallow */ }
533
- throw txErr;
534
- }
523
+ });
535
524
  }
536
525
  catch (err) {
537
526
  const msg = err instanceof Error ? err.message : String(err);
@@ -83,12 +83,9 @@ export function executeReplan(basePath, mid, sid, capture) {
83
83
  // Also write replan_triggered_at column for DB-backed detection
84
84
  try {
85
85
  const req = createRequire(import.meta.url);
86
- const { isDbAvailable, _getAdapter } = req("./gsd-db.js");
86
+ const { isDbAvailable, setSliceReplanTriggeredAt } = req("./gsd-db.js");
87
87
  if (isDbAvailable()) {
88
- const adapter = _getAdapter();
89
- if (adapter) {
90
- adapter.prepare("UPDATE slices SET replan_triggered_at = :ts WHERE milestone_id = :mid AND id = :sid").run({ ":ts": ts, ":mid": mid, ":sid": sid });
91
- }
88
+ setSliceReplanTriggeredAt(mid, sid, ts);
92
89
  }
93
90
  }
94
91
  catch {
@@ -1,4 +1,4 @@
1
- import { _getAdapter, transaction, } from "./gsd-db.js";
1
+ import { _getAdapter, readTransaction, restoreManifest, } from "./gsd-db.js";
2
2
  import { atomicWriteSync } from "./atomic-write.js";
3
3
  import { readFileSync, existsSync, mkdirSync } from "node:fs";
4
4
  import { join } from "node:path";
@@ -41,8 +41,7 @@ export function snapshotState() {
41
41
  const db = requireDb();
42
42
  // Wrap all reads in a deferred transaction so the snapshot is consistent
43
43
  // (all SELECTs see the same DB state even if a concurrent write lands between them).
44
- db.exec("BEGIN DEFERRED");
45
- try {
44
+ return readTransaction(() => {
46
45
  const rawMilestones = db.prepare("SELECT * FROM milestones ORDER BY id").all();
47
46
  const milestones = rawMilestones.map((r) => ({
48
47
  id: r["id"],
@@ -146,74 +145,14 @@ export function snapshotState() {
146
145
  decisions,
147
146
  verification_evidence,
148
147
  };
149
- db.exec("COMMIT");
150
148
  return result;
151
- }
152
- catch (err) {
153
- try {
154
- db.exec("ROLLBACK");
155
- }
156
- catch { /* ignore rollback failure */ }
157
- throw err;
158
- }
159
- }
160
- // ─── restore ─────────────────────────────────────────────────────────────
161
- /**
162
- * Atomically replace all workflow state from a manifest.
163
- * Runs inside a transaction — if any insert fails, no tables are modified.
164
- * Only touches engine tables + decisions. Does NOT modify artifacts or memories.
165
- */
166
- function restore(manifest) {
167
- const db = requireDb();
168
- transaction(() => {
169
- // Clear engine tables (order matters for foreign-key-like consistency)
170
- db.exec("DELETE FROM verification_evidence");
171
- db.exec("DELETE FROM tasks");
172
- db.exec("DELETE FROM slices");
173
- db.exec("DELETE FROM milestones");
174
- db.exec("DELETE FROM decisions WHERE 1=1");
175
- // Restore milestones
176
- const msStmt = db.prepare(`INSERT INTO milestones (id, title, status, depends_on, created_at, completed_at,
177
- vision, success_criteria, key_risks, proof_strategy,
178
- verification_contract, verification_integration, verification_operational, verification_uat,
179
- definition_of_done, requirement_coverage, boundary_map_markdown)
180
- VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`);
181
- for (const m of manifest.milestones) {
182
- msStmt.run(m.id, m.title, m.status, JSON.stringify(m.depends_on), m.created_at, m.completed_at, m.vision, JSON.stringify(m.success_criteria), JSON.stringify(m.key_risks), JSON.stringify(m.proof_strategy), m.verification_contract, m.verification_integration, m.verification_operational, m.verification_uat, JSON.stringify(m.definition_of_done), m.requirement_coverage, m.boundary_map_markdown);
183
- }
184
- // Restore slices
185
- const slStmt = db.prepare(`INSERT INTO slices (milestone_id, id, title, status, risk, depends, demo,
186
- created_at, completed_at, full_summary_md, full_uat_md,
187
- goal, success_criteria, proof_level, integration_closure, observability_impact,
188
- sequence, replan_triggered_at)
189
- VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`);
190
- for (const s of manifest.slices) {
191
- slStmt.run(s.milestone_id, s.id, s.title, s.status, s.risk, JSON.stringify(s.depends), s.demo, s.created_at, s.completed_at, s.full_summary_md, s.full_uat_md, s.goal, s.success_criteria, s.proof_level, s.integration_closure, s.observability_impact, s.sequence, s.replan_triggered_at);
192
- }
193
- // Restore tasks
194
- const tkStmt = db.prepare(`INSERT INTO tasks (milestone_id, slice_id, id, title, status,
195
- one_liner, narrative, verification_result, duration, completed_at,
196
- blocker_discovered, deviations, known_issues, key_files, key_decisions,
197
- full_summary_md, description, estimate, files, verify,
198
- inputs, expected_output, observability_impact, sequence)
199
- VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`);
200
- for (const t of manifest.tasks) {
201
- tkStmt.run(t.milestone_id, t.slice_id, t.id, t.title, t.status, t.one_liner, t.narrative, t.verification_result, t.duration, t.completed_at, t.blocker_discovered ? 1 : 0, t.deviations, t.known_issues, JSON.stringify(t.key_files), JSON.stringify(t.key_decisions), t.full_summary_md, t.description, t.estimate, JSON.stringify(t.files), t.verify, JSON.stringify(t.inputs), JSON.stringify(t.expected_output), t.observability_impact, t.sequence);
202
- }
203
- // Restore decisions
204
- const dcStmt = db.prepare(`INSERT INTO decisions (seq, id, when_context, scope, decision, choice, rationale, revisable, made_by, superseded_by)
205
- VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`);
206
- for (const d of manifest.decisions) {
207
- dcStmt.run(d.seq, d.id, d.when_context, d.scope, d.decision, d.choice, d.rationale, d.revisable, d.made_by, d.superseded_by);
208
- }
209
- // Restore verification evidence
210
- const evStmt = db.prepare(`INSERT INTO verification_evidence (task_id, slice_id, milestone_id, command, exit_code, verdict, duration_ms, created_at)
211
- VALUES (?, ?, ?, ?, ?, ?, ?, ?)`);
212
- for (const e of manifest.verification_evidence) {
213
- evStmt.run(e.task_id, e.slice_id, e.milestone_id, e.command, e.exit_code, e.verdict, e.duration_ms, e.created_at);
214
- }
215
149
  });
216
150
  }
151
+ // ─── restore ─────────────────────────────────────────────────────────────
152
+ //
153
+ // The actual restore() implementation lives in gsd-db.ts (single-writer
154
+ // invariant). This module only orchestrates reading the manifest file
155
+ // and handing it to the writer.
217
156
  // ─── writeManifest ───────────────────────────────────────────────────────
218
157
  /**
219
158
  * Write current DB state to .gsd/state-manifest.json via atomicWriteSync.
@@ -258,6 +197,6 @@ export function bootstrapFromManifest(basePath) {
258
197
  if (!manifest) {
259
198
  return false;
260
199
  }
261
- restore(manifest);
200
+ restoreManifest(manifest);
262
201
  return true;
263
202
  }
@@ -4,7 +4,7 @@
4
4
  // Populates data into the already-existing v10 schema tables.
5
5
  import { existsSync, readdirSync, readFileSync } from "node:fs";
6
6
  import { join } from "node:path";
7
- import { _getAdapter, transaction } from "./gsd-db.js";
7
+ import { _getAdapter, bulkInsertLegacyHierarchy } from "./gsd-db.js";
8
8
  import { parseRoadmap, parsePlan } from "./parsers-legacy.js";
9
9
  import { logWarning } from "./workflow-logger.js";
10
10
  // ─── needsAutoMigration ───────────────────────────────────────────────────
@@ -174,27 +174,26 @@ export function migrateFromMarkdown(basePath) {
174
174
  process.stderr.write("workflow-migration: no milestones collected, nothing to insert\n");
175
175
  return;
176
176
  }
177
- const placeholders = migratedMilestoneIds.map(() => "?").join(",");
178
- transaction(() => {
179
- // Clear existing data to handle stale DB shape (DELETE ... IN (...))
180
- db.prepare(`DELETE FROM tasks WHERE milestone_id IN (${placeholders})`).run(...migratedMilestoneIds);
181
- db.prepare(`DELETE FROM slices WHERE milestone_id IN (${placeholders})`).run(...migratedMilestoneIds);
182
- db.prepare(`DELETE FROM milestones WHERE id IN (${placeholders})`).run(...migratedMilestoneIds);
183
- // Insert milestones
184
- const insertMilestone = db.prepare("INSERT INTO milestones (id, title, status, created_at) VALUES (?, ?, ?, ?)");
185
- for (const m of milestoneInserts) {
186
- insertMilestone.run(m.id, m.title, m.status, now);
187
- }
188
- // Insert slices (using v10 column names: depends, sequence)
189
- const insertSlice = db.prepare("INSERT INTO slices (id, milestone_id, title, status, risk, depends, sequence, created_at) VALUES (?, ?, ?, ?, ?, ?, ?, ?)");
190
- for (const s of sliceInserts) {
191
- insertSlice.run(s.id, s.milestoneId, s.title, s.status, s.risk, "[]", s.sequence, now);
192
- }
193
- // Insert tasks (using v10 column names: sequence, blocker_discovered, full_summary_md)
194
- const insertTask = db.prepare("INSERT INTO tasks (id, slice_id, milestone_id, title, description, status, estimate, files, sequence) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)");
195
- for (const t of taskInserts) {
196
- insertTask.run(t.id, t.sliceId, t.milestoneId, t.title, "", t.status, "", "[]", t.sequence);
197
- }
177
+ bulkInsertLegacyHierarchy({
178
+ milestones: milestoneInserts,
179
+ slices: sliceInserts.map(s => ({
180
+ id: s.id,
181
+ milestoneId: s.milestoneId,
182
+ title: s.title,
183
+ status: s.status,
184
+ risk: s.risk,
185
+ sequence: s.sequence,
186
+ })),
187
+ tasks: taskInserts.map(t => ({
188
+ id: t.id,
189
+ sliceId: t.sliceId,
190
+ milestoneId: t.milestoneId,
191
+ title: t.title,
192
+ status: t.status,
193
+ sequence: t.sequence,
194
+ })),
195
+ clearMilestoneIds: migratedMilestoneIds,
196
+ createdAt: now,
198
197
  });
199
198
  }
200
199
  // ─── validateMigration ────────────────────────────────────────────────────
@@ -303,7 +303,10 @@ export async function renderStateProjection(basePath) {
303
303
  try {
304
304
  adapter.prepare("SELECT 1").get();
305
305
  }
306
- catch {
306
+ catch (err) {
307
+ logWarning("projection", "renderStateProjection: DB handle probe failed, skipping render", {
308
+ error: err.message,
309
+ });
307
310
  return;
308
311
  }
309
312
  const state = await deriveState(basePath);
@@ -2,7 +2,7 @@ import { join } from "node:path";
2
2
  import { mkdirSync, existsSync, readFileSync, unlinkSync } from "node:fs";
3
3
  import { logWarning, logError } from "./workflow-logger.js";
4
4
  import { readEvents, findForkPoint, getSessionId } from "./workflow-events.js";
5
- import { transaction, updateTaskStatus, updateSliceStatus, updateMilestoneStatus, getSliceTasks, insertMilestone, _getAdapter, getMilestoneSlices, insertVerificationEvidence, upsertDecision, openDatabase, setTaskBlockerDiscovered, } from "./gsd-db.js";
5
+ import { transaction, updateTaskStatus, updateSliceStatus, updateMilestoneStatus, getSliceTasks, insertMilestone, getMilestoneSlices, insertVerificationEvidence, upsertDecision, openDatabase, setTaskBlockerDiscovered, insertOrIgnoreSlice, insertOrIgnoreTask, } from "./gsd-db.js";
6
6
  import { isClosedStatus } from "./status-guards.js";
7
7
  import { invalidateStateCache } from "./state.js";
8
8
  import { clearPathCache } from "./paths.js";
@@ -132,11 +132,12 @@ function replayEvents(events) {
132
132
  const milestoneId = p["milestoneId"];
133
133
  const sliceId = p["sliceId"];
134
134
  if (milestoneId && sliceId) {
135
- const adapter = _getAdapter();
136
- if (adapter) {
137
- adapter.prepare(`INSERT OR IGNORE INTO slices (milestone_id, id, title, status, created_at)
138
- VALUES (:mid, :sid, :title, 'pending', :ts)`).run({ ":mid": milestoneId, ":sid": sliceId, ":title": p["title"] ?? sliceId, ":ts": event.ts });
139
- }
135
+ insertOrIgnoreSlice({
136
+ milestoneId,
137
+ sliceId,
138
+ title: p["title"] ?? sliceId,
139
+ createdAt: event.ts,
140
+ });
140
141
  }
141
142
  break;
142
143
  }
@@ -148,11 +149,13 @@ function replayEvents(events) {
148
149
  const sliceId = p["sliceId"];
149
150
  const taskId = p["taskId"];
150
151
  if (milestoneId && sliceId && taskId) {
151
- const adapter = _getAdapter();
152
- if (adapter) {
153
- adapter.prepare(`INSERT OR IGNORE INTO tasks (milestone_id, slice_id, id, title, status, created_at)
154
- VALUES (:mid, :sid, :tid, :title, 'pending', :ts)`).run({ ":mid": milestoneId, ":sid": sliceId, ":tid": taskId, ":title": p["title"] ?? taskId, ":ts": event.ts });
155
- }
152
+ insertOrIgnoreTask({
153
+ milestoneId,
154
+ sliceId,
155
+ taskId,
156
+ title: p["title"] ?? taskId,
157
+ createdAt: event.ts,
158
+ });
156
159
  }
157
160
  break;
158
161
  }