gsd-pi 2.74.0-dev.2b524c3 → 2.74.0-dev.703eabc

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 (484) hide show
  1. package/dist/cli.js +85 -0
  2. package/dist/headless-query.js +4 -1
  3. package/dist/help-text.js +23 -0
  4. package/dist/resources/extensions/gsd/activity-log.js +16 -0
  5. package/dist/resources/extensions/gsd/auto/detect-stuck.js +11 -4
  6. package/dist/resources/extensions/gsd/auto/loop.js +147 -10
  7. package/dist/resources/extensions/gsd/auto/phases.js +158 -4
  8. package/dist/resources/extensions/gsd/auto/session.js +10 -0
  9. package/dist/resources/extensions/gsd/auto-dispatch.js +11 -1
  10. package/dist/resources/extensions/gsd/auto-model-selection.js +51 -5
  11. package/dist/resources/extensions/gsd/auto-post-unit.js +220 -17
  12. package/dist/resources/extensions/gsd/auto-prompts.js +12 -0
  13. package/dist/resources/extensions/gsd/auto-unit-closeout.js +18 -0
  14. package/dist/resources/extensions/gsd/auto-verification.js +100 -2
  15. package/dist/resources/extensions/gsd/auto.js +36 -4
  16. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +30 -8
  17. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +45 -4
  18. package/dist/resources/extensions/gsd/commands/catalog.js +26 -1
  19. package/dist/resources/extensions/gsd/commands/handlers/ops.js +25 -0
  20. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +68 -9
  21. package/dist/resources/extensions/gsd/commands-add-tests.js +111 -0
  22. package/dist/resources/extensions/gsd/commands-backlog.js +140 -0
  23. package/dist/resources/extensions/gsd/commands-do.js +79 -0
  24. package/dist/resources/extensions/gsd/commands-extract-learnings.js +225 -0
  25. package/dist/resources/extensions/gsd/commands-maintenance.js +6 -6
  26. package/dist/resources/extensions/gsd/commands-pr-branch.js +180 -0
  27. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +1 -1
  28. package/dist/resources/extensions/gsd/commands-session-report.js +82 -0
  29. package/dist/resources/extensions/gsd/commands-ship.js +187 -0
  30. package/dist/resources/extensions/gsd/db-writer.js +3 -5
  31. package/dist/resources/extensions/gsd/docs/preferences-reference.md +14 -1
  32. package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +144 -0
  33. package/dist/resources/extensions/gsd/ecosystem/loader.js +145 -0
  34. package/dist/resources/extensions/gsd/git-service.js +49 -1
  35. package/dist/resources/extensions/gsd/graph-context.js +157 -0
  36. package/dist/resources/extensions/gsd/gsd-db.js +581 -2
  37. package/dist/resources/extensions/gsd/guided-flow.js +23 -0
  38. package/dist/resources/extensions/gsd/index.js +15 -2
  39. package/dist/resources/extensions/gsd/init-wizard.js +1 -0
  40. package/dist/resources/extensions/gsd/journal.js +27 -0
  41. package/dist/resources/extensions/gsd/md-importer.js +3 -4
  42. package/dist/resources/extensions/gsd/memory-store.js +19 -51
  43. package/dist/resources/extensions/gsd/metrics.js +19 -0
  44. package/dist/resources/extensions/gsd/milestone-validation-gates.js +13 -12
  45. package/dist/resources/extensions/gsd/native-git-bridge.js +7 -4
  46. package/dist/resources/extensions/gsd/parallel-orchestrator.js +33 -1
  47. package/dist/resources/extensions/gsd/preferences-models.js +20 -3
  48. package/dist/resources/extensions/gsd/preferences-types.js +1 -0
  49. package/dist/resources/extensions/gsd/preferences-validation.js +108 -2
  50. package/dist/resources/extensions/gsd/preferences.js +26 -0
  51. package/dist/resources/extensions/gsd/prompts/add-tests.md +35 -0
  52. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +12 -2
  53. package/dist/resources/extensions/gsd/state.js +5 -1
  54. package/dist/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  55. package/dist/resources/extensions/gsd/tools/complete-slice.js +20 -0
  56. package/dist/resources/extensions/gsd/tools/validate-milestone.js +39 -4
  57. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +3 -14
  58. package/dist/resources/extensions/gsd/triage-resolution.js +2 -5
  59. package/dist/resources/extensions/gsd/unit-ownership.js +1 -1
  60. package/dist/resources/extensions/gsd/uok/audit-toggle.js +7 -0
  61. package/dist/resources/extensions/gsd/uok/audit.js +40 -0
  62. package/dist/resources/extensions/gsd/uok/contracts.js +1 -0
  63. package/dist/resources/extensions/gsd/uok/execution-graph.js +179 -0
  64. package/dist/resources/extensions/gsd/uok/flags.js +29 -0
  65. package/dist/resources/extensions/gsd/uok/gate-runner.js +109 -0
  66. package/dist/resources/extensions/gsd/uok/gitops.js +53 -0
  67. package/dist/resources/extensions/gsd/uok/kernel.js +80 -0
  68. package/dist/resources/extensions/gsd/uok/loop-adapter.js +133 -0
  69. package/dist/resources/extensions/gsd/uok/model-policy.js +66 -0
  70. package/dist/resources/extensions/gsd/uok/plan-v2.js +132 -0
  71. package/dist/resources/extensions/gsd/workflow-logger.js +22 -0
  72. package/dist/resources/extensions/gsd/workflow-manifest.js +8 -69
  73. package/dist/resources/extensions/gsd/workflow-migration.js +21 -22
  74. package/dist/resources/extensions/gsd/workflow-projections.js +4 -1
  75. package/dist/resources/extensions/gsd/workflow-reconcile.js +14 -11
  76. package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
  77. package/dist/tsconfig.extensions.tsbuildinfo +1 -0
  78. package/dist/web/standalone/.next/BUILD_ID +1 -1
  79. package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
  80. package/dist/web/standalone/.next/build-manifest.json +2 -2
  81. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  82. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  83. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  85. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  86. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  87. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  88. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  89. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  90. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  91. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  92. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  93. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  94. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  95. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  96. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  97. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  98. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  99. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +1 -1
  100. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +1 -1
  101. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
  102. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
  103. package/dist/web/standalone/.next/server/app/index.html +1 -1
  104. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  105. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  106. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  107. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  108. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  109. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  110. package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
  111. package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
  112. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  113. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  114. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  115. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  116. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  117. package/package.json +3 -2
  118. package/packages/daemon/package.json +2 -2
  119. package/packages/mcp-server/dist/index.d.ts +3 -0
  120. package/packages/mcp-server/dist/index.d.ts.map +1 -1
  121. package/packages/mcp-server/dist/index.js +3 -0
  122. package/packages/mcp-server/dist/index.js.map +1 -1
  123. package/packages/mcp-server/dist/readers/graph.d.ts +87 -0
  124. package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -0
  125. package/packages/mcp-server/dist/readers/graph.js +655 -0
  126. package/packages/mcp-server/dist/readers/graph.js.map +1 -0
  127. package/packages/mcp-server/dist/readers/index.d.ts +2 -0
  128. package/packages/mcp-server/dist/readers/index.d.ts.map +1 -1
  129. package/packages/mcp-server/dist/readers/index.js +1 -0
  130. package/packages/mcp-server/dist/readers/index.js.map +1 -1
  131. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  132. package/packages/mcp-server/dist/server.js +65 -0
  133. package/packages/mcp-server/dist/server.js.map +1 -1
  134. package/packages/mcp-server/package.json +2 -2
  135. package/packages/mcp-server/src/index.ts +15 -0
  136. package/packages/mcp-server/src/readers/graph.test.ts +604 -0
  137. package/packages/mcp-server/src/readers/graph.ts +855 -0
  138. package/packages/mcp-server/src/readers/index.ts +12 -0
  139. package/packages/mcp-server/src/server.ts +83 -0
  140. package/packages/mcp-server/tsconfig.json +1 -0
  141. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -0
  142. package/packages/native/package.json +2 -2
  143. package/packages/native/tsconfig.tsbuildinfo +1 -0
  144. package/packages/pi-agent-core/package.json +1 -1
  145. package/packages/pi-agent-core/tsconfig.json +1 -0
  146. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -0
  147. package/packages/pi-ai/dist/index.d.ts +1 -9
  148. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  149. package/packages/pi-ai/dist/index.js +1 -9
  150. package/packages/pi-ai/dist/index.js.map +1 -1
  151. package/packages/pi-ai/dist/models/capability-patches.d.ts +19 -0
  152. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -0
  153. package/packages/pi-ai/dist/models/capability-patches.js +36 -0
  154. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -0
  155. package/packages/pi-ai/dist/{models.custom.d.ts → models/custom.d.ts} +1 -1
  156. package/packages/pi-ai/dist/models/custom.d.ts.map +1 -0
  157. package/packages/pi-ai/dist/{models.custom.js → models/custom.js} +4 -4
  158. package/packages/pi-ai/dist/models/custom.js.map +1 -0
  159. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts +1482 -0
  160. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts.map +1 -0
  161. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js +1484 -0
  162. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js.map +1 -0
  163. package/packages/pi-ai/dist/models/generated/anthropic.d.ts +377 -0
  164. package/packages/pi-ai/dist/models/generated/anthropic.d.ts.map +1 -0
  165. package/packages/pi-ai/dist/models/generated/anthropic.js +379 -0
  166. package/packages/pi-ai/dist/models/generated/anthropic.js.map +1 -0
  167. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts +700 -0
  168. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts.map +1 -0
  169. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js +702 -0
  170. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js.map +1 -0
  171. package/packages/pi-ai/dist/models/generated/cerebras.d.ts +71 -0
  172. package/packages/pi-ai/dist/models/generated/cerebras.d.ts.map +1 -0
  173. package/packages/pi-ai/dist/models/generated/cerebras.js +73 -0
  174. package/packages/pi-ai/dist/models/generated/cerebras.js.map +1 -0
  175. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts +590 -0
  176. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts.map +1 -0
  177. package/packages/pi-ai/dist/models/generated/github-copilot.js +444 -0
  178. package/packages/pi-ai/dist/models/generated/github-copilot.js.map +1 -0
  179. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts +156 -0
  180. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts.map +1 -0
  181. package/packages/pi-ai/dist/models/generated/google-antigravity.js +158 -0
  182. package/packages/pi-ai/dist/models/generated/google-antigravity.js.map +1 -0
  183. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts +105 -0
  184. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts.map +1 -0
  185. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js +107 -0
  186. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js.map +1 -0
  187. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts +207 -0
  188. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts.map +1 -0
  189. package/packages/pi-ai/dist/models/generated/google-vertex.js +209 -0
  190. package/packages/pi-ai/dist/models/generated/google-vertex.js.map +1 -0
  191. package/packages/pi-ai/dist/models/generated/google.d.ts +462 -0
  192. package/packages/pi-ai/dist/models/generated/google.d.ts.map +1 -0
  193. package/packages/pi-ai/dist/models/generated/google.js +464 -0
  194. package/packages/pi-ai/dist/models/generated/google.js.map +1 -0
  195. package/packages/pi-ai/dist/models/generated/groq.d.ts +309 -0
  196. package/packages/pi-ai/dist/models/generated/groq.d.ts.map +1 -0
  197. package/packages/pi-ai/dist/models/generated/groq.js +311 -0
  198. package/packages/pi-ai/dist/models/generated/groq.js.map +1 -0
  199. package/packages/pi-ai/dist/models/generated/huggingface.d.ts +383 -0
  200. package/packages/pi-ai/dist/models/generated/huggingface.d.ts.map +1 -0
  201. package/packages/pi-ai/dist/models/generated/huggingface.js +347 -0
  202. package/packages/pi-ai/dist/models/generated/huggingface.js.map +1 -0
  203. package/packages/pi-ai/dist/{models.generated.d.ts → models/generated/index.d.ts} +1 -1
  204. package/packages/pi-ai/dist/{models.generated.d.ts.map → models/generated/index.d.ts.map} +1 -1
  205. package/packages/pi-ai/dist/models/generated/index.js +51 -0
  206. package/packages/pi-ai/dist/models/generated/index.js.map +1 -0
  207. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts +37 -0
  208. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts.map +1 -0
  209. package/packages/pi-ai/dist/models/generated/kimi-coding.js +39 -0
  210. package/packages/pi-ai/dist/models/generated/kimi-coding.js.map +1 -0
  211. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts +105 -0
  212. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts.map +1 -0
  213. package/packages/pi-ai/dist/models/generated/minimax-cn.js +107 -0
  214. package/packages/pi-ai/dist/models/generated/minimax-cn.js.map +1 -0
  215. package/packages/pi-ai/dist/models/generated/minimax.d.ts +105 -0
  216. package/packages/pi-ai/dist/models/generated/minimax.d.ts.map +1 -0
  217. package/packages/pi-ai/dist/models/generated/minimax.js +107 -0
  218. package/packages/pi-ai/dist/models/generated/minimax.js.map +1 -0
  219. package/packages/pi-ai/dist/models/generated/mistral.d.ts +445 -0
  220. package/packages/pi-ai/dist/models/generated/mistral.d.ts.map +1 -0
  221. package/packages/pi-ai/dist/models/generated/mistral.js +447 -0
  222. package/packages/pi-ai/dist/models/generated/mistral.js.map +1 -0
  223. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +139 -0
  224. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -0
  225. package/packages/pi-ai/dist/models/generated/openai-codex.js +141 -0
  226. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -0
  227. package/packages/pi-ai/dist/models/generated/openai.d.ts +700 -0
  228. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -0
  229. package/packages/pi-ai/dist/models/generated/openai.js +702 -0
  230. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -0
  231. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts +122 -0
  232. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts.map +1 -0
  233. package/packages/pi-ai/dist/models/generated/opencode-go.js +124 -0
  234. package/packages/pi-ai/dist/models/generated/opencode-go.js.map +1 -0
  235. package/packages/pi-ai/dist/models/generated/opencode.d.ts +530 -0
  236. package/packages/pi-ai/dist/models/generated/opencode.d.ts.map +1 -0
  237. package/packages/pi-ai/dist/models/generated/opencode.js +532 -0
  238. package/packages/pi-ai/dist/models/generated/opencode.js.map +1 -0
  239. package/packages/pi-ai/dist/models/generated/openrouter.d.ts +4270 -0
  240. package/packages/pi-ai/dist/models/generated/openrouter.d.ts.map +1 -0
  241. package/packages/pi-ai/dist/models/generated/openrouter.js +4272 -0
  242. package/packages/pi-ai/dist/models/generated/openrouter.js.map +1 -0
  243. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts +2604 -0
  244. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts.map +1 -0
  245. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js +2606 -0
  246. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js.map +1 -0
  247. package/packages/pi-ai/dist/models/generated/xai.d.ts +411 -0
  248. package/packages/pi-ai/dist/models/generated/xai.d.ts.map +1 -0
  249. package/packages/pi-ai/dist/models/generated/xai.js +413 -0
  250. package/packages/pi-ai/dist/models/generated/xai.js.map +1 -0
  251. package/packages/pi-ai/dist/models/generated/zai.d.ts +276 -0
  252. package/packages/pi-ai/dist/models/generated/zai.d.ts.map +1 -0
  253. package/packages/pi-ai/dist/models/generated/zai.js +239 -0
  254. package/packages/pi-ai/dist/models/generated/zai.js.map +1 -0
  255. package/packages/pi-ai/dist/models/index.d.ts +27 -0
  256. package/packages/pi-ai/dist/models/index.d.ts.map +1 -0
  257. package/packages/pi-ai/dist/models/index.js +80 -0
  258. package/packages/pi-ai/dist/models/index.js.map +1 -0
  259. package/packages/pi-ai/dist/models.d.ts +1 -36
  260. package/packages/pi-ai/dist/models.d.ts.map +1 -1
  261. package/packages/pi-ai/dist/models.generated.test.js +1 -2
  262. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  263. package/packages/pi-ai/dist/models.js +3 -112
  264. package/packages/pi-ai/dist/models.js.map +1 -1
  265. package/packages/pi-ai/dist/models.test.js +6 -5
  266. package/packages/pi-ai/dist/models.test.js.map +1 -1
  267. package/packages/pi-ai/package.json +1 -1
  268. package/packages/pi-ai/scripts/generate-models.ts +74 -40
  269. package/packages/pi-ai/src/index.ts +1 -9
  270. package/packages/pi-ai/src/models/capability-patches.ts +40 -0
  271. package/packages/pi-ai/src/{models.custom.ts → models/custom.ts} +4 -4
  272. package/packages/pi-ai/src/models/generated/amazon-bedrock.ts +1486 -0
  273. package/packages/pi-ai/src/models/generated/anthropic.ts +381 -0
  274. package/packages/pi-ai/src/models/generated/azure-openai-responses.ts +704 -0
  275. package/packages/pi-ai/src/models/generated/cerebras.ts +75 -0
  276. package/packages/pi-ai/src/models/generated/github-copilot.ts +446 -0
  277. package/packages/pi-ai/src/models/generated/google-antigravity.ts +160 -0
  278. package/packages/pi-ai/src/models/generated/google-gemini-cli.ts +109 -0
  279. package/packages/pi-ai/src/models/generated/google-vertex.ts +211 -0
  280. package/packages/pi-ai/src/models/generated/google.ts +466 -0
  281. package/packages/pi-ai/src/models/generated/groq.ts +313 -0
  282. package/packages/pi-ai/src/models/generated/huggingface.ts +349 -0
  283. package/packages/pi-ai/src/models/generated/index.ts +52 -0
  284. package/packages/pi-ai/src/models/generated/kimi-coding.ts +41 -0
  285. package/packages/pi-ai/src/models/generated/minimax-cn.ts +109 -0
  286. package/packages/pi-ai/src/models/generated/minimax.ts +109 -0
  287. package/packages/pi-ai/src/models/generated/mistral.ts +449 -0
  288. package/packages/pi-ai/src/models/generated/openai-codex.ts +143 -0
  289. package/packages/pi-ai/src/models/generated/openai.ts +704 -0
  290. package/packages/pi-ai/src/models/generated/opencode-go.ts +126 -0
  291. package/packages/pi-ai/src/models/generated/opencode.ts +534 -0
  292. package/packages/pi-ai/src/models/generated/openrouter.ts +4274 -0
  293. package/packages/pi-ai/src/models/generated/vercel-ai-gateway.ts +2608 -0
  294. package/packages/pi-ai/src/models/generated/xai.ts +415 -0
  295. package/packages/pi-ai/src/models/generated/zai.ts +241 -0
  296. package/packages/pi-ai/src/models/index.ts +106 -0
  297. package/packages/pi-ai/src/models.generated.test.ts +1 -2
  298. package/packages/pi-ai/src/models.test.ts +6 -5
  299. package/packages/pi-ai/src/models.ts +3 -153
  300. package/packages/pi-ai/tsconfig.json +1 -0
  301. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -0
  302. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  303. package/packages/pi-coding-agent/dist/core/agent-session.js +8 -2
  304. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  305. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +472 -0
  306. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  307. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts +2 -0
  308. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts.map +1 -0
  309. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js +52 -0
  310. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js.map +1 -0
  311. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  312. package/packages/pi-coding-agent/dist/core/model-registry.js +2 -2
  313. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  314. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +11 -0
  315. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  316. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +1 -0
  317. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  318. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +23 -9
  319. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
  320. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts +11 -0
  321. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -0
  322. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +47 -0
  323. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -0
  324. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  325. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +51 -8
  326. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  327. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.d.ts.map +1 -1
  328. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js +22 -22
  329. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js.map +1 -1
  330. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  331. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +232 -18
  332. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  333. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts +2 -0
  334. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts.map +1 -0
  335. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +38 -0
  336. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -0
  337. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +13 -0
  338. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  339. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +53 -6
  340. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  341. package/packages/pi-coding-agent/src/core/agent-session.ts +12 -6
  342. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +612 -0
  343. package/packages/pi-coding-agent/src/core/model-registry-env-fallback.test.ts +59 -0
  344. package/packages/pi-coding-agent/src/core/model-registry.ts +2 -1
  345. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +19 -0
  346. package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +25 -10
  347. package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +67 -0
  348. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +66 -7
  349. package/packages/pi-coding-agent/src/modes/interactive/components/user-message.ts +23 -26
  350. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +298 -41
  351. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +44 -0
  352. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +73 -6
  353. package/packages/pi-coding-agent/src/types/ambient-modules.d.ts +69 -0
  354. package/packages/pi-coding-agent/tsconfig.json +3 -2
  355. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -0
  356. package/packages/pi-tui/package.json +1 -1
  357. package/packages/pi-tui/tsconfig.json +1 -0
  358. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -0
  359. package/packages/rpc-client/package.json +1 -1
  360. package/packages/rpc-client/tsconfig.json +1 -0
  361. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -0
  362. package/src/resources/extensions/gsd/activity-log.ts +21 -0
  363. package/src/resources/extensions/gsd/auto/detect-stuck.ts +12 -4
  364. package/src/resources/extensions/gsd/auto/loop-deps.ts +10 -0
  365. package/src/resources/extensions/gsd/auto/loop.ts +159 -10
  366. package/src/resources/extensions/gsd/auto/phases.ts +191 -4
  367. package/src/resources/extensions/gsd/auto/session.ts +10 -0
  368. package/src/resources/extensions/gsd/auto-dispatch.ts +16 -6
  369. package/src/resources/extensions/gsd/auto-model-selection.ts +66 -5
  370. package/src/resources/extensions/gsd/auto-post-unit.ts +238 -18
  371. package/src/resources/extensions/gsd/auto-prompts.ts +13 -0
  372. package/src/resources/extensions/gsd/auto-unit-closeout.ts +25 -1
  373. package/src/resources/extensions/gsd/auto-verification.ts +129 -2
  374. package/src/resources/extensions/gsd/auto.ts +41 -2
  375. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +38 -8
  376. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +56 -3
  377. package/src/resources/extensions/gsd/commands/catalog.ts +26 -1
  378. package/src/resources/extensions/gsd/commands/handlers/ops.ts +25 -0
  379. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +74 -9
  380. package/src/resources/extensions/gsd/commands-add-tests.ts +137 -0
  381. package/src/resources/extensions/gsd/commands-backlog.ts +182 -0
  382. package/src/resources/extensions/gsd/commands-do.ts +109 -0
  383. package/src/resources/extensions/gsd/commands-extract-learnings.ts +304 -0
  384. package/src/resources/extensions/gsd/commands-maintenance.ts +6 -6
  385. package/src/resources/extensions/gsd/commands-pr-branch.ts +234 -0
  386. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +1 -1
  387. package/src/resources/extensions/gsd/commands-session-report.ts +101 -0
  388. package/src/resources/extensions/gsd/commands-ship.ts +219 -0
  389. package/src/resources/extensions/gsd/db-writer.ts +3 -5
  390. package/src/resources/extensions/gsd/docs/preferences-reference.md +14 -1
  391. package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +228 -0
  392. package/src/resources/extensions/gsd/ecosystem/loader.ts +201 -0
  393. package/src/resources/extensions/gsd/git-service.ts +68 -0
  394. package/src/resources/extensions/gsd/graph-context.ts +212 -0
  395. package/src/resources/extensions/gsd/gsd-db.ts +788 -3
  396. package/src/resources/extensions/gsd/guided-flow.ts +32 -0
  397. package/src/resources/extensions/gsd/index.ts +18 -2
  398. package/src/resources/extensions/gsd/init-wizard.ts +3 -2
  399. package/src/resources/extensions/gsd/journal.ts +30 -0
  400. package/src/resources/extensions/gsd/md-importer.ts +3 -5
  401. package/src/resources/extensions/gsd/memory-store.ts +31 -62
  402. package/src/resources/extensions/gsd/metrics.ts +26 -0
  403. package/src/resources/extensions/gsd/milestone-validation-gates.ts +13 -14
  404. package/src/resources/extensions/gsd/native-git-bridge.ts +11 -12
  405. package/src/resources/extensions/gsd/parallel-orchestrator.ts +40 -1
  406. package/src/resources/extensions/gsd/preferences-models.ts +20 -3
  407. package/src/resources/extensions/gsd/preferences-types.ts +32 -0
  408. package/src/resources/extensions/gsd/preferences-validation.ts +107 -2
  409. package/src/resources/extensions/gsd/preferences.ts +28 -0
  410. package/src/resources/extensions/gsd/prompts/add-tests.md +35 -0
  411. package/src/resources/extensions/gsd/session-lock.ts +14 -2
  412. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +20 -1
  413. package/src/resources/extensions/gsd/state.ts +9 -2
  414. package/src/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  415. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +7 -3
  416. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +20 -0
  417. package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +7 -3
  418. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +6 -2
  419. package/src/resources/extensions/gsd/tests/commands-backlog.test.ts +158 -0
  420. package/src/resources/extensions/gsd/tests/commands-do.test.ts +127 -0
  421. package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +340 -0
  422. package/src/resources/extensions/gsd/tests/commands-pr-branch.test.ts +68 -0
  423. package/src/resources/extensions/gsd/tests/commands-session-report.test.ts +82 -0
  424. package/src/resources/extensions/gsd/tests/commands-ship.test.ts +71 -0
  425. package/src/resources/extensions/gsd/tests/commands-workflow-custom.test.ts +14 -0
  426. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  427. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  428. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +154 -0
  429. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +10 -7
  430. package/src/resources/extensions/gsd/tests/graph-context.test.ts +337 -0
  431. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
  432. package/src/resources/extensions/gsd/tests/health-widget.test.ts +1 -1
  433. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +68 -1
  434. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -2
  435. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -3
  436. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +140 -0
  437. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  438. package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +2 -1
  439. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +40 -1
  440. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +1 -1
  441. package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +180 -0
  442. package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -5
  443. package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +101 -0
  444. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +85 -0
  445. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +69 -0
  446. package/src/resources/extensions/gsd/tests/uok-flags.test.ts +39 -0
  447. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +70 -0
  448. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +85 -0
  449. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +35 -0
  450. package/src/resources/extensions/gsd/tests/uok-model-policy.test.ts +89 -0
  451. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +167 -0
  452. package/src/resources/extensions/gsd/tests/uok-preferences.test.ts +42 -0
  453. package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +39 -0
  454. package/src/resources/extensions/gsd/tests/workflow-logger-wiring.test.ts +223 -0
  455. package/src/resources/extensions/gsd/tools/complete-slice.ts +26 -0
  456. package/src/resources/extensions/gsd/tools/validate-milestone.ts +48 -3
  457. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +3 -11
  458. package/src/resources/extensions/gsd/triage-resolution.ts +2 -7
  459. package/src/resources/extensions/gsd/types.ts +14 -1
  460. package/src/resources/extensions/gsd/unit-ownership.ts +2 -2
  461. package/src/resources/extensions/gsd/uok/audit-toggle.ts +9 -0
  462. package/src/resources/extensions/gsd/uok/audit.ts +51 -0
  463. package/src/resources/extensions/gsd/uok/contracts.ts +135 -0
  464. package/src/resources/extensions/gsd/uok/execution-graph.ts +241 -0
  465. package/src/resources/extensions/gsd/uok/flags.ts +45 -0
  466. package/src/resources/extensions/gsd/uok/gate-runner.ts +146 -0
  467. package/src/resources/extensions/gsd/uok/gitops.ts +75 -0
  468. package/src/resources/extensions/gsd/uok/kernel.ts +105 -0
  469. package/src/resources/extensions/gsd/uok/loop-adapter.ts +162 -0
  470. package/src/resources/extensions/gsd/uok/model-policy.ts +112 -0
  471. package/src/resources/extensions/gsd/uok/plan-v2.ts +156 -0
  472. package/src/resources/extensions/gsd/workflow-logger.ts +27 -1
  473. package/src/resources/extensions/gsd/workflow-manifest.ts +9 -104
  474. package/src/resources/extensions/gsd/workflow-migration.ts +21 -29
  475. package/src/resources/extensions/gsd/workflow-projections.ts +8 -1
  476. package/src/resources/extensions/gsd/workflow-reconcile.ts +15 -15
  477. package/src/resources/extensions/ttsr/ttsr-manager.ts +10 -5
  478. package/packages/pi-ai/dist/models.custom.d.ts.map +0 -1
  479. package/packages/pi-ai/dist/models.custom.js.map +0 -1
  480. package/packages/pi-ai/dist/models.generated.js +0 -14343
  481. package/packages/pi-ai/dist/models.generated.js.map +0 -1
  482. package/packages/pi-ai/src/models.generated.ts +0 -14345
  483. /package/dist/web/standalone/.next/static/{YzIEI9sxJy4t5xgClF08g → 3U-oZ5FT59BM7sm2GInic}/_buildManifest.js +0 -0
  484. /package/dist/web/standalone/.next/static/{YzIEI9sxJy4t5xgClF08g → 3U-oZ5FT59BM7sm2GInic}/_ssgManifest.js +0 -0
