gsd-pi 2.73.1 → 2.74.0-dev.0306a2e

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 (640) 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/activity-log.js +16 -0
  12. package/dist/resources/extensions/gsd/auto/detect-stuck.js +11 -4
  13. package/dist/resources/extensions/gsd/auto/loop.js +147 -10
  14. package/dist/resources/extensions/gsd/auto/phases.js +173 -13
  15. package/dist/resources/extensions/gsd/auto/session.js +10 -0
  16. package/dist/resources/extensions/gsd/auto-dispatch.js +22 -4
  17. package/dist/resources/extensions/gsd/auto-model-selection.js +105 -16
  18. package/dist/resources/extensions/gsd/auto-post-unit.js +254 -15
  19. package/dist/resources/extensions/gsd/auto-prompts.js +12 -0
  20. package/dist/resources/extensions/gsd/auto-start.js +23 -6
  21. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +13 -0
  22. package/dist/resources/extensions/gsd/auto-unit-closeout.js +18 -0
  23. package/dist/resources/extensions/gsd/auto-verification.js +186 -3
  24. package/dist/resources/extensions/gsd/auto.js +65 -12
  25. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +30 -8
  26. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +41 -2
  27. package/dist/resources/extensions/gsd/commands/catalog.js +26 -1
  28. package/dist/resources/extensions/gsd/commands/handlers/ops.js +25 -0
  29. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +68 -9
  30. package/dist/resources/extensions/gsd/commands-add-tests.js +111 -0
  31. package/dist/resources/extensions/gsd/commands-backlog.js +140 -0
  32. package/dist/resources/extensions/gsd/commands-do.js +79 -0
  33. package/dist/resources/extensions/gsd/commands-extract-learnings.js +225 -0
  34. package/dist/resources/extensions/gsd/commands-handlers.js +8 -2
  35. package/dist/resources/extensions/gsd/commands-maintenance.js +6 -6
  36. package/dist/resources/extensions/gsd/commands-pr-branch.js +180 -0
  37. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +1 -1
  38. package/dist/resources/extensions/gsd/commands-session-report.js +82 -0
  39. package/dist/resources/extensions/gsd/commands-ship.js +187 -0
  40. package/dist/resources/extensions/gsd/db-writer.js +3 -5
  41. package/dist/resources/extensions/gsd/docs/preferences-reference.md +15 -2
  42. package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +144 -0
  43. package/dist/resources/extensions/gsd/ecosystem/loader.js +145 -0
  44. package/dist/resources/extensions/gsd/git-service.js +49 -1
  45. package/dist/resources/extensions/gsd/graph-context.js +157 -0
  46. package/dist/resources/extensions/gsd/gsd-db.js +581 -2
  47. package/dist/resources/extensions/gsd/guided-flow.js +23 -0
  48. package/dist/resources/extensions/gsd/index.js +15 -2
  49. package/dist/resources/extensions/gsd/init-wizard.js +1 -0
  50. package/dist/resources/extensions/gsd/journal.js +27 -0
  51. package/dist/resources/extensions/gsd/md-importer.js +3 -4
  52. package/dist/resources/extensions/gsd/memory-store.js +19 -51
  53. package/dist/resources/extensions/gsd/metrics.js +19 -0
  54. package/dist/resources/extensions/gsd/milestone-validation-gates.js +13 -12
  55. package/dist/resources/extensions/gsd/native-git-bridge.js +7 -4
  56. package/dist/resources/extensions/gsd/notification-widget.js +2 -2
  57. package/dist/resources/extensions/gsd/parallel-orchestrator.js +33 -1
  58. package/dist/resources/extensions/gsd/preferences-models.js +63 -3
  59. package/dist/resources/extensions/gsd/preferences-types.js +2 -0
  60. package/dist/resources/extensions/gsd/preferences-validation.js +130 -2
  61. package/dist/resources/extensions/gsd/preferences.js +26 -0
  62. package/dist/resources/extensions/gsd/prompts/add-tests.md +35 -0
  63. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +12 -2
  64. package/dist/resources/extensions/gsd/state.js +66 -15
  65. package/dist/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  66. package/dist/resources/extensions/gsd/tools/complete-slice.js +20 -0
  67. package/dist/resources/extensions/gsd/tools/validate-milestone.js +39 -4
  68. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +3 -14
  69. package/dist/resources/extensions/gsd/triage-resolution.js +2 -5
  70. package/dist/resources/extensions/gsd/unit-ownership.js +1 -1
  71. package/dist/resources/extensions/gsd/uok/audit-toggle.js +7 -0
  72. package/dist/resources/extensions/gsd/uok/audit.js +40 -0
  73. package/dist/resources/extensions/gsd/uok/contracts.js +1 -0
  74. package/dist/resources/extensions/gsd/uok/execution-graph.js +179 -0
  75. package/dist/resources/extensions/gsd/uok/flags.js +29 -0
  76. package/dist/resources/extensions/gsd/uok/gate-runner.js +109 -0
  77. package/dist/resources/extensions/gsd/uok/gitops.js +53 -0
  78. package/dist/resources/extensions/gsd/uok/kernel.js +80 -0
  79. package/dist/resources/extensions/gsd/uok/loop-adapter.js +133 -0
  80. package/dist/resources/extensions/gsd/uok/model-policy.js +66 -0
  81. package/dist/resources/extensions/gsd/uok/plan-v2.js +132 -0
  82. package/dist/resources/extensions/gsd/workflow-logger.js +22 -0
  83. package/dist/resources/extensions/gsd/workflow-manifest.js +8 -69
  84. package/dist/resources/extensions/gsd/workflow-migration.js +21 -22
  85. package/dist/resources/extensions/gsd/workflow-projections.js +4 -1
  86. package/dist/resources/extensions/gsd/workflow-reconcile.js +14 -11
  87. package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
  88. package/dist/tsconfig.extensions.tsbuildinfo +1 -0
  89. package/dist/update-check.d.ts +1 -0
  90. package/dist/update-check.js +13 -5
  91. package/dist/update-cmd.js +4 -3
  92. package/dist/web/standalone/.next/BUILD_ID +1 -1
  93. package/dist/web/standalone/.next/app-path-routes-manifest.json +10 -10
  94. package/dist/web/standalone/.next/build-manifest.json +3 -3
  95. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  96. package/dist/web/standalone/.next/required-server-files.json +3 -3
  97. package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
  98. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  99. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  100. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  102. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  103. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  104. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  105. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  106. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  107. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  108. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  109. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  110. package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
  111. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
  112. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  113. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +3 -3
  114. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  115. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  116. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  117. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  118. package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
  119. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  120. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
  121. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  122. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
  123. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  124. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
  125. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  126. package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
  127. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  128. package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
  129. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  130. package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
  131. package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
  132. package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
  133. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  134. package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
  135. package/dist/web/standalone/.next/server/app/api/experimental/route.js +2 -2
  136. package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -1
  137. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  138. package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
  139. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  140. package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
  141. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  142. package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
  143. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  144. package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
  145. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  146. package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
  147. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  148. package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
  149. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  150. package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
  151. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  152. package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
  153. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  154. package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
  155. package/dist/web/standalone/.next/server/app/api/notifications/route.js +2 -2
  156. package/dist/web/standalone/.next/server/app/api/notifications/route_client-reference-manifest.js +1 -1
  157. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  158. package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
  159. package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
  160. package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
  161. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  162. package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  163. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  164. package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
  165. package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +2 -2
  166. package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
  167. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  168. package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
  169. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  170. package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
  171. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  172. package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
  173. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  174. package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
  175. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  176. package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
  177. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  178. package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
  179. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  180. package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
  181. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  182. package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
  183. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  184. package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
  185. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +2 -2
  186. package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
  187. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
  188. package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
  189. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
  190. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
  191. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +4 -4
  192. package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
  193. package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js +1 -1
  194. package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
  195. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  196. package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
  197. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  198. package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  199. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  200. package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
  201. package/dist/web/standalone/.next/server/app/index.html +1 -1
  202. package/dist/web/standalone/.next/server/app/index.rsc +4 -4
  203. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  204. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
  205. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  206. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +3 -3
  207. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  208. package/dist/web/standalone/.next/server/app/page.js +2 -2
  209. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  210. package/dist/web/standalone/.next/server/app-paths-manifest.json +10 -10
  211. package/dist/web/standalone/.next/server/chunks/63.js +3 -3
  212. package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
  213. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  214. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  215. package/dist/web/standalone/.next/server/middleware.js +2 -2
  216. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  217. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  218. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  219. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  220. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  221. package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-2f24283c162b6ab3.js → page-f2a7482d42a5614b.js} +1 -1
  222. package/dist/web/standalone/.next/static/chunks/app/{layout-9ecfd95f343793f0.js → layout-a16c7a7ecdf0c2cf.js} +1 -1
  223. package/dist/web/standalone/.next/static/chunks/app/page-f1e30ab6bb269149.js +1 -0
  224. package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +1 -0
  225. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +1 -0
  226. package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
  227. package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
  228. package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
  229. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
  230. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
  231. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
  232. package/dist/web/standalone/server.js +1 -1
  233. package/package.json +3 -3
  234. package/packages/daemon/package.json +2 -2
  235. package/packages/mcp-server/dist/index.d.ts +3 -0
  236. package/packages/mcp-server/dist/index.d.ts.map +1 -1
  237. package/packages/mcp-server/dist/index.js +3 -0
  238. package/packages/mcp-server/dist/index.js.map +1 -1
  239. package/packages/mcp-server/dist/readers/graph.d.ts +87 -0
  240. package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -0
  241. package/packages/mcp-server/dist/readers/graph.js +655 -0
  242. package/packages/mcp-server/dist/readers/graph.js.map +1 -0
  243. package/packages/mcp-server/dist/readers/index.d.ts +2 -0
  244. package/packages/mcp-server/dist/readers/index.d.ts.map +1 -1
  245. package/packages/mcp-server/dist/readers/index.js +1 -0
  246. package/packages/mcp-server/dist/readers/index.js.map +1 -1
  247. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  248. package/packages/mcp-server/dist/server.js +65 -0
  249. package/packages/mcp-server/dist/server.js.map +1 -1
  250. package/packages/mcp-server/package.json +2 -2
  251. package/packages/mcp-server/src/index.ts +15 -0
  252. package/packages/mcp-server/src/readers/graph.test.ts +604 -0
  253. package/packages/mcp-server/src/readers/graph.ts +855 -0
  254. package/packages/mcp-server/src/readers/index.ts +12 -0
  255. package/packages/mcp-server/src/server.ts +83 -0
  256. package/packages/mcp-server/tsconfig.json +1 -0
  257. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -0
  258. package/packages/native/package.json +2 -2
  259. package/packages/native/tsconfig.tsbuildinfo +1 -0
  260. package/packages/pi-agent-core/package.json +1 -1
  261. package/packages/pi-agent-core/tsconfig.json +1 -0
  262. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -0
  263. package/packages/pi-ai/dist/index.d.ts +2 -9
  264. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  265. package/packages/pi-ai/dist/index.js +2 -9
  266. package/packages/pi-ai/dist/index.js.map +1 -1
  267. package/packages/pi-ai/dist/models/capability-patches.d.ts +19 -0
  268. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -0
  269. package/packages/pi-ai/dist/models/capability-patches.js +36 -0
  270. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -0
  271. package/packages/pi-ai/dist/{models.custom.d.ts → models/custom.d.ts} +1 -1
  272. package/packages/pi-ai/dist/models/custom.d.ts.map +1 -0
  273. package/packages/pi-ai/dist/{models.custom.js → models/custom.js} +4 -4
  274. package/packages/pi-ai/dist/models/custom.js.map +1 -0
  275. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts +1482 -0
  276. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts.map +1 -0
  277. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js +1484 -0
  278. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js.map +1 -0
  279. package/packages/pi-ai/dist/models/generated/anthropic.d.ts +377 -0
  280. package/packages/pi-ai/dist/models/generated/anthropic.d.ts.map +1 -0
  281. package/packages/pi-ai/dist/models/generated/anthropic.js +379 -0
  282. package/packages/pi-ai/dist/models/generated/anthropic.js.map +1 -0
  283. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts +700 -0
  284. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts.map +1 -0
  285. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js +702 -0
  286. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js.map +1 -0
  287. package/packages/pi-ai/dist/models/generated/cerebras.d.ts +71 -0
  288. package/packages/pi-ai/dist/models/generated/cerebras.d.ts.map +1 -0
  289. package/packages/pi-ai/dist/models/generated/cerebras.js +73 -0
  290. package/packages/pi-ai/dist/models/generated/cerebras.js.map +1 -0
  291. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts +590 -0
  292. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts.map +1 -0
  293. package/packages/pi-ai/dist/models/generated/github-copilot.js +444 -0
  294. package/packages/pi-ai/dist/models/generated/github-copilot.js.map +1 -0
  295. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts +156 -0
  296. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts.map +1 -0
  297. package/packages/pi-ai/dist/models/generated/google-antigravity.js +158 -0
  298. package/packages/pi-ai/dist/models/generated/google-antigravity.js.map +1 -0
  299. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts +105 -0
  300. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts.map +1 -0
  301. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js +107 -0
  302. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js.map +1 -0
  303. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts +207 -0
  304. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts.map +1 -0
  305. package/packages/pi-ai/dist/models/generated/google-vertex.js +209 -0
  306. package/packages/pi-ai/dist/models/generated/google-vertex.js.map +1 -0
  307. package/packages/pi-ai/dist/models/generated/google.d.ts +462 -0
  308. package/packages/pi-ai/dist/models/generated/google.d.ts.map +1 -0
  309. package/packages/pi-ai/dist/models/generated/google.js +464 -0
  310. package/packages/pi-ai/dist/models/generated/google.js.map +1 -0
  311. package/packages/pi-ai/dist/models/generated/groq.d.ts +309 -0
  312. package/packages/pi-ai/dist/models/generated/groq.d.ts.map +1 -0
  313. package/packages/pi-ai/dist/models/generated/groq.js +311 -0
  314. package/packages/pi-ai/dist/models/generated/groq.js.map +1 -0
  315. package/packages/pi-ai/dist/models/generated/huggingface.d.ts +383 -0
  316. package/packages/pi-ai/dist/models/generated/huggingface.d.ts.map +1 -0
  317. package/packages/pi-ai/dist/models/generated/huggingface.js +347 -0
  318. package/packages/pi-ai/dist/models/generated/huggingface.js.map +1 -0
  319. package/packages/pi-ai/dist/{models.generated.d.ts → models/generated/index.d.ts} +1 -1
  320. package/packages/pi-ai/dist/{models.generated.d.ts.map → models/generated/index.d.ts.map} +1 -1
  321. package/packages/pi-ai/dist/models/generated/index.js +51 -0
  322. package/packages/pi-ai/dist/models/generated/index.js.map +1 -0
  323. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts +37 -0
  324. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts.map +1 -0
  325. package/packages/pi-ai/dist/models/generated/kimi-coding.js +39 -0
  326. package/packages/pi-ai/dist/models/generated/kimi-coding.js.map +1 -0
  327. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts +105 -0
  328. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts.map +1 -0
  329. package/packages/pi-ai/dist/models/generated/minimax-cn.js +107 -0
  330. package/packages/pi-ai/dist/models/generated/minimax-cn.js.map +1 -0
  331. package/packages/pi-ai/dist/models/generated/minimax.d.ts +105 -0
  332. package/packages/pi-ai/dist/models/generated/minimax.d.ts.map +1 -0
  333. package/packages/pi-ai/dist/models/generated/minimax.js +107 -0
  334. package/packages/pi-ai/dist/models/generated/minimax.js.map +1 -0
  335. package/packages/pi-ai/dist/models/generated/mistral.d.ts +445 -0
  336. package/packages/pi-ai/dist/models/generated/mistral.d.ts.map +1 -0
  337. package/packages/pi-ai/dist/models/generated/mistral.js +447 -0
  338. package/packages/pi-ai/dist/models/generated/mistral.js.map +1 -0
  339. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +139 -0
  340. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -0
  341. package/packages/pi-ai/dist/models/generated/openai-codex.js +141 -0
  342. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -0
  343. package/packages/pi-ai/dist/models/generated/openai.d.ts +700 -0
  344. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -0
  345. package/packages/pi-ai/dist/models/generated/openai.js +702 -0
  346. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -0
  347. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts +122 -0
  348. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts.map +1 -0
  349. package/packages/pi-ai/dist/models/generated/opencode-go.js +124 -0
  350. package/packages/pi-ai/dist/models/generated/opencode-go.js.map +1 -0
  351. package/packages/pi-ai/dist/models/generated/opencode.d.ts +530 -0
  352. package/packages/pi-ai/dist/models/generated/opencode.d.ts.map +1 -0
  353. package/packages/pi-ai/dist/models/generated/opencode.js +532 -0
  354. package/packages/pi-ai/dist/models/generated/opencode.js.map +1 -0
  355. package/packages/pi-ai/dist/models/generated/openrouter.d.ts +4270 -0
  356. package/packages/pi-ai/dist/models/generated/openrouter.d.ts.map +1 -0
  357. package/packages/pi-ai/dist/models/generated/openrouter.js +4272 -0
  358. package/packages/pi-ai/dist/models/generated/openrouter.js.map +1 -0
  359. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts +2604 -0
  360. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts.map +1 -0
  361. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js +2606 -0
  362. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js.map +1 -0
  363. package/packages/pi-ai/dist/models/generated/xai.d.ts +411 -0
  364. package/packages/pi-ai/dist/models/generated/xai.d.ts.map +1 -0
  365. package/packages/pi-ai/dist/models/generated/xai.js +413 -0
  366. package/packages/pi-ai/dist/models/generated/xai.js.map +1 -0
  367. package/packages/pi-ai/dist/models/generated/zai.d.ts +276 -0
  368. package/packages/pi-ai/dist/models/generated/zai.d.ts.map +1 -0
  369. package/packages/pi-ai/dist/models/generated/zai.js +239 -0
  370. package/packages/pi-ai/dist/models/generated/zai.js.map +1 -0
  371. package/packages/pi-ai/dist/models/index.d.ts +27 -0
  372. package/packages/pi-ai/dist/models/index.d.ts.map +1 -0
  373. package/packages/pi-ai/dist/models/index.js +80 -0
  374. package/packages/pi-ai/dist/models/index.js.map +1 -0
  375. package/packages/pi-ai/dist/models.d.ts +1 -36
  376. package/packages/pi-ai/dist/models.d.ts.map +1 -1
  377. package/packages/pi-ai/dist/models.generated.test.js +1 -2
  378. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  379. package/packages/pi-ai/dist/models.js +3 -112
  380. package/packages/pi-ai/dist/models.js.map +1 -1
  381. package/packages/pi-ai/dist/models.test.js +6 -5
  382. package/packages/pi-ai/dist/models.test.js.map +1 -1
  383. package/packages/pi-ai/dist/utils/overflow.d.ts.map +1 -1
  384. package/packages/pi-ai/dist/utils/overflow.js +12 -0
  385. package/packages/pi-ai/dist/utils/overflow.js.map +1 -1
  386. package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts +2 -0
  387. package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts.map +1 -0
  388. package/packages/pi-ai/dist/utils/tests/overflow.test.js +50 -0
  389. package/packages/pi-ai/dist/utils/tests/overflow.test.js.map +1 -0
  390. package/packages/pi-ai/package.json +1 -1
  391. package/packages/pi-ai/scripts/generate-models.ts +74 -40
  392. package/packages/pi-ai/src/index.ts +5 -9
  393. package/packages/pi-ai/src/models/capability-patches.ts +40 -0
  394. package/packages/pi-ai/src/{models.custom.ts → models/custom.ts} +4 -4
  395. package/packages/pi-ai/src/models/generated/amazon-bedrock.ts +1486 -0
  396. package/packages/pi-ai/src/models/generated/anthropic.ts +381 -0
  397. package/packages/pi-ai/src/models/generated/azure-openai-responses.ts +704 -0
  398. package/packages/pi-ai/src/models/generated/cerebras.ts +75 -0
  399. package/packages/pi-ai/src/models/generated/github-copilot.ts +446 -0
  400. package/packages/pi-ai/src/models/generated/google-antigravity.ts +160 -0
  401. package/packages/pi-ai/src/models/generated/google-gemini-cli.ts +109 -0
  402. package/packages/pi-ai/src/models/generated/google-vertex.ts +211 -0
  403. package/packages/pi-ai/src/models/generated/google.ts +466 -0
  404. package/packages/pi-ai/src/models/generated/groq.ts +313 -0
  405. package/packages/pi-ai/src/models/generated/huggingface.ts +349 -0
  406. package/packages/pi-ai/src/models/generated/index.ts +52 -0
  407. package/packages/pi-ai/src/models/generated/kimi-coding.ts +41 -0
  408. package/packages/pi-ai/src/models/generated/minimax-cn.ts +109 -0
  409. package/packages/pi-ai/src/models/generated/minimax.ts +109 -0
  410. package/packages/pi-ai/src/models/generated/mistral.ts +449 -0
  411. package/packages/pi-ai/src/models/generated/openai-codex.ts +143 -0
  412. package/packages/pi-ai/src/models/generated/openai.ts +704 -0
  413. package/packages/pi-ai/src/models/generated/opencode-go.ts +126 -0
  414. package/packages/pi-ai/src/models/generated/opencode.ts +534 -0
  415. package/packages/pi-ai/src/models/generated/openrouter.ts +4274 -0
  416. package/packages/pi-ai/src/models/generated/vercel-ai-gateway.ts +2608 -0
  417. package/packages/pi-ai/src/models/generated/xai.ts +415 -0
  418. package/packages/pi-ai/src/models/generated/zai.ts +241 -0
  419. package/packages/pi-ai/src/models/index.ts +106 -0
  420. package/packages/pi-ai/src/models.generated.test.ts +1 -2
  421. package/packages/pi-ai/src/models.test.ts +6 -5
  422. package/packages/pi-ai/src/models.ts +3 -153
  423. package/packages/pi-ai/src/utils/overflow.ts +14 -1
  424. package/packages/pi-ai/src/utils/tests/overflow.test.ts +58 -0
  425. package/packages/pi-ai/tsconfig.json +1 -0
  426. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -0
  427. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +721 -8
  428. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  429. package/packages/pi-coding-agent/dist/core/compaction/utils.js +5 -5
  430. package/packages/pi-coding-agent/dist/core/compaction/utils.js.map +1 -1
  431. package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts +2 -0
  432. package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts.map +1 -0
  433. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js +45 -0
  434. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js.map +1 -0
  435. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts +2 -0
  436. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts.map +1 -0
  437. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js +52 -0
  438. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js.map +1 -0
  439. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  440. package/packages/pi-coding-agent/dist/core/model-registry.js +2 -2
  441. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  442. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +12 -2
  443. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  444. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +65 -28
  445. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
  446. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts +2 -1
  447. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -1
  448. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js +9 -3
  449. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js.map +1 -1
  450. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts +2 -0
  451. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts.map +1 -0
  452. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +52 -0
  453. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -0
  454. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  455. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +305 -20
  456. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  457. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts +2 -0
  458. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts.map +1 -0
  459. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +38 -0
  460. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -0
  461. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +13 -0
  462. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  463. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +59 -6
  464. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  465. package/packages/pi-coding-agent/package.json +1 -1
  466. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +884 -8
  467. package/packages/pi-coding-agent/src/core/compaction/utils.ts +5 -5
  468. package/packages/pi-coding-agent/src/core/compaction-utils.test.ts +50 -0
  469. package/packages/pi-coding-agent/src/core/model-registry-env-fallback.test.ts +59 -0
  470. package/packages/pi-coding-agent/src/core/model-registry.ts +2 -1
  471. package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +78 -32
  472. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +73 -0
  473. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.ts +9 -3
  474. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +381 -39
  475. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +44 -0
  476. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +79 -6
  477. package/packages/pi-coding-agent/src/types/ambient-modules.d.ts +69 -0
  478. package/packages/pi-coding-agent/tsconfig.json +3 -2
  479. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -0
  480. package/packages/pi-tui/dist/__tests__/tui.test.js +60 -1
  481. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  482. package/packages/pi-tui/dist/tui.d.ts +8 -0
  483. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  484. package/packages/pi-tui/dist/tui.js +32 -3
  485. package/packages/pi-tui/dist/tui.js.map +1 -1
  486. package/packages/pi-tui/package.json +1 -1
  487. package/packages/pi-tui/src/__tests__/tui.test.ts +76 -1
  488. package/packages/pi-tui/src/tui.ts +31 -3
  489. package/packages/pi-tui/tsconfig.json +1 -0
  490. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -0
  491. package/packages/rpc-client/package.json +1 -1
  492. package/packages/rpc-client/tsconfig.json +1 -0
  493. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -0
  494. package/pkg/package.json +1 -1
  495. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +107 -5
  496. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +111 -2
  497. package/src/resources/extensions/gsd/activity-log.ts +21 -0
  498. package/src/resources/extensions/gsd/auto/detect-stuck.ts +12 -4
  499. package/src/resources/extensions/gsd/auto/loop-deps.ts +10 -0
  500. package/src/resources/extensions/gsd/auto/loop.ts +159 -10
  501. package/src/resources/extensions/gsd/auto/phases.ts +213 -13
  502. package/src/resources/extensions/gsd/auto/session.ts +10 -0
  503. package/src/resources/extensions/gsd/auto-dispatch.ts +26 -10
  504. package/src/resources/extensions/gsd/auto-model-selection.ts +151 -16
  505. package/src/resources/extensions/gsd/auto-post-unit.ts +278 -16
  506. package/src/resources/extensions/gsd/auto-prompts.ts +13 -0
  507. package/src/resources/extensions/gsd/auto-start.ts +30 -6
  508. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +17 -0
  509. package/src/resources/extensions/gsd/auto-unit-closeout.ts +25 -1
  510. package/src/resources/extensions/gsd/auto-verification.ts +225 -3
  511. package/src/resources/extensions/gsd/auto.ts +72 -16
  512. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +38 -8
  513. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +52 -2
  514. package/src/resources/extensions/gsd/commands/catalog.ts +26 -1
  515. package/src/resources/extensions/gsd/commands/handlers/ops.ts +25 -0
  516. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +74 -9
  517. package/src/resources/extensions/gsd/commands-add-tests.ts +137 -0
  518. package/src/resources/extensions/gsd/commands-backlog.ts +182 -0
  519. package/src/resources/extensions/gsd/commands-do.ts +109 -0
  520. package/src/resources/extensions/gsd/commands-extract-learnings.ts +304 -0
  521. package/src/resources/extensions/gsd/commands-handlers.ts +8 -2
  522. package/src/resources/extensions/gsd/commands-maintenance.ts +6 -6
  523. package/src/resources/extensions/gsd/commands-pr-branch.ts +234 -0
  524. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +1 -1
  525. package/src/resources/extensions/gsd/commands-session-report.ts +101 -0
  526. package/src/resources/extensions/gsd/commands-ship.ts +219 -0
  527. package/src/resources/extensions/gsd/db-writer.ts +3 -5
  528. package/src/resources/extensions/gsd/docs/preferences-reference.md +15 -2
  529. package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +228 -0
  530. package/src/resources/extensions/gsd/ecosystem/loader.ts +201 -0
  531. package/src/resources/extensions/gsd/git-service.ts +68 -0
  532. package/src/resources/extensions/gsd/graph-context.ts +212 -0
  533. package/src/resources/extensions/gsd/gsd-db.ts +788 -3
  534. package/src/resources/extensions/gsd/guided-flow.ts +32 -0
  535. package/src/resources/extensions/gsd/index.ts +18 -2
  536. package/src/resources/extensions/gsd/init-wizard.ts +3 -2
  537. package/src/resources/extensions/gsd/journal.ts +30 -0
  538. package/src/resources/extensions/gsd/md-importer.ts +3 -5
  539. package/src/resources/extensions/gsd/memory-store.ts +31 -62
  540. package/src/resources/extensions/gsd/metrics.ts +26 -0
  541. package/src/resources/extensions/gsd/milestone-validation-gates.ts +13 -14
  542. package/src/resources/extensions/gsd/native-git-bridge.ts +11 -12
  543. package/src/resources/extensions/gsd/notification-widget.ts +2 -2
  544. package/src/resources/extensions/gsd/parallel-orchestrator.ts +40 -1
  545. package/src/resources/extensions/gsd/preferences-models.ts +61 -3
  546. package/src/resources/extensions/gsd/preferences-types.ts +44 -0
  547. package/src/resources/extensions/gsd/preferences-validation.ts +130 -2
  548. package/src/resources/extensions/gsd/preferences.ts +28 -0
  549. package/src/resources/extensions/gsd/prompts/add-tests.md +35 -0
  550. package/src/resources/extensions/gsd/session-lock.ts +14 -2
  551. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +20 -1
  552. package/src/resources/extensions/gsd/state.ts +80 -17
  553. package/src/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  554. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +9 -5
  555. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +20 -0
  556. package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +53 -0
  557. package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +7 -3
  558. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +51 -2
  559. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +6 -2
  560. package/src/resources/extensions/gsd/tests/commands-backlog.test.ts +158 -0
  561. package/src/resources/extensions/gsd/tests/commands-do.test.ts +127 -0
  562. package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +340 -0
  563. package/src/resources/extensions/gsd/tests/commands-pr-branch.test.ts +68 -0
  564. package/src/resources/extensions/gsd/tests/commands-session-report.test.ts +82 -0
  565. package/src/resources/extensions/gsd/tests/commands-ship.test.ts +71 -0
  566. package/src/resources/extensions/gsd/tests/commands-workflow-custom.test.ts +14 -0
  567. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +142 -0
  568. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  569. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  570. package/src/resources/extensions/gsd/tests/completed-at-reconcile.test.ts +42 -0
  571. package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -2
  572. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +3 -2
  573. package/src/resources/extensions/gsd/tests/derive-state-helpers.test.ts +68 -8
  574. package/src/resources/extensions/gsd/tests/derive-state.test.ts +3 -3
  575. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +154 -0
  576. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +10 -7
  577. package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +137 -1
  578. package/src/resources/extensions/gsd/tests/graph-context.test.ts +337 -0
  579. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
  580. package/src/resources/extensions/gsd/tests/health-widget.test.ts +1 -1
  581. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +4 -2
  582. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +68 -1
  583. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -2
  584. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -3
  585. package/src/resources/extensions/gsd/tests/model-isolation.test.ts +91 -2
  586. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +140 -0
  587. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  588. package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +2 -1
  589. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +40 -1
  590. package/src/resources/extensions/gsd/tests/preferences.test.ts +47 -0
  591. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +1 -1
  592. package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +180 -0
  593. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +5 -7
  594. package/src/resources/extensions/gsd/tests/token-profile.test.ts +9 -6
  595. package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +101 -0
  596. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +85 -0
  597. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +69 -0
  598. package/src/resources/extensions/gsd/tests/uok-flags.test.ts +39 -0
  599. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +70 -0
  600. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +85 -0
  601. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +35 -0
  602. package/src/resources/extensions/gsd/tests/uok-model-policy.test.ts +89 -0
  603. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +167 -0
  604. package/src/resources/extensions/gsd/tests/uok-preferences.test.ts +42 -0
  605. package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +179 -0
  606. package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +39 -0
  607. package/src/resources/extensions/gsd/tests/workflow-logger-wiring.test.ts +223 -0
  608. package/src/resources/extensions/gsd/tools/complete-slice.ts +26 -0
  609. package/src/resources/extensions/gsd/tools/validate-milestone.ts +48 -3
  610. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +3 -11
  611. package/src/resources/extensions/gsd/triage-resolution.ts +2 -7
  612. package/src/resources/extensions/gsd/types.ts +14 -1
  613. package/src/resources/extensions/gsd/unit-ownership.ts +2 -2
  614. package/src/resources/extensions/gsd/uok/audit-toggle.ts +9 -0
  615. package/src/resources/extensions/gsd/uok/audit.ts +51 -0
  616. package/src/resources/extensions/gsd/uok/contracts.ts +135 -0
  617. package/src/resources/extensions/gsd/uok/execution-graph.ts +241 -0
  618. package/src/resources/extensions/gsd/uok/flags.ts +45 -0
  619. package/src/resources/extensions/gsd/uok/gate-runner.ts +146 -0
  620. package/src/resources/extensions/gsd/uok/gitops.ts +75 -0
  621. package/src/resources/extensions/gsd/uok/kernel.ts +105 -0
  622. package/src/resources/extensions/gsd/uok/loop-adapter.ts +162 -0
  623. package/src/resources/extensions/gsd/uok/model-policy.ts +112 -0
  624. package/src/resources/extensions/gsd/uok/plan-v2.ts +156 -0
  625. package/src/resources/extensions/gsd/workflow-logger.ts +27 -1
  626. package/src/resources/extensions/gsd/workflow-manifest.ts +9 -104
  627. package/src/resources/extensions/gsd/workflow-migration.ts +21 -29
  628. package/src/resources/extensions/gsd/workflow-projections.ts +8 -1
  629. package/src/resources/extensions/gsd/workflow-reconcile.ts +15 -15
  630. package/src/resources/extensions/ttsr/ttsr-manager.ts +10 -5
  631. package/dist/web/standalone/.next/static/chunks/app/page-7115e62689b5fd84.js +0 -1
  632. package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +0 -1
  633. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +0 -1
  634. package/packages/pi-ai/dist/models.custom.d.ts.map +0 -1
  635. package/packages/pi-ai/dist/models.custom.js.map +0 -1
  636. package/packages/pi-ai/dist/models.generated.js +0 -14343
  637. package/packages/pi-ai/dist/models.generated.js.map +0 -1
  638. package/packages/pi-ai/src/models.generated.ts +0 -14345
  639. /package/dist/web/standalone/.next/static/{Qr27MOHx0lxRGnJvlhxxu → tqdo0yKKYz6fJXQnIgbdx}/_buildManifest.js +0 -0
  640. /package/dist/web/standalone/.next/static/{Qr27MOHx0lxRGnJvlhxxu → tqdo0yKKYz6fJXQnIgbdx}/_ssgManifest.js +0 -0
