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.
- package/dist/cli.js +85 -0
- package/dist/headless-query.js +4 -1
- package/dist/help-text.js +23 -0
- package/dist/resources/extensions/gsd/activity-log.js +16 -0
- package/dist/resources/extensions/gsd/auto/detect-stuck.js +11 -4
- package/dist/resources/extensions/gsd/auto/loop.js +147 -10
- package/dist/resources/extensions/gsd/auto/phases.js +158 -4
- package/dist/resources/extensions/gsd/auto/session.js +10 -0
- package/dist/resources/extensions/gsd/auto-dispatch.js +11 -1
- package/dist/resources/extensions/gsd/auto-model-selection.js +51 -5
- package/dist/resources/extensions/gsd/auto-post-unit.js +220 -17
- package/dist/resources/extensions/gsd/auto-prompts.js +12 -0
- package/dist/resources/extensions/gsd/auto-unit-closeout.js +18 -0
- package/dist/resources/extensions/gsd/auto-verification.js +100 -2
- package/dist/resources/extensions/gsd/auto.js +36 -4
- package/dist/resources/extensions/gsd/bootstrap/register-extension.js +30 -8
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +45 -4
- package/dist/resources/extensions/gsd/commands/catalog.js +26 -1
- package/dist/resources/extensions/gsd/commands/handlers/ops.js +25 -0
- package/dist/resources/extensions/gsd/commands/handlers/workflow.js +68 -9
- package/dist/resources/extensions/gsd/commands-add-tests.js +111 -0
- package/dist/resources/extensions/gsd/commands-backlog.js +140 -0
- package/dist/resources/extensions/gsd/commands-do.js +79 -0
- package/dist/resources/extensions/gsd/commands-extract-learnings.js +225 -0
- package/dist/resources/extensions/gsd/commands-maintenance.js +6 -6
- package/dist/resources/extensions/gsd/commands-pr-branch.js +180 -0
- package/dist/resources/extensions/gsd/commands-prefs-wizard.js +1 -1
- package/dist/resources/extensions/gsd/commands-session-report.js +82 -0
- package/dist/resources/extensions/gsd/commands-ship.js +187 -0
- package/dist/resources/extensions/gsd/db-writer.js +3 -5
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +14 -1
- package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +144 -0
- package/dist/resources/extensions/gsd/ecosystem/loader.js +145 -0
- package/dist/resources/extensions/gsd/git-service.js +49 -1
- package/dist/resources/extensions/gsd/graph-context.js +157 -0
- package/dist/resources/extensions/gsd/gsd-db.js +581 -2
- package/dist/resources/extensions/gsd/guided-flow.js +23 -0
- package/dist/resources/extensions/gsd/index.js +15 -2
- package/dist/resources/extensions/gsd/init-wizard.js +1 -0
- package/dist/resources/extensions/gsd/journal.js +27 -0
- package/dist/resources/extensions/gsd/md-importer.js +3 -4
- package/dist/resources/extensions/gsd/memory-store.js +19 -51
- package/dist/resources/extensions/gsd/metrics.js +19 -0
- package/dist/resources/extensions/gsd/milestone-validation-gates.js +13 -12
- package/dist/resources/extensions/gsd/native-git-bridge.js +7 -4
- package/dist/resources/extensions/gsd/parallel-orchestrator.js +33 -1
- package/dist/resources/extensions/gsd/preferences-models.js +20 -3
- package/dist/resources/extensions/gsd/preferences-types.js +1 -0
- package/dist/resources/extensions/gsd/preferences-validation.js +108 -2
- package/dist/resources/extensions/gsd/preferences.js +26 -0
- package/dist/resources/extensions/gsd/prompts/add-tests.md +35 -0
- package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +12 -2
- package/dist/resources/extensions/gsd/state.js +5 -1
- package/dist/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
- package/dist/resources/extensions/gsd/tools/complete-slice.js +20 -0
- package/dist/resources/extensions/gsd/tools/validate-milestone.js +39 -4
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +3 -14
- package/dist/resources/extensions/gsd/triage-resolution.js +2 -5
- package/dist/resources/extensions/gsd/unit-ownership.js +1 -1
- package/dist/resources/extensions/gsd/uok/audit-toggle.js +7 -0
- package/dist/resources/extensions/gsd/uok/audit.js +40 -0
- package/dist/resources/extensions/gsd/uok/contracts.js +1 -0
- package/dist/resources/extensions/gsd/uok/execution-graph.js +179 -0
- package/dist/resources/extensions/gsd/uok/flags.js +29 -0
- package/dist/resources/extensions/gsd/uok/gate-runner.js +109 -0
- package/dist/resources/extensions/gsd/uok/gitops.js +53 -0
- package/dist/resources/extensions/gsd/uok/kernel.js +80 -0
- package/dist/resources/extensions/gsd/uok/loop-adapter.js +133 -0
- package/dist/resources/extensions/gsd/uok/model-policy.js +66 -0
- package/dist/resources/extensions/gsd/uok/plan-v2.js +132 -0
- package/dist/resources/extensions/gsd/workflow-logger.js +22 -0
- package/dist/resources/extensions/gsd/workflow-manifest.js +8 -69
- package/dist/resources/extensions/gsd/workflow-migration.js +21 -22
- package/dist/resources/extensions/gsd/workflow-projections.js +4 -1
- package/dist/resources/extensions/gsd/workflow-reconcile.js +14 -11
- package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
- package/dist/tsconfig.extensions.tsbuildinfo +1 -0
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
- package/dist/web/standalone/.next/build-manifest.json +2 -2
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
- package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/package.json +3 -2
- package/packages/daemon/package.json +2 -2
- package/packages/mcp-server/dist/index.d.ts +3 -0
- package/packages/mcp-server/dist/index.d.ts.map +1 -1
- package/packages/mcp-server/dist/index.js +3 -0
- package/packages/mcp-server/dist/index.js.map +1 -1
- package/packages/mcp-server/dist/readers/graph.d.ts +87 -0
- package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -0
- package/packages/mcp-server/dist/readers/graph.js +655 -0
- package/packages/mcp-server/dist/readers/graph.js.map +1 -0
- package/packages/mcp-server/dist/readers/index.d.ts +2 -0
- package/packages/mcp-server/dist/readers/index.d.ts.map +1 -1
- package/packages/mcp-server/dist/readers/index.js +1 -0
- package/packages/mcp-server/dist/readers/index.js.map +1 -1
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +65 -0
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/package.json +2 -2
- package/packages/mcp-server/src/index.ts +15 -0
- package/packages/mcp-server/src/readers/graph.test.ts +604 -0
- package/packages/mcp-server/src/readers/graph.ts +855 -0
- package/packages/mcp-server/src/readers/index.ts +12 -0
- package/packages/mcp-server/src/server.ts +83 -0
- package/packages/mcp-server/tsconfig.json +1 -0
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -0
- package/packages/native/package.json +2 -2
- package/packages/native/tsconfig.tsbuildinfo +1 -0
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-agent-core/tsconfig.json +1 -0
- package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -0
- package/packages/pi-ai/dist/index.d.ts +1 -9
- package/packages/pi-ai/dist/index.d.ts.map +1 -1
- package/packages/pi-ai/dist/index.js +1 -9
- package/packages/pi-ai/dist/index.js.map +1 -1
- package/packages/pi-ai/dist/models/capability-patches.d.ts +19 -0
- package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/capability-patches.js +36 -0
- package/packages/pi-ai/dist/models/capability-patches.js.map +1 -0
- package/packages/pi-ai/dist/{models.custom.d.ts → models/custom.d.ts} +1 -1
- package/packages/pi-ai/dist/models/custom.d.ts.map +1 -0
- package/packages/pi-ai/dist/{models.custom.js → models/custom.js} +4 -4
- package/packages/pi-ai/dist/models/custom.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts +1482 -0
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.js +1484 -0
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/anthropic.d.ts +377 -0
- package/packages/pi-ai/dist/models/generated/anthropic.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/anthropic.js +379 -0
- package/packages/pi-ai/dist/models/generated/anthropic.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts +700 -0
- package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/azure-openai-responses.js +702 -0
- package/packages/pi-ai/dist/models/generated/azure-openai-responses.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/cerebras.d.ts +71 -0
- package/packages/pi-ai/dist/models/generated/cerebras.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/cerebras.js +73 -0
- package/packages/pi-ai/dist/models/generated/cerebras.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/github-copilot.d.ts +590 -0
- package/packages/pi-ai/dist/models/generated/github-copilot.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/github-copilot.js +444 -0
- package/packages/pi-ai/dist/models/generated/github-copilot.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts +156 -0
- package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-antigravity.js +158 -0
- package/packages/pi-ai/dist/models/generated/google-antigravity.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts +105 -0
- package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-gemini-cli.js +107 -0
- package/packages/pi-ai/dist/models/generated/google-gemini-cli.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-vertex.d.ts +207 -0
- package/packages/pi-ai/dist/models/generated/google-vertex.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/google-vertex.js +209 -0
- package/packages/pi-ai/dist/models/generated/google-vertex.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/google.d.ts +462 -0
- package/packages/pi-ai/dist/models/generated/google.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/google.js +464 -0
- package/packages/pi-ai/dist/models/generated/google.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/groq.d.ts +309 -0
- package/packages/pi-ai/dist/models/generated/groq.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/groq.js +311 -0
- package/packages/pi-ai/dist/models/generated/groq.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/huggingface.d.ts +383 -0
- package/packages/pi-ai/dist/models/generated/huggingface.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/huggingface.js +347 -0
- package/packages/pi-ai/dist/models/generated/huggingface.js.map +1 -0
- package/packages/pi-ai/dist/{models.generated.d.ts → models/generated/index.d.ts} +1 -1
- package/packages/pi-ai/dist/{models.generated.d.ts.map → models/generated/index.d.ts.map} +1 -1
- package/packages/pi-ai/dist/models/generated/index.js +51 -0
- package/packages/pi-ai/dist/models/generated/index.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts +37 -0
- package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/kimi-coding.js +39 -0
- package/packages/pi-ai/dist/models/generated/kimi-coding.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts +105 -0
- package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/minimax-cn.js +107 -0
- package/packages/pi-ai/dist/models/generated/minimax-cn.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/minimax.d.ts +105 -0
- package/packages/pi-ai/dist/models/generated/minimax.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/minimax.js +107 -0
- package/packages/pi-ai/dist/models/generated/minimax.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/mistral.d.ts +445 -0
- package/packages/pi-ai/dist/models/generated/mistral.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/mistral.js +447 -0
- package/packages/pi-ai/dist/models/generated/mistral.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +139 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.js +141 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/openai.d.ts +700 -0
- package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/openai.js +702 -0
- package/packages/pi-ai/dist/models/generated/openai.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/opencode-go.d.ts +122 -0
- package/packages/pi-ai/dist/models/generated/opencode-go.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/opencode-go.js +124 -0
- package/packages/pi-ai/dist/models/generated/opencode-go.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/opencode.d.ts +530 -0
- package/packages/pi-ai/dist/models/generated/opencode.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/opencode.js +532 -0
- package/packages/pi-ai/dist/models/generated/opencode.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/openrouter.d.ts +4270 -0
- package/packages/pi-ai/dist/models/generated/openrouter.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/openrouter.js +4272 -0
- package/packages/pi-ai/dist/models/generated/openrouter.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts +2604 -0
- package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js +2606 -0
- package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/xai.d.ts +411 -0
- package/packages/pi-ai/dist/models/generated/xai.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/xai.js +413 -0
- package/packages/pi-ai/dist/models/generated/xai.js.map +1 -0
- package/packages/pi-ai/dist/models/generated/zai.d.ts +276 -0
- package/packages/pi-ai/dist/models/generated/zai.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/generated/zai.js +239 -0
- package/packages/pi-ai/dist/models/generated/zai.js.map +1 -0
- package/packages/pi-ai/dist/models/index.d.ts +27 -0
- package/packages/pi-ai/dist/models/index.d.ts.map +1 -0
- package/packages/pi-ai/dist/models/index.js +80 -0
- package/packages/pi-ai/dist/models/index.js.map +1 -0
- package/packages/pi-ai/dist/models.d.ts +1 -36
- package/packages/pi-ai/dist/models.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.test.js +1 -2
- package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
- package/packages/pi-ai/dist/models.js +3 -112
- package/packages/pi-ai/dist/models.js.map +1 -1
- package/packages/pi-ai/dist/models.test.js +6 -5
- package/packages/pi-ai/dist/models.test.js.map +1 -1
- package/packages/pi-ai/package.json +1 -1
- package/packages/pi-ai/scripts/generate-models.ts +74 -40
- package/packages/pi-ai/src/index.ts +1 -9
- package/packages/pi-ai/src/models/capability-patches.ts +40 -0
- package/packages/pi-ai/src/{models.custom.ts → models/custom.ts} +4 -4
- package/packages/pi-ai/src/models/generated/amazon-bedrock.ts +1486 -0
- package/packages/pi-ai/src/models/generated/anthropic.ts +381 -0
- package/packages/pi-ai/src/models/generated/azure-openai-responses.ts +704 -0
- package/packages/pi-ai/src/models/generated/cerebras.ts +75 -0
- package/packages/pi-ai/src/models/generated/github-copilot.ts +446 -0
- package/packages/pi-ai/src/models/generated/google-antigravity.ts +160 -0
- package/packages/pi-ai/src/models/generated/google-gemini-cli.ts +109 -0
- package/packages/pi-ai/src/models/generated/google-vertex.ts +211 -0
- package/packages/pi-ai/src/models/generated/google.ts +466 -0
- package/packages/pi-ai/src/models/generated/groq.ts +313 -0
- package/packages/pi-ai/src/models/generated/huggingface.ts +349 -0
- package/packages/pi-ai/src/models/generated/index.ts +52 -0
- package/packages/pi-ai/src/models/generated/kimi-coding.ts +41 -0
- package/packages/pi-ai/src/models/generated/minimax-cn.ts +109 -0
- package/packages/pi-ai/src/models/generated/minimax.ts +109 -0
- package/packages/pi-ai/src/models/generated/mistral.ts +449 -0
- package/packages/pi-ai/src/models/generated/openai-codex.ts +143 -0
- package/packages/pi-ai/src/models/generated/openai.ts +704 -0
- package/packages/pi-ai/src/models/generated/opencode-go.ts +126 -0
- package/packages/pi-ai/src/models/generated/opencode.ts +534 -0
- package/packages/pi-ai/src/models/generated/openrouter.ts +4274 -0
- package/packages/pi-ai/src/models/generated/vercel-ai-gateway.ts +2608 -0
- package/packages/pi-ai/src/models/generated/xai.ts +415 -0
- package/packages/pi-ai/src/models/generated/zai.ts +241 -0
- package/packages/pi-ai/src/models/index.ts +106 -0
- package/packages/pi-ai/src/models.generated.test.ts +1 -2
- package/packages/pi-ai/src/models.test.ts +6 -5
- package/packages/pi-ai/src/models.ts +3 -153
- package/packages/pi-ai/tsconfig.json +1 -0
- package/packages/pi-ai/tsconfig.tsbuildinfo +1 -0
- package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session.js +8 -2
- package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +472 -0
- package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js +52 -0
- package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry.js +2 -2
- package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +11 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +23 -9
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts +11 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +47 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +51 -8
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js +22 -22
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +232 -18
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +38 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +13 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +53 -6
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/src/core/agent-session.ts +12 -6
- package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +612 -0
- package/packages/pi-coding-agent/src/core/model-registry-env-fallback.test.ts +59 -0
- package/packages/pi-coding-agent/src/core/model-registry.ts +2 -1
- package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +19 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +25 -10
- package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +67 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +66 -7
- package/packages/pi-coding-agent/src/modes/interactive/components/user-message.ts +23 -26
- package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +298 -41
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +44 -0
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +73 -6
- package/packages/pi-coding-agent/src/types/ambient-modules.d.ts +69 -0
- package/packages/pi-coding-agent/tsconfig.json +3 -2
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -0
- package/packages/pi-tui/package.json +1 -1
- package/packages/pi-tui/tsconfig.json +1 -0
- package/packages/pi-tui/tsconfig.tsbuildinfo +1 -0
- package/packages/rpc-client/package.json +1 -1
- package/packages/rpc-client/tsconfig.json +1 -0
- package/packages/rpc-client/tsconfig.tsbuildinfo +1 -0
- package/src/resources/extensions/gsd/activity-log.ts +21 -0
- package/src/resources/extensions/gsd/auto/detect-stuck.ts +12 -4
- package/src/resources/extensions/gsd/auto/loop-deps.ts +10 -0
- package/src/resources/extensions/gsd/auto/loop.ts +159 -10
- package/src/resources/extensions/gsd/auto/phases.ts +191 -4
- package/src/resources/extensions/gsd/auto/session.ts +10 -0
- package/src/resources/extensions/gsd/auto-dispatch.ts +16 -6
- package/src/resources/extensions/gsd/auto-model-selection.ts +66 -5
- package/src/resources/extensions/gsd/auto-post-unit.ts +238 -18
- package/src/resources/extensions/gsd/auto-prompts.ts +13 -0
- package/src/resources/extensions/gsd/auto-unit-closeout.ts +25 -1
- package/src/resources/extensions/gsd/auto-verification.ts +129 -2
- package/src/resources/extensions/gsd/auto.ts +41 -2
- package/src/resources/extensions/gsd/bootstrap/register-extension.ts +38 -8
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +56 -3
- package/src/resources/extensions/gsd/commands/catalog.ts +26 -1
- package/src/resources/extensions/gsd/commands/handlers/ops.ts +25 -0
- package/src/resources/extensions/gsd/commands/handlers/workflow.ts +74 -9
- package/src/resources/extensions/gsd/commands-add-tests.ts +137 -0
- package/src/resources/extensions/gsd/commands-backlog.ts +182 -0
- package/src/resources/extensions/gsd/commands-do.ts +109 -0
- package/src/resources/extensions/gsd/commands-extract-learnings.ts +304 -0
- package/src/resources/extensions/gsd/commands-maintenance.ts +6 -6
- package/src/resources/extensions/gsd/commands-pr-branch.ts +234 -0
- package/src/resources/extensions/gsd/commands-prefs-wizard.ts +1 -1
- package/src/resources/extensions/gsd/commands-session-report.ts +101 -0
- package/src/resources/extensions/gsd/commands-ship.ts +219 -0
- package/src/resources/extensions/gsd/db-writer.ts +3 -5
- package/src/resources/extensions/gsd/docs/preferences-reference.md +14 -1
- package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +228 -0
- package/src/resources/extensions/gsd/ecosystem/loader.ts +201 -0
- package/src/resources/extensions/gsd/git-service.ts +68 -0
- package/src/resources/extensions/gsd/graph-context.ts +212 -0
- package/src/resources/extensions/gsd/gsd-db.ts +788 -3
- package/src/resources/extensions/gsd/guided-flow.ts +32 -0
- package/src/resources/extensions/gsd/index.ts +18 -2
- package/src/resources/extensions/gsd/init-wizard.ts +3 -2
- package/src/resources/extensions/gsd/journal.ts +30 -0
- package/src/resources/extensions/gsd/md-importer.ts +3 -5
- package/src/resources/extensions/gsd/memory-store.ts +31 -62
- package/src/resources/extensions/gsd/metrics.ts +26 -0
- package/src/resources/extensions/gsd/milestone-validation-gates.ts +13 -14
- package/src/resources/extensions/gsd/native-git-bridge.ts +11 -12
- package/src/resources/extensions/gsd/parallel-orchestrator.ts +40 -1
- package/src/resources/extensions/gsd/preferences-models.ts +20 -3
- package/src/resources/extensions/gsd/preferences-types.ts +32 -0
- package/src/resources/extensions/gsd/preferences-validation.ts +107 -2
- package/src/resources/extensions/gsd/preferences.ts +28 -0
- package/src/resources/extensions/gsd/prompts/add-tests.md +35 -0
- package/src/resources/extensions/gsd/session-lock.ts +14 -2
- package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +20 -1
- package/src/resources/extensions/gsd/state.ts +9 -2
- package/src/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +7 -3
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +20 -0
- package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +7 -3
- package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +6 -2
- package/src/resources/extensions/gsd/tests/commands-backlog.test.ts +158 -0
- package/src/resources/extensions/gsd/tests/commands-do.test.ts +127 -0
- package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +340 -0
- package/src/resources/extensions/gsd/tests/commands-pr-branch.test.ts +68 -0
- package/src/resources/extensions/gsd/tests/commands-session-report.test.ts +82 -0
- package/src/resources/extensions/gsd/tests/commands-ship.test.ts +71 -0
- package/src/resources/extensions/gsd/tests/commands-workflow-custom.test.ts +14 -0
- package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +154 -0
- package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +10 -7
- package/src/resources/extensions/gsd/tests/graph-context.test.ts +337 -0
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/health-widget.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/journal-integration.test.ts +68 -1
- package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -2
- package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -3
- package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +140 -0
- package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
- package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +40 -1
- package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +180 -0
- package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -5
- package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +101 -0
- package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +85 -0
- package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +69 -0
- package/src/resources/extensions/gsd/tests/uok-flags.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +70 -0
- package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +85 -0
- package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +35 -0
- package/src/resources/extensions/gsd/tests/uok-model-policy.test.ts +89 -0
- package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +167 -0
- package/src/resources/extensions/gsd/tests/uok-preferences.test.ts +42 -0
- package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/workflow-logger-wiring.test.ts +223 -0
- package/src/resources/extensions/gsd/tools/complete-slice.ts +26 -0
- package/src/resources/extensions/gsd/tools/validate-milestone.ts +48 -3
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +3 -11
- package/src/resources/extensions/gsd/triage-resolution.ts +2 -7
- package/src/resources/extensions/gsd/types.ts +14 -1
- package/src/resources/extensions/gsd/unit-ownership.ts +2 -2
- package/src/resources/extensions/gsd/uok/audit-toggle.ts +9 -0
- package/src/resources/extensions/gsd/uok/audit.ts +51 -0
- package/src/resources/extensions/gsd/uok/contracts.ts +135 -0
- package/src/resources/extensions/gsd/uok/execution-graph.ts +241 -0
- package/src/resources/extensions/gsd/uok/flags.ts +45 -0
- package/src/resources/extensions/gsd/uok/gate-runner.ts +146 -0
- package/src/resources/extensions/gsd/uok/gitops.ts +75 -0
- package/src/resources/extensions/gsd/uok/kernel.ts +105 -0
- package/src/resources/extensions/gsd/uok/loop-adapter.ts +162 -0
- package/src/resources/extensions/gsd/uok/model-policy.ts +112 -0
- package/src/resources/extensions/gsd/uok/plan-v2.ts +156 -0
- package/src/resources/extensions/gsd/workflow-logger.ts +27 -1
- package/src/resources/extensions/gsd/workflow-manifest.ts +9 -104
- package/src/resources/extensions/gsd/workflow-migration.ts +21 -29
- package/src/resources/extensions/gsd/workflow-projections.ts +8 -1
- package/src/resources/extensions/gsd/workflow-reconcile.ts +15 -15
- package/src/resources/extensions/ttsr/ttsr-manager.ts +10 -5
- package/packages/pi-ai/dist/models.custom.d.ts.map +0 -1
- package/packages/pi-ai/dist/models.custom.js.map +0 -1
- package/packages/pi-ai/dist/models.generated.js +0 -14343
- package/packages/pi-ai/dist/models.generated.js.map +0 -1
- package/packages/pi-ai/src/models.generated.ts +0 -14345
- /package/dist/web/standalone/.next/static/{YzIEI9sxJy4t5xgClF08g → 3U-oZ5FT59BM7sm2GInic}/_buildManifest.js +0 -0
- /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
|
-
//
|
|
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 =
|
|
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
|
+
}
|