@@ -4,6 +4,21 @@
4
4
  //
5
5
  // Exposes a unified sync API for decisions and requirements storage.
6
6
  // Schema is initialized on first open with WAL mode for file-backed DBs.
7
+ //
8
+ // ─── Single-writer invariant ─────────────────────────────────────────────
9
+ // This file is the ONLY place in the codebase that issues write SQL
10
+ // (INSERT / UPDATE / DELETE / REPLACE / BEGIN-COMMIT transactions) against
11
+ // the engine database at `.gsd/gsd.db`. All other modules must call the
12
+ // typed wrappers exported here. The structural test
13
+ // `tests/single-writer-invariant.test.ts` fails CI if a new bypass appears.
14
+ //
15
+ // `_getAdapter()` is retained for read-only SELECTs in query modules
16
+ // (context-store, memory-store queries, doctor checks, projections).
17
+ // Do NOT use it for writes — add a wrapper here instead.
18
+ //
19
+ // The separate `.gsd/unit-claims.db` managed by `unit-ownership.ts` is an
20
+ // intentionally independent store for cross-worktree claim races and is
21
+ // excluded from this invariant.
7
22
 
8
23
  import { createRequire } from "node:module";
9
24
  import { existsSync, copyFileSync, mkdirSync, realpathSync } from "node:fs";
