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