@@ -0,0 +1,142 @@
1
+ /**
2
+ * complete-milestone-false-merge.test.ts — Regression test for #4175.
3
+ *
4
+ * Before the fix, a failed complete-milestone unit could leave a stub
5
+ * SUMMARY blocker placeholder on disk. stopAuto's SUMMARY-presence check
6
+ * then treated the milestone as complete and merged the worktree branch
7
+ * into main — emitting a misleading metadata-only merge warning for a
8
+ * milestone that was never legitimately finished.
9
+ *
10
+ * The fix has three cooperating parts:
11
+ * 1. stopAuto uses DB status (authoritative) instead of SUMMARY presence
12
+ * when the project DB is available.
13
+ * 2. postUnitPreVerification pauses auto-mode for complete-milestone
14
+ * after retries are exhausted instead of writing a blocker placeholder.
15
+ * 3. recoverTimedOutUnit pauses for complete-milestone instead of
16
+ * writing a blocker placeholder.
17
+ *
18
+ * This test guards all three via source inspection so a future refactor
19
+ * cannot silently reintroduce the false-merge path.
20
+ */
21
+
22
+ import test from "node:test";
23
+ import assert from "node:assert/strict";
24
+ import { readFileSync } from "node:fs";
25
+ import { join } from "node:path";
26
+
27
+ const gsdDir = join(import.meta.dirname, "..");
28
+ const autoSrc = readFileSync(join(gsdDir, "auto.ts"), "utf-8");
29
+ const postUnitSrc = readFileSync(join(gsdDir, "auto-post-unit.ts"), "utf-8");
30
+ const timeoutSrc = readFileSync(join(gsdDir, "auto-timeout-recovery.ts"), "utf-8");
31
+
32
+ test("#4175: stopAuto uses DB status as the authoritative milestone-complete signal", () => {
33
+ const step4Idx = autoSrc.indexOf("Step 4: Auto-worktree exit");
34
+ assert.ok(step4Idx !== -1, "Step 4 comment exists in stopAuto");
35
+ const step5Idx = autoSrc.indexOf("Step 5:", step4Idx);
36
+ const step4Block = autoSrc.slice(step4Idx, step5Idx);
37
+
38
+ assert.ok(
39
+ step4Block.includes("isDbAvailable()"),
40
+ "Step 4 should branch on isDbAvailable() so DB is consulted when present",
41
+ );
42
+ assert.ok(
43
+ step4Block.includes("getMilestone(s.currentMilestoneId)"),
44
+ "Step 4 should read authoritative milestone status via getMilestone()",
45
+ );
46
+ assert.ok(
47
+ /status\s*===\s*"complete"/.test(step4Block),
48
+ 'Step 4 should compare the DB row status to "complete"',
49
+ );
50
+ });
51
+
52
+ test("#4175: stopAuto imports getMilestone from gsd-db", () => {
53
+ assert.ok(
54
+ /import\s*\{[^}]*\bgetMilestone\b[^}]*\}\s*from\s*"\.\/gsd-db\.js"/.test(autoSrc),
55
+ "auto.ts should import getMilestone from ./gsd-db.js",
56
+ );
57
+ });
58
+
59
+ test("#4175: stopAuto still falls back to SUMMARY presence when DB is unavailable", () => {
60
+ const step4Idx = autoSrc.indexOf("Step 4: Auto-worktree exit");
61
+ const step5Idx = autoSrc.indexOf("Step 5:", step4Idx);
62
+ const step4Block = autoSrc.slice(step4Idx, step5Idx);
63
+
64
+ assert.ok(
65
+ step4Block.includes("resolveMilestoneFile"),
66
+ "Step 4 should keep SUMMARY-file resolution for DB-unavailable projects",
67
+ );
68
+ assert.ok(
69
+ step4Block.includes("preserveBranch"),
70
+ "Step 4 should still preserve branch for incomplete milestones (fallback path)",
71
+ );
72
+ });
73
+
74
+ test("#4175: postUnitPreVerification pauses complete-milestone after retries exhausted", () => {
75
+ // The pause branch must live inside the retries-exhausted block, above the
76
+ // writeBlockerPlaceholder call — otherwise the stub SUMMARY is still written.
77
+ const retriesExhaustedIdx = postUnitSrc.indexOf(
78
+ "if (attempt > MAX_VERIFICATION_RETRIES)",
79
+ );
80
+ assert.ok(
81
+ retriesExhaustedIdx !== -1,
82
+ "retries-exhausted guard exists in postUnitPreVerification",
83
+ );
84
+
85
+ const blockerCallIdx = postUnitSrc.indexOf("writeBlockerPlaceholder", retriesExhaustedIdx);
86
+ assert.ok(
87
+ blockerCallIdx !== -1,
88
+ "blocker placeholder call still exists for non-milestone units",
89
+ );
90
+
91
+ const exhaustedBlock = postUnitSrc.slice(retriesExhaustedIdx, blockerCallIdx);
92
+
93
+ assert.ok(
94
+ /s\.currentUnit\.type\s*===\s*"complete-milestone"/.test(exhaustedBlock),
95
+ "retries-exhausted block should specifically handle complete-milestone",
96
+ );
97
+ assert.ok(
98
+ /pauseAuto\s*\(\s*ctx\s*,\s*pi\s*\)/.test(exhaustedBlock),
99
+ "complete-milestone path should call pauseAuto instead of falling through",
100
+ );
101
+ // The pause branch must return so execution never reaches writeBlockerPlaceholder.
102
+ assert.ok(
103
+ /return\s+"dispatched"\s*;/.test(exhaustedBlock),
104
+ "complete-milestone pause branch should return before the placeholder call",
105
+ );
106
+ });
107
+
108
+ test("#4175: recoverTimedOutUnit pauses complete-milestone instead of writing a blocker placeholder", () => {
109
+ // The complete-milestone pause branch must sit immediately above the
110
+ // "retries exhausted" writeBlockerPlaceholder call so a failed
111
+ // complete-milestone never produces a stub SUMMARY. Anchor on the
112
+ // comment that precedes that specific placeholder call rather than the
113
+ // function's earlier writeBlockerPlaceholder use sites or its import.
114
+ // Use lastIndexOf so we find the final retries-exhausted block in
115
+ // recoverTimedOutUnit, not an earlier helper with the same comment.
116
+ const exhaustedAnchor = "Retries exhausted — write a blocker placeholder";
117
+ const exhaustedIdx = timeoutSrc.lastIndexOf(exhaustedAnchor);
118
+ assert.ok(
119
+ exhaustedIdx !== -1,
120
+ "retries-exhausted blocker-placeholder path still exists for non-milestone units",
121
+ );
122
+
123
+ const guardIdx = timeoutSrc.lastIndexOf(
124
+ 'unitType === "complete-milestone"',
125
+ exhaustedIdx,
126
+ );
127
+ assert.ok(
128
+ guardIdx !== -1,
129
+ "complete-milestone guard should appear above the retries-exhausted placeholder call",
130
+ );
131
+
132
+ const guardBlock = timeoutSrc.slice(guardIdx, exhaustedIdx);
133
+ assert.ok(
134
+ /return\s+"paused"\s*;/.test(guardBlock),
135
+ "complete-milestone guard should return 'paused' before the placeholder call",
136
+ );
137
+ // The guard itself must not call writeBlockerPlaceholder.
138
+ assert.ok(
139
+ !guardBlock.includes("writeBlockerPlaceholder"),
140
+ "complete-milestone guard must not write a blocker placeholder",
141
+ );
142
+ });
@@ -125,9 +125,9 @@ console.log('\n=== complete-slice: schema v6 migration ===');
125
125
 