@@ -12,6 +27,10 @@ import type { Decision, Requirement, GateRow, GateId, GateScope, GateStatus, Gat
12
27
  import { GSDError, GSD_STALE_STATE } from "./errors.js";
13
28
  import { getGateIdsForTurn, type OwnerTurn } from "./gate-registry.js";
14
29
  import { logError, logWarning } from "./workflow-logger.js";
30
+ // Type-only import to avoid a circular runtime dep. The runtime side of
31
+ // workflow-manifest.ts depends on this file, but the StateManifest type is
32
+ // pure structure with no runtime coupling.
33
+ import type { StateManifest } from "./workflow-manifest.js";
15
34
 
16
35
  const _require = createRequire(import.meta.url);
17
36
 
@@ -35,8 +54,8 @@ let loadAttempted = false;
35
54
 
36
55
  function suppressSqliteWarning(): void {
37
56
  const origEmit = process.emit;
38
- // @ts-expect-error overriding process.emit for warning filter
39
- process.emit = function (event: string, ...args: unknown[]): boolean {
57
+ // Override via loose cast: Node's overloaded emit signature is not directly assignable.
58
+ (process as any).emit = function (event: string, ...args: unknown[]): boolean {
40
59
  if (
41
60
  event === "warning" &&
42
61
  args[0] &&
@@ -161,7 +180,7 @@ function openRawDb(path: string): unknown {
161
180
  return new Database(path);
162
181
  }
163
182
 
164
- const SCHEMA_VERSION = 14;
183
+ const SCHEMA_VERSION = 15;
165
184
 
166
185
  function indexExists(db: DbAdapter, name: string): boolean {
167
186
  return !!db.prepare(
@@ -424,6 +443,70 @@ function initSchema(db: DbAdapter, fileBacked: boolean): void {
424
443
  )
425
444
  `);
426
445
 
446
+ db.exec(`
447
+ CREATE TABLE IF NOT EXISTS gate_runs (
448
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
449
+ trace_id TEXT NOT NULL,
450
+ turn_id TEXT NOT NULL,
451
+ gate_id TEXT NOT NULL,
452
+ gate_type TEXT NOT NULL DEFAULT '',
453
+ unit_type TEXT DEFAULT NULL,
454
+ unit_id TEXT DEFAULT NULL,
455
+ milestone_id TEXT DEFAULT NULL,
456
+ slice_id TEXT DEFAULT NULL,
457
+ task_id TEXT DEFAULT NULL,
458
+ outcome TEXT NOT NULL DEFAULT 'pass',
459
+ failure_class TEXT NOT NULL DEFAULT 'none',
460
+ rationale TEXT NOT NULL DEFAULT '',
461
+ findings TEXT NOT NULL DEFAULT '',
462
+ attempt INTEGER NOT NULL DEFAULT 1,
463
+ max_attempts INTEGER NOT NULL DEFAULT 1,
464
+ retryable INTEGER NOT NULL DEFAULT 0,
465
+ evaluated_at TEXT NOT NULL DEFAULT ''
466
+ )
467
+ `);
468
+
469
+ db.exec(`
470
+ CREATE TABLE IF NOT EXISTS turn_git_transactions (
471
+ trace_id TEXT NOT NULL,
472
+ turn_id TEXT NOT NULL,
473
+ unit_type TEXT DEFAULT NULL,
474
+ unit_id TEXT DEFAULT NULL,
475
+ stage TEXT NOT NULL DEFAULT 'turn-start',
476
+ action TEXT NOT NULL DEFAULT 'status-only',
477
+ push INTEGER NOT NULL DEFAULT 0,
478
+ status TEXT NOT NULL DEFAULT 'ok',
479
+ error TEXT DEFAULT NULL,
480
+ metadata_json TEXT NOT NULL DEFAULT '{}',
481
+ updated_at TEXT NOT NULL DEFAULT '',
482
+ PRIMARY KEY (trace_id, turn_id, stage)
483
+ )
484
+ `);
485
+
486
+ db.exec(`
487
+ CREATE TABLE IF NOT EXISTS audit_events (
488
+ event_id TEXT PRIMARY KEY,
489
+ trace_id TEXT NOT NULL,
490
+ turn_id TEXT DEFAULT NULL,
491
+ caused_by TEXT DEFAULT NULL,
492
+ category TEXT NOT NULL,
493
+ type TEXT NOT NULL,
494
+ ts TEXT NOT NULL,
495
+ payload_json TEXT NOT NULL DEFAULT '{}'
496
+ )
497
+ `);
498
+
499
+ db.exec(`
500
+ CREATE TABLE IF NOT EXISTS audit_turn_index (
501
+ trace_id TEXT NOT NULL,
502
+ turn_id TEXT NOT NULL,
503
+ first_ts TEXT NOT NULL,
504
+ last_ts TEXT NOT NULL,
505
+ event_count INTEGER NOT NULL DEFAULT 0,
506
+ PRIMARY KEY (trace_id, turn_id)
507
+ )
508
+ `);
509
+
427
510
  db.exec("CREATE INDEX IF NOT EXISTS idx_memories_active ON memories(superseded_by)");
428
511
  db.exec("CREATE INDEX IF NOT EXISTS idx_replan_history_milestone ON replan_history(milestone_id, created_at)");
429
512
 
@@ -437,6 +520,11 @@ function initSchema(db: DbAdapter, fileBacked: boolean): void {
437
520
 
438
521
  // v14 index — slice dependency lookups
439
522
  db.exec("CREATE INDEX IF NOT EXISTS idx_slice_deps_target ON slice_dependencies(milestone_id, depends_on_slice_id)");
523
+ db.exec("CREATE INDEX IF NOT EXISTS idx_gate_runs_turn ON gate_runs(trace_id, turn_id)");
524
+ db.exec("CREATE INDEX IF NOT EXISTS idx_gate_runs_lookup ON gate_runs(milestone_id, slice_id, task_id, gate_id)");
525
+ db.exec("CREATE INDEX IF NOT EXISTS idx_turn_git_tx_turn ON turn_git_transactions(trace_id, turn_id)");
526
+ db.exec("CREATE INDEX IF NOT EXISTS idx_audit_events_trace ON audit_events(trace_id, ts)");
527
+ db.exec("CREATE INDEX IF NOT EXISTS idx_audit_events_turn ON audit_events(trace_id, turn_id, ts)");
440
528
 
441
529
  db.exec(`CREATE VIEW IF NOT EXISTS active_decisions AS SELECT * FROM decisions WHERE superseded_by IS NULL`);
442
530
  db.exec(`CREATE VIEW IF NOT EXISTS active_requirements AS SELECT * FROM requirements WHERE superseded_by IS NULL`);
@@ -791,6 +879,78 @@ function migrateSchema(db: DbAdapter): void {
791
879
  });
792
880
  }
793
881
 
882
+ if (currentVersion < 15) {
883
+ db.exec(`
884
+ CREATE TABLE IF NOT EXISTS gate_runs (
885
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
886
+ trace_id TEXT NOT NULL,
887
+ turn_id TEXT NOT NULL,
888
+ gate_id TEXT NOT NULL,
889
+ gate_type TEXT NOT NULL DEFAULT '',
890
+ unit_type TEXT DEFAULT NULL,
891
+ unit_id TEXT DEFAULT NULL,
892
+ milestone_id TEXT DEFAULT NULL,
893
+ slice_id TEXT DEFAULT NULL,
894
+ task_id TEXT DEFAULT NULL,
895
+ outcome TEXT NOT NULL DEFAULT 'pass',
896
+ failure_class TEXT NOT NULL DEFAULT 'none',
897
+ rationale TEXT NOT NULL DEFAULT '',
898
+ findings TEXT NOT NULL DEFAULT '',
899
+ attempt INTEGER NOT NULL DEFAULT 1,
900
+ max_attempts INTEGER NOT NULL DEFAULT 1,
901
+ retryable INTEGER NOT NULL DEFAULT 0,
902
+ evaluated_at TEXT NOT NULL DEFAULT ''
903
+ )
904
+ `);
905
+ db.exec(`
906
+ CREATE TABLE IF NOT EXISTS turn_git_transactions (
907
+ trace_id TEXT NOT NULL,
908
+ turn_id TEXT NOT NULL,
909
+ unit_type TEXT DEFAULT NULL,
910
+ unit_id TEXT DEFAULT NULL,
911
+ stage TEXT NOT NULL DEFAULT 'turn-start',
912
+ action TEXT NOT NULL DEFAULT 'status-only',
913
+ push INTEGER NOT NULL DEFAULT 0,
914
+ status TEXT NOT NULL DEFAULT 'ok',
915
+ error TEXT DEFAULT NULL,
916
+ metadata_json TEXT NOT NULL DEFAULT '{}',
917
+ updated_at TEXT NOT NULL DEFAULT '',
918
+ PRIMARY KEY (trace_id, turn_id, stage)
919
+ )
920
+ `);
921
+ db.exec(`
922
+ CREATE TABLE IF NOT EXISTS audit_events (
923
+ event_id TEXT PRIMARY KEY,
924
+ trace_id TEXT NOT NULL,
925
+ turn_id TEXT DEFAULT NULL,
926
+ caused_by TEXT DEFAULT NULL,
927
+ category TEXT NOT NULL,
928
+ type TEXT NOT NULL,
929
+ ts TEXT NOT NULL,
930
+ payload_json TEXT NOT NULL DEFAULT '{}'
931
+ )
932
+ `);
933
+ db.exec(`
934
+ CREATE TABLE IF NOT EXISTS audit_turn_index (
935
+ trace_id TEXT NOT NULL,
936
+ turn_id TEXT NOT NULL,
937
+ first_ts TEXT NOT NULL,
938
+ last_ts TEXT NOT NULL,
939
+ event_count INTEGER NOT NULL DEFAULT 0,
940
+ PRIMARY KEY (trace_id, turn_id)
941
+ )
942
+ `);
943
+ db.exec("CREATE INDEX IF NOT EXISTS idx_gate_runs_turn ON gate_runs(trace_id, turn_id)");
944
+ db.exec("CREATE INDEX IF NOT EXISTS idx_gate_runs_lookup ON gate_runs(milestone_id, slice_id, task_id, gate_id)");
945
+ db.exec("CREATE INDEX IF NOT EXISTS idx_turn_git_tx_turn ON turn_git_transactions(trace_id, turn_id)");
946
+ db.exec("CREATE INDEX IF NOT EXISTS idx_audit_events_trace ON audit_events(trace_id, ts)");
947
+ db.exec("CREATE INDEX IF NOT EXISTS idx_audit_events_turn ON audit_events(trace_id, turn_id, ts)");
948
+ db.prepare("INSERT INTO schema_version (version, applied_at) VALUES (:version, :applied_at)").run({
949
+ ":version": 15,
950
+ ":applied_at": new Date().toISOString(),
951
+ });
952
+ }
953
+
794
954
  db.exec("COMMIT");
795
955
  } catch (err) {
796
956
  db.exec("ROLLBACK");
@@ -922,6 +1082,48 @@ export function transaction<T>(fn: () => T): T {
922
1082
  }
923
1083
  }
924
1084
 
1085
+ /**
1086
+ * Wrap a block of reads in a DEFERRED transaction so that all SELECTs observe
1087
+ * a consistent snapshot of the DB even if a concurrent writer commits between
1088
+ * them. Use this for multi-query read flows (e.g. tool executors that query
1089
+ * milestone + slices + counts and want one snapshot). Re-entrant — if already
1090
+ * inside a transaction, runs fn() without starting a nested one.
1091
+ */
1092
+ export function readTransaction<T>(fn: () => T): T {
1093
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
1094
+
1095
+ if (_txDepth > 0) {
1096
+ _txDepth++;
1097
+ try {
1098
+ return fn();
1099
+ } finally {
1100
+ _txDepth--;
1101
+ }
1102
+ }
1103
+
1104
+ _txDepth++;
1105
+ currentDb.exec("BEGIN DEFERRED");
1106
+ try {
1107
+ const result = fn();
1108
+ currentDb.exec("COMMIT");
1109
+ return result;
1110
+ } catch (err) {
1111
+ try {
1112
+ currentDb.exec("ROLLBACK");
1113
+ } catch (rollbackErr) {
1114
+ // A failed ROLLBACK after a failed read is a split-brain signal —
1115
+ // the transaction is in an indeterminate state. Surface it via the
1116
+ // logger instead of swallowing it.
1117
+ logError("db", "snapshotState ROLLBACK failed", {
1118
+ error: (rollbackErr as Error).message,
1119
+ });
1120
+ }
1121
+ throw err;
1122
+ } finally {
1123
+ _txDepth--;
1124
+ }
1125
+ }
1126
+
925
1127
  export function insertDecision(d: Omit<Decision, "seq">): void {
926
1128
  if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
927
1129
  currentDb.prepare(
@@ -2226,6 +2428,9 @@ export function deleteMilestone(milestoneId: string): void {
2226
2428
  currentDb!.prepare(
2227
2429
  `DELETE FROM quality_gates WHERE milestone_id = :mid`,
2228
2430
  ).run({ ":mid": milestoneId });
2431
+ currentDb!.prepare(
2432
+ `DELETE FROM gate_runs WHERE milestone_id = :mid`,
2433
+ ).run({ ":mid": milestoneId });
2229
2434
  currentDb!.prepare(
2230
2435
  `DELETE FROM tasks WHERE milestone_id = :mid`,
2231
2436
  ).run({ ":mid": milestoneId });
@@ -2359,6 +2564,30 @@ export function saveGateResult(g: {
2359
2564
  ":findings": g.findings,
2360
2565
  ":evaluated_at": new Date().toISOString(),
2361
2566
  });
2567
+
2568
+ const outcome =
2569
+ g.verdict === "pass"
2570
+ ? "pass"
2571
+ : g.verdict === "omitted"
2572
+ ? "manual-attention"
2573
+ : "fail";
2574
+ insertGateRun({
2575
+ traceId: `quality-gate:${g.milestoneId}:${g.sliceId}`,
2576
+ turnId: `gate:${g.gateId}:${g.taskId ?? "slice"}`,
2577
+ gateId: g.gateId,
2578
+ gateType: "quality-gate",
2579
+ milestoneId: g.milestoneId,
2580
+ sliceId: g.sliceId,
2581
+ taskId: g.taskId ?? undefined,
2582
+ outcome,
2583
+ failureClass: outcome === "fail" ? "verification" : outcome === "manual-attention" ? "manual-attention" : "none",
2584
+ rationale: g.rationale,
2585
+ findings: g.findings,
2586
+ attempt: 1,
2587
+ maxAttempts: 1,
2588
+ retryable: false,
2589
+ evaluatedAt: new Date().toISOString(),
2590
+ });
2362
2591
  }
2363
2592
 
2364
2593
  export function getPendingGates(milestoneId: string, sliceId: string, scope?: GateScope): GateRow[] {
@@ -2451,3 +2680,559 @@ export function getPendingGateCountForTurn(
2451
2680
  ): number {
2452
2681
  return getPendingGatesForTurn(milestoneId, sliceId, turn).length;
2453
2682
  }
2683
+
2684
+ export function insertGateRun(entry: {
2685
+ traceId: string;
2686
+ turnId: string;
2687
+ gateId: string;
2688
+ gateType: string;
2689
+ unitType?: string;
2690
+ unitId?: string;
2691
+ milestoneId?: string;
2692
+ sliceId?: string;
2693
+ taskId?: string;
2694
+ outcome: "pass" | "fail" | "retry" | "manual-attention";
2695
+ failureClass: "none" | "policy" | "input" | "execution" | "artifact" | "verification" | "closeout" | "git" | "timeout" | "manual-attention" | "unknown";
2696
+ rationale?: string;
2697
+ findings?: string;
2698
+ attempt: number;
2699
+ maxAttempts: number;
2700
+ retryable: boolean;
2701
+ evaluatedAt: string;
2702
+ }): void {
2703
+ if (!currentDb) return;
2704
+ currentDb.prepare(
2705
+ `INSERT INTO gate_runs (
2706
+ trace_id, turn_id, gate_id, gate_type, unit_type, unit_id, milestone_id, slice_id, task_id,
2707
+ outcome, failure_class, rationale, findings, attempt, max_attempts, retryable, evaluated_at
2708
+ ) VALUES (
2709
+ :trace_id, :turn_id, :gate_id, :gate_type, :unit_type, :unit_id, :milestone_id, :slice_id, :task_id,
2710
+ :outcome, :failure_class, :rationale, :findings, :attempt, :max_attempts, :retryable, :evaluated_at
2711
+ )`,
2712
+ ).run({
2713
+ ":trace_id": entry.traceId,
2714
+ ":turn_id": entry.turnId,
2715
+ ":gate_id": entry.gateId,
2716
+ ":gate_type": entry.gateType,
2717
+ ":unit_type": entry.unitType ?? null,
2718
+ ":unit_id": entry.unitId ?? null,
2719
+ ":milestone_id": entry.milestoneId ?? null,
2720
+ ":slice_id": entry.sliceId ?? null,
2721
+ ":task_id": entry.taskId ?? null,
2722
+ ":outcome": entry.outcome,
2723
+ ":failure_class": entry.failureClass,
2724
+ ":rationale": entry.rationale ?? "",
2725
+ ":findings": entry.findings ?? "",
2726
+ ":attempt": entry.attempt,
2727
+ ":max_attempts": entry.maxAttempts,
2728
+ ":retryable": entry.retryable ? 1 : 0,
2729
+ ":evaluated_at": entry.evaluatedAt,
2730
+ });
2731
+ }
2732
+
2733
+ export function upsertTurnGitTransaction(entry: {
2734
+ traceId: string;
2735
+ turnId: string;
2736
+ unitType?: string;
2737
+ unitId?: string;
2738
+ stage: string;
2739
+ action: "commit" | "snapshot" | "status-only";
2740
+ push: boolean;
2741
+ status: "ok" | "failed";
2742
+ error?: string;
2743
+ metadata?: Record<string, unknown>;
2744
+ updatedAt: string;
2745
+ }): void {
2746
+ if (!currentDb) return;
2747
+ currentDb.prepare(
2748
+ `INSERT OR REPLACE INTO turn_git_transactions (
2749
+ trace_id, turn_id, unit_type, unit_id, stage, action, push, status, error, metadata_json, updated_at
2750
+ ) VALUES (
2751
+ :trace_id, :turn_id, :unit_type, :unit_id, :stage, :action, :push, :status, :error, :metadata_json, :updated_at
2752
+ )`,
2753
+ ).run({
2754
+ ":trace_id": entry.traceId,
2755
+ ":turn_id": entry.turnId,
2756
+ ":unit_type": entry.unitType ?? null,
2757
+ ":unit_id": entry.unitId ?? null,
2758
+ ":stage": entry.stage,
2759
+ ":action": entry.action,
2760
+ ":push": entry.push ? 1 : 0,
2761
+ ":status": entry.status,
2762
+ ":error": entry.error ?? null,
2763
+ ":metadata_json": JSON.stringify(entry.metadata ?? {}),
2764
+ ":updated_at": entry.updatedAt,
2765
+ });
2766
+ }
2767
+
2768
+ export function insertAuditEvent(entry: {
2769
+ eventId: string;
2770
+ traceId: string;
2771
+ turnId?: string;
2772
+ causedBy?: string;
2773
+ category: string;
2774
+ type: string;
2775
+ ts: string;
2776
+ payload: Record<string, unknown>;
2777
+ }): void {
2778
+ if (!currentDb) return;
2779
+ transaction(() => {
2780
+ currentDb!.prepare(
2781
+ `INSERT OR IGNORE INTO audit_events (
2782
+ event_id, trace_id, turn_id, caused_by, category, type, ts, payload_json
2783
+ ) VALUES (
2784
+ :event_id, :trace_id, :turn_id, :caused_by, :category, :type, :ts, :payload_json
2785
+ )`,
2786
+ ).run({
2787
+ ":event_id": entry.eventId,
2788
+ ":trace_id": entry.traceId,
2789
+ ":turn_id": entry.turnId ?? null,
2790
+ ":caused_by": entry.causedBy ?? null,
2791
+ ":category": entry.category,
2792
+ ":type": entry.type,
2793
+ ":ts": entry.ts,
2794
+ ":payload_json": JSON.stringify(entry.payload ?? {}),
2795
+ });
2796
+
2797
+ if (entry.turnId) {
2798
+ const row = currentDb!.prepare(
2799
+ `SELECT event_count, first_ts, last_ts
2800
+ FROM audit_turn_index
2801
+ WHERE trace_id = :trace_id AND turn_id = :turn_id`,
2802
+ ).get({
2803
+ ":trace_id": entry.traceId,
2804
+ ":turn_id": entry.turnId,
2805
+ });
2806
+ if (row) {
2807
+ currentDb!.prepare(
2808
+ `UPDATE audit_turn_index
2809
+ SET first_ts = CASE WHEN :ts < first_ts THEN :ts ELSE first_ts END,
2810
+ last_ts = CASE WHEN :ts > last_ts THEN :ts ELSE last_ts END,
2811
+ event_count = event_count + 1
2812
+ WHERE trace_id = :trace_id AND turn_id = :turn_id`,
2813
+ ).run({
2814
+ ":trace_id": entry.traceId,
2815
+ ":turn_id": entry.turnId,
2816
+ ":ts": entry.ts,
2817
+ });
2818
+ } else {
2819
+ currentDb!.prepare(
2820
+ `INSERT INTO audit_turn_index (trace_id, turn_id, first_ts, last_ts, event_count)
2821
+ VALUES (:trace_id, :turn_id, :first_ts, :last_ts, :event_count)`,
2822
+ ).run({
2823
+ ":trace_id": entry.traceId,
2824
+ ":turn_id": entry.turnId,
2825
+ ":first_ts": entry.ts,
2826
+ ":last_ts": entry.ts,
2827
+ ":event_count": 1,
2828
+ });
2829
+ }
2830
+ }
2831
+ });
2832
+ }
2833
+
2834
+ // ─── Single-writer bypass wrappers ───────────────────────────────────────
2835
+ // These wrappers exist so modules outside this file never need to call
2836
+ // `_getAdapter()` for writes. Each one is a byte-equivalent replacement for
2837
+ // a raw prepare/run previously issued from another module. Keep them
2838
+ // minimal and direct — they exist to hold SQL text in one place, not to
2839
+ // add new behavior.
2840
+
2841
+ /** Delete a decision row by id. Used by db-writer.ts rollback on disk-write failure. */
2842
+ export function deleteDecisionById(id: string): void {
2843
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
2844
+ currentDb.prepare("DELETE FROM decisions WHERE id = :id").run({ ":id": id });
2845
+ }
2846
+
2847
+ /** Delete a requirement row by id. Used by db-writer.ts rollback on disk-write failure. */
2848
+ export function deleteRequirementById(id: string): void {
2849
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
2850
+ currentDb.prepare("DELETE FROM requirements WHERE id = :id").run({ ":id": id });
2851
+ }
2852
+
2853
+ /** Delete an artifact row by path. Used by db-writer.ts rollback on disk-write failure. */
2854
+ export function deleteArtifactByPath(path: string): void {
2855
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
2856
+ currentDb.prepare("DELETE FROM artifacts WHERE path = :path").run({ ":path": path });
2857
+ }
2858
+
2859
+ /**
2860
+ * Drop all rows from tasks/slices/milestones in dependency order inside a
2861
+ * transaction. Used by `gsd recover` to rebuild engine state from markdown.
2862
+ */
2863
+ export function clearEngineHierarchy(): void {
2864
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
2865
+ transaction(() => {
2866
+ currentDb!.exec("DELETE FROM tasks");
2867
+ currentDb!.exec("DELETE FROM slices");
2868
+ currentDb!.exec("DELETE FROM milestones");
2869
+ });
2870
+ }
2871
+
2872
+ /**
2873
+ * INSERT OR IGNORE a slice during event replay (workflow-reconcile.ts).
2874
+ * Strict insert-or-ignore semantics are required here to avoid the
2875
+ * `insertSlice` ON CONFLICT path that could downgrade an already-completed
2876
+ * slice back to 'pending'.
2877
+ */
2878
+ export function insertOrIgnoreSlice(args: {
2879
+ milestoneId: string;
2880
+ sliceId: string;
2881
+ title: string;
2882
+ createdAt: string;
2883
+ }): void {
2884
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
2885
+ currentDb.prepare(
2886
+ `INSERT OR IGNORE INTO slices (milestone_id, id, title, status, created_at)
2887
+ VALUES (:mid, :sid, :title, 'pending', :ts)`,
2888
+ ).run({
2889
+ ":mid": args.milestoneId,
2890
+ ":sid": args.sliceId,
2891
+ ":title": args.title,
2892
+ ":ts": args.createdAt,
2893
+ });
2894
+ }
2895
+
2896
+ /**
2897
+ * INSERT OR IGNORE a task during event replay (workflow-reconcile.ts).
2898
+ * Same rationale as `insertOrIgnoreSlice`.
2899
+ */
2900
+ export function insertOrIgnoreTask(args: {
2901
+ milestoneId: string;
2902
+ sliceId: string;
2903
+ taskId: string;
2904
+ title: string;
2905
+ createdAt: string;
2906
+ }): void {
2907
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
2908
+ currentDb.prepare(
2909
+ `INSERT OR IGNORE INTO tasks (milestone_id, slice_id, id, title, status, created_at)
2910
+ VALUES (:mid, :sid, :tid, :title, 'pending', :ts)`,
2911
+ ).run({
2912
+ ":mid": args.milestoneId,
2913
+ ":sid": args.sliceId,
2914
+ ":tid": args.taskId,
2915
+ ":title": args.title,
2916
+ ":ts": args.createdAt,
2917
+ });
2918
+ }
2919
+
2920
+ /**
2921
+ * Stamp the `replan_triggered_at` column on a slice. Used by triage-resolution
2922
+ * when a user capture requests a replan so the dispatcher can detect the
2923
+ * trigger via DB in addition to the on-disk REPLAN-TRIGGER.md marker.
2924
+ */
2925
+ export function setSliceReplanTriggeredAt(milestoneId: string, sliceId: string, ts: string): void {
2926
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
2927
+ currentDb.prepare(
2928
+ "UPDATE slices SET replan_triggered_at = :ts WHERE milestone_id = :mid AND id = :sid",
2929
+ ).run({ ":ts": ts, ":mid": milestoneId, ":sid": sliceId });
2930
+ }
2931
+
2932
+ /**
2933
+ * INSERT OR REPLACE a quality_gates row. Used by milestone-validation-gates.ts
2934
+ * to persist milestone-level (MV*) gate outcomes after validate-milestone runs.
2935
+ */
2936
+ export function upsertQualityGate(g: {
2937
+ milestoneId: string;
2938
+ sliceId: string;
2939
+ gateId: string;
2940
+ scope: string;
2941
+ taskId: string;
2942
+ status: string;
2943
+ verdict: string;
2944
+ rationale: string;
2945
+ findings: string;
2946
+ evaluatedAt: string;
2947
+ }): void {
2948
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
2949
+ currentDb.prepare(
2950
+ `INSERT OR REPLACE INTO quality_gates
2951
+ (milestone_id, slice_id, gate_id, scope, task_id, status, verdict, rationale, findings, evaluated_at)
2952
+ VALUES (:mid, :sid, :gid, :scope, :tid, :status, :verdict, :rationale, :findings, :evaluated_at)`,
2953
+ ).run({
2954
+ ":mid": g.milestoneId,
2955
+ ":sid": g.sliceId,
2956
+ ":gid": g.gateId,
2957
+ ":scope": g.scope,
2958
+ ":tid": g.taskId,
2959
+ ":status": g.status,
2960
+ ":verdict": g.verdict,
2961
+ ":rationale": g.rationale,
2962
+ ":findings": g.findings,
2963
+ ":evaluated_at": g.evaluatedAt,
2964
+ });
2965
+ }
2966
+
2967
+ /**
2968
+ * Atomically replace all workflow state from a manifest. Lifted verbatim from
2969
+ * workflow-manifest.ts so the single-writer invariant holds. Only touches
2970
+ * engine tables + decisions. Does NOT modify artifacts or memories.
2971
+ */
2972
+ export function restoreManifest(manifest: StateManifest): void {
2973
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
2974
+ const db = currentDb;
2975
+
2976
+ transaction(() => {
2977
+ // Clear engine tables (order matters for foreign-key-like consistency)
2978
+ db.exec("DELETE FROM verification_evidence");
2979
+ db.exec("DELETE FROM tasks");
2980
+ db.exec("DELETE FROM slices");
2981
+ db.exec("DELETE FROM milestones");
2982
+ db.exec("DELETE FROM decisions WHERE 1=1");
2983
+
2984
+ // Restore milestones
2985
+ const msStmt = db.prepare(
2986
+ `INSERT INTO milestones (id, title, status, depends_on, created_at, completed_at,
2987
+ vision, success_criteria, key_risks, proof_strategy,
2988
+ verification_contract, verification_integration, verification_operational, verification_uat,
2989
+ definition_of_done, requirement_coverage, boundary_map_markdown)
2990
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
2991
+ );
2992
+ for (const m of manifest.milestones) {
2993
+ msStmt.run(
2994
+ m.id, m.title, m.status,
2995
+ JSON.stringify(m.depends_on), m.created_at, m.completed_at,
2996
+ m.vision, JSON.stringify(m.success_criteria), JSON.stringify(m.key_risks),
2997
+ JSON.stringify(m.proof_strategy),
2998
+ m.verification_contract, m.verification_integration, m.verification_operational, m.verification_uat,
2999
+ JSON.stringify(m.definition_of_done), m.requirement_coverage, m.boundary_map_markdown,
3000
+ );
3001
+ }
3002
+
3003
+ // Restore slices
3004
+ const slStmt = db.prepare(
3005
+ `INSERT INTO slices (milestone_id, id, title, status, risk, depends, demo,
3006
+ created_at, completed_at, full_summary_md, full_uat_md,
3007
+ goal, success_criteria, proof_level, integration_closure, observability_impact,
3008
+ sequence, replan_triggered_at)
3009
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
3010
+ );
3011
+ for (const s of manifest.slices) {
3012
+ slStmt.run(
3013
+ s.milestone_id, s.id, s.title, s.status, s.risk,
3014
+ JSON.stringify(s.depends), s.demo,
3015
+ s.created_at, s.completed_at, s.full_summary_md, s.full_uat_md,
3016
+ s.goal, s.success_criteria, s.proof_level, s.integration_closure, s.observability_impact,
3017
+ s.sequence, s.replan_triggered_at,
3018
+ );
3019
+ }
3020
+
3021
+ // Restore tasks
3022
+ const tkStmt = db.prepare(
3023
+ `INSERT INTO tasks (milestone_id, slice_id, id, title, status,
3024
+ one_liner, narrative, verification_result, duration, completed_at,
3025
+ blocker_discovered, deviations, known_issues, key_files, key_decisions,
3026
+ full_summary_md, description, estimate, files, verify,
3027
+ inputs, expected_output, observability_impact, sequence)
3028
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
3029
+ );
3030
+ for (const t of manifest.tasks) {
3031
+ tkStmt.run(
3032
+ t.milestone_id, t.slice_id, t.id, t.title, t.status,
3033
+ t.one_liner, t.narrative, t.verification_result, t.duration, t.completed_at,
3034
+ t.blocker_discovered ? 1 : 0, t.deviations, t.known_issues,
3035
+ JSON.stringify(t.key_files), JSON.stringify(t.key_decisions),
3036
+ t.full_summary_md, t.description, t.estimate, JSON.stringify(t.files), t.verify,
3037
+ JSON.stringify(t.inputs), JSON.stringify(t.expected_output),
3038
+ t.observability_impact, t.sequence,
3039
+ );
3040
+ }
3041
+
3042
+ // Restore decisions
3043
+ const dcStmt = db.prepare(
3044
+ `INSERT INTO decisions (seq, id, when_context, scope, decision, choice, rationale, revisable, made_by, superseded_by)
3045
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
3046
+ );
3047
+ for (const d of manifest.decisions) {
3048
+ 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);
3049
+ }
3050
+
3051
+ // Restore verification evidence
3052
+ const evStmt = db.prepare(
3053
+ `INSERT INTO verification_evidence (task_id, slice_id, milestone_id, command, exit_code, verdict, duration_ms, created_at)
3054
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?)`,
3055
+ );
3056
+ for (const e of manifest.verification_evidence) {
3057
+ evStmt.run(e.task_id, e.slice_id, e.milestone_id, e.command, e.exit_code, e.verdict, e.duration_ms, e.created_at);
3058
+ }
3059
+ });
3060
+ }
3061
+
3062
+ // ─── Legacy markdown → DB bulk migration ─────────────────────────────────
3063
+
3064
+ export interface LegacyMilestoneInsert {
3065
+ id: string;
3066
+ title: string;
3067
+ status: string;
3068
+ }
3069
+
3070
+ export interface LegacySliceInsert {
3071
+ id: string;
3072
+ milestoneId: string;
3073
+ title: string;
3074
+ status: string;
3075
+ risk: string;
3076
+ sequence: number;
3077
+ }
3078
+
3079
+ export interface LegacyTaskInsert {
3080
+ id: string;
3081
+ sliceId: string;
3082
+ milestoneId: string;
3083
+ title: string;
3084
+ status: string;
3085
+ sequence: number;
3086
+ }
3087
+
3088
+ /**
3089
+ * Bulk delete + insert a legacy milestone hierarchy for markdown → DB migration.
3090
+ * Used by workflow-migration.ts to populate engine tables from parsed ROADMAP/PLAN
3091
+ * files. All operations run inside a single transaction.
3092
+ */
3093
+ export function bulkInsertLegacyHierarchy(payload: {
3094
+ milestones: LegacyMilestoneInsert[];
3095
+ slices: LegacySliceInsert[];
3096
+ tasks: LegacyTaskInsert[];
3097
+ clearMilestoneIds: string[];
3098
+ createdAt: string;
3099
+ }): void {
3100
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
3101
+ const db = currentDb;
3102
+ const { milestones, slices, tasks, clearMilestoneIds, createdAt } = payload;
3103
+
3104
+ if (clearMilestoneIds.length === 0) return;
3105
+ const placeholders = clearMilestoneIds.map(() => "?").join(",");
3106
+
3107
+ transaction(() => {
3108
+ db.prepare(`DELETE FROM tasks WHERE milestone_id IN (${placeholders})`).run(...clearMilestoneIds);
3109
+ db.prepare(`DELETE FROM slices WHERE milestone_id IN (${placeholders})`).run(...clearMilestoneIds);
3110
+ db.prepare(`DELETE FROM milestones WHERE id IN (${placeholders})`).run(...clearMilestoneIds);
3111
+
3112
+ const insertMilestone = db.prepare(
3113
+ "INSERT INTO milestones (id, title, status, created_at) VALUES (?, ?, ?, ?)",
3114
+ );
3115
+ for (const m of milestones) {
3116
+ insertMilestone.run(m.id, m.title, m.status, createdAt);
3117
+ }
3118
+
3119
+ const insertSliceStmt = db.prepare(
3120
+ "INSERT INTO slices (id, milestone_id, title, status, risk, depends, sequence, created_at) VALUES (?, ?, ?, ?, ?, ?, ?, ?)",
3121
+ );
3122
+ for (const s of slices) {
3123
+ insertSliceStmt.run(s.id, s.milestoneId, s.title, s.status, s.risk, "[]", s.sequence, createdAt);
3124
+ }
3125
+
3126
+ const insertTaskStmt = db.prepare(
3127
+ "INSERT INTO tasks (id, slice_id, milestone_id, title, description, status, estimate, files, sequence) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)",
3128
+ );
3129
+ for (const t of tasks) {
3130
+ insertTaskStmt.run(t.id, t.sliceId, t.milestoneId, t.title, "", t.status, "", "[]", t.sequence);
3131
+ }
3132
+ });
3133
+ }
3134
+
3135
+ // ─── Memory store writers ────────────────────────────────────────────────
3136
+ // All memory writes go through gsd-db.ts so the single-writer invariant
3137
+ // holds. These are direct pass-throughs to the SQL previously in
3138
+ // memory-store.ts — same bindings, same behavior.
3139
+
3140
+ export function insertMemoryRow(args: {
3141
+ id: string;
3142
+ category: string;
3143
+ content: string;
3144
+ confidence: number;
3145
+ sourceUnitType: string | null;
3146
+ sourceUnitId: string | null;
3147
+ createdAt: string;
3148
+ updatedAt: string;
3149
+ }): void {
3150
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
3151
+ currentDb.prepare(
3152
+ `INSERT INTO memories (id, category, content, confidence, source_unit_type, source_unit_id, created_at, updated_at)
3153
+ VALUES (:id, :category, :content, :confidence, :source_unit_type, :source_unit_id, :created_at, :updated_at)`,
3154
+ ).run({
3155
+ ":id": args.id,
3156
+ ":category": args.category,
3157
+ ":content": args.content,
3158
+ ":confidence": args.confidence,
3159
+ ":source_unit_type": args.sourceUnitType,
3160
+ ":source_unit_id": args.sourceUnitId,
3161
+ ":created_at": args.createdAt,
3162
+ ":updated_at": args.updatedAt,
3163
+ });
3164
+ }
3165
+
3166
+ export function rewriteMemoryId(placeholderId: string, realId: string): void {
3167
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
3168
+ currentDb.prepare("UPDATE memories SET id = :real_id WHERE id = :placeholder").run({
3169
+ ":real_id": realId,
3170
+ ":placeholder": placeholderId,
3171
+ });
3172
+ }
3173
+
3174
+ export function updateMemoryContentRow(
3175
+ id: string,
3176
+ content: string,
3177
+ confidence: number | undefined,
3178
+ updatedAt: string,
3179
+ ): void {
3180
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
3181
+ if (confidence != null) {
3182
+ currentDb.prepare(
3183
+ "UPDATE memories SET content = :content, confidence = :confidence, updated_at = :updated_at WHERE id = :id",
3184
+ ).run({ ":content": content, ":confidence": confidence, ":updated_at": updatedAt, ":id": id });
3185
+ } else {
3186
+ currentDb.prepare(
3187
+ "UPDATE memories SET content = :content, updated_at = :updated_at WHERE id = :id",
3188
+ ).run({ ":content": content, ":updated_at": updatedAt, ":id": id });
3189
+ }
3190
+ }
3191
+
3192
+ export function incrementMemoryHitCount(id: string, updatedAt: string): void {
3193
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
3194
+ currentDb.prepare(
3195
+ "UPDATE memories SET hit_count = hit_count + 1, updated_at = :updated_at WHERE id = :id",
3196
+ ).run({ ":updated_at": updatedAt, ":id": id });
3197
+ }
3198
+
3199
+ export function supersedeMemoryRow(oldId: string, newId: string, updatedAt: string): void {
3200
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
3201
+ currentDb.prepare(
3202
+ "UPDATE memories SET superseded_by = :new_id, updated_at = :updated_at WHERE id = :old_id",
3203
+ ).run({ ":new_id": newId, ":updated_at": updatedAt, ":old_id": oldId });
3204
+ }
3205
+
3206
+ export function markMemoryUnitProcessed(
3207
+ unitKey: string,
3208
+ activityFile: string,
3209
+ processedAt: string,
3210
+ ): void {
3211
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
3212
+ currentDb.prepare(
3213
+ `INSERT OR IGNORE INTO memory_processed_units (unit_key, activity_file, processed_at)
3214
+ VALUES (:key, :file, :at)`,
3215
+ ).run({ ":key": unitKey, ":file": activityFile, ":at": processedAt });
3216
+ }
3217
+
3218
+ export function decayMemoriesBefore(cutoffTs: string, now: string): void {
3219
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
3220
+ currentDb.prepare(
3221
+ `UPDATE memories
3222
+ SET confidence = MAX(0.1, confidence - 0.1), updated_at = :now
3223
+ WHERE superseded_by IS NULL AND updated_at < :cutoff AND confidence > 0.1`,
3224
+ ).run({ ":now": now, ":cutoff": cutoffTs });
3225
+ }
3226
+
3227
+ export function supersedeLowestRankedMemories(limit: number, now: string): void {
3228
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
3229
+ currentDb.prepare(
3230
+ `UPDATE memories SET superseded_by = 'CAP_EXCEEDED', updated_at = :now
3231
+ WHERE id IN (
3232
+ SELECT id FROM memories
3233
+ WHERE superseded_by IS NULL
3234
+ ORDER BY (confidence * (1.0 + hit_count * 0.1)) ASC
3235
+ LIMIT :limit
3236
+ )`,
3237
+ ).run({ ":now": now, ":limit": limit });
3238
+ }