gsd-pi 2.77.0 → 2.78.0
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/README.md +18 -36
- package/dist/claude-cli-check.js +5 -1
- package/dist/cli-web-branch.d.ts +1 -0
- package/dist/cli-web-branch.js +3 -0
- package/dist/cli.js +38 -2
- package/dist/extension-discovery.d.ts +6 -0
- package/dist/extension-discovery.js +37 -0
- package/dist/extension-registry.d.ts +3 -0
- package/dist/extension-sort.d.ts +18 -0
- package/dist/extension-sort.js +114 -0
- package/dist/extension-validator.d.ts +47 -0
- package/dist/extension-validator.js +127 -0
- package/dist/headless.js +49 -4
- package/dist/loader.js +35 -7
- package/dist/provider-migrations.d.ts +18 -0
- package/dist/provider-migrations.js +14 -0
- package/dist/resource-loader.d.ts +40 -0
- package/dist/resource-loader.js +32 -13
- package/dist/resources/extensions/browser-tools/capture.js +9 -0
- package/dist/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +8 -59
- package/dist/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +36 -24
- package/dist/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +69 -71
- package/dist/resources/extensions/browser-tools/tools/forms.js +5 -1
- package/dist/resources/extensions/browser-tools/tools/intent.js +5 -1
- package/dist/resources/extensions/claude-code-cli/readiness.js +5 -1
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +552 -67
- package/dist/resources/extensions/cmux/index.js +20 -0
- package/dist/resources/extensions/github-sync/templates.js +103 -0
- package/dist/resources/extensions/google-search/extension-manifest.json +5 -4
- package/dist/resources/extensions/google-search/index.js +3 -375
- package/dist/resources/extensions/gsd/abandon-detect.js +44 -0
- package/dist/resources/extensions/gsd/auto/loop.js +124 -2
- package/dist/resources/extensions/gsd/auto/phases.js +57 -39
- package/dist/resources/extensions/gsd/auto/resolve.js +24 -0
- package/dist/resources/extensions/gsd/auto/run-unit.js +10 -2
- package/dist/resources/extensions/gsd/auto/session.js +6 -2
- package/dist/resources/extensions/gsd/auto/turn-epoch.js +95 -0
- package/dist/resources/extensions/gsd/auto-dispatch.js +201 -38
- package/dist/resources/extensions/gsd/auto-loop.js +1 -1
- package/dist/resources/extensions/gsd/auto-model-selection.js +124 -4
- package/dist/resources/extensions/gsd/auto-post-unit.js +215 -64
- package/dist/resources/extensions/gsd/auto-prompts.js +372 -104
- package/dist/resources/extensions/gsd/auto-recovery.js +210 -24
- package/dist/resources/extensions/gsd/auto-start.js +122 -30
- package/dist/resources/extensions/gsd/auto-timeout-recovery.js +11 -5
- package/dist/resources/extensions/gsd/auto-tool-tracking.js +47 -7
- package/dist/resources/extensions/gsd/auto-unit-closeout.js +11 -2
- package/dist/resources/extensions/gsd/auto-worktree.js +180 -34
- package/dist/resources/extensions/gsd/auto.js +107 -35
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +19 -1
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +209 -0
- package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +5 -6
- package/dist/resources/extensions/gsd/bootstrap/register-extension.js +11 -0
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +7 -3
- package/dist/resources/extensions/gsd/bootstrap/system-context.js +11 -6
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +127 -9
- package/dist/resources/extensions/gsd/commands/handlers/workflow.js +31 -4
- package/dist/resources/extensions/gsd/commands-cmux.js +9 -6
- package/dist/resources/extensions/gsd/commands-extensions.js +634 -43
- package/dist/resources/extensions/gsd/component-loader.js +447 -0
- package/dist/resources/extensions/gsd/component-types.js +69 -0
- package/dist/resources/extensions/gsd/context-store.js +23 -7
- package/dist/resources/extensions/gsd/detection.js +49 -1
- package/dist/resources/extensions/gsd/dispatch-guard.js +29 -3
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -1
- package/dist/resources/extensions/gsd/file-lock.js +49 -9
- package/dist/resources/extensions/gsd/forensics.js +106 -0
- package/dist/resources/extensions/gsd/gate-registry.js +2 -2
- package/dist/resources/extensions/gsd/git-constants.js +28 -1
- package/dist/resources/extensions/gsd/git-self-heal.js +27 -0
- package/dist/resources/extensions/gsd/git-service.js +127 -2
- package/dist/resources/extensions/gsd/gitignore.js +1 -0
- package/dist/resources/extensions/gsd/gsd-db.js +6 -3
- package/dist/resources/extensions/gsd/guided-flow-queue.js +4 -1
- package/dist/resources/extensions/gsd/guided-flow.js +39 -13
- package/dist/resources/extensions/gsd/journal.js +17 -2
- package/dist/resources/extensions/gsd/memory-extractor.js +7 -1
- package/dist/resources/extensions/gsd/milestone-actions.js +15 -0
- package/dist/resources/extensions/gsd/milestone-scope-classifier.js +299 -0
- package/dist/resources/extensions/gsd/milestone-summary-classifier.js +37 -0
- package/dist/resources/extensions/gsd/model-cost-table.js +3 -0
- package/dist/resources/extensions/gsd/model-router.js +6 -0
- package/dist/resources/extensions/gsd/native-git-bridge.js +34 -4
- package/dist/resources/extensions/gsd/notifications.js +30 -16
- package/dist/resources/extensions/gsd/preferences-validation.js +23 -0
- package/dist/resources/extensions/gsd/prompt-cache-optimizer.js +4 -0
- package/dist/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
- package/dist/resources/extensions/gsd/prompts/discuss-headless.md +23 -4
- package/dist/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +15 -2
- package/dist/resources/extensions/gsd/prompts/system.md +1 -0
- package/dist/resources/extensions/gsd/reports.js +5 -4
- package/dist/resources/extensions/gsd/safety/git-checkpoint.js +11 -0
- package/dist/resources/extensions/gsd/service-tier.js +5 -2
- package/dist/resources/extensions/gsd/session-lock.js +19 -10
- package/dist/resources/extensions/gsd/skill-manifest.js +168 -0
- package/dist/resources/extensions/gsd/slice-cadence.js +238 -0
- package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +278 -8
- package/dist/resources/extensions/gsd/state-transition-matrix.js +118 -0
- package/dist/resources/extensions/gsd/state.js +69 -58
- package/dist/resources/extensions/gsd/sync-lock.js +98 -42
- package/dist/resources/extensions/gsd/tools/complete-slice.js +21 -0
- package/dist/resources/extensions/gsd/tools/complete-task.js +31 -0
- package/dist/resources/extensions/gsd/tools/validate-milestone.js +7 -2
- package/dist/resources/extensions/gsd/unit-context-composer.js +147 -0
- package/dist/resources/extensions/gsd/unit-context-manifest.js +370 -0
- package/dist/resources/extensions/gsd/uok/audit.js +18 -2
- package/dist/resources/extensions/gsd/uok/dispatch-envelope.js +33 -0
- package/dist/resources/extensions/gsd/uok/execution-graph.js +10 -0
- package/dist/resources/extensions/gsd/uok/gate-runner.js +53 -5
- package/dist/resources/extensions/gsd/uok/gitops.js +2 -1
- package/dist/resources/extensions/gsd/uok/loop-adapter.js +37 -10
- package/dist/resources/extensions/gsd/uok/parity-report.js +58 -0
- package/dist/resources/extensions/gsd/uok/plan-v2.js +10 -4
- package/dist/resources/extensions/gsd/uok/writer.js +82 -0
- package/dist/resources/extensions/gsd/workflow-logger.js +10 -2
- package/dist/resources/extensions/gsd/workflow-mcp.js +6 -0
- package/dist/resources/extensions/gsd/worktree-manager.js +86 -8
- package/dist/resources/extensions/gsd/worktree-resolver.js +86 -7
- package/dist/resources/extensions/gsd/worktree-telemetry.js +198 -0
- package/dist/resources/extensions/mcp-client/auth.js +10 -1
- package/dist/resources/extensions/mcp-client/index.js +121 -10
- package/dist/resources/extensions/ollama/index.js +5 -1
- package/dist/resources/extensions/remote-questions/manager.js +11 -5
- package/dist/resources/extensions/shared/cmux-events.js +12 -0
- package/dist/resources/extensions/shared/rtk-session-stats.js +1 -2
- package/dist/resources/skills/create-skill/SKILL.md +2 -2
- package/dist/resources/skills/create-skill/references/gsd-skill-ecosystem.md +4 -4
- package/dist/resources/skills/create-skill/workflows/audit-skill.md +4 -4
- package/dist/resources/skills/create-skill/workflows/create-new-skill.md +5 -5
- 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 +12 -12
- package/dist/web/standalone/.next/build-manifest.json +3 -3
- package/dist/web/standalone/.next/react-loadable-manifest.json +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.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/boot/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +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.js.nft.json +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.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/notifications/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/events/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +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.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +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.js.nft.json +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.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
- package/dist/web/standalone/.next/server/chunks/1926.js +1 -0
- package/dist/web/standalone/.next/server/chunks/6897.js +2 -2
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-manifest.json +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/static/chunks/2826.e9f5195e91f9cad2.js +11 -0
- package/dist/web/standalone/.next/static/chunks/{webpack-5fc74f13a25fa1bb.js → webpack-2e68521d7c82f7c2.js} +1 -1
- package/package.json +17 -16
- package/packages/daemon/package.json +2 -2
- package/packages/daemon/src/logger.ts +4 -3
- package/packages/mcp-server/README.md +3 -3
- package/packages/mcp-server/dist/env-writer.d.ts +1 -0
- package/packages/mcp-server/dist/env-writer.d.ts.map +1 -1
- package/packages/mcp-server/dist/env-writer.js +74 -6
- package/packages/mcp-server/dist/env-writer.js.map +1 -1
- package/packages/mcp-server/dist/server.d.ts +24 -0
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +111 -87
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +15 -6
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +7 -2
- package/packages/mcp-server/src/env-writer.test.ts +79 -1
- package/packages/mcp-server/src/env-writer.ts +76 -6
- package/packages/mcp-server/src/mcp-server.test.ts +25 -3
- package/packages/mcp-server/src/readers/graph.test.ts +87 -15
- package/packages/mcp-server/src/readers/readers.test.ts +5 -1
- package/packages/mcp-server/src/secure-env-collect.test.ts +232 -237
- package/packages/mcp-server/src/server.ts +158 -105
- package/packages/mcp-server/src/workflow-tools.test.ts +85 -0
- package/packages/mcp-server/src/workflow-tools.ts +19 -6
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
- package/packages/native/package.json +7 -2
- package/packages/native/src/__tests__/_test-coverage-guard.test.mjs +98 -0
- package/packages/native/src/__tests__/clipboard.test.mjs +69 -23
- package/packages/native/src/__tests__/module-compat.test.mjs +59 -27
- package/packages/native/src/__tests__/ps.test.mjs +14 -8
- package/packages/native/src/__tests__/stream-process.test.mjs +23 -2
- package/packages/native/src/__tests__/truncate.test.mjs +17 -2
- package/packages/native/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-agent-core/package.json +6 -1
- package/packages/pi-agent-core/src/agent-loop.test.ts +226 -31
- package/packages/pi-agent-core/src/agent.test.ts +96 -102
- package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/capability-patches.js +9 -2
- package/packages/pi-ai/dist/models/capability-patches.js.map +1 -1
- package/packages/pi-ai/dist/models/generated/index.d.ts +34 -0
- package/packages/pi-ai/dist/models/generated/index.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +17 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/generated/openai-codex.js +17 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -1
- package/packages/pi-ai/dist/models/generated/openai.d.ts +17 -0
- package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/generated/openai.js +17 -0
- package/packages/pi-ai/dist/models/generated/openai.js.map +1 -1
- package/packages/pi-ai/dist/models.generated.test.js +43 -70
- package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
- package/packages/pi-ai/dist/models.test.js +36 -11
- package/packages/pi-ai/dist/models.test.js.map +1 -1
- package/packages/pi-ai/package.json +6 -1
- package/packages/pi-ai/scripts/generate-models.ts +44 -0
- package/packages/pi-ai/src/models/capability-patches.ts +10 -2
- package/packages/pi-ai/src/models/generated/openai-codex.ts +17 -0
- package/packages/pi-ai/src/models/generated/openai.ts +17 -0
- package/packages/pi-ai/src/models.generated.test.ts +46 -73
- package/packages/pi-ai/src/models.test.ts +48 -11
- package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +96 -32
- package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js +75 -12
- package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js +99 -31
- package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts +25 -0
- package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction/compaction.js +105 -6
- package/packages/pi-coding-agent/dist/core/compaction/compaction.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js +230 -28
- package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction/utils.d.ts +30 -2
- package/packages/pi-coding-agent/dist/core/compaction/utils.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction/utils.js +113 -12
- package/packages/pi-coding-agent/dist/core/compaction/utils.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction-orchestrator.d.ts +1 -0
- package/packages/pi-coding-agent/dist/core/compaction-orchestrator.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction-orchestrator.js +29 -18
- package/packages/pi-coding-agent/dist/core/compaction-orchestrator.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.js +130 -0
- package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/compaction-utils.test.js +56 -1
- package/packages/pi-coding-agent/dist/core/compaction-utils.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/discovery-cache.test.js +8 -15
- package/packages/pi-coding-agent/dist/core/discovery-cache.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.d.ts +25 -0
- package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.js +109 -0
- package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/extensions/extension-registry.d.ts +67 -0
- package/packages/pi-coding-agent/dist/core/extensions/extension-registry.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/extensions/extension-registry.js +167 -0
- package/packages/pi-coding-agent/dist/core/extensions/extension-registry.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +8 -2
- package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/loader.js +85 -8
- package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +7 -0
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js +41 -4
- package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +19 -2
- package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js +76 -18
- package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/resource-loader.js +1 -1
- package/packages/pi-coding-agent/dist/core/resource-loader.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.js +2 -6
- package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.test.js +5 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts +18 -0
- package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/retryable-error-regex.js +18 -0
- package/packages/pi-coding-agent/dist/core/retryable-error-regex.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/sdk.d.ts +1 -0
- package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.js +4 -1
- package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.test.js +19 -1
- package/packages/pi-coding-agent/dist/core/sdk.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/system-prompt.d.ts +20 -0
- package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/system-prompt.js +19 -5
- package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/tools/path-utils.test.js +2 -1
- package/packages/pi-coding-agent/dist/core/tools/path-utils.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/index.d.ts +1 -0
- package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/index.js +1 -0
- package/packages/pi-coding-agent/dist/index.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/provider-display-name.test.js +15 -6
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/provider-display-name.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +36 -5
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +20 -13
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +14 -5
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts +7 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js +31 -9
- package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +30 -12
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +18 -3
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +139 -0
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +4 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +105 -13
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js +130 -0
- package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js.map +1 -0
- package/packages/pi-coding-agent/package.json +6 -1
- package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +113 -37
- package/packages/pi-coding-agent/src/core/agent-session-model-switch.test.ts +89 -17
- package/packages/pi-coding-agent/src/core/agent-session-tool-refresh.test.ts +112 -43
- package/packages/pi-coding-agent/src/core/compaction/compaction.test.ts +368 -28
- package/packages/pi-coding-agent/src/core/compaction/compaction.ts +122 -6
- package/packages/pi-coding-agent/src/core/compaction/utils.ts +111 -13
- package/packages/pi-coding-agent/src/core/compaction-orchestrator.test.ts +154 -0
- package/packages/pi-coding-agent/src/core/compaction-orchestrator.ts +32 -18
- package/packages/pi-coding-agent/src/core/compaction-utils.test.ts +68 -1
- package/packages/pi-coding-agent/src/core/discovery-cache.test.ts +9 -18
- package/packages/pi-coding-agent/src/core/extensions/extension-discovery.ts +119 -0
- package/packages/pi-coding-agent/src/core/extensions/extension-registry.ts +222 -0
- package/packages/pi-coding-agent/src/core/extensions/loader.ts +82 -11
- package/packages/pi-coding-agent/src/core/extensions/types.ts +8 -0
- package/packages/pi-coding-agent/src/core/lsp/lsp-integration.test.ts +48 -4
- package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +22 -2
- package/packages/pi-coding-agent/src/core/resource-loader-cache-reset.test.ts +93 -28
- package/packages/pi-coding-agent/src/core/resource-loader.ts +1 -1
- package/packages/pi-coding-agent/src/core/retry-handler.test.ts +5 -1
- package/packages/pi-coding-agent/src/core/retry-handler.ts +2 -8
- package/packages/pi-coding-agent/src/core/retryable-error-regex.ts +18 -0
- package/packages/pi-coding-agent/src/core/sdk.test.ts +25 -1
- package/packages/pi-coding-agent/src/core/sdk.ts +10 -3
- package/packages/pi-coding-agent/src/core/system-prompt.ts +38 -4
- package/packages/pi-coding-agent/src/core/tools/path-utils.test.ts +2 -1
- package/packages/pi-coding-agent/src/index.ts +1 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/provider-display-name.test.ts +17 -7
- package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +49 -3
- package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +26 -20
- package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +14 -5
- package/packages/pi-coding-agent/src/modes/interactive/components/model-selector.ts +45 -11
- package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +48 -9
- package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +160 -1
- package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +20 -3
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-state.ts +2 -0
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +119 -13
- package/packages/pi-coding-agent/src/tests/system-prompt-skill-filter.test.ts +157 -0
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-tui/dist/__tests__/autocomplete.test.js +31 -14
- package/packages/pi-tui/dist/__tests__/autocomplete.test.js.map +1 -1
- package/packages/pi-tui/dist/__tests__/overlay-layout.test.js +128 -17
- package/packages/pi-tui/dist/__tests__/overlay-layout.test.js.map +1 -1
- package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js +51 -6
- package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js.map +1 -1
- package/packages/pi-tui/dist/__tests__/tui.test.js +18 -30
- package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
- package/packages/pi-tui/dist/components/__tests__/input.test.js +10 -3
- package/packages/pi-tui/dist/components/__tests__/input.test.js.map +1 -1
- package/packages/pi-tui/dist/components/__tests__/loader.test.js +53 -9
- package/packages/pi-tui/dist/components/__tests__/loader.test.js.map +1 -1
- package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js +6 -2
- package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js.map +1 -1
- package/packages/pi-tui/dist/components/editor.d.ts +14 -0
- package/packages/pi-tui/dist/components/editor.d.ts.map +1 -1
- package/packages/pi-tui/dist/components/editor.js +19 -0
- package/packages/pi-tui/dist/components/editor.js.map +1 -1
- package/packages/pi-tui/dist/components/image.test.js +6 -5
- package/packages/pi-tui/dist/components/image.test.js.map +1 -1
- package/packages/pi-tui/dist/editor-component.d.ts +2 -0
- package/packages/pi-tui/dist/editor-component.d.ts.map +1 -1
- package/packages/pi-tui/dist/editor-component.js.map +1 -1
- package/packages/pi-tui/dist/stdin-buffer.d.ts +7 -0
- package/packages/pi-tui/dist/stdin-buffer.d.ts.map +1 -1
- package/packages/pi-tui/dist/stdin-buffer.js +20 -0
- package/packages/pi-tui/dist/stdin-buffer.js.map +1 -1
- package/packages/pi-tui/package.json +6 -1
- package/packages/pi-tui/src/__tests__/autocomplete.test.ts +46 -15
- package/packages/pi-tui/src/__tests__/overlay-layout.test.ts +140 -17
- package/packages/pi-tui/src/__tests__/stdin-buffer.test.ts +62 -6
- package/packages/pi-tui/src/__tests__/tui.test.ts +18 -37
- package/packages/pi-tui/src/components/__tests__/input.test.ts +19 -3
- package/packages/pi-tui/src/components/__tests__/loader.test.ts +112 -35
- package/packages/pi-tui/src/components/__tests__/markdown-maxlines.test.ts +9 -2
- package/packages/pi-tui/src/components/editor.ts +22 -0
- package/packages/pi-tui/src/components/image.test.ts +10 -5
- package/packages/pi-tui/src/editor-component.ts +3 -0
- package/packages/pi-tui/src/stdin-buffer.ts +26 -0
- package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
- package/packages/rpc-client/dist/rpc-client.test.js +101 -51
- package/packages/rpc-client/dist/rpc-client.test.js.map +1 -1
- package/packages/rpc-client/package.json +6 -1
- package/packages/rpc-client/src/rpc-client.test.ts +109 -52
- package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
- package/pkg/package.json +1 -1
- package/scripts/install.js +526 -0
- package/scripts/lib/workspace-manifest.cjs +86 -0
- package/scripts/link-workspace-packages.cjs +5 -17
- package/scripts/postinstall.js +9 -178
- package/src/resources/extensions/browser-tools/capture.ts +12 -0
- package/src/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +8 -59
- package/src/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +36 -24
- package/src/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +69 -71
- package/src/resources/extensions/browser-tools/tools/forms.ts +5 -1
- package/src/resources/extensions/browser-tools/tools/intent.ts +5 -1
- package/src/resources/extensions/claude-code-cli/readiness.ts +5 -1
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +602 -73
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +1028 -91
- package/src/resources/extensions/cmux/index.ts +35 -10
- package/src/resources/extensions/github-sync/templates.ts +151 -0
- package/src/resources/extensions/github-sync/tests/cli.test.ts +76 -7
- package/src/resources/extensions/github-sync/tests/templates.test.ts +92 -1
- package/src/resources/extensions/google-search/extension-manifest.json +5 -4
- package/src/resources/extensions/google-search/index.ts +9 -470
- package/src/resources/extensions/gsd/abandon-detect.ts +62 -0
- package/src/resources/extensions/gsd/auto/loop-deps.ts +1 -1
- package/src/resources/extensions/gsd/auto/loop.ts +142 -2
- package/src/resources/extensions/gsd/auto/phases.ts +62 -38
- package/src/resources/extensions/gsd/auto/resolve.ts +29 -0
- package/src/resources/extensions/gsd/auto/run-unit.ts +16 -2
- package/src/resources/extensions/gsd/auto/session.ts +7 -2
- package/src/resources/extensions/gsd/auto/turn-epoch.ts +108 -0
- package/src/resources/extensions/gsd/auto/types.ts +1 -1
- package/src/resources/extensions/gsd/auto-dispatch.ts +214 -37
- package/src/resources/extensions/gsd/auto-loop.ts +1 -1
- package/src/resources/extensions/gsd/auto-model-selection.ts +131 -4
- package/src/resources/extensions/gsd/auto-post-unit.ts +226 -73
- package/src/resources/extensions/gsd/auto-prompts.ts +385 -93
- package/src/resources/extensions/gsd/auto-recovery.ts +240 -25
- package/src/resources/extensions/gsd/auto-start.ts +146 -14
- package/src/resources/extensions/gsd/auto-timeout-recovery.ts +12 -5
- package/src/resources/extensions/gsd/auto-tool-tracking.ts +51 -7
- package/src/resources/extensions/gsd/auto-unit-closeout.ts +14 -3
- package/src/resources/extensions/gsd/auto-worktree.ts +190 -31
- package/src/resources/extensions/gsd/auto.ts +127 -41
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +20 -1
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +221 -0
- package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +6 -6
- package/src/resources/extensions/gsd/bootstrap/register-extension.ts +11 -0
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +7 -3
- package/src/resources/extensions/gsd/bootstrap/system-context.ts +13 -9
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +158 -9
- package/src/resources/extensions/gsd/commands/handlers/workflow.ts +27 -8
- package/src/resources/extensions/gsd/commands-cmux.ts +10 -6
- package/src/resources/extensions/gsd/commands-extensions.ts +747 -41
- package/src/resources/extensions/gsd/component-loader.ts +598 -0
- package/src/resources/extensions/gsd/component-types.ts +362 -0
- package/src/resources/extensions/gsd/context-store.ts +25 -8
- package/src/resources/extensions/gsd/detection.ts +58 -1
- package/src/resources/extensions/gsd/dispatch-guard.ts +26 -2
- package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -1
- package/src/resources/extensions/gsd/file-lock.ts +84 -11
- package/src/resources/extensions/gsd/forensics.ts +118 -1
- package/src/resources/extensions/gsd/gate-registry.ts +2 -2
- package/src/resources/extensions/gsd/git-constants.ts +30 -1
- package/src/resources/extensions/gsd/git-self-heal.ts +31 -0
- package/src/resources/extensions/gsd/git-service.ts +150 -2
- package/src/resources/extensions/gsd/gitignore.ts +1 -0
- package/src/resources/extensions/gsd/gsd-db.ts +6 -3
- package/src/resources/extensions/gsd/guided-flow-queue.ts +4 -1
- package/src/resources/extensions/gsd/guided-flow.ts +57 -14
- package/src/resources/extensions/gsd/journal.ts +38 -3
- package/src/resources/extensions/gsd/memory-extractor.ts +11 -3
- package/src/resources/extensions/gsd/milestone-actions.ts +18 -0
- package/src/resources/extensions/gsd/milestone-scope-classifier.ts +366 -0
- package/src/resources/extensions/gsd/milestone-summary-classifier.ts +42 -0
- package/src/resources/extensions/gsd/model-cost-table.ts +3 -0
- package/src/resources/extensions/gsd/model-router.ts +6 -0
- package/src/resources/extensions/gsd/native-git-bridge.ts +34 -4
- package/src/resources/extensions/gsd/notifications.ts +27 -15
- package/src/resources/extensions/gsd/preferences-validation.ts +21 -0
- package/src/resources/extensions/gsd/prompt-cache-optimizer.ts +4 -0
- package/src/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
- package/src/resources/extensions/gsd/prompts/discuss-headless.md +23 -4
- package/src/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
- package/src/resources/extensions/gsd/prompts/plan-slice.md +15 -2
- package/src/resources/extensions/gsd/prompts/system.md +1 -0
- package/src/resources/extensions/gsd/reports.ts +5 -4
- package/src/resources/extensions/gsd/safety/git-checkpoint.ts +15 -0
- package/src/resources/extensions/gsd/service-tier.ts +5 -2
- package/src/resources/extensions/gsd/session-lock.ts +20 -10
- package/src/resources/extensions/gsd/skill-manifest.ts +175 -0
- package/src/resources/extensions/gsd/slice-cadence.ts +299 -0
- package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +309 -8
- package/src/resources/extensions/gsd/state-transition-matrix.ts +152 -0
- package/src/resources/extensions/gsd/state.ts +76 -66
- package/src/resources/extensions/gsd/sync-lock.ts +97 -39
- package/src/resources/extensions/gsd/tests/artifact-retry-cap.test.ts +270 -0
- package/src/resources/extensions/gsd/tests/artifacts-table-preserved-on-cache-invalidate.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +341 -0
- package/src/resources/extensions/gsd/tests/auto-discuss-milestone-deadlock-4973.test.ts +264 -0
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +135 -285
- package/src/resources/extensions/gsd/tests/auto-mode-guards.test.ts +79 -0
- package/src/resources/extensions/gsd/tests/auto-model-selection-tool-poisoning.test.ts +742 -0
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +78 -0
- package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +61 -0
- package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +166 -0
- package/src/resources/extensions/gsd/tests/auto-remediate-slice-status.test.ts +4 -1
- package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +8 -194
- package/src/resources/extensions/gsd/tests/auto-start-clean-runtime-db-gated.test.ts +64 -0
- package/src/resources/extensions/gsd/tests/auto-start-cold-db-bootstrap.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/auto-start-needs-discussion.test.ts +15 -58
- package/src/resources/extensions/gsd/tests/auto-start-worktree-db-path.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/auto-thinking-restore.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/auto-warning-noise-regression.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/bootstrap-derive-state-db-open.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/cache-staleness-regression.test.ts +17 -21
- package/src/resources/extensions/gsd/tests/canonical-milestone-root.test.ts +108 -0
- package/src/resources/extensions/gsd/tests/cmux.test.ts +5 -9
- package/src/resources/extensions/gsd/tests/complete-milestone-excerpt.test.ts +263 -0
- package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +25 -0
- package/src/resources/extensions/gsd/tests/complete-slice-composer.test.ts +192 -0
- package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +16 -8
- package/src/resources/extensions/gsd/tests/component-loader.test.ts +589 -0
- package/src/resources/extensions/gsd/tests/component-types.test.ts +127 -0
- package/src/resources/extensions/gsd/tests/context-store.test.ts +79 -0
- package/src/resources/extensions/gsd/tests/copy-planning-artifacts-samepath.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/crash-recovery.test.ts +50 -1
- package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +159 -0
- package/src/resources/extensions/gsd/tests/db-access-guardrails.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/derive-state-db-disk-reconcile.test.ts +40 -0
- package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +91 -3
- package/src/resources/extensions/gsd/tests/derive-state.test.ts +4 -4
- package/src/resources/extensions/gsd/tests/discuss-slice-structured-questions.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/discuss-tool-scope-leak.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +14 -9
- package/src/resources/extensions/gsd/tests/dispatch-guard-summary-db-mismatch.test.ts +77 -0
- package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +25 -0
- package/src/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +14 -0
- package/src/resources/extensions/gsd/tests/dispatcher-stuck-planning.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +4 -3
- package/src/resources/extensions/gsd/tests/empty-content-abort-loop.test.ts +4 -3
- package/src/resources/extensions/gsd/tests/execution-entry-missing-context-4671.test.ts +173 -0
- package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +139 -129
- package/src/resources/extensions/gsd/tests/file-lock.test.ts +86 -12
- package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +8 -104
- package/src/resources/extensions/gsd/tests/gate-state-canonicalization.test.ts +102 -0
- package/src/resources/extensions/gsd/tests/gate-storage.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/google-search-stub.test.ts +131 -0
- package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +117 -0
- package/src/resources/extensions/gsd/tests/hook-key-parsing.test.ts +4 -55
- package/src/resources/extensions/gsd/tests/integration/all-milestones-complete-merge.test.ts +7 -56
- package/src/resources/extensions/gsd/tests/integration/auto-recovery.test.ts +20 -0
- package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +30 -0
- package/src/resources/extensions/gsd/tests/integration/doctor-proactive.test.ts +18 -2
- package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/integration/queue-completed-milestone-perf.test.ts +10 -4
- package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +144 -7
- package/src/resources/extensions/gsd/tests/integration/state-machine-live-validation.test.ts +4 -0
- package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +2 -16
- package/src/resources/extensions/gsd/tests/integration/worktree-e2e.test.ts +11 -0
- package/src/resources/extensions/gsd/tests/interactive-routing-bypass.test.ts +9 -3
- package/src/resources/extensions/gsd/tests/interrupted-session-ui.test.ts +6 -9
- package/src/resources/extensions/gsd/tests/journal-integration.test.ts +64 -0
- package/src/resources/extensions/gsd/tests/knowledge.test.ts +93 -1
- package/src/resources/extensions/gsd/tests/mcp-client-security.test.ts +47 -0
- package/src/resources/extensions/gsd/tests/memory-extractor.test.ts +5 -15
- package/src/resources/extensions/gsd/tests/merge-conflict-stops-loop.test.ts +227 -55
- package/src/resources/extensions/gsd/tests/milestone-scope-classifier.test.ts +187 -0
- package/src/resources/extensions/gsd/tests/milestone-status-authoritative.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/milestone-summary-classifier.test.ts +30 -0
- package/src/resources/extensions/gsd/tests/milestone-transition-state-rebuild.test.ts +4 -2
- package/src/resources/extensions/gsd/tests/model-cost-table.test.ts +9 -1
- package/src/resources/extensions/gsd/tests/model-router.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +6 -48
- package/src/resources/extensions/gsd/tests/notification-widget.test.ts +6 -3
- package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +59 -2
- package/src/resources/extensions/gsd/tests/parallel-commit-scope.test.ts +5 -0
- package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +273 -130
- package/src/resources/extensions/gsd/tests/parallel-skill-prompt-integration.test.ts +150 -0
- package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +301 -0
- package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +32 -1
- package/src/resources/extensions/gsd/tests/preferences-worktree-sync.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/prompt-cache-optimizer.test.ts +12 -0
- package/src/resources/extensions/gsd/tests/prompt-step-ordering.test.ts +15 -4
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +54 -41
- package/src/resources/extensions/gsd/tests/queue-auto-guard.test.ts +213 -0
- package/src/resources/extensions/gsd/tests/queue-draft-detection.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/queued-discuss-fast-path.test.ts +4 -5
- package/src/resources/extensions/gsd/tests/quick-auto-guard.test.ts +13 -7
- package/src/resources/extensions/gsd/tests/ready-phrase-no-files-4573.test.ts +75 -2
- package/src/resources/extensions/gsd/tests/reassess-default-optin.test.ts +132 -0
- package/src/resources/extensions/gsd/tests/recovery-attempts-reset.test.ts +8 -40
- package/src/resources/extensions/gsd/tests/regex-hardening.test.ts +136 -256
- package/src/resources/extensions/gsd/tests/require-slice-discussion-dispatch.test.ts +170 -0
- package/src/resources/extensions/gsd/tests/research-milestone-composer.test.ts +114 -0
- package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +6 -3
- package/src/resources/extensions/gsd/tests/rewrite-docs-abandon-detect.test.ts +195 -0
- package/src/resources/extensions/gsd/tests/run-uat-composer.test.ts +148 -0
- package/src/resources/extensions/gsd/tests/service-tier.test.ts +4 -0
- package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +29 -0
- package/src/resources/extensions/gsd/tests/sidecar-queue.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/silent-catch-diagnostics.test.ts +55 -95
- package/src/resources/extensions/gsd/tests/single-writer-v3-tool-surface.test.ts +158 -0
- package/src/resources/extensions/gsd/tests/skill-activation.test.ts +120 -1
- package/src/resources/extensions/gsd/tests/skill-manifest.test.ts +112 -0
- package/src/resources/extensions/gsd/tests/slice-cadence.test.ts +242 -0
- package/src/resources/extensions/gsd/tests/slice-context-injection.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +164 -1
- package/src/resources/extensions/gsd/tests/smart-entry-draft.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/stale-dirlistcache-4648.test.ts +112 -0
- package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +29 -5
- package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +44 -0
- package/src/resources/extensions/gsd/tests/stop-auto-race-null-unit.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/structured-data-formatter.test.ts +11 -92
- package/src/resources/extensions/gsd/tests/stuck-detection-coverage.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/subagent-model-dispatch.test.ts +7 -6
- package/src/resources/extensions/gsd/tests/survivor-branch-complete.test.ts +102 -101
- package/src/resources/extensions/gsd/tests/sync-lock.test.ts +31 -0
- package/src/resources/extensions/gsd/tests/sync-worktree-skip-current.test.ts +4 -3
- package/src/resources/extensions/gsd/tests/test-helpers.test.ts +98 -0
- package/src/resources/extensions/gsd/tests/test-helpers.ts +153 -0
- package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -1
- package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +61 -1
- package/src/resources/extensions/gsd/tests/tool-naming.test.ts +8 -1
- package/src/resources/extensions/gsd/tests/triage-resolution.test.ts +50 -2
- package/src/resources/extensions/gsd/tests/turn-epoch.test.ts +162 -0
- package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +355 -0
- package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +258 -0
- package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +51 -0
- package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +16 -0
- package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +75 -0
- package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +49 -26
- package/src/resources/extensions/gsd/tests/uok-loop-adapter-writer.test.ts +65 -0
- package/src/resources/extensions/gsd/tests/uok-parity-report.test.ts +42 -0
- package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +19 -2
- package/src/resources/extensions/gsd/tests/uok-writer.test.ts +75 -0
- package/src/resources/extensions/gsd/tests/validate-extension-package.test.ts +168 -0
- package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +139 -5
- package/src/resources/extensions/gsd/tests/verify-artifact-tightened.test.ts +144 -80
- package/src/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +20 -54
- package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +342 -277
- package/src/resources/extensions/gsd/tests/worker-model-override.test.ts +37 -29
- package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +25 -2
- package/src/resources/extensions/gsd/tests/worktree-db.test.ts +226 -266
- package/src/resources/extensions/gsd/tests/worktree-health-monorepo.test.ts +103 -67
- package/src/resources/extensions/gsd/tests/worktree-nested-git-safety.test.ts +92 -90
- package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +238 -59
- package/src/resources/extensions/gsd/tests/worktree-sync-overwrite-loop.test.ts +113 -161
- package/src/resources/extensions/gsd/tests/worktree-telemetry.test.ts +210 -0
- package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +262 -0
- package/src/resources/extensions/gsd/tests/write-gate-predicates.test.ts +186 -0
- package/src/resources/extensions/gsd/tests/write-gate.test.ts +7 -5
- package/src/resources/extensions/gsd/tests/zombie-gsd-state.test.ts +80 -96
- package/src/resources/extensions/gsd/tools/complete-slice.ts +38 -0
- package/src/resources/extensions/gsd/tools/complete-task.ts +49 -0
- package/src/resources/extensions/gsd/tools/validate-milestone.ts +8 -2
- package/src/resources/extensions/gsd/types.ts +3 -3
- package/src/resources/extensions/gsd/unit-context-composer.ts +218 -0
- package/src/resources/extensions/gsd/unit-context-manifest.ts +574 -0
- package/src/resources/extensions/gsd/uok/audit.ts +20 -2
- package/src/resources/extensions/gsd/uok/contracts.ts +65 -0
- package/src/resources/extensions/gsd/uok/dispatch-envelope.ts +56 -0
- package/src/resources/extensions/gsd/uok/execution-graph.ts +22 -0
- package/src/resources/extensions/gsd/uok/gate-runner.ts +65 -5
- package/src/resources/extensions/gsd/uok/gitops.ts +6 -1
- package/src/resources/extensions/gsd/uok/loop-adapter.ts +45 -10
- package/src/resources/extensions/gsd/uok/parity-report.ts +84 -0
- package/src/resources/extensions/gsd/uok/plan-v2.ts +13 -5
- package/src/resources/extensions/gsd/uok/writer.ts +113 -0
- package/src/resources/extensions/gsd/workflow-logger.ts +22 -3
- package/src/resources/extensions/gsd/workflow-mcp.ts +6 -0
- package/src/resources/extensions/gsd/worktree-manager.ts +109 -7
- package/src/resources/extensions/gsd/worktree-resolver.ts +96 -9
- package/src/resources/extensions/gsd/worktree-telemetry.ts +322 -0
- package/src/resources/extensions/mcp-client/auth.ts +12 -1
- package/src/resources/extensions/mcp-client/index.ts +132 -11
- package/src/resources/extensions/mcp-client/tests/server-name-spaces.test.ts +70 -36
- package/src/resources/extensions/ollama/index.ts +5 -1
- package/src/resources/extensions/ollama/ollama-auth-mode.test.ts +123 -15
- package/src/resources/extensions/ollama/ollama-status-indicator.test.ts +206 -19
- package/src/resources/extensions/remote-questions/manager.ts +36 -4
- package/src/resources/extensions/remote-questions/tests/command-polling.test.ts +200 -190
- package/src/resources/extensions/shared/cmux-events.ts +59 -0
- package/src/resources/extensions/shared/rtk-session-stats.ts +1 -2
- package/src/resources/extensions/shared/tests/interview-preview.test.ts +11 -3
- package/src/resources/extensions/voice/tests/linux-ready.test.ts +129 -113
- package/src/resources/skills/create-skill/SKILL.md +2 -2
- package/src/resources/skills/create-skill/references/gsd-skill-ecosystem.md +4 -4
- package/src/resources/skills/create-skill/workflows/audit-skill.md +4 -4
- package/src/resources/skills/create-skill/workflows/create-new-skill.md +5 -5
- package/dist/web/standalone/.next/server/chunks/7461.js +0 -1
- package/dist/web/standalone/.next/static/chunks/2826.e59e8578e2e28639.js +0 -9
- package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts +0 -2
- package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts.map +0 -1
- package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js +0 -289
- package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js.map +0 -1
- package/packages/pi-ai/src/utils/oauth/oauth-providers.test.ts +0 -363
- package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +0 -143
- package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +0 -142
- package/src/resources/extensions/gsd/tests/dashboard-model-label-ordering.test.ts +0 -107
- package/src/resources/extensions/gsd/tests/find-missing-summaries-closed.test.ts +0 -48
- package/src/resources/extensions/gsd/tests/forensics-context-persist.test.ts +0 -159
- package/src/resources/extensions/gsd/tests/forensics-db-completion.test.ts +0 -96
- package/src/resources/extensions/gsd/tests/forensics-dedup.test.ts +0 -79
- package/src/resources/extensions/gsd/tests/forensics-hook-key-parse.test.ts +0 -74
- package/src/resources/extensions/gsd/tests/forensics-journal.test.ts +0 -162
- package/src/resources/extensions/gsd/tests/gitignore-bg-shell.test.ts +0 -38
- package/src/resources/extensions/gsd/tests/gsd-no-project-error.test.ts +0 -73
- package/src/resources/extensions/gsd/tests/idle-watchdog-stall-override.test.ts +0 -125
- package/src/resources/extensions/gsd/tests/import-done-milestones.test.ts +0 -42
- /package/dist/web/standalone/.next/static/{pV-mPo7rYGb5JBC09C8GG → C1zT2kEfoLhDdbWPWKrXd}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{pV-mPo7rYGb5JBC09C8GG → C1zT2kEfoLhDdbWPWKrXd}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,118 @@
|
|
|
1
|
+
export const STATE_TRANSITION_MATRIX = [
|
|
2
|
+
{
|
|
3
|
+
from: "needs-discussion",
|
|
4
|
+
event: "context-ready",
|
|
5
|
+
guard: "CONTEXT artifact exists or PRD/context express path produced context",
|
|
6
|
+
to: "researching",
|
|
7
|
+
onFail: "needs-discussion",
|
|
8
|
+
reasonCode: "state",
|
|
9
|
+
},
|
|
10
|
+
{
|
|
11
|
+
from: "researching",
|
|
12
|
+
event: "research-ready",
|
|
13
|
+
guard: "RESEARCH artifact exists or research is explicitly skipped",
|
|
14
|
+
to: "planning",
|
|
15
|
+
onFail: "researching",
|
|
16
|
+
reasonCode: "state",
|
|
17
|
+
},
|
|
18
|
+
{
|
|
19
|
+
from: "planning",
|
|
20
|
+
event: "plan-ready",
|
|
21
|
+
guard: "ROADMAP/PLAN artifacts exist and plan gate passes",
|
|
22
|
+
to: "executing",
|
|
23
|
+
onFail: "replanning-slice",
|
|
24
|
+
reasonCode: "state",
|
|
25
|
+
},
|
|
26
|
+
{
|
|
27
|
+
from: "executing",
|
|
28
|
+
event: "task-dispatched",
|
|
29
|
+
guard: "task inputs are ready and dependencies are closed",
|
|
30
|
+
to: "executing",
|
|
31
|
+
onFail: "blocked",
|
|
32
|
+
reasonCode: "dependency",
|
|
33
|
+
},
|
|
34
|
+
{
|
|
35
|
+
from: "executing",
|
|
36
|
+
event: "slice-complete",
|
|
37
|
+
guard: "all slice tasks are closed and verification gate passes",
|
|
38
|
+
to: "summarizing",
|
|
39
|
+
onFail: "validating-milestone",
|
|
40
|
+
reasonCode: "state",
|
|
41
|
+
},
|
|
42
|
+
{
|
|
43
|
+
from: "summarizing",
|
|
44
|
+
event: "summary-ready",
|
|
45
|
+
guard: "SUMMARY artifact exists for the completed work unit",
|
|
46
|
+
to: "validating-milestone",
|
|
47
|
+
onFail: "summarizing",
|
|
48
|
+
reasonCode: "state",
|
|
49
|
+
},
|
|
50
|
+
{
|
|
51
|
+
from: "validating-milestone",
|
|
52
|
+
event: "validation-pass",
|
|
53
|
+
guard: "validation verdict is terminal and not remediation-required",
|
|
54
|
+
to: "completing-milestone",
|
|
55
|
+
onFail: "blocked",
|
|
56
|
+
reasonCode: "state",
|
|
57
|
+
},
|
|
58
|
+
{
|
|
59
|
+
from: "blocked",
|
|
60
|
+
event: "recovery-plan-ready",
|
|
61
|
+
guard: "reassessment produced an executable next action",
|
|
62
|
+
to: "executing",
|
|
63
|
+
onFail: "blocked",
|
|
64
|
+
reasonCode: "recovery",
|
|
65
|
+
},
|
|
66
|
+
{
|
|
67
|
+
from: "replanning-slice",
|
|
68
|
+
event: "replan-ready",
|
|
69
|
+
guard: "replacement slice/task plan exists and plan gate passes",
|
|
70
|
+
to: "executing",
|
|
71
|
+
onFail: "blocked",
|
|
72
|
+
reasonCode: "recovery",
|
|
73
|
+
},
|
|
74
|
+
{
|
|
75
|
+
from: "completing-milestone",
|
|
76
|
+
event: "closeout-complete",
|
|
77
|
+
guard: "closeout gate passes and git transaction succeeds",
|
|
78
|
+
to: "complete",
|
|
79
|
+
onFail: "blocked",
|
|
80
|
+
reasonCode: "state",
|
|
81
|
+
},
|
|
82
|
+
{
|
|
83
|
+
from: "*",
|
|
84
|
+
event: "manual-block",
|
|
85
|
+
guard: "operator or hard gate requested manual attention",
|
|
86
|
+
to: "blocked",
|
|
87
|
+
onFail: "manual-attention",
|
|
88
|
+
reasonCode: "manual",
|
|
89
|
+
},
|
|
90
|
+
{
|
|
91
|
+
from: "*",
|
|
92
|
+
event: "retryable-failure",
|
|
93
|
+
guard: "retry budget remains for failure class",
|
|
94
|
+
to: "executing",
|
|
95
|
+
onFail: "blocked",
|
|
96
|
+
reasonCode: "retry",
|
|
97
|
+
},
|
|
98
|
+
];
|
|
99
|
+
export function findTransition(from, event) {
|
|
100
|
+
return STATE_TRANSITION_MATRIX.find((entry) => (entry.from === from || entry.from === "*") && entry.event === event);
|
|
101
|
+
}
|
|
102
|
+
export function validateTransitionMatrix(requiredEvents) {
|
|
103
|
+
const seen = new Set();
|
|
104
|
+
const duplicateKeys = [];
|
|
105
|
+
for (const entry of STATE_TRANSITION_MATRIX) {
|
|
106
|
+
const key = `${entry.from}:${entry.event}`;
|
|
107
|
+
if (seen.has(key))
|
|
108
|
+
duplicateKeys.push(key);
|
|
109
|
+
seen.add(key);
|
|
110
|
+
}
|
|
111
|
+
const availableEvents = new Set(STATE_TRANSITION_MATRIX.map((entry) => entry.event));
|
|
112
|
+
const missingEvents = requiredEvents.filter((event) => !availableEvents.has(event));
|
|
113
|
+
return {
|
|
114
|
+
ok: missingEvents.length === 0 && duplicateKeys.length === 0,
|
|
115
|
+
missingEvents,
|
|
116
|
+
duplicateKeys,
|
|
117
|
+
};
|
|
118
|
+
}
|
|
@@ -15,6 +15,7 @@ import { logWarning, logError } from './workflow-logger.js';
|
|
|
15
15
|
import { extractVerdict } from './verdict-parser.js';
|
|
16
16
|
import { loadEffectiveGSDPreferences } from './preferences.js';
|
|
17
17
|
import { detectPendingEscalation } from './escalation.js';
|
|
18
|
+
import { isTerminalMilestoneSummaryContent } from './milestone-summary-classifier.js';
|
|
18
19
|
import { isDbAvailable, wasDbOpenAttempted, getAllMilestones, getMilestone, getMilestoneSlices, getSliceTasks, getReplanHistory, getSlice, insertMilestone, insertSlice, insertTask, updateSliceStatus, updateTaskStatus, getPendingGateCountForTurn, autoHealSketchFlags, } from './gsd-db.js';
|
|
19
20
|
/**
|
|
20
21
|
* A "ghost" milestone directory contains only META.json (and no substantive
|
|
@@ -81,6 +82,10 @@ export function isMilestoneComplete(roadmap) {
|
|
|
81
82
|
export function isValidationTerminal(validationContent) {
|
|
82
83
|
return extractVerdict(validationContent) != null;
|
|
83
84
|
}
|
|
85
|
+
async function isTerminalMilestoneSummaryFile(path, loader) {
|
|
86
|
+
const content = await loader(path);
|
|
87
|
+
return content != null && isTerminalMilestoneSummaryContent(content);
|
|
88
|
+
}
|
|
84
89
|
const CACHE_TTL_MS = 100;
|
|
85
90
|
let _stateCache = null;
|
|
86
91
|
// ── Telemetry counters for derive-path observability ────────────────────────
|
|
@@ -138,18 +143,19 @@ export async function getActiveMilestoneId(basePath) {
|
|
|
138
143
|
const content = roadmapFile ? await loadFile(roadmapFile) : null;
|
|
139
144
|
if (!content) {
|
|
140
145
|
const summaryFile = resolveMilestoneFile(basePath, mid, "SUMMARY");
|
|
141
|
-
if (summaryFile)
|
|
146
|
+
if (summaryFile && await isTerminalMilestoneSummaryFile(summaryFile, loadFile))
|
|
142
147
|
continue;
|
|
143
148
|
if (isGhostMilestone(basePath, mid))
|
|
144
149
|
continue;
|
|
145
150
|
return mid;
|
|
146
151
|
}
|
|
147
152
|
const roadmap = parseRoadmap(content);
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
+
const summaryFile = resolveMilestoneFile(basePath, mid, "SUMMARY");
|
|
154
|
+
if (summaryFile && await isTerminalMilestoneSummaryFile(summaryFile, loadFile))
|
|
155
|
+
continue;
|
|
156
|
+
if (!isMilestoneComplete(roadmap))
|
|
157
|
+
return mid;
|
|
158
|
+
return mid;
|
|
153
159
|
}
|
|
154
160
|
return null;
|
|
155
161
|
}
|
|
@@ -182,7 +188,7 @@ export async function deriveState(basePath) {
|
|
|
182
188
|
let synced = false;
|
|
183
189
|
for (const diskId of diskIds) {
|
|
184
190
|
if (!isGhostMilestone(basePath, diskId)) {
|
|
185
|
-
insertMilestone(
|
|
191
|
+
insertMilestone(diskMilestoneInsert(basePath, diskId));
|
|
186
192
|
synced = true;
|
|
187
193
|
}
|
|
188
194
|
}
|
|
@@ -240,6 +246,36 @@ function extractContextTitle(content, fallback) {
|
|
|
240
246
|
// isStatusDone replaced by isClosedStatus from status-guards.ts (single source of truth).
|
|
241
247
|
// Alias kept for backward compatibility within this file.
|
|
242
248
|
const isStatusDone = isClosedStatus;
|
|
249
|
+
function loadSync(path) {
|
|
250
|
+
if (!path)
|
|
251
|
+
return null;
|
|
252
|
+
try {
|
|
253
|
+
return readFileSync(path, "utf-8");
|
|
254
|
+
}
|
|
255
|
+
catch {
|
|
256
|
+
return null;
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
function diskMilestoneInsert(basePath, mid) {
|
|
260
|
+
const contextContent = loadSync(resolveMilestoneFile(basePath, mid, "CONTEXT"));
|
|
261
|
+
const draftContent = !contextContent ? loadSync(resolveMilestoneFile(basePath, mid, "CONTEXT-DRAFT")) : null;
|
|
262
|
+
const roadmapContent = loadSync(resolveMilestoneFile(basePath, mid, "ROADMAP"));
|
|
263
|
+
const summaryContent = loadSync(resolveMilestoneFile(basePath, mid, "SUMMARY"));
|
|
264
|
+
const roadmap = roadmapContent ? parseRoadmap(roadmapContent) : null;
|
|
265
|
+
const summary = summaryContent ? parseSummary(summaryContent) : null;
|
|
266
|
+
const summaryTerminal = summaryContent != null && isTerminalMilestoneSummaryContent(summaryContent);
|
|
267
|
+
const parked = resolveMilestoneFile(basePath, mid, "PARKED") !== null;
|
|
268
|
+
return {
|
|
269
|
+
id: mid,
|
|
270
|
+
title: roadmap
|
|
271
|
+
? stripMilestonePrefix(roadmap.title)
|
|
272
|
+
: (contextContent || draftContent)
|
|
273
|
+
? extractContextTitle(contextContent || draftContent, mid)
|
|
274
|
+
: (summary?.title || mid),
|
|
275
|
+
status: parked ? "parked" : summaryTerminal ? "complete" : "active",
|
|
276
|
+
depends_on: parseContextDependsOn(contextContent ?? draftContent),
|
|
277
|
+
};
|
|
278
|
+
}
|
|
243
279
|
/**
|
|
244
280
|
* Derive GSD state from the milestones/slices/tasks DB tables.
|
|
245
281
|
* Flag files (PARKED, VALIDATION, CONTINUE, REPLAN, REPLAN-TRIGGER, CONTEXT-DRAFT)
|
|
@@ -255,7 +291,7 @@ function reconcileDiskToDb(basePath) {
|
|
|
255
291
|
let synced = false;
|
|
256
292
|
for (const diskId of diskIds) {
|
|
257
293
|
if (!dbIdSet.has(diskId) && !isGhostMilestone(basePath, diskId)) {
|
|
258
|
-
insertMilestone(
|
|
294
|
+
insertMilestone(diskMilestoneInsert(basePath, diskId));
|
|
259
295
|
synced = true;
|
|
260
296
|
}
|
|
261
297
|
}
|
|
@@ -550,9 +586,6 @@ function resolveSliceDependencies(activeMilestoneSlices) {
|
|
|
550
586
|
return { activeSlice: null, activeSliceRow: null };
|
|
551
587
|
}
|
|
552
588
|
}
|
|
553
|
-
// First pass: find a slice with ALL dependencies satisfied (strict)
|
|
554
|
-
let bestFallback = null;
|
|
555
|
-
let bestFallbackSatisfied = -1;
|
|
556
589
|
for (const s of activeMilestoneSlices) {
|
|
557
590
|
if (isStatusDone(s.status))
|
|
558
591
|
continue;
|
|
@@ -561,36 +594,25 @@ function resolveSliceDependencies(activeMilestoneSlices) {
|
|
|
561
594
|
if (s.depends.every(dep => doneSliceIds.has(dep))) {
|
|
562
595
|
return { activeSlice: { id: s.id, title: s.title }, activeSliceRow: s };
|
|
563
596
|
}
|
|
564
|
-
// Track the slice with the most satisfied dependencies as fallback
|
|
565
|
-
const satisfied = s.depends.filter(dep => doneSliceIds.has(dep)).length;
|
|
566
|
-
if (satisfied > bestFallbackSatisfied || (satisfied === bestFallbackSatisfied && !bestFallback)) {
|
|
567
|
-
bestFallback = s;
|
|
568
|
-
bestFallbackSatisfied = satisfied;
|
|
569
|
-
}
|
|
570
|
-
}
|
|
571
|
-
// Fallback: if no slice has all deps met but there ARE incomplete non-deferred
|
|
572
|
-
// slices, pick the one with the most deps satisfied. This prevents hard-blocking
|
|
573
|
-
// when dependency metadata is stale (e.g. after reassessment added/removed slices)
|
|
574
|
-
// or when deps reference slices from previous milestones.
|
|
575
|
-
if (bestFallback) {
|
|
576
|
-
const unmet = bestFallback.depends.filter(dep => !doneSliceIds.has(dep));
|
|
577
|
-
logWarning("state", `No slice has all deps satisfied — falling back to ${bestFallback.id} ` +
|
|
578
|
-
`(${bestFallbackSatisfied}/${bestFallback.depends.length} deps met, ` +
|
|
579
|
-
`unmet: ${unmet.join(", ")})`, { mid: activeMilestoneSlices[0]?.milestone_id, sid: bestFallback.id });
|
|
580
|
-
return { activeSlice: { id: bestFallback.id, title: bestFallback.title }, activeSliceRow: bestFallback };
|
|
581
597
|
}
|
|
582
598
|
return { activeSlice: null, activeSliceRow: null };
|
|
583
599
|
}
|
|
584
600
|
async function reconcileSliceTasks(basePath, milestoneId, sliceId, planFile) {
|
|
585
601
|
let tasks = getSliceTasks(milestoneId, sliceId);
|
|
586
|
-
|
|
602
|
+
// #3600/#4974: import missing plan-file tasks even when the DB already has
|
|
603
|
+
// a partial task set. Existing DB task statuses stay authoritative.
|
|
604
|
+
if (planFile) {
|
|
587
605
|
try {
|
|
588
606
|
const planContent = await loadFile(planFile);
|
|
589
607
|
if (planContent) {
|
|
590
608
|
const diskPlan = parsePlan(planContent);
|
|
591
609
|
if (diskPlan.tasks.length > 0) {
|
|
610
|
+
const dbTaskIds = new Set(tasks.map(t => t.id));
|
|
611
|
+
let inserted = 0;
|
|
592
612
|
for (let i = 0; i < diskPlan.tasks.length; i++) {
|
|
593
613
|
const t = diskPlan.tasks[i];
|
|
614
|
+
if (dbTaskIds.has(t.id))
|
|
615
|
+
continue;
|
|
594
616
|
try {
|
|
595
617
|
insertTask({
|
|
596
618
|
id: t.id,
|
|
@@ -600,13 +622,16 @@ async function reconcileSliceTasks(basePath, milestoneId, sliceId, planFile) {
|
|
|
600
622
|
status: t.done ? 'complete' : 'pending',
|
|
601
623
|
sequence: i + 1,
|
|
602
624
|
});
|
|
625
|
+
inserted++;
|
|
603
626
|
}
|
|
604
627
|
catch (insertErr) {
|
|
605
628
|
logWarning("reconcile", `failed to insert task ${t.id} from plan file: ${insertErr instanceof Error ? insertErr.message : String(insertErr)}`);
|
|
606
629
|
}
|
|
607
630
|
}
|
|
608
|
-
|
|
609
|
-
|
|
631
|
+
if (inserted > 0) {
|
|
632
|
+
tasks = getSliceTasks(milestoneId, sliceId);
|
|
633
|
+
logWarning("reconcile", `imported ${inserted} missing task(s) from plan file for ${milestoneId}/${sliceId}`, { mid: milestoneId, sid: sliceId });
|
|
634
|
+
}
|
|
610
635
|
}
|
|
611
636
|
}
|
|
612
637
|
}
|
|
@@ -988,7 +1013,7 @@ export async function _deriveStateImpl(basePath) {
|
|
|
988
1013
|
const rc = rf ? await cachedLoadFile(rf) : null;
|
|
989
1014
|
if (!rc) {
|
|
990
1015
|
const sf = resolveMilestoneFile(basePath, mid, "SUMMARY");
|
|
991
|
-
if (sf)
|
|
1016
|
+
if (sf && await isTerminalMilestoneSummaryFile(sf, cachedLoadFile))
|
|
992
1017
|
completeMilestoneIds.add(mid);
|
|
993
1018
|
continue;
|
|
994
1019
|
}
|
|
@@ -998,12 +1023,12 @@ export async function _deriveStateImpl(basePath) {
|
|
|
998
1023
|
// Summary is the terminal artifact — if it exists, the milestone is
|
|
999
1024
|
// complete even when roadmap checkboxes weren't ticked (#864).
|
|
1000
1025
|
const sf = resolveMilestoneFile(basePath, mid, "SUMMARY");
|
|
1001
|
-
if (sf)
|
|
1026
|
+
if (sf && await isTerminalMilestoneSummaryFile(sf, cachedLoadFile))
|
|
1002
1027
|
completeMilestoneIds.add(mid);
|
|
1003
1028
|
continue;
|
|
1004
1029
|
}
|
|
1005
1030
|
const sf = resolveMilestoneFile(basePath, mid, "SUMMARY");
|
|
1006
|
-
if (sf)
|
|
1031
|
+
if (sf && await isTerminalMilestoneSummaryFile(sf, cachedLoadFile))
|
|
1007
1032
|
completeMilestoneIds.add(mid);
|
|
1008
1033
|
}
|
|
1009
1034
|
// Phase 2: Build registry using cached roadmaps (no re-parsing or re-reading)
|
|
@@ -1028,12 +1053,14 @@ export async function _deriveStateImpl(basePath) {
|
|
|
1028
1053
|
const summaryFile = resolveMilestoneFile(basePath, mid, "SUMMARY");
|
|
1029
1054
|
if (summaryFile) {
|
|
1030
1055
|
const summaryContent = await cachedLoadFile(summaryFile);
|
|
1031
|
-
|
|
1032
|
-
|
|
1033
|
-
|
|
1034
|
-
|
|
1035
|
-
|
|
1036
|
-
|
|
1056
|
+
if (summaryContent != null && isTerminalMilestoneSummaryContent(summaryContent)) {
|
|
1057
|
+
const summaryTitle = summaryContent
|
|
1058
|
+
? (parseSummary(summaryContent).title || mid)
|
|
1059
|
+
: mid;
|
|
1060
|
+
registry.push({ id: mid, title: summaryTitle, status: 'complete' });
|
|
1061
|
+
completeMilestoneIds.add(mid);
|
|
1062
|
+
continue;
|
|
1063
|
+
}
|
|
1037
1064
|
}
|
|
1038
1065
|
// Ghost milestone (only META.json, no CONTEXT/ROADMAP/SUMMARY) — skip entirely
|
|
1039
1066
|
if (isGhostMilestone(basePath, mid))
|
|
@@ -1087,7 +1114,7 @@ export async function _deriveStateImpl(basePath) {
|
|
|
1087
1114
|
const verdict = validationContent ? extractVerdict(validationContent) : undefined;
|
|
1088
1115
|
// needs-remediation is terminal but requires re-validation (#3596)
|
|
1089
1116
|
const needsRevalidation = !validationTerminal || verdict === 'needs-remediation';
|
|
1090
|
-
if (summaryFile) {
|
|
1117
|
+
if (summaryFile && await isTerminalMilestoneSummaryFile(summaryFile, cachedLoadFile)) {
|
|
1091
1118
|
// Summary exists → milestone is complete regardless of validation state.
|
|
1092
1119
|
// The summary is the terminal artifact (#864).
|
|
1093
1120
|
registry.push({ id: mid, title, status: 'complete' });
|
|
@@ -1118,7 +1145,7 @@ export async function _deriveStateImpl(basePath) {
|
|
|
1118
1145
|
// Roadmap slices not all checked — but if a summary exists, the milestone
|
|
1119
1146
|
// is still complete. The summary is the terminal artifact (#864).
|
|
1120
1147
|
const summaryFile = resolveMilestoneFile(basePath, mid, "SUMMARY");
|
|
1121
|
-
if (summaryFile) {
|
|
1148
|
+
if (summaryFile && await isTerminalMilestoneSummaryFile(summaryFile, cachedLoadFile)) {
|
|
1122
1149
|
registry.push({ id: mid, title, status: 'complete' });
|
|
1123
1150
|
}
|
|
1124
1151
|
else if (!activeMilestoneFound) {
|
|
@@ -1387,8 +1414,6 @@ export async function _deriveStateImpl(basePath) {
|
|
|
1387
1414
|
}
|
|
1388
1415
|
}
|
|
1389
1416
|
else {
|
|
1390
|
-
let bestFallbackLegacy = null;
|
|
1391
|
-
let bestFallbackLegacySatisfied = -1;
|
|
1392
1417
|
for (const s of activeRoadmap.slices) {
|
|
1393
1418
|
if (s.done)
|
|
1394
1419
|
continue;
|
|
@@ -1396,20 +1421,6 @@ export async function _deriveStateImpl(basePath) {
|
|
|
1396
1421
|
activeSlice = { id: s.id, title: s.title };
|
|
1397
1422
|
break;
|
|
1398
1423
|
}
|
|
1399
|
-
// Track best fallback
|
|
1400
|
-
const satisfied = s.depends.filter(dep => doneSliceIds.has(dep)).length;
|
|
1401
|
-
if (satisfied > bestFallbackLegacySatisfied) {
|
|
1402
|
-
bestFallbackLegacy = s;
|
|
1403
|
-
bestFallbackLegacySatisfied = satisfied;
|
|
1404
|
-
}
|
|
1405
|
-
}
|
|
1406
|
-
// Fallback: if no slice has all deps met, pick the one with the most deps satisfied
|
|
1407
|
-
if (!activeSlice && bestFallbackLegacy) {
|
|
1408
|
-
const unmet = bestFallbackLegacy.depends.filter(dep => !doneSliceIds.has(dep));
|
|
1409
|
-
logWarning("state", `No slice has all deps satisfied — falling back to ${bestFallbackLegacy.id} ` +
|
|
1410
|
-
`(${bestFallbackLegacySatisfied}/${bestFallbackLegacy.depends.length} deps met, ` +
|
|
1411
|
-
`unmet: ${unmet.join(", ")})`);
|
|
1412
|
-
activeSlice = { id: bestFallbackLegacy.id, title: bestFallbackLegacy.title };
|
|
1413
1424
|
}
|
|
1414
1425
|
}
|
|
1415
1426
|
if (!activeSlice) {
|
|
@@ -1,10 +1,9 @@
|
|
|
1
1
|
// GSD Extension — Advisory Sync Lock
|
|
2
2
|
// Prevents concurrent worktree syncs from colliding via a simple file lock.
|
|
3
|
-
// Stale locks (mtime > 60s) are
|
|
4
|
-
// to 5 seconds then skips non-fatally.
|
|
5
|
-
import { existsSync, statSync, unlinkSync } from "node:fs";
|
|
6
|
-
import { join } from "node:path";
|
|
7
|
-
import { atomicWriteSync } from "./atomic-write.js";
|
|
3
|
+
// Stale locks (mtime > 60s, owner PID confirmed dead) are overridden. Lock
|
|
4
|
+
// acquisition waits up to 5 seconds then skips non-fatally.
|
|
5
|
+
import { closeSync, existsSync, mkdirSync, openSync, readFileSync, statSync, unlinkSync, writeSync } from "node:fs";
|
|
6
|
+
import { dirname, join } from "node:path";
|
|
8
7
|
const STALE_THRESHOLD_MS = 60_000; // 60 seconds
|
|
9
8
|
const DEFAULT_TIMEOUT_MS = 5_000; // 5 seconds
|
|
10
9
|
const SPIN_INTERVAL_MS = 100; // 100ms polling interval
|
|
@@ -17,60 +16,117 @@ function lockFilePath(basePath) {
|
|
|
17
16
|
function sleepSync(ms) {
|
|
18
17
|
Atomics.wait(SLEEP_VIEW, 0, 0, ms);
|
|
19
18
|
}
|
|
19
|
+
/** True if the given PID is alive in the current process namespace. */
|
|
20
|
+
function isPidAlive(pid) {
|
|
21
|
+
try {
|
|
22
|
+
process.kill(pid, 0);
|
|
23
|
+
return true;
|
|
24
|
+
}
|
|
25
|
+
catch (err) {
|
|
26
|
+
if (err.code === "EPERM")
|
|
27
|
+
return true;
|
|
28
|
+
return false;
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Atomically create the lock file with `O_EXCL` semantics. Returns true on
|
|
33
|
+
* exclusive create, false if the file already existed. Any other error
|
|
34
|
+
* propagates.
|
|
35
|
+
*/
|
|
36
|
+
function tryCreateLockFile(lp, payload) {
|
|
37
|
+
// Ensure parent dir exists (`atomicWriteSync` previously did this implicitly).
|
|
38
|
+
try {
|
|
39
|
+
mkdirSync(dirname(lp), { recursive: true });
|
|
40
|
+
}
|
|
41
|
+
catch {
|
|
42
|
+
/* best-effort */
|
|
43
|
+
}
|
|
44
|
+
let fd;
|
|
45
|
+
try {
|
|
46
|
+
// "wx" → O_WRONLY | O_CREAT | O_EXCL — atomic create-if-not-exists on POSIX.
|
|
47
|
+
fd = openSync(lp, "wx");
|
|
48
|
+
}
|
|
49
|
+
catch (err) {
|
|
50
|
+
const code = err.code;
|
|
51
|
+
if (code === "EEXIST")
|
|
52
|
+
return false;
|
|
53
|
+
throw err;
|
|
54
|
+
}
|
|
55
|
+
try {
|
|
56
|
+
writeSync(fd, payload);
|
|
57
|
+
}
|
|
58
|
+
finally {
|
|
59
|
+
closeSync(fd);
|
|
60
|
+
}
|
|
61
|
+
return true;
|
|
62
|
+
}
|
|
20
63
|
/**
|
|
21
64
|
* Acquire an advisory sync lock for the given basePath.
|
|
22
65
|
* Returns { acquired: true } on success, { acquired: false } after timeout.
|
|
23
66
|
*
|
|
24
|
-
*
|
|
25
|
-
*
|
|
26
|
-
*
|
|
67
|
+
* Replaces a non-atomic `existsSync` + `atomicWriteSync` (write-temp+rename,
|
|
68
|
+
* which is not exclusive-create) sequence that allowed two callers to both
|
|
69
|
+
* believe they had acquired the lock, corrupting the event log.
|
|
70
|
+
* (Issue #4980 CRIT-4)
|
|
71
|
+
*
|
|
72
|
+
* Stale-lock override now also verifies the recorded owner PID is dead
|
|
73
|
+
* before stealing — prevents a slow event-loop pause (>60s under heavy I/O)
|
|
74
|
+
* from making a legitimately-held lock appear stale and get stolen.
|
|
75
|
+
* (Issue #4980 M-concurrency-3)
|
|
27
76
|
*/
|
|
28
77
|
export function acquireSyncLock(basePath, timeoutMs = DEFAULT_TIMEOUT_MS) {
|
|
29
78
|
const lp = lockFilePath(basePath);
|
|
30
79
|
const deadline = Date.now() + timeoutMs;
|
|
80
|
+
const lockData = JSON.stringify({ pid: process.pid, acquired_at: new Date().toISOString() }, null, 2);
|
|
31
81
|
while (true) {
|
|
32
|
-
//
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
82
|
+
// First try the fast path: atomic create. No check-then-write race.
|
|
83
|
+
try {
|
|
84
|
+
if (tryCreateLockFile(lp, lockData)) {
|
|
85
|
+
return { acquired: true };
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
catch {
|
|
89
|
+
/* unexpected — fall through to retry */
|
|
90
|
+
}
|
|
91
|
+
// File exists. Decide whether to steal (stale + owner dead) or wait.
|
|
92
|
+
let canSteal = false;
|
|
93
|
+
try {
|
|
94
|
+
const stat = statSync(lp);
|
|
95
|
+
const age = Date.now() - stat.mtimeMs;
|
|
96
|
+
if (age > STALE_THRESHOLD_MS) {
|
|
97
|
+
// Verify the recorded owner PID is dead before stealing.
|
|
98
|
+
let ownerAlive = false;
|
|
99
|
+
try {
|
|
100
|
+
const data = JSON.parse(readFileSync(lp, "utf-8"));
|
|
101
|
+
if (typeof data.pid === "number" && data.pid !== process.pid) {
|
|
102
|
+
ownerAlive = isPidAlive(data.pid);
|
|
42
103
|
}
|
|
43
|
-
catch { /* race: already removed */ }
|
|
44
104
|
}
|
|
45
|
-
|
|
46
|
-
// Lock
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
}
|
|
50
|
-
sleepSync(SPIN_INTERVAL_MS);
|
|
51
|
-
continue;
|
|
105
|
+
catch {
|
|
106
|
+
// Lock contents unreadable — be conservative and steal only on age.
|
|
107
|
+
// A garbage lock file that we cannot parse is safer to remove than
|
|
108
|
+
// to leave wedging the lock indefinitely.
|
|
52
109
|
}
|
|
110
|
+
canSteal = !ownerAlive;
|
|
53
111
|
}
|
|
54
|
-
catch {
|
|
55
|
-
// stat failed (file removed between exists check and stat) — try to acquire
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
// Lock file does not exist (or was just removed) — try to write it
|
|
59
|
-
try {
|
|
60
|
-
const lockData = {
|
|
61
|
-
pid: process.pid,
|
|
62
|
-
acquired_at: new Date().toISOString(),
|
|
63
|
-
};
|
|
64
|
-
atomicWriteSync(lp, JSON.stringify(lockData, null, 2));
|
|
65
|
-
return { acquired: true };
|
|
66
112
|
}
|
|
67
113
|
catch {
|
|
68
|
-
//
|
|
69
|
-
|
|
70
|
-
|
|
114
|
+
// stat failed (file removed between exists and stat) — retry create.
|
|
115
|
+
continue;
|
|
116
|
+
}
|
|
117
|
+
if (canSteal) {
|
|
118
|
+
try {
|
|
119
|
+
unlinkSync(lp);
|
|
71
120
|
}
|
|
72
|
-
|
|
121
|
+
catch { /* race: already removed */ }
|
|
122
|
+
// Loop back to retry create.
|
|
123
|
+
continue;
|
|
124
|
+
}
|
|
125
|
+
// Lock is held and not stale (or owner is alive) — wait or give up.
|
|
126
|
+
if (Date.now() >= deadline) {
|
|
127
|
+
return { acquired: false };
|
|
73
128
|
}
|
|
129
|
+
sleepSync(SPIN_INTERVAL_MS);
|
|
74
130
|
}
|
|
75
131
|
}
|
|
76
132
|
/**
|
|
@@ -16,6 +16,7 @@ import { checkOwnership, sliceUnitKey } from "../unit-ownership.js";
|
|
|
16
16
|
import { saveFile, clearParseCache } from "../files.js";
|
|
17
17
|
import { invalidateStateCache } from "../state.js";
|
|
18
18
|
import { renderRoadmapCheckboxes } from "../markdown-renderer.js";
|
|
19
|
+
import { isStaleWrite } from "../auto/turn-epoch.js";
|
|
19
20
|
import { renderAllProjections } from "../workflow-projections.js";
|
|
20
21
|
import { writeManifest } from "../workflow-manifest.js";
|
|
21
22
|
import { appendEvent } from "../workflow-events.js";
|
|
@@ -228,6 +229,10 @@ export async function handleCompleteSlice(params, basePath) {
|
|
|
228
229
|
}
|
|
229
230
|
const slice = getSlice(params.milestoneId, params.sliceId);
|
|
230
231
|
if (slice && isClosedStatus(slice.status)) {
|
|
232
|
+
if (isStaleWrite("complete-slice")) {
|
|
233
|
+
guardError = "__stale_duplicate__";
|
|
234
|
+
return;
|
|
235
|
+
}
|
|
231
236
|
guardError = `slice ${params.sliceId} is already complete — use gsd_slice_reopen first if you need to redo it`;
|
|
232
237
|
return;
|
|
233
238
|
}
|
|
@@ -248,6 +253,22 @@ export async function handleCompleteSlice(params, basePath) {
|
|
|
248
253
|
insertSlice({ id: params.sliceId, milestoneId: params.milestoneId, title: params.sliceId });
|
|
249
254
|
updateSliceStatus(params.milestoneId, params.sliceId, "complete", completedAt);
|
|
250
255
|
});
|
|
256
|
+
if (guardError === "__stale_duplicate__") {
|
|
257
|
+
// Stale duplicate from a turn superseded by timeout recovery. Return a
|
|
258
|
+
// non-mutating success so the orphaned LLM tool call unwinds quietly.
|
|
259
|
+
const sliceDir = resolveSlicePath(basePath, params.milestoneId, params.sliceId);
|
|
260
|
+
const staleSummaryPath = sliceDir
|
|
261
|
+
? join(sliceDir, `${params.sliceId}-SUMMARY.md`)
|
|
262
|
+
: join(basePath, ".gsd", "milestones", params.milestoneId, "slices", params.sliceId, `${params.sliceId}-SUMMARY.md`);
|
|
263
|
+
return {
|
|
264
|
+
sliceId: params.sliceId,
|
|
265
|
+
milestoneId: params.milestoneId,
|
|
266
|
+
summaryPath: staleSummaryPath,
|
|
267
|
+
uatPath: staleSummaryPath.replace(/-SUMMARY\.md$/, "-UAT.md"),
|
|
268
|
+
duplicate: true,
|
|
269
|
+
stale: true,
|
|
270
|
+
};
|
|
271
|
+
}
|
|
251
272
|
if (guardError) {
|
|
252
273
|
return { error: guardError };
|
|
253
274
|
}
|
|
@@ -21,6 +21,7 @@ import { writeManifest } from "../workflow-manifest.js";
|
|
|
21
21
|
import { appendEvent } from "../workflow-events.js";
|
|
22
22
|
import { logWarning, logError } from "../workflow-logger.js";
|
|
23
23
|
import { loadEffectiveGSDPreferences } from "../preferences.js";
|
|
24
|
+
import { isStaleWrite } from "../auto/turn-epoch.js";
|
|
24
25
|
import { buildEscalationArtifact, writeEscalationArtifact } from "../escalation.js";
|
|
25
26
|
/**
|
|
26
27
|
* Map an execute-task-owned gate id to the CompleteTaskParams field whose
|
|
@@ -167,6 +168,18 @@ export async function handleCompleteTask(params, basePath) {
|
|
|
167
168
|
}
|
|
168
169
|
const existingTask = getTask(params.milestoneId, params.sliceId, params.taskId);
|
|
169
170
|
if (existingTask && isClosedStatus(existingTask.status)) {
|
|
171
|
+
// Stale-turn path: a timed-out turn that was superseded by recovery
|
|
172
|
+
// can still reach this code when its LLM call eventually returns and
|
|
173
|
+
// invokes gsd_complete_task. Returning an error would produce noisy
|
|
174
|
+
// "already complete — use reopen first" logs in the orphaned turn.
|
|
175
|
+
// Instead, signal the duplicate via a non-mutating success shape that
|
|
176
|
+
// callers can detect via `duplicate: true` / `stale: true`.
|
|
177
|
+
if (isStaleWrite("complete-task")) {
|
|
178
|
+
// Sentinel handled below — outside the transaction — so we don't
|
|
179
|
+
// render SUMMARY.md or flip plan checkboxes for a stale duplicate.
|
|
180
|
+
guardError = "__stale_duplicate__";
|
|
181
|
+
return;
|
|
182
|
+
}
|
|
170
183
|
guardError = `task ${params.taskId} is already complete — use gsd_task_reopen first if you need to redo it`;
|
|
171
184
|
return;
|
|
172
185
|
}
|
|
@@ -201,6 +214,24 @@ export async function handleCompleteTask(params, basePath) {
|
|
|
201
214
|
});
|
|
202
215
|
}
|
|
203
216
|
});
|
|
217
|
+
if (guardError === "__stale_duplicate__") {
|
|
218
|
+
// Orphaned-turn duplicate: the task is already complete from the
|
|
219
|
+
// superseded turn's earlier (real) call. Return a non-mutating success
|
|
220
|
+
// so the stale LLM tool call unwinds cleanly. summaryPath is synthesized
|
|
221
|
+
// from the existing on-disk layout; no file is written.
|
|
222
|
+
const tasksDir = resolveTasksDir(basePath, params.milestoneId, params.sliceId);
|
|
223
|
+
const staleSummaryPath = tasksDir
|
|
224
|
+
? join(tasksDir, `${params.taskId}-SUMMARY.md`)
|
|
225
|
+
: join(basePath, ".gsd", "milestones", params.milestoneId, "slices", params.sliceId, "tasks", `${params.taskId}-SUMMARY.md`);
|
|
226
|
+
return {
|
|
227
|
+
taskId: params.taskId,
|
|
228
|
+
sliceId: params.sliceId,
|
|
229
|
+
milestoneId: params.milestoneId,
|
|
230
|
+
summaryPath: staleSummaryPath,
|
|
231
|
+
duplicate: true,
|
|
232
|
+
stale: true,
|
|
233
|
+
};
|
|
234
|
+
}
|
|
204
235
|
if (guardError) {
|
|
205
236
|
return { error: guardError };
|
|
206
237
|
}
|