126
126
  const adapter = _getAdapter()!;
127
127
 
128
- // Verify schema version is current (v14 after indexes + slice_dependencies)
128
+ // Verify schema version is current (v15 with UOK projection tables)
129
129
  const versionRow = adapter.prepare('SELECT MAX(version) as v FROM schema_version').get();
130
- assertEq(versionRow?.['v'], 14, 'schema version should be 14');
130
+ assertEq(versionRow?.['v'], 15, 'schema version should be 15');
131
131
 
132
132
  // Verify slices table has full_summary_md and full_uat_md columns
133
133
  const cols = adapter.prepare("PRAGMA table_info(slices)").all();
@@ -109,9 +109,9 @@ console.log('\n=== complete-task: schema v5 migration ===');
109
109
 
110
110
  const adapter = _getAdapter()!;
111
111
 
112
- // Verify schema version is current (v14 after indexes + slice_dependencies)
112
+ // Verify schema version is current (v15 with UOK projection tables)
113
113
  const versionRow = adapter.prepare('SELECT MAX(version) as v FROM schema_version').get();
114
- assertEq(versionRow?.['v'], 14, 'schema version should be 14');
114
+ assertEq(versionRow?.['v'], 15, 'schema version should be 15');
115
115
 
116
116
  // Verify all 4 new tables exist
