@opengsd/gsd-pi 1.3.0-dev.65546769 → 1.3.0-dev.72e3af2a
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/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +11 -2
- package/dist/resources/extensions/google-cli/stream-adapter.js +82 -15
- package/dist/resources/extensions/gsd/artifact-verification.js +427 -0
- package/dist/resources/extensions/gsd/auto/orchestrator.js +12 -3
- package/dist/resources/extensions/gsd/auto/session.js +3 -0
- package/dist/resources/extensions/gsd/auto-artifact-paths.js +28 -1
- package/dist/resources/extensions/gsd/auto-dispatch.js +20 -19
- package/dist/resources/extensions/gsd/auto-prompts.js +26 -11
- package/dist/resources/extensions/gsd/auto-recovery.js +6 -507
- package/dist/resources/extensions/gsd/auto-runtime-state.js +4 -5
- package/dist/resources/extensions/gsd/auto-timeout-recovery.js +3 -3
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +103 -13
- package/dist/resources/extensions/gsd/bootstrap/core-session-tools.js +38 -0
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +6 -1
- package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +2 -0
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +10 -19
- package/dist/resources/extensions/gsd/bootstrap/system-context.js +46 -19
- package/dist/resources/extensions/gsd/bootstrap/tool-call-loop-guard.js +68 -10
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +1 -1
- package/dist/resources/extensions/gsd/commands-context.js +19 -1
- package/dist/resources/extensions/gsd/commands-prefs-wizard.js +16 -10
- package/dist/resources/extensions/gsd/commands-worktree.js +12 -10
- package/dist/resources/extensions/gsd/dashboard-overlay.js +32 -3
- package/dist/resources/extensions/gsd/db/queries.js +60 -0
- package/dist/resources/extensions/gsd/db-workspace.js +55 -3
- package/dist/resources/extensions/gsd/doctor-providers.js +92 -8
- package/dist/resources/extensions/gsd/exec-sandbox.js +45 -9
- package/dist/resources/extensions/gsd/forensics.js +2 -32
- package/dist/resources/extensions/gsd/git-service.js +4 -4
- package/dist/resources/extensions/gsd/guided-flow-queue.js +66 -5
- package/dist/resources/extensions/gsd/health-widget.js +55 -29
- package/dist/resources/extensions/gsd/layout-policy.js +3 -1
- package/dist/resources/extensions/gsd/markdown-renderer.js +8 -9
- package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +44 -21
- package/dist/resources/extensions/gsd/migration-auto-check.js +22 -0
- package/dist/resources/extensions/gsd/milestone-ids.js +32 -2
- package/dist/resources/extensions/gsd/milestone-implementation-evidence.js +26 -20
- package/dist/resources/extensions/gsd/prompts/code-review.md +6 -4
- package/dist/resources/extensions/gsd/quick.js +45 -2
- package/dist/resources/extensions/gsd/session-forensics.js +11 -1
- package/dist/resources/extensions/gsd/skills/gsd-headless/references/commands.md +1 -1
- package/dist/resources/extensions/gsd/state/derive/cache.js +28 -0
- package/dist/resources/extensions/gsd/state/derive/db-open.js +39 -0
- package/dist/resources/extensions/gsd/state/derive/from-db.js +452 -0
- package/dist/resources/extensions/gsd/state/derive/index.js +75 -0
- package/dist/resources/extensions/gsd/state/derive/interrupted-work.js +21 -0
- package/dist/resources/extensions/gsd/state-reconciliation/drift/stale-render.js +45 -2
- package/dist/resources/extensions/gsd/state-reconciliation/index.js +48 -23
- package/dist/resources/extensions/gsd/state-reconciliation/registry.js +32 -28
- package/dist/resources/extensions/gsd/state.js +12 -611
- package/dist/resources/extensions/gsd/tools/complete-slice.js +2 -2
- package/dist/resources/extensions/gsd/tools/complete-task.js +43 -14
- package/dist/resources/extensions/gsd/tools/exec-tool.js +7 -2
- package/dist/resources/extensions/gsd/unit-context-composer.js +23 -7
- package/dist/resources/extensions/gsd/unit-registry.js +32 -4
- package/dist/resources/extensions/gsd/unmerged-milestone-guard.js +33 -3
- package/dist/resources/extensions/gsd/validation-block-guard.js +9 -4
- package/dist/resources/extensions/gsd/workflow-projections.js +19 -14
- package/dist/resources/extensions/gsd/workspace-git-preflight.js +30 -1
- package/dist/resources/extensions/gsd/worktree-manager.js +44 -2
- package/dist/tsconfig.extensions.tsbuildinfo +1 -1
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +10 -10
- 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/react-loadable-manifest.json +9 -9
- 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_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 +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/visualizer/route.js +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +2 -2
- 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 +2 -2
- 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/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +10 -10
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +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/{2659.b7b129ee6a769448.js → 2659.58e950899a9bb82f.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/2772.a7c1fcc69a4685ef.js +1 -0
- package/dist/web/standalone/.next/static/chunks/{3616.3c60753b8ffcbd2e.js → 3616.61a2af74bb8833c8.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/{4283.8e446784528ed9dc.js → 4283.d0d9e0a955e441cb.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/{5826.a46ecdd1cfe8dabc.js → 5826.5421d66c72b9f34e.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/{8785.481aa5869991b760.js → 8785.e29b3134cab1d153.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/8937.640dc9c2aaa1dfad.js +10 -0
- package/dist/web/standalone/.next/static/chunks/app/{page-6644fc6ee8ca1247.js → page-72a856634ad14c10.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/webpack-9c401904f87ded16.js +1 -0
- package/dist/web/standalone/node_modules/node-pty/build/Makefile +1 -1
- package/package.json +1 -1
- package/packages/cloud-mcp-gateway/package.json +2 -2
- package/packages/contracts/dist/workflow.d.ts +1 -0
- package/packages/contracts/dist/workflow.d.ts.map +1 -1
- package/packages/contracts/dist/workflow.js +2 -0
- package/packages/contracts/dist/workflow.js.map +1 -1
- package/packages/contracts/package.json +1 -1
- package/packages/daemon/package.json +4 -4
- package/packages/gsd-agent-core/dist/agent-session.d.ts +1 -0
- package/packages/gsd-agent-core/dist/agent-session.d.ts.map +1 -1
- package/packages/gsd-agent-core/dist/agent-session.js +3 -0
- package/packages/gsd-agent-core/dist/agent-session.js.map +1 -1
- package/packages/gsd-agent-core/dist/extension-ui-snapshot.d.ts +41 -0
- package/packages/gsd-agent-core/dist/extension-ui-snapshot.d.ts.map +1 -0
- package/packages/gsd-agent-core/dist/extension-ui-snapshot.js +62 -0
- package/packages/gsd-agent-core/dist/extension-ui-snapshot.js.map +1 -0
- package/packages/gsd-agent-core/dist/index.d.ts +2 -0
- package/packages/gsd-agent-core/dist/index.d.ts.map +1 -1
- package/packages/gsd-agent-core/dist/index.js +2 -0
- package/packages/gsd-agent-core/dist/index.js.map +1 -1
- package/packages/gsd-agent-core/dist/session/agent-session-events.js +1 -1
- package/packages/gsd-agent-core/dist/session/agent-session-events.js.map +1 -1
- package/packages/gsd-agent-core/dist/session/agent-session-host.d.ts +1 -0
- package/packages/gsd-agent-core/dist/session/agent-session-host.d.ts.map +1 -1
- package/packages/gsd-agent-core/dist/session/agent-session-host.js.map +1 -1
- package/packages/gsd-agent-core/dist/session/agent-session-prompt.d.ts +5 -0
- package/packages/gsd-agent-core/dist/session/agent-session-prompt.d.ts.map +1 -1
- package/packages/gsd-agent-core/dist/session/agent-session-prompt.js +60 -3
- package/packages/gsd-agent-core/dist/session/agent-session-prompt.js.map +1 -1
- package/packages/gsd-agent-core/dist/transcript-store.d.ts +58 -0
- package/packages/gsd-agent-core/dist/transcript-store.d.ts.map +1 -0
- package/packages/gsd-agent-core/dist/transcript-store.js +132 -0
- package/packages/gsd-agent-core/dist/transcript-store.js.map +1 -0
- package/packages/gsd-agent-core/package.json +5 -5
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts +2 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js +25 -11
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller-latency.d.ts +4 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller-latency.d.ts.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller-latency.js +7 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller-latency.js.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts +3 -24
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +26 -829
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-handoff-filter.d.ts +58 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-handoff-filter.d.ts.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-handoff-filter.js +312 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-handoff-filter.js.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-pinned-zone.d.ts +31 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-pinned-zone.d.ts.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-pinned-zone.js +130 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-pinned-zone.js.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-segment-walker.d.ts +15 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-segment-walker.d.ts.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-segment-walker.js +258 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-segment-walker.js.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-tool-rollup.d.ts +13 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-tool-rollup.d.ts.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-tool-rollup.js +118 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-tool-rollup.js.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-state.d.ts +9 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-state.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-state.js +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-state.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-ui-state.d.ts +54 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-ui-state.d.ts.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-ui-state.js +20 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-ui-state.js.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.d.ts +4 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.js +9 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/streaming-render-state.d.ts +56 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/streaming-render-state.d.ts.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/streaming-render-state.js +44 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/streaming-render-state.js.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/tui-transcript-tracker.d.ts +5 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/tui-transcript-tracker.d.ts.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/tui-transcript-tracker.js +77 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/tui-transcript-tracker.js.map +1 -0
- package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js +18 -0
- package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js.map +1 -1
- package/packages/gsd-agent-modes/package.json +7 -7
- package/packages/mcp-server/README.md +1 -1
- package/packages/mcp-server/dist/server.d.ts +1 -1
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +3 -3
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts +13 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +34 -20
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +4 -4
- package/packages/native/package.json +1 -1
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-ai/package.json +1 -1
- package/packages/pi-coding-agent/README.md +3 -2
- package/packages/pi-coding-agent/dist/core/session-manager-context.d.ts +9 -0
- package/packages/pi-coding-agent/dist/core/session-manager-context.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/session-manager-context.js +94 -0
- package/packages/pi-coding-agent/dist/core/session-manager-context.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/session-manager-list.d.ts +8 -0
- package/packages/pi-coding-agent/dist/core/session-manager-list.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/session-manager-list.js +244 -0
- package/packages/pi-coding-agent/dist/core/session-manager-list.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/session-manager-migration.d.ts +12 -0
- package/packages/pi-coding-agent/dist/core/session-manager-migration.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/session-manager-migration.js +84 -0
- package/packages/pi-coding-agent/dist/core/session-manager-migration.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/session-manager-types.d.ts +135 -0
- package/packages/pi-coding-agent/dist/core/session-manager-types.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/session-manager-types.js +2 -0
- package/packages/pi-coding-agent/dist/core/session-manager-types.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/session-manager.d.ts +6 -154
- package/packages/pi-coding-agent/dist/core/session-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/session-manager.js +22 -459
- package/packages/pi-coding-agent/dist/core/session-manager.js.map +1 -1
- package/packages/pi-coding-agent/dist/theme/theme-schema.d.ts +75 -75
- package/packages/pi-coding-agent/dist/theme/theme-schema.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/theme/theme-schema.js +1 -1
- package/packages/pi-coding-agent/dist/theme/theme-schema.js.map +1 -1
- package/packages/pi-coding-agent/dist/theme/theme.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/theme/theme.js +11 -7
- package/packages/pi-coding-agent/dist/theme/theme.js.map +1 -1
- package/packages/pi-coding-agent/package.json +7 -7
- package/packages/pi-tui/package.json +2 -2
- package/packages/rpc-client/package.json +2 -2
- package/pkg/dist/theme/theme-schema.d.ts +75 -75
- package/pkg/dist/theme/theme-schema.d.ts.map +1 -1
- package/pkg/dist/theme/theme-schema.js +1 -1
- package/pkg/dist/theme/theme-schema.js.map +1 -1
- package/pkg/dist/theme/theme.d.ts.map +1 -1
- package/pkg/dist/theme/theme.js +11 -7
- package/pkg/dist/theme/theme.js.map +1 -1
- package/pkg/package.json +1 -1
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +20 -2
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +80 -0
- package/src/resources/extensions/google-cli/stream-adapter.ts +106 -19
- package/src/resources/extensions/gsd/artifact-verification.ts +464 -0
- package/src/resources/extensions/gsd/auto/orchestrator.ts +25 -11
- package/src/resources/extensions/gsd/auto/session.ts +5 -0
- package/src/resources/extensions/gsd/auto-artifact-paths.ts +47 -1
- package/src/resources/extensions/gsd/auto-dispatch.ts +21 -23
- package/src/resources/extensions/gsd/auto-prompts.ts +38 -12
- package/src/resources/extensions/gsd/auto-recovery.ts +10 -508
- package/src/resources/extensions/gsd/auto-runtime-state.ts +4 -5
- package/src/resources/extensions/gsd/auto-timeout-recovery.ts +3 -2
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +125 -12
- package/src/resources/extensions/gsd/bootstrap/core-session-tools.ts +43 -0
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +6 -1
- package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +2 -0
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +11 -19
- package/src/resources/extensions/gsd/bootstrap/system-context.ts +52 -18
- package/src/resources/extensions/gsd/bootstrap/tool-call-loop-guard.ts +74 -10
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +1 -1
- package/src/resources/extensions/gsd/commands-context.ts +18 -1
- package/src/resources/extensions/gsd/commands-prefs-wizard.ts +14 -9
- package/src/resources/extensions/gsd/commands-worktree.ts +12 -10
- package/src/resources/extensions/gsd/dashboard-overlay.ts +32 -3
- package/src/resources/extensions/gsd/db/queries.ts +79 -0
- package/src/resources/extensions/gsd/db-workspace.ts +61 -3
- package/src/resources/extensions/gsd/doctor-providers.ts +103 -9
- package/src/resources/extensions/gsd/exec-sandbox.ts +49 -9
- package/src/resources/extensions/gsd/forensics.ts +2 -33
- package/src/resources/extensions/gsd/git-service.ts +5 -5
- package/src/resources/extensions/gsd/guided-flow-queue.ts +89 -4
- package/src/resources/extensions/gsd/health-widget.ts +69 -32
- package/src/resources/extensions/gsd/layout-policy.ts +2 -1
- package/src/resources/extensions/gsd/markdown-renderer.ts +8 -11
- package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +51 -19
- package/src/resources/extensions/gsd/migration-auto-check.ts +23 -0
- package/src/resources/extensions/gsd/milestone-ids.ts +31 -2
- package/src/resources/extensions/gsd/milestone-implementation-evidence.ts +35 -21
- package/src/resources/extensions/gsd/prompts/code-review.md +6 -4
- package/src/resources/extensions/gsd/quick.ts +43 -2
- package/src/resources/extensions/gsd/session-forensics.ts +11 -1
- package/src/resources/extensions/gsd/skills/gsd-headless/references/commands.md +1 -1
- package/src/resources/extensions/gsd/state/derive/cache.ts +46 -0
- package/src/resources/extensions/gsd/state/derive/db-open.ts +45 -0
- package/src/resources/extensions/gsd/state/derive/from-db.ts +561 -0
- package/src/resources/extensions/gsd/state/derive/index.ts +104 -0
- package/src/resources/extensions/gsd/state/derive/interrupted-work.ts +31 -0
- package/src/resources/extensions/gsd/state-reconciliation/drift/stale-render.ts +81 -7
- package/src/resources/extensions/gsd/state-reconciliation/index.ts +50 -24
- package/src/resources/extensions/gsd/state-reconciliation/registry.ts +43 -28
- package/src/resources/extensions/gsd/state.ts +32 -732
- package/src/resources/extensions/gsd/tests/auto-artifact-paths.test.ts +98 -1
- package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +111 -1
- package/src/resources/extensions/gsd/tests/commands-context.test.ts +26 -0
- package/src/resources/extensions/gsd/tests/commands-gsd-core.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/commands-worktree-clean.test.ts +80 -0
- package/src/resources/extensions/gsd/tests/complete-slice.test.ts +11 -0
- package/src/resources/extensions/gsd/tests/complete-task-rollback-evidence.test.ts +48 -8
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +75 -0
- package/src/resources/extensions/gsd/tests/dashboard-overlay.test.ts +55 -2
- package/src/resources/extensions/gsd/tests/dispatch-rule-coverage.test.ts +26 -1
- package/src/resources/extensions/gsd/tests/doctor-forensics-db-open-regression.test.ts +70 -2
- package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +107 -0
- package/src/resources/extensions/gsd/tests/exec-graceful-kill.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/exec-tool.test.ts +45 -1
- package/src/resources/extensions/gsd/tests/forensics-error-filter.test.ts +88 -0
- package/src/resources/extensions/gsd/tests/guided-discuss-milestone-prompt-rendering.test.ts +42 -0
- package/src/resources/extensions/gsd/tests/health-widget.test.ts +268 -3
- package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +119 -1
- package/src/resources/extensions/gsd/tests/integration/queue-active-milestone-context-budget.test.ts +93 -0
- package/src/resources/extensions/gsd/tests/integration/quick-branch-lifecycle.test.ts +56 -9
- package/src/resources/extensions/gsd/tests/knowledge-cold-start.test.ts +14 -0
- package/src/resources/extensions/gsd/tests/memory-consolidation-scanner.test.ts +78 -0
- package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +70 -0
- package/src/resources/extensions/gsd/tests/orchestrator-logs.test.ts +43 -1
- package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +26 -0
- package/src/resources/extensions/gsd/tests/parsers-legacy-importers.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +54 -1
- package/src/resources/extensions/gsd/tests/progressive-planning.test.ts +50 -14
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +195 -1
- package/src/resources/extensions/gsd/tests/read-uat-gate-verdict.test.ts +185 -0
- package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +87 -0
- package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +7 -0
- package/src/resources/extensions/gsd/tests/stalled-tool-recovery.test.ts +56 -0
- package/src/resources/extensions/gsd/tests/state-reconciliation-drift.test.ts +191 -0
- package/src/resources/extensions/gsd/tests/tool-call-loop-guard.test.ts +151 -0
- package/src/resources/extensions/gsd/tests/tool-param-optionality.test.ts +26 -0
- package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +193 -14
- package/src/resources/extensions/gsd/tests/unmerged-milestone-guard.test.ts +25 -0
- package/src/resources/extensions/gsd/tests/validation-block-guard.test.ts +79 -0
- package/src/resources/extensions/gsd/tests/verify-artifact-tightened.test.ts +66 -0
- package/src/resources/extensions/gsd/tests/workspace-git-preflight.test.ts +151 -2
- package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +39 -0
- package/src/resources/extensions/gsd/tools/complete-slice.ts +2 -2
- package/src/resources/extensions/gsd/tools/complete-task.ts +53 -15
- package/src/resources/extensions/gsd/tools/exec-tool.ts +7 -3
- package/src/resources/extensions/gsd/unit-context-composer.ts +33 -7
- package/src/resources/extensions/gsd/unit-registry.ts +32 -4
- package/src/resources/extensions/gsd/unmerged-milestone-guard.ts +41 -5
- package/src/resources/extensions/gsd/validation-block-guard.ts +13 -7
- package/src/resources/extensions/gsd/workflow-projections.ts +20 -14
- package/src/resources/extensions/gsd/workspace-git-preflight.ts +31 -0
- package/src/resources/extensions/gsd/worktree-manager.ts +41 -1
- package/dist/web/standalone/.next/static/chunks/2772.bfa657f49f955239.js +0 -1
- package/dist/web/standalone/.next/static/chunks/796.e0bdc932325d7e03.js +0 -10
- package/dist/web/standalone/.next/static/chunks/webpack-f46ea08200a0227e.js +0 -1
- /package/dist/web/standalone/.next/static/{BTKtGFF1Y-hvVJEGhBRo9 → O7xDYXO0r4zFhIzY1hrWV}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{BTKtGFF1Y-hvVJEGhBRo9 → O7xDYXO0r4zFhIzY1hrWV}/_ssgManifest.js +0 -0
|
@@ -13,6 +13,7 @@ import {
|
|
|
13
13
|
diffWorktreeNumstat,
|
|
14
14
|
getWorktreeGSDDiff,
|
|
15
15
|
getWorktreeLog,
|
|
16
|
+
mergeWorktreeToMain,
|
|
16
17
|
worktreeBranchName,
|
|
17
18
|
worktreePath,
|
|
18
19
|
pruneEphemeralGhostWorktreeDirectories,
|
|
@@ -333,6 +334,44 @@ describe("getWorktreeLog", () => {
|
|
|
333
334
|
});
|
|
334
335
|
});
|
|
335
336
|
|
|
337
|
+
// ─── mergeWorktreeToMain ─────────────────────────────────────────────────────
|
|
338
|
+
|
|
339
|
+
describe("mergeWorktreeToMain", () => {
|
|
340
|
+
let base: string;
|
|
341
|
+
let wtPath: string;
|
|
342
|
+
|
|
343
|
+
beforeEach(() => {
|
|
344
|
+
base = makeBaseRepo();
|
|
345
|
+
wtPath = worktreePath(base, "M900");
|
|
346
|
+
mkdirSync(join(base, ".gsd-worktrees"), { recursive: true });
|
|
347
|
+
run(`git worktree add -b milestone/M900 "${wtPath}"`, base);
|
|
348
|
+
});
|
|
349
|
+
|
|
350
|
+
afterEach(() => { rmSync(base, { recursive: true, force: true }); });
|
|
351
|
+
|
|
352
|
+
test("cleans failed milestone squash state and deletes orphan branch", () => {
|
|
353
|
+
writeFileSync(join(wtPath, "README.md"), "# Milestone change\n", "utf-8");
|
|
354
|
+
run("git add README.md", wtPath);
|
|
355
|
+
run('git commit -m "feat: milestone change"', wtPath);
|
|
356
|
+
|
|
357
|
+
writeFileSync(join(base, "README.md"), "# Main change\n", "utf-8");
|
|
358
|
+
run("git add README.md", base);
|
|
359
|
+
run('git commit -m "feat: main change"', base);
|
|
360
|
+
|
|
361
|
+
assert.throws(
|
|
362
|
+
() => mergeWorktreeToMain(base, "M900", "feat: merge M900", "milestone/M900"),
|
|
363
|
+
/Merge conflicts detected/,
|
|
364
|
+
);
|
|
365
|
+
|
|
366
|
+
assert.equal(run("git status --porcelain", base), "", "failed squash cleanup should leave main clean");
|
|
367
|
+
assert.ok(!existsSync(wtPath), "failed milestone worktree should be removed");
|
|
368
|
+
assert.ok(
|
|
369
|
+
!run("git branch", base).includes("milestone/M900"),
|
|
370
|
+
"failed milestone branch should be deleted after worktree removal",
|
|
371
|
+
);
|
|
372
|
+
});
|
|
373
|
+
});
|
|
374
|
+
|
|
336
375
|
// ─── removeWorktree ───────────────────────────────────────────────────────────
|
|
337
376
|
|
|
338
377
|
describe("removeWorktree", () => {
|
|
@@ -30,7 +30,7 @@ import { classifyUatContent, escalatesArtifactUatToBrowser } from "../uat-policy
|
|
|
30
30
|
import { invalidateStateCache } from "../state.js";
|
|
31
31
|
import { renderRoadmapFromDb, roadmapRenderMarksSliceDone } from "../markdown-renderer.js";
|
|
32
32
|
import { isStaleWrite } from "../auto/turn-epoch.js";
|
|
33
|
-
import {
|
|
33
|
+
import { renderMilestoneShellProjections } from "../workflow-projections.js";
|
|
34
34
|
import { writeManifest } from "../workflow-manifest.js";
|
|
35
35
|
import { appendEvent } from "../workflow-events.js";
|
|
36
36
|
import { logWarning, logError } from "../workflow-logger.js";
|
|
@@ -540,7 +540,7 @@ export async function handleCompleteSlice(
|
|
|
540
540
|
// Separate try/catch per step so a projection failure doesn't prevent
|
|
541
541
|
// the event log entry (critical for worktree reconciliation).
|
|
542
542
|
try {
|
|
543
|
-
await
|
|
543
|
+
await renderMilestoneShellProjections(artifactBasePath, params.milestoneId);
|
|
544
544
|
} catch (projErr) {
|
|
545
545
|
logWarning("tool", `complete-slice projection warning for ${params.milestoneId}/${params.sliceId}: ${(projErr as Error).message}`);
|
|
546
546
|
}
|
|
@@ -6,11 +6,12 @@
|
|
|
6
6
|
*
|
|
7
7
|
* Validates inputs, writes task row and rendered SUMMARY.md to DB in a
|
|
8
8
|
* transaction, then renders projections to disk and invalidates caches.
|
|
9
|
-
*
|
|
10
|
-
* back
|
|
9
|
+
* If the critical task summary / plan projection write fails, the DB
|
|
10
|
+
* completion is compensated back to pending so DB state does not drift ahead
|
|
11
|
+
* of PLAN.md.
|
|
11
12
|
*/
|
|
12
13
|
|
|
13
|
-
import { existsSync } from "node:fs";
|
|
14
|
+
import { existsSync, unlinkSync } from "node:fs";
|
|
14
15
|
import { join } from "node:path";
|
|
15
16
|
|
|
16
17
|
import type { CompleteTaskParams, EscalationArtifact } from "../types.js";
|
|
@@ -29,6 +30,7 @@ import {
|
|
|
29
30
|
saveGateResult,
|
|
30
31
|
getPendingGatesForTurn,
|
|
31
32
|
} from "../gsd-db.js";
|
|
33
|
+
import { getWorkflowDatabasePath, ensureWorkflowDbAtPath } from "../db-workspace.js";
|
|
32
34
|
import { getGatesForTurn } from "../gate-registry.js";
|
|
33
35
|
import { gsdProjectionRoot, clearPathCache, resolveMilestonePath, resolveSlicePath } from "../paths.js";
|
|
34
36
|
import { resolveCanonicalMilestoneRoot } from "../worktree-manager.js";
|
|
@@ -36,8 +38,10 @@ import { checkOwnership, taskUnitKey } from "../unit-ownership.js";
|
|
|
36
38
|
import { saveFile, clearParseCache } from "../files.js";
|
|
37
39
|
import { invalidateStateCache } from "../state.js";
|
|
38
40
|
import { renderPlanCheckboxes } from "../markdown-renderer.js";
|
|
39
|
-
import {
|
|
40
|
-
|
|
41
|
+
import {
|
|
42
|
+
renderMilestoneShellProjections,
|
|
43
|
+
renderSummaryContent,
|
|
44
|
+
} from "../workflow-projections.js";
|
|
41
45
|
import { writeManifest } from "../workflow-manifest.js";
|
|
42
46
|
import { appendEvent } from "../workflow-events.js";
|
|
43
47
|
import { logWarning, logError } from "../workflow-logger.js";
|
|
@@ -131,7 +135,7 @@ async function repairMissingTaskSummaryProjection(
|
|
|
131
135
|
clearParseCache();
|
|
132
136
|
|
|
133
137
|
try {
|
|
134
|
-
await
|
|
138
|
+
await renderMilestoneShellProjections(artifactBasePath, taskRow.milestone_id);
|
|
135
139
|
} catch (projErr) {
|
|
136
140
|
logWarning("tool", `complete-task repair projection warning: ${(projErr as Error).message}`);
|
|
137
141
|
}
|
|
@@ -267,6 +271,7 @@ export async function handleCompleteTask(
|
|
|
267
271
|
let guardError: string | null = null;
|
|
268
272
|
let summaryMd = "";
|
|
269
273
|
let repairTaskSummaryRow: TaskRow | null = null;
|
|
274
|
+
const rollbackDbPath = getWorkflowDatabasePath();
|
|
270
275
|
|
|
271
276
|
// ── ADR-011 Phase 2: validate escalation payload BEFORE any side effects ─
|
|
272
277
|
// Building the artifact runs the full shape validation (2-4 options, unique
|
|
@@ -423,8 +428,6 @@ export async function handleCompleteTask(
|
|
|
423
428
|
return { error: guardError };
|
|
424
429
|
}
|
|
425
430
|
|
|
426
|
-
let projectionStale = false;
|
|
427
|
-
|
|
428
431
|
// Resolve and write summary to disk
|
|
429
432
|
const summaryPath = taskSummaryPath(
|
|
430
433
|
artifactBasePath,
|
|
@@ -438,12 +441,48 @@ export async function handleCompleteTask(
|
|
|
438
441
|
|
|
439
442
|
// Toggle or regenerate the plan projection from DB. Missing projection
|
|
440
443
|
// files are rebuilt by the renderer instead of being skipped.
|
|
441
|
-
|
|
444
|
+
if (!ensureWorkflowDbAtPath(rollbackDbPath)) {
|
|
445
|
+
throw new Error(`database unavailable before plan projection render for ${params.milestoneId}/${params.sliceId}`);
|
|
446
|
+
}
|
|
447
|
+
const wrotePlan = await renderPlanCheckboxes(artifactBasePath, params.milestoneId, params.sliceId);
|
|
448
|
+
if (!wrotePlan) {
|
|
449
|
+
throw new Error(`plan projection write returned false for ${params.milestoneId}/${params.sliceId}`);
|
|
450
|
+
}
|
|
442
451
|
} catch (renderErr) {
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
452
|
+
logWarning(
|
|
453
|
+
"projection",
|
|
454
|
+
`complete_task projection write failed for ${params.milestoneId}/${params.sliceId}/${params.taskId}`,
|
|
455
|
+
{ error: (renderErr as Error).message },
|
|
456
|
+
);
|
|
457
|
+
let rollbackSucceeded = false;
|
|
458
|
+
try {
|
|
459
|
+
ensureWorkflowDbAtPath(rollbackDbPath);
|
|
460
|
+
deleteVerificationEvidence(params.milestoneId, params.sliceId, params.taskId);
|
|
461
|
+
updateTaskStatus(params.milestoneId, params.sliceId, params.taskId, "pending");
|
|
462
|
+
invalidateStateCache();
|
|
463
|
+
rollbackSucceeded = true;
|
|
464
|
+
} catch (rollbackErr) {
|
|
465
|
+
logWarning(
|
|
466
|
+
"projection",
|
|
467
|
+
`complete_task rollback failed after projection write failure for ${params.milestoneId}/${params.sliceId}/${params.taskId}: ${(rollbackErr as Error).message}`,
|
|
468
|
+
);
|
|
469
|
+
}
|
|
470
|
+
try {
|
|
471
|
+
if (existsSync(summaryPath)) unlinkSync(summaryPath);
|
|
472
|
+
} catch (summaryErr) {
|
|
473
|
+
logWarning(
|
|
474
|
+
"projection",
|
|
475
|
+
`complete_task could not remove SUMMARY.md after projection write failure for ${params.milestoneId}/${params.sliceId}/${params.taskId}: ${(summaryErr as Error).message}`,
|
|
476
|
+
);
|
|
477
|
+
}
|
|
478
|
+
// Clear path/parse caches regardless of rollback outcome so stale
|
|
479
|
+
// entries from the failed write attempt don't leak into subsequent calls.
|
|
480
|
+
clearPathCache();
|
|
481
|
+
clearParseCache();
|
|
482
|
+
const returnMsg = rollbackSucceeded
|
|
483
|
+
? `complete_task projection write failed for ${params.milestoneId}/${params.sliceId}/${params.taskId}; rolled completion back to pending`
|
|
484
|
+
: `complete_task projection write failed for ${params.milestoneId}/${params.sliceId}/${params.taskId}; rollback also failed — task may remain complete with stale plan`;
|
|
485
|
+
return { error: returnMsg };
|
|
447
486
|
}
|
|
448
487
|
|
|
449
488
|
// ── Close gates owned by execute-task (Q5/Q6/Q7) for this task ────────
|
|
@@ -569,7 +608,7 @@ export async function handleCompleteTask(
|
|
|
569
608
|
// Separate try/catch per step so a projection failure doesn't prevent
|
|
570
609
|
// the event log entry (critical for worktree reconciliation).
|
|
571
610
|
try {
|
|
572
|
-
await
|
|
611
|
+
await renderMilestoneShellProjections(artifactBasePath, params.milestoneId);
|
|
573
612
|
} catch (projErr) {
|
|
574
613
|
logWarning("tool", `complete-task projection warning: ${(projErr as Error).message}`);
|
|
575
614
|
}
|
|
@@ -597,6 +636,5 @@ export async function handleCompleteTask(
|
|
|
597
636
|
milestoneId: params.milestoneId,
|
|
598
637
|
summaryPath,
|
|
599
638
|
...(escalationMetadata ? { escalation: escalationMetadata } : {}),
|
|
600
|
-
...(projectionStale ? { stale: true } : {}),
|
|
601
639
|
};
|
|
602
640
|
}
|
|
@@ -31,6 +31,7 @@ export interface ExecToolDeps {
|
|
|
31
31
|
env?: NodeJS.ProcessEnv;
|
|
32
32
|
now?: () => Date;
|
|
33
33
|
generateId?: () => string;
|
|
34
|
+
signal?: AbortSignal;
|
|
34
35
|
}
|
|
35
36
|
|
|
36
37
|
export type UatExecIntent =
|
|
@@ -74,7 +75,7 @@ const UAT_EXEC_INTENT_ALIASES: Record<string, UatExecIntent> = {
|
|
|
74
75
|
export function buildExecOptions(
|
|
75
76
|
baseDir: string,
|
|
76
77
|
cfg: ContextModeConfig | undefined,
|
|
77
|
-
extras?: Pick<ExecSandboxOptions, "env" | "now" | "generateId">,
|
|
78
|
+
extras?: Pick<ExecSandboxOptions, "env" | "now" | "generateId" | "signal">,
|
|
78
79
|
): ExecSandboxOptions {
|
|
79
80
|
const allowlist = Array.isArray(cfg?.exec_env_allowlist) ? cfg!.exec_env_allowlist! : EXEC_DEFAULTS.envAllowlist;
|
|
80
81
|
const stdoutCap = clampNumber(
|
|
@@ -209,7 +210,7 @@ export async function executeGsdExec(
|
|
|
209
210
|
const opts = buildExecOptions(
|
|
210
211
|
deps.baseDir,
|
|
211
212
|
deps.preferences?.context_mode,
|
|
212
|
-
{ env: deps.env, now: deps.now, generateId: deps.generateId },
|
|
213
|
+
{ env: deps.env, now: deps.now, generateId: deps.generateId, signal: deps.signal },
|
|
213
214
|
);
|
|
214
215
|
const run = deps.run ?? runExecSandbox;
|
|
215
216
|
|
|
@@ -308,6 +309,7 @@ function formatResult(result: ExecSandboxResult): ToolExecutionResult {
|
|
|
308
309
|
exit_code: result.exit_code,
|
|
309
310
|
signal: result.signal,
|
|
310
311
|
timed_out: result.timed_out,
|
|
312
|
+
aborted: result.aborted === true,
|
|
311
313
|
force_resolved: result.force_resolved,
|
|
312
314
|
duration_ms: result.duration_ms,
|
|
313
315
|
stdout_bytes: result.stdout_bytes,
|
|
@@ -318,13 +320,15 @@ function formatResult(result: ExecSandboxResult): ToolExecutionResult {
|
|
|
318
320
|
stderr_path: result.stderr_path,
|
|
319
321
|
meta_path: result.meta_path,
|
|
320
322
|
},
|
|
321
|
-
isError: result.timed_out || result.signal !== null || result.exit_code !== 0,
|
|
323
|
+
isError: result.aborted === true || result.timed_out || result.signal !== null || result.exit_code !== 0,
|
|
322
324
|
};
|
|
323
325
|
}
|
|
324
326
|
|
|
325
327
|
function formatExit(result: ExecSandboxResult): string {
|
|
326
328
|
// force_resolved means a non-closing (D-state) child was force-resolved past its
|
|
327
329
|
// hard deadline rather than observed exiting; distinguish it from a clean timeout.
|
|
330
|
+
if (result.aborted && result.force_resolved) return "aborted(force-killed)";
|
|
331
|
+
if (result.aborted) return "aborted";
|
|
328
332
|
if (result.force_resolved) return "timeout(force-killed)";
|
|
329
333
|
if (result.timed_out) return "timeout";
|
|
330
334
|
if (result.signal) return `signal:${result.signal}`;
|
|
@@ -135,14 +135,38 @@ const CONTEXT_MODE_GUIDANCE_BY_LANE: Record<Exclude<ContextModePolicy, "none">,
|
|
|
135
135
|
"Use `gsd_resume` for prior context, `gsd_exec_search` for saved evidence, and `gsd_exec` for noisy doc validation commands.",
|
|
136
136
|
};
|
|
137
137
|
|
|
138
|
-
// Per-unit overrides win over the lane default.
|
|
139
|
-
//
|
|
140
|
-
//
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
138
|
+
// Per-unit overrides win over the lane default. Some units intentionally run
|
|
139
|
+
// with narrower tool contracts than their shared Context Mode lane, so their
|
|
140
|
+
// guidance must name only tools the unit can actually call.
|
|
141
|
+
export const CONTEXT_MODE_GUIDANCE_BY_UNIT: Readonly<Record<string, string>> = {
|
|
142
|
+
"discuss-milestone":
|
|
143
|
+
"Use `ask_user_questions` to continue the milestone interview, then persist outcomes with `gsd_summary_save`, `gsd_decision_save`, `gsd_requirement_save`, `gsd_requirement_update`, `gsd_plan_milestone`, or `gsd_milestone_generate_id` as appropriate.",
|
|
144
|
+
"discuss-slice":
|
|
145
|
+
"Use `ask_user_questions` to continue the slice interview, then persist outcomes with `gsd_summary_save` or `gsd_decision_save` as appropriate.",
|
|
146
|
+
"discuss-project":
|
|
147
|
+
"Use `ask_user_questions` to continue the project interview, then persist outcomes with `gsd_summary_save`, `gsd_decision_save`, or `gsd_requirement_save` as appropriate.",
|
|
148
|
+
"discuss-requirements":
|
|
149
|
+
"Use `ask_user_questions` to continue the requirements interview, then persist outcomes with `gsd_requirement_save` or `gsd_summary_save` as appropriate.",
|
|
150
|
+
"replan-slice":
|
|
151
|
+
"Use `gsd_replan_slice` to persist the revised slice plan, and `gsd_decision_save` for planning decisions that need durable rationale.",
|
|
152
|
+
"reassess-roadmap":
|
|
153
|
+
"Use `gsd_milestone_status` to inspect current milestone state, then `gsd_reassess_roadmap` to persist the roadmap reassessment.",
|
|
144
154
|
"run-uat":
|
|
145
155
|
"Use `gsd_uat_exec` for acceptance checks so evidence is typed as UAT-owned, and `gsd_resume` after compaction or resume.",
|
|
156
|
+
"research-project":
|
|
157
|
+
"Dispatch parallel scout subagents for stack, features, architecture, and pitfalls research; each writes one file under `.gsd/research/` (`STACK.md`, `FEATURES.md`, `ARCHITECTURE.md`, `PITFALLS.md`).",
|
|
158
|
+
"gate-evaluate":
|
|
159
|
+
"Use `subagent` to dispatch tester agents, then persist each gate with `gsd_save_gate_result`; rely on testers for verification evidence.",
|
|
160
|
+
};
|
|
161
|
+
|
|
162
|
+
// Per-unit guidance for the nested render mode (renderMode: "nested"), used when this
|
|
163
|
+
// unit's Context Mode line is embedded into a subagent prompt — e.g. the tester prompts
|
|
164
|
+
// dispatched by gate-evaluate. Must instruct the subagent on what IT should do, not
|
|
165
|
+
// re-state the parent coordinator's dispatch instructions. Falls back to
|
|
166
|
+
// CONTEXT_MODE_GUIDANCE_BY_UNIT then the lane default when no nested entry exists.
|
|
167
|
+
export const CONTEXT_MODE_NESTED_GUIDANCE_BY_UNIT: Readonly<Record<string, string>> = {
|
|
168
|
+
"gate-evaluate":
|
|
169
|
+
"Run verification checks to answer the gate question, then persist the verdict with `gsd_save_gate_result`.",
|
|
146
170
|
};
|
|
147
171
|
|
|
148
172
|
/**
|
|
@@ -160,7 +184,9 @@ export function composeContextModeInstructions(
|
|
|
160
184
|
|
|
161
185
|
const lane = CONTEXT_MODE_LANE_LABELS[manifest.contextMode];
|
|
162
186
|
const guidance =
|
|
163
|
-
|
|
187
|
+
(opts.renderMode === "nested" ? CONTEXT_MODE_NESTED_GUIDANCE_BY_UNIT[unitType] : undefined)
|
|
188
|
+
?? CONTEXT_MODE_GUIDANCE_BY_UNIT[unitType]
|
|
189
|
+
?? CONTEXT_MODE_GUIDANCE_BY_LANE[manifest.contextMode];
|
|
164
190
|
if (opts.renderMode === "nested") {
|
|
165
191
|
return `Context Mode (${lane} lane): ${guidance}`;
|
|
166
192
|
}
|
|
@@ -98,7 +98,13 @@ export const UNIT_REGISTRY = {
|
|
|
98
98
|
scopeClass: "standard",
|
|
99
99
|
phaseChain: ["research"],
|
|
100
100
|
toolContract: {
|
|
101
|
-
allowedGsdTools: [
|
|
101
|
+
allowedGsdTools: [
|
|
102
|
+
"gsd_summary_save",
|
|
103
|
+
"gsd_decision_save",
|
|
104
|
+
"gsd_exec",
|
|
105
|
+
"gsd_exec_search",
|
|
106
|
+
"gsd_resume",
|
|
107
|
+
],
|
|
102
108
|
requiredWorkflowTools: ["gsd_summary_save"],
|
|
103
109
|
},
|
|
104
110
|
},
|
|
@@ -154,7 +160,15 @@ export const UNIT_REGISTRY = {
|
|
|
154
160
|
scopeClass: "section-close",
|
|
155
161
|
phaseChain: ["validation", "planning"],
|
|
156
162
|
toolContract: {
|
|
157
|
-
allowedGsdTools: [
|
|
163
|
+
allowedGsdTools: [
|
|
164
|
+
"gsd_milestone_status",
|
|
165
|
+
"gsd_exec",
|
|
166
|
+
"gsd_exec_search",
|
|
167
|
+
"gsd_resume",
|
|
168
|
+
"gsd_validate_milestone",
|
|
169
|
+
"gsd_reassess_roadmap",
|
|
170
|
+
"subagent",
|
|
171
|
+
],
|
|
158
172
|
requiredWorkflowTools: ["gsd_milestone_status", "gsd_validate_milestone", "gsd_reassess_roadmap"],
|
|
159
173
|
},
|
|
160
174
|
},
|
|
@@ -165,6 +179,9 @@ export const UNIT_REGISTRY = {
|
|
|
165
179
|
toolContract: {
|
|
166
180
|
allowedGsdTools: [
|
|
167
181
|
"gsd_milestone_status",
|
|
182
|
+
"gsd_exec",
|
|
183
|
+
"gsd_exec_search",
|
|
184
|
+
"gsd_resume",
|
|
168
185
|
"gsd_requirement_update",
|
|
169
186
|
"gsd_summary_save",
|
|
170
187
|
"gsd_complete_milestone",
|
|
@@ -244,6 +261,8 @@ export const UNIT_REGISTRY = {
|
|
|
244
261
|
allowedGsdTools: [
|
|
245
262
|
"gsd_milestone_status",
|
|
246
263
|
"gsd_exec",
|
|
264
|
+
"gsd_exec_search",
|
|
265
|
+
"gsd_resume",
|
|
247
266
|
"gsd_slice_complete",
|
|
248
267
|
"gsd_task_reopen",
|
|
249
268
|
"gsd_replan_slice",
|
|
@@ -326,7 +345,13 @@ export const UNIT_REGISTRY = {
|
|
|
326
345
|
scopeClass: "execute-task",
|
|
327
346
|
phaseChain: ["execution"],
|
|
328
347
|
toolContract: {
|
|
329
|
-
allowedGsdTools: [
|
|
348
|
+
allowedGsdTools: [
|
|
349
|
+
"gsd_milestone_status",
|
|
350
|
+
"gsd_task_complete",
|
|
351
|
+
"gsd_exec",
|
|
352
|
+
"gsd_summary_save",
|
|
353
|
+
"gsd_decision_save",
|
|
354
|
+
],
|
|
330
355
|
requiredWorkflowTools: ["gsd_task_complete", "gsd_summary_save"],
|
|
331
356
|
},
|
|
332
357
|
},
|
|
@@ -413,12 +438,15 @@ export const UNIT_REGISTRY = {
|
|
|
413
438
|
requiredWorkflowTools: ["ask_user_questions"],
|
|
414
439
|
},
|
|
415
440
|
},
|
|
441
|
+
// research-project dispatches 4 parallel scout subagents (Task calls); each scout
|
|
442
|
+
// writes one file under .gsd/research/ directly. The parent coordinator does not
|
|
443
|
+
// call gsd_summary_save or gsd_decision_save — the scouts own their own output.
|
|
416
444
|
"research-project": {
|
|
417
445
|
kind: "primary",
|
|
418
446
|
scopeClass: "standard",
|
|
419
447
|
phaseChain: ["research"],
|
|
420
448
|
toolContract: {
|
|
421
|
-
allowedGsdTools: [
|
|
449
|
+
allowedGsdTools: [],
|
|
422
450
|
requiredWorkflowTools: [],
|
|
423
451
|
},
|
|
424
452
|
},
|
|
@@ -1,6 +1,8 @@
|
|
|
1
1
|
// Project/App: gsd-pi
|
|
2
2
|
// File Purpose: Block new workflow entry when completed milestone branches are still unmerged.
|
|
3
3
|
|
|
4
|
+
import { execFileSync } from "node:child_process";
|
|
5
|
+
|
|
4
6
|
import {
|
|
5
7
|
nativeBranchExists,
|
|
6
8
|
nativeDetectMainBranch,
|
|
@@ -12,17 +14,23 @@ import { ensureDbOpen } from "./bootstrap/dynamic-tools.js";
|
|
|
12
14
|
import { getAllMilestones } from "./gsd-db.js";
|
|
13
15
|
import { resolveMilestoneIntegrationBranch } from "./git-service.js";
|
|
14
16
|
import { loadEffectiveGSDPreferences } from "./preferences.js";
|
|
15
|
-
import { captureRootDirtySnapshot, type RootDirtyEntry } from "./root-write-leak-guard.js";
|
|
16
17
|
import { isClosedStatus } from "./status-guards.js";
|
|
17
18
|
|
|
19
|
+
export interface UnmergedMilestoneDirtyEntry {
|
|
20
|
+
path: string;
|
|
21
|
+
status: string;
|
|
22
|
+
}
|
|
23
|
+
|
|
18
24
|
export interface UnmergedMilestoneBlocker {
|
|
19
25
|
milestoneId: string;
|
|
20
26
|
branch: string;
|
|
21
27
|
integrationBranch: string;
|
|
22
28
|
files: string[];
|
|
23
|
-
dirtyOverlap:
|
|
29
|
+
dirtyOverlap: UnmergedMilestoneDirtyEntry[];
|
|
24
30
|
}
|
|
25
31
|
|
|
32
|
+
type UnmergedMilestoneDirtySnapshot = Map<string, UnmergedMilestoneDirtyEntry>;
|
|
33
|
+
|
|
26
34
|
const BLOCKED_COMMANDS = new Set([
|
|
27
35
|
"auto",
|
|
28
36
|
"next",
|
|
@@ -67,6 +75,32 @@ function isRuntimePath(path: string): boolean {
|
|
|
67
75
|
return path === ".gsd" || path.startsWith(".gsd/");
|
|
68
76
|
}
|
|
69
77
|
|
|
78
|
+
function captureDirtyPathStatusSnapshot(rootPath: string): UnmergedMilestoneDirtySnapshot {
|
|
79
|
+
const snapshot: UnmergedMilestoneDirtySnapshot = new Map();
|
|
80
|
+
let status = "";
|
|
81
|
+
try {
|
|
82
|
+
status = execFileSync("git", ["status", "--porcelain", "--untracked-files=all"], {
|
|
83
|
+
cwd: rootPath,
|
|
84
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
85
|
+
encoding: "utf-8",
|
|
86
|
+
});
|
|
87
|
+
} catch {
|
|
88
|
+
return snapshot;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
for (const line of status.split("\n")) {
|
|
92
|
+
if (!line.trim()) continue;
|
|
93
|
+
const code = line.slice(0, 2);
|
|
94
|
+
const path = line.slice(3).replace(/^"|"$/g, "");
|
|
95
|
+
if (!path || isRuntimePath(path)) continue;
|
|
96
|
+
snapshot.set(path, {
|
|
97
|
+
path,
|
|
98
|
+
status: code.trim() || code,
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
return snapshot;
|
|
102
|
+
}
|
|
103
|
+
|
|
70
104
|
function formatCommandLabel(attemptedCommand: string): string {
|
|
71
105
|
const trimmed = attemptedCommand.trim();
|
|
72
106
|
return trimmed ? `/gsd ${trimmed}` : "/gsd";
|
|
@@ -132,7 +166,7 @@ export async function findUnmergedCompletedMilestones(base: string): Promise<Unm
|
|
|
132
166
|
await ensureDbOpen(base);
|
|
133
167
|
|
|
134
168
|
const blockers: UnmergedMilestoneBlocker[] = [];
|
|
135
|
-
|
|
169
|
+
let dirtyByPath: UnmergedMilestoneDirtySnapshot | null = null;
|
|
136
170
|
|
|
137
171
|
for (const milestone of getAllMilestones()) {
|
|
138
172
|
if (!isClosedStatus(milestone.status)) continue;
|
|
@@ -158,14 +192,16 @@ export async function findUnmergedCompletedMilestones(base: string): Promise<Unm
|
|
|
158
192
|
const uniqueFiles = [...new Set(files)].sort();
|
|
159
193
|
if (uniqueFiles.length === 0) continue;
|
|
160
194
|
|
|
195
|
+
if (!dirtyByPath) dirtyByPath = captureDirtyPathStatusSnapshot(base);
|
|
196
|
+
const dirtySnapshot = dirtyByPath;
|
|
161
197
|
blockers.push({
|
|
162
198
|
milestoneId: milestone.id,
|
|
163
199
|
branch,
|
|
164
200
|
integrationBranch,
|
|
165
201
|
files: uniqueFiles,
|
|
166
202
|
dirtyOverlap: uniqueFiles
|
|
167
|
-
.map((path) =>
|
|
168
|
-
.filter((entry): entry is
|
|
203
|
+
.map((path) => dirtySnapshot.get(path))
|
|
204
|
+
.filter((entry): entry is UnmergedMilestoneDirtyEntry => Boolean(entry)),
|
|
169
205
|
});
|
|
170
206
|
}
|
|
171
207
|
|
|
@@ -7,7 +7,7 @@ import { getAutoWorktreePath, isInAutoWorktree } from "./auto-worktree.js";
|
|
|
7
7
|
import { ensureDbOpen } from "./bootstrap/dynamic-tools.js";
|
|
8
8
|
import { refreshWorkflowDatabaseFromDisk } from "./db-workspace.js";
|
|
9
9
|
import { getIsolationMode } from "./preferences.js";
|
|
10
|
-
import { deriveState } from "./state.js";
|
|
10
|
+
import { deriveState, invalidateStateCache } from "./state.js";
|
|
11
11
|
import type { GSDState } from "./types.js";
|
|
12
12
|
import { detectWorktreeName } from "./worktree.js";
|
|
13
13
|
|
|
@@ -133,12 +133,7 @@ export function formatValidationBlockedMessage(
|
|
|
133
133
|
].join("\n\n");
|
|
134
134
|
}
|
|
135
135
|
|
|
136
|
-
|
|
137
|
-
base: string,
|
|
138
|
-
attemptedCommand = "",
|
|
139
|
-
): Promise<string | null> {
|
|
140
|
-
await ensureDbOpen(base);
|
|
141
|
-
refreshWorkflowDatabaseFromDisk();
|
|
136
|
+
async function deriveValidationBlockState(base: string): Promise<GSDState> {
|
|
142
137
|
let state = await deriveState(base);
|
|
143
138
|
|
|
144
139
|
if (
|
|
@@ -153,5 +148,16 @@ export async function getValidationBlockMessageForBase(
|
|
|
153
148
|
}
|
|
154
149
|
}
|
|
155
150
|
|
|
151
|
+
return state;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
export async function getValidationBlockMessageForBase(
|
|
155
|
+
base: string,
|
|
156
|
+
attemptedCommand = "",
|
|
157
|
+
): Promise<string | null> {
|
|
158
|
+
await ensureDbOpen(base);
|
|
159
|
+
refreshWorkflowDatabaseFromDisk();
|
|
160
|
+
invalidateStateCache();
|
|
161
|
+
const state = await deriveValidationBlockState(base);
|
|
156
162
|
return formatValidationBlockedMessage(state, attemptedCommand);
|
|
157
163
|
}
|
|
@@ -451,16 +451,14 @@ export async function renderStateProjection(basePath: string): Promise<void> {
|
|
|
451
451
|
}
|
|
452
452
|
}
|
|
453
453
|
|
|
454
|
-
// ───
|
|
454
|
+
// ─── renderMilestoneShellProjections ─────────────────────────────────────
|
|
455
455
|
|
|
456
456
|
/**
|
|
457
|
-
* Regenerate
|
|
458
|
-
*
|
|
457
|
+
* Regenerate milestone-level projections (roadmap, top-level views, STATE.md).
|
|
458
|
+
* Does not touch slice PLAN.md or task SUMMARY.md — those have authoritative
|
|
459
|
+
* renderers in plan-slice / complete-task.
|
|
459
460
|
*/
|
|
460
|
-
export async function
|
|
461
|
-
// Delegate to the authoritative roadmap renderer — the reduced
|
|
462
|
-
// renderRoadmapProjection omits sections like ## Boundary Map and would
|
|
463
|
-
// clobber the output written by plan-milestone / reassess-roadmap.
|
|
461
|
+
export async function renderMilestoneShellProjections(basePath: string, milestoneId: string): Promise<void> {
|
|
464
462
|
try {
|
|
465
463
|
await renderRoadmapFromDb(basePath, milestoneId);
|
|
466
464
|
} catch (err) {
|
|
@@ -476,6 +474,21 @@ export async function renderAllProjections(basePath: string, milestoneId: string
|
|
|
476
474
|
} catch (err) {
|
|
477
475
|
logWarning("projection", `renderTopLevelQueueFromDb failed: ${(err as Error).message}`);
|
|
478
476
|
}
|
|
477
|
+
try {
|
|
478
|
+
await renderStateProjection(basePath);
|
|
479
|
+
} catch (err) {
|
|
480
|
+
logWarning("projection", `renderStateProjection failed: ${(err as Error).message}`);
|
|
481
|
+
}
|
|
482
|
+
}
|
|
483
|
+
|
|
484
|
+
// ─── renderAllProjections ───────────────────────────────────────────────
|
|
485
|
+
|
|
486
|
+
/**
|
|
487
|
+
* Regenerate all projection files for a milestone from DB state.
|
|
488
|
+
* All calls are wrapped in try/catch — projection failure is non-fatal per D-02.
|
|
489
|
+
*/
|
|
490
|
+
export async function renderAllProjections(basePath: string, milestoneId: string): Promise<void> {
|
|
491
|
+
await renderMilestoneShellProjections(basePath, milestoneId);
|
|
479
492
|
|
|
480
493
|
// Query all slices for this milestone
|
|
481
494
|
const sliceRows = getMilestoneSlices(milestoneId);
|
|
@@ -498,13 +511,6 @@ export async function renderAllProjections(basePath: string, milestoneId: string
|
|
|
498
511
|
}
|
|
499
512
|
}
|
|
500
513
|
}
|
|
501
|
-
|
|
502
|
-
// Render STATE.md
|
|
503
|
-
try {
|
|
504
|
-
await renderStateProjection(basePath);
|
|
505
|
-
} catch (err) {
|
|
506
|
-
logWarning("projection", `renderStateProjection failed: ${(err as Error).message}`);
|
|
507
|
-
}
|
|
508
514
|
}
|
|
509
515
|
|
|
510
516
|
// ─── regenerateIfMissing ────────────────────────────────────────────────
|
|
@@ -7,6 +7,7 @@ import { getAutoWorktreePath } from "./auto-worktree.js";
|
|
|
7
7
|
import { isSafeToAutoResolve } from "./auto-worktree.js";
|
|
8
8
|
import { ensureDbOpen } from "./bootstrap/dynamic-tools.js";
|
|
9
9
|
import {
|
|
10
|
+
listMergeStateBlockers,
|
|
10
11
|
probeGitConflictState,
|
|
11
12
|
reconcileGitConflictsOnSignal,
|
|
12
13
|
type GitConflictProbeResult,
|
|
@@ -30,6 +31,9 @@ export type WorkspaceGitReadyResult =
|
|
|
30
31
|
targets: string[];
|
|
31
32
|
};
|
|
32
33
|
|
|
34
|
+
const CLEAN_TARGET_CACHE_TTL_MS = 10_000;
|
|
35
|
+
const cleanTargetProbeCache = new Map<string, number>();
|
|
36
|
+
|
|
33
37
|
function normalizeTargetPath(path: string): string {
|
|
34
38
|
try {
|
|
35
39
|
return realpathSync(path);
|
|
@@ -110,6 +114,25 @@ function formatBlockReason(
|
|
|
110
114
|
|
|
111
115
|
async function ensureTargetGitReady(target: string): Promise<WorkspaceGitReadyResult> {
|
|
112
116
|
const fixesApplied: string[] = [];
|
|
117
|
+
const cacheKey = normalizeTargetPath(target);
|
|
118
|
+
const cachedCleanAt = cleanTargetProbeCache.get(cacheKey);
|
|
119
|
+
if (cachedCleanAt !== undefined) {
|
|
120
|
+
if (Date.now() - cachedCleanAt < CLEAN_TARGET_CACHE_TTL_MS) {
|
|
121
|
+
// Merge-state markers (MERGE_HEAD, rebase-apply, rebase-merge) are the most
|
|
122
|
+
// common way a repo transitions from clean to dirty within the TTL window,
|
|
123
|
+
// including when a non-git folder is initialized as a repo mid-TTL and a
|
|
124
|
+
// merge immediately introduces conflicts. Check them here — they are pure
|
|
125
|
+
// existsSync calls with no git subprocess — and fall through to a full probe
|
|
126
|
+
// only when markers are present.
|
|
127
|
+
if (listMergeStateBlockers(cacheKey).length === 0) {
|
|
128
|
+
return { ok: true, fixesApplied };
|
|
129
|
+
}
|
|
130
|
+
cleanTargetProbeCache.delete(cacheKey);
|
|
131
|
+
} else {
|
|
132
|
+
cleanTargetProbeCache.delete(cacheKey);
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
113
136
|
let probe = probeGitConflictState(target);
|
|
114
137
|
|
|
115
138
|
for (let attempt = 0; attempt < 3 && probe.status === "dirty"; attempt++) {
|
|
@@ -129,6 +152,7 @@ async function ensureTargetGitReady(target: string): Promise<WorkspaceGitReadyRe
|
|
|
129
152
|
}
|
|
130
153
|
|
|
131
154
|
if (probe.status === "unknown") {
|
|
155
|
+
cleanTargetProbeCache.delete(cacheKey);
|
|
132
156
|
return {
|
|
133
157
|
ok: false,
|
|
134
158
|
reason: formatBlockReason("unrecoverable", [], [target]),
|
|
@@ -141,6 +165,7 @@ async function ensureTargetGitReady(target: string): Promise<WorkspaceGitReadyRe
|
|
|
141
165
|
|
|
142
166
|
const conflictedPaths = productConflictPaths(probe);
|
|
143
167
|
if (conflictedPaths.length > 0 || probe.checkFailures.length > 0) {
|
|
168
|
+
cleanTargetProbeCache.delete(cacheKey);
|
|
144
169
|
return {
|
|
145
170
|
ok: false,
|
|
146
171
|
reason: formatBlockReason("product-conflicts", conflictedPaths, [target]),
|
|
@@ -151,6 +176,12 @@ async function ensureTargetGitReady(target: string): Promise<WorkspaceGitReadyRe
|
|
|
151
176
|
};
|
|
152
177
|
}
|
|
153
178
|
|
|
179
|
+
if (probe.status === "clean") {
|
|
180
|
+
cleanTargetProbeCache.set(cacheKey, Date.now());
|
|
181
|
+
} else {
|
|
182
|
+
cleanTargetProbeCache.delete(cacheKey);
|
|
183
|
+
}
|
|
184
|
+
|
|
154
185
|
return { ok: true, fixesApplied };
|
|
155
186
|
}
|
|
156
187
|
|