117
117
  const tables = adapter.prepare(
@@ -0,0 +1,42 @@
1
+ /**
2
+ * Regression test for #4129: tasks.completed_at stays NULL when status is
3
+ * reconciled to 'complete' via the file-existence path in state.ts.
4
+ *
5
+ * Root cause: reconcileSliceTasks called
6
+ * updateTaskStatus(milestoneId, sliceId, t.id, "complete")
7
+ * without a completedAt timestamp, so the column stays NULL.
8
+ *
9
+ * Fix: pass new Date().toISOString() as the 5th argument.
10
+ */
11
+
12
+ import { describe, test } from "node:test";
13
+ import assert from "node:assert/strict";
14
+ import { readFileSync } from "node:fs";
15
+ import { join, dirname } from "node:path";
16
+ import { fileURLToPath } from "node:url";
17
+
18
+ const __dirname = dirname(fileURLToPath(import.meta.url));
19
+ const stateSource = readFileSync(join(__dirname, "..", "state.ts"), "utf-8");
20
+
21
+ describe("completed-at reconcile (#4129)", () => {
22
+ test("reconcileSliceTasks passes a completedAt timestamp when setting status to complete", () => {
23
+ // Before the fix, state.ts had:
24
+ // updateTaskStatus(milestoneId, sliceId, t.id, "complete")
25
+ // which leaves completed_at NULL in the DB.
26
+ // After the fix, a timestamp must be passed as the 5th argument.
27
+ assert.doesNotMatch(
28
+ stateSource,
29
+ /updateTaskStatus\(\s*milestoneId\s*,\s*sliceId\s*,\s*t\.id\s*,\s*["']complete["']\s*\)/,
30
+ "updateTaskStatus must not be called without a completedAt timestamp when reconciling tasks to 'complete' (#4129)",
31
+ );
32
+ });
33
+
34
+ test("reconcileSliceTasks passes new Date().toISOString() as the completedAt argument", () => {
35
+ // Positive assertion: the fixed call must include a timestamp.
36
+ assert.match(
37
+ stateSource,
38
+ /updateTaskStatus\(\s*milestoneId\s*,\s*sliceId\s*,\s*t\.id\s*,\s*["']complete["']\s*,\s*new Date\(\)\.toISOString\(\)\s*\)/,
39
+ "reconcileSliceTasks must pass new Date().toISOString() as completedAt when setting task status to 'complete' (#4129)",
40
+ );
41
+ });
42
+ });
@@ -351,8 +351,9 @@ skills_used: []
351
351
  const dbState = await deriveStateFromDb(base);
352
352
 
353
353
  assertStatesEqual(dbState, fileState, 'E-blocked');
354
- assert.deepStrictEqual(dbState.phase, 'blocked', 'E-blocked: phase is blocked');
355
- assert.ok(dbState.blockers.length > 0, 'E-blocked: has blockers');
354
+ // With partial-dep fallback, circular deps no longer block — fallback picks first eligible slice
355
+ assert.deepStrictEqual(dbState.phase, 'planning', 'E-blocked: phase is planning (fallback picks a slice)');
356
+ assert.ok(dbState.activeSlice !== null, 'E-blocked: activeSlice is set via fallback');
356
357
 
357
358
  closeDatabase();
358
359
  } finally {
@@ -616,9 +616,10 @@ describe('derive-state-db', async () => {
616
616
  invalidateStateCache();
617
617
  const dbState = await deriveStateFromDb(base);
618
618
 
619
- assert.deepStrictEqual(dbState.phase, 'blocked', 'blocked-db: phase is blocked');
619
+ // With partial-dep fallback, circular deps no longer block — fallback picks first eligible slice
620
+ assert.deepStrictEqual(dbState.phase, 'planning', 'blocked-db: phase is planning (fallback picks a slice)');
620
621
  assert.deepStrictEqual(dbState.phase, fileState.phase, 'blocked-db: phase matches filesystem');
621
- assert.ok(dbState.blockers.length > 0, 'blocked-db: has blockers');
622
+ assert.ok(dbState.activeSlice !== null, 'blocked-db: activeSlice is set via fallback');
622
623
 
623
624
  closeDatabase();
624
625
  } finally {
@@ -307,27 +307,87 @@ describe('derive-state-helpers', () => {
307
307
  }
308
308
  });
309
309
 
310
- // ─── buildCompletenessSet: SUMMARY-on-disk marks complete ───────────
311
- test('buildCompletenessSet: milestone with SUMMARY on disk treated as complete', async () => {
310
+ // ─── buildCompletenessSet: DB status is authoritative ──────────────
311
+ test('buildCompletenessSet: DB status=complete marks milestone complete', async () => {
312
312
  const base = createFixtureBase();
313
313
  try {
314
- // M001 has summary on disk but DB status is still 'active'
315
314
  writeFile(base, 'milestones/M001/M001-ROADMAP.md', ROADMAP_CONTENT);
316
315
  writeFile(base, 'milestones/M001/M001-SUMMARY.md', '# M001 Summary\n\nDone.');
317
- // M002 is the real active milestone
318
316
  writeFile(base, 'milestones/M002/M002-CONTEXT.md', '# M002\n\nActive.');
319
317
 
320
318
  openDatabase(':memory:');
321
- insertMilestone({ id: 'M001', title: 'First', status: 'active' });
319
+ insertMilestone({ id: 'M001', title: 'First', status: 'complete' });
322
320
  insertMilestone({ id: 'M002', title: 'Second', status: 'active' });
323
321
 
324
322
  invalidateStateCache();
325
323
  const state = await deriveStateFromDb(base);
326
324
 
327
- // M001 should be complete (summary on disk), M002 should be active
328
325
  const m1 = state.registry.find(e => e.id === 'M001');
329
- assert.equal(m1?.status, 'complete', 'summary-disk: M001 marked complete via disk SUMMARY');
330
- assert.equal(state.activeMilestone?.id, 'M002', 'summary-disk: M002 is active');
326
+ assert.equal(m1?.status, 'complete', 'DB status=complete registry entry complete');
327
+ assert.equal(state.activeMilestone?.id, 'M002', 'M002 is the active milestone');
328
+ } finally {
329
+ closeDatabase();
330
+ cleanup(base);
331
+ }
332
+ });
333
+
334
+ // ─── Regression #4179: orphan SUMMARY must NOT flip DB-active milestone ───
335
+ // A crashed complete-milestone turn (or stale/manual SUMMARY.md) can leave
336
+ // a milestone SUMMARY on disk while the DB row still reads 'active'. The
337
+ // read-side of state derivation must NOT treat the orphan SUMMARY as a
338
+ // completion signal, or the auto-loop advances and merges work that was
339
+ // never actually finished (same failure class as #4175, read-side twin).
340
+ test('buildCompletenessSet (#4179): orphan SUMMARY on disk does not mark DB-active milestone complete', async () => {
341
+ const base = createFixtureBase();
342
+ try {
343
+ writeFile(base, 'milestones/M001/M001-ROADMAP.md', ROADMAP_CONTENT);
344
+ writeFile(base, 'milestones/M001/M001-SUMMARY.md', '# M001 Orphan Summary\n\nLeft over from crashed turn.');
345
+
346
+ openDatabase(':memory:');
347
+ insertMilestone({ id: 'M001', title: 'First', status: 'active' });
348
+ // Slice still in-flight — auto should resume, not merge.
349
+ insertSlice({ id: 'S01', milestoneId: 'M001', title: 'First', status: 'active', risk: 'low', depends: [] });
350
+ insertSlice({ id: 'S02', milestoneId: 'M001', title: 'Second', status: 'pending', risk: 'low', depends: ['S01'] });
351
+ insertTask({ id: 'T01', sliceId: 'S01', milestoneId: 'M001', title: 'In-flight', status: 'pending' });
352
+
353
+ invalidateStateCache();
354
+ const state = await deriveStateFromDb(base);
355
+
356
+ const m1 = state.registry.find(e => e.id === 'M001');
357
+ assert.notEqual(m1?.status, 'complete', 'orphan SUMMARY must not mark milestone complete');
358
+ assert.equal(m1?.status, 'active', 'M001 remains active — DB is authoritative');
359
+ assert.equal(state.activeMilestone?.id, 'M001', 'M001 is still the active milestone');
360
+ assert.notEqual(state.phase, 'completing-milestone', 'must not short-circuit into completion');
361
+ } finally {
362
+ closeDatabase();
363
+ cleanup(base);
364
+ }
365
+ });
366
+
367
+ // Regression #4179 (companion): DB-active milestone with all slices done +
368
+ // validation terminal + orphan SUMMARY must still flow through completing-milestone
369
+ // (re-runs complete-milestone), not be reported as already-complete.
370
+ test('buildRegistryAndFindActive (#4179): orphan SUMMARY with validation-terminal falls through to completing-milestone', async () => {
371
+ const base = createFixtureBase();
372
+ try {
373
+ writeFile(base, 'milestones/M001/M001-ROADMAP.md', ROADMAP_CONTENT);
374
+ writeFile(base, 'milestones/M001/slices/S01/S01-PLAN.md', PLAN_CONTENT);
375
+ writeFile(base, 'milestones/M001/slices/S02/S02-PLAN.md', PLAN_CONTENT);
376
+ writeFile(base, 'milestones/M001/M001-VALIDATION.md', '---\nverdict: passed\n---\n# Validation\nAll good.');
377
+ writeFile(base, 'milestones/M001/M001-SUMMARY.md', '# M001 Orphan Summary\n\nLeft over.');
378
+
379
+ openDatabase(':memory:');
380
+ insertMilestone({ id: 'M001', title: 'First', status: 'active' });
381
+ insertSlice({ id: 'S01', milestoneId: 'M001', title: 'First', status: 'complete', risk: 'low', depends: [] });
382
+ insertSlice({ id: 'S02', milestoneId: 'M001', title: 'Second', status: 'complete', risk: 'low', depends: ['S01'] });
383
+
384
+ invalidateStateCache();
385
+ const state = await deriveStateFromDb(base);
386
+
387
+ const m1 = state.registry.find(e => e.id === 'M001');
388
+ assert.equal(m1?.status, 'active', 'M001 stays active despite orphan SUMMARY + validation-terminal');
389
+ assert.equal(state.activeMilestone?.id, 'M001', 'M001 is still the active milestone');
390
+ assert.equal(state.phase, 'completing-milestone', 'phase flows through completing-milestone (re-run)');
331
391
  } finally {
332
392
  closeDatabase();
333
393
  cleanup(base);
@@ -446,9 +446,9 @@ Continue from step 2.
446
446
 
447
447
  const state2 = await deriveState(base2);
448
448
 
449
- assert.deepStrictEqual(state2.phase, 'blocked', 'blocked-B: phase is blocked');
450
- assert.deepStrictEqual(state2.activeSlice, null, 'blocked-B: activeSlice is null');
451
- assert.ok(state2.blockers.length > 0, 'blocked-B: blockers array is non-empty');
449
+ // With partial-dep fallback, S01 is picked despite unmet dep on S99
450
+ assert.deepStrictEqual(state2.phase, 'planning', 'blocked-B: phase is planning (fallback picks S01)');
451
+ assert.deepStrictEqual(state2.activeSlice?.id, 'S01', 'blocked-B: activeSlice is S01 via fallback');
452
452
  } finally {
453
453
  cleanup(base2);
454
454
  }
@@ -0,0 +1,154 @@
1
+ // Structural contracts for GSD extension bootstrap isolation.
2
+ //
3
+ // The /gsd command must survive failures in the full extension bootstrap
4
+ // (register-extension.ts). This guards against the regression where a
5
+ // Windows-specific import failure in register-shortcuts.ts silently
6
+ // prevented /gsd from being registered at all (#4168, #4172).
7
+
8
+ import { describe, test } from "node:test";
9
+ import assert from "node:assert/strict";
10
+ import { readFileSync } from "node:fs";
11
+ import { join, dirname } from "node:path";
12
+ import { fileURLToPath } from "node:url";
13
+
14
+ const __dirname = dirname(fileURLToPath(import.meta.url));
15
+ const indexSrc = readFileSync(join(__dirname, "../index.ts"), "utf-8");
16
+ const registerExtSrc = readFileSync(
17
+ join(__dirname, "../bootstrap/register-extension.ts"),
18
+ "utf-8",
19
+ );
20
+
21
+ // ─── index.ts: core /gsd command must be registered before full bootstrap ─────
22
+
23
+ describe("index.ts bootstrap isolation", () => {
24
+ test("imports registerGSDCommand from commands/index.js separately", () => {
25
+ assert.ok(
26
+ indexSrc.includes('./commands/index.js"') || indexSrc.includes("./commands/index.js'"),
27
+ "index.ts must import registerGSDCommand from ./commands/index.js",
28
+ );
29
+ });
30
+
31
+ test("calls registerGSDCommand before importing register-extension.js", () => {
32
+ const gsdCommandCallPos = indexSrc.indexOf("registerGSDCommand(pi)");
33
+ const bootstrapImportPos = indexSrc.indexOf(
34
+ './bootstrap/register-extension.js"',
35
+ );
36
+
37
+ assert.ok(gsdCommandCallPos >= 0, "must call registerGSDCommand(pi)");
38
+ assert.ok(bootstrapImportPos >= 0, "must import register-extension.js");
39
+ assert.ok(
40
+ gsdCommandCallPos < bootstrapImportPos,
41
+ "registerGSDCommand(pi) must be called BEFORE importing register-extension.js",
42
+ );
43
+ });
44
+
45
+ test("wraps register-extension.js import in try-catch", () => {
46
+ // The dynamic import of register-extension.js must be inside a try block
47
+ const tryPos = indexSrc.indexOf("try {");
48
+ const bootstrapImportPos = indexSrc.indexOf(
49
+ './bootstrap/register-extension.js"',
50
+ );
51
+ const catchPos = indexSrc.indexOf("catch (err)");
52
+
53
+ assert.ok(tryPos >= 0, "must have try block");
54
+ assert.ok(catchPos >= 0, "must have catch block");
55
+ assert.ok(
56
+ tryPos < bootstrapImportPos && bootstrapImportPos < catchPos,
57
+ "register-extension.js import must be wrapped in try-catch",
58
+ );
59
+ });
60
+
61
+ test("logs warning on bootstrap failure via workflow-logger", () => {
62
+ assert.ok(
63
+ indexSrc.includes("logWarning"),
64
+ "must use logWarning when bootstrap fails",
65
+ );
66
+ assert.ok(
67
+ indexSrc.includes("Extension setup partially failed"),
68
+ "warning message must indicate partial failure with /gsd still available",
69
+ );
70
+ });
71
+ });
72
+
73
+ // ─── register-extension.ts: no double-registration + defensive wrapping ───────
74
+
75
+ describe("register-extension.ts defensive registration", () => {
76
+ test("does NOT import or call registerGSDCommand (avoids double-registration)", () => {
77
+ // registerGSDCommand is now called by index.ts, not register-extension.ts
78
+ assert.ok(
79
+ !registerExtSrc.includes("import { registerGSDCommand }"),
80
+ "register-extension.ts must NOT import registerGSDCommand",
81
+ );
82
+
83
+ // Check the function body of registerGsdExtension doesn't call it
84
+ const funcBodyStart = registerExtSrc.indexOf(
85
+ "export function registerGsdExtension",
86
+ );
87
+ const funcBody = registerExtSrc.slice(funcBodyStart);
88
+ assert.ok(
89
+ !funcBody.includes("registerGSDCommand(pi)"),
90
+ "registerGsdExtension must NOT call registerGSDCommand(pi)",
91
+ );
92
+ });
93
+
94
+ test("still registers worktree, exit, and kill commands", () => {
95
+ const funcBodyStart = registerExtSrc.indexOf(
96
+ "export function registerGsdExtension",
97
+ );
98
+ const funcBody = registerExtSrc.slice(funcBodyStart);
99
+
100
+ assert.ok(
101
+ funcBody.includes("registerWorktreeCommand(pi)"),
102
+ "must register worktree command",
103
+ );
104
+ assert.ok(
105
+ funcBody.includes("registerExitCommand(pi)"),
106
+ "must register exit command",
107
+ );
108
+ assert.ok(
109
+ funcBody.includes('"kill"'),
110
+ "must register kill command",
111
+ );
112
+ });
113
+
114
+ test("wraps non-critical registrations in individual try-catch blocks", () => {
115
+ const funcBodyStart = registerExtSrc.indexOf(
116
+ "export function registerGsdExtension",
117
+ );
118
+ const funcBody = registerExtSrc.slice(funcBodyStart);
119
+
120
+ // Each non-critical registration should be wrapped with error handling
121
+ const registrationNames = [
122
+ "dynamic-tools",
123
+ "db-tools",
124
+ "journal-tools",
125
+ "query-tools",
126
+ "shortcuts",
127
+ "hooks",
128
+ ];
129
+
130
+ for (const name of registrationNames) {
131
+ assert.ok(
132
+ funcBody.includes(`"${name}"`),
133
+ `non-critical registration "${name}" must be present`,
134
+ );
135
+ }
136
+
137
+ // Must have try-catch inside the registration loop
138
+ assert.ok(
139
+ funcBody.includes("try {") && funcBody.includes("catch (err)"),
140
+ "must have try-catch for non-critical registrations",
141
+ );
142
+ });
143
+
144
+ test("logs warning when a non-critical registration fails", () => {
145
+ assert.ok(
146
+ registerExtSrc.includes("Failed to register"),
147
+ "must log descriptive warning for individual registration failures",
148
+ );
149
+ assert.ok(
150
+ registerExtSrc.includes("logWarning"),
151
+ "must use logWarning from workflow-logger",
152
+ );
153
+ });
154
+ });
@@ -26,6 +26,14 @@ import { MAX_FINALIZE_TIMEOUTS } from "../auto/types.ts";
26
26
 
27
27
  const { assertTrue, assertEq, report } = createTestContext();
28
28
 
29
+ function getRunFinalizeBody(phasesSource: string): string {
30
+ const fnIdx = phasesSource.indexOf("export async function runFinalize(");
31
+ assertTrue(fnIdx > 0, "runFinalize function should exist in phases.ts");
32
+
33
+ const nextExportIdx = phasesSource.indexOf("\nexport ", fnIdx + 1);
34
+ return phasesSource.slice(fnIdx, nextExportIdx > fnIdx ? nextExportIdx : undefined);
35
+ }
36
+
29
37
  // ═══ Test: withTimeout resolves when inner promise resolves promptly ══════════
30
38
 
31
39
  {
@@ -145,11 +153,7 @@ const { assertTrue, assertEq, report } = createTestContext();
145
153
  "utf-8",
146
154
  );
147
155
 
148
- // Find the runFinalize function body
149
- const fnIdx = phasesSource.indexOf("export async function runFinalize(");
150
- assertTrue(fnIdx > 0, "runFinalize function should exist in phases.ts");
151
-
152
- const fnBody = phasesSource.slice(fnIdx, fnIdx + 8000);
156
+ const fnBody = getRunFinalizeBody(phasesSource);
153
157
 
154
158
  // postUnitPreVerification must be wrapped in withTimeout
155
159
  const preTimeoutIdx = fnBody.indexOf("withTimeout(");
@@ -207,8 +211,7 @@ const { assertTrue, assertEq, report } = createTestContext();
207
211
  "utf-8",
208
212
  );
209
213
 
210
- const fnIdx = phasesSource.indexOf("export async function runFinalize(");
211
- const fnBody = phasesSource.slice(fnIdx, fnIdx + 8000);
214
+ const fnBody = getRunFinalizeBody(phasesSource);
212
215
 
213
216
  // Both timeout handlers should increment consecutiveFinalizeTimeouts
214
217
  const incrementCount = (fnBody.match(/consecutiveFinalizeTimeouts\+\+/g) || []).length;