gsd-pi 2.82.0-dev.725028083 → 2.82.0-dev.9d5798940
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 +3 -2
- package/dist/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/GSD-WORKFLOW.md +3 -1
- package/dist/resources/extensions/claude-code-cli/partial-builder.js +2 -1
- package/dist/resources/extensions/cmux/index.js +5 -0
- package/dist/resources/extensions/gsd/auto/orchestrator.js +113 -6
- package/dist/resources/extensions/gsd/auto-dispatch.js +13 -6
- package/dist/resources/extensions/gsd/auto-post-unit.js +232 -126
- package/dist/resources/extensions/gsd/auto-prompts.js +2 -2
- package/dist/resources/extensions/gsd/auto-recovery.js +31 -1
- package/dist/resources/extensions/gsd/auto-start.js +7 -3
- package/dist/resources/extensions/gsd/auto-verification.js +28 -22
- package/dist/resources/extensions/gsd/auto-worktree.js +96 -0
- package/dist/resources/extensions/gsd/auto.js +128 -52
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +4 -1
- package/dist/resources/extensions/gsd/bootstrap/subagent-input.js +16 -7
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +16 -2
- package/dist/resources/extensions/gsd/clean-root-preflight.js +170 -8
- package/dist/resources/extensions/gsd/commands/catalog.js +4 -1
- package/dist/resources/extensions/gsd/commands/handlers/core.js +37 -0
- package/dist/resources/extensions/gsd/commands-bootstrap.js +5 -0
- package/dist/resources/extensions/gsd/db/unit-dispatches.js +2 -2
- package/dist/resources/extensions/gsd/export-html.js +27 -425
- package/dist/resources/extensions/gsd/md-importer.js +1 -1
- package/dist/resources/extensions/gsd/migrate/command.js +5 -0
- package/dist/resources/extensions/gsd/migrate/preview.js +9 -0
- package/dist/resources/extensions/gsd/migrate/transformer.js +51 -4
- package/dist/resources/extensions/gsd/migrate/writer.js +11 -1
- package/dist/resources/extensions/gsd/milestone-actions.js +11 -4
- package/dist/resources/extensions/gsd/native-git-bridge.js +8 -3
- package/dist/resources/extensions/gsd/state-reconciliation/drift/merge-state.js +6 -1
- package/dist/resources/extensions/gsd/tools/plan-slice.js +2 -1
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +119 -0
- package/dist/resources/extensions/gsd/unit-context-manifest.js +32 -10
- package/dist/resources/extensions/gsd/verification-verdict.js +26 -0
- package/dist/resources/extensions/gsd/worktree-lifecycle.js +49 -9
- package/dist/resources/extensions/shared/html-shell.js +388 -0
- package/dist/resources/extensions/subagent/index.js +448 -78
- package/dist/resources/extensions/subagent/launch.js +77 -0
- package/dist/resources/extensions/subagent/run-store.js +148 -0
- package/dist/resources/extensions/visual-brief/artifact-policy.js +29 -0
- package/dist/resources/extensions/visual-brief/extension-manifest.json +8 -0
- package/dist/resources/extensions/visual-brief/index.js +5 -0
- package/dist/resources/extensions/visual-brief/page-contract.js +124 -0
- package/dist/resources/extensions/visual-brief/prompts.js +140 -0
- 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 +15 -15
- 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 +3 -3
- package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
- package/dist/web/standalone/.next/server/app/_not-found/page.js.nft.json +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 +4 -7
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +4 -7
- 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 +4 -5
- 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 +2 -5
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +4 -7
- 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 +4 -7
- 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 +4 -5
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -5
- package/dist/web/standalone/.next/server/app/page.js +2 -2
- package/dist/web/standalone/.next/server/app/page.js.nft.json +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 +15 -15
- package/dist/web/standalone/.next/server/chunks/4266.js +2 -0
- 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/next-font-manifest.js +1 -1
- package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/dist/web/standalone/.next/static/chunks/2973.33f26573894b6153.js +2 -0
- package/dist/web/standalone/.next/static/chunks/{8359.e059d86b255fce1c.js → 8359.7eb3bb8f8ecf4c01.js} +2 -2
- package/dist/web/standalone/.next/static/chunks/app/layout-8c10ec293ae0f1d5.js +1 -0
- package/dist/web/standalone/.next/static/chunks/{webpack-de742b64187e13fe.js → webpack-9a4db269f9ed63ad.js} +1 -1
- package/dist/web/standalone/.next/static/css/746ee28c929d1880.css +1 -0
- package/package.json +3 -3
- package/packages/contracts/dist/rpc.test.js +7 -0
- package/packages/contracts/dist/rpc.test.js.map +1 -1
- package/packages/contracts/dist/workflow.d.ts +21 -0
- package/packages/contracts/dist/workflow.d.ts.map +1 -1
- package/packages/contracts/dist/workflow.js +24 -0
- package/packages/contracts/dist/workflow.js.map +1 -1
- package/packages/contracts/src/rpc.test.ts +8 -0
- package/packages/contracts/src/workflow.ts +24 -0
- package/packages/mcp-server/README.md +13 -4
- package/packages/mcp-server/dist/workflow-tools.d.ts +0 -3
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +80 -0
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/src/workflow-tools.test.ts +22 -0
- package/packages/mcp-server/src/workflow-tools.ts +168 -0
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-tui/dist/tui.d.ts.map +1 -1
- package/packages/pi-tui/dist/tui.js +5 -0
- package/packages/pi-tui/dist/tui.js.map +1 -1
- package/packages/pi-tui/src/tui.ts +6 -0
- package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
- package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
- package/src/resources/GSD-WORKFLOW.md +3 -1
- package/src/resources/extensions/claude-code-cli/partial-builder.ts +2 -1
- package/src/resources/extensions/claude-code-cli/tests/partial-builder.test.ts +19 -2
- package/src/resources/extensions/cmux/index.ts +6 -0
- package/src/resources/extensions/gsd/auto/contracts.ts +46 -11
- package/src/resources/extensions/gsd/auto/orchestrator.ts +118 -6
- package/src/resources/extensions/gsd/auto-dispatch.ts +14 -6
- package/src/resources/extensions/gsd/auto-post-unit.ts +265 -138
- package/src/resources/extensions/gsd/auto-prompts.ts +2 -2
- package/src/resources/extensions/gsd/auto-recovery.ts +29 -0
- package/src/resources/extensions/gsd/auto-start.ts +7 -3
- package/src/resources/extensions/gsd/auto-verification.ts +36 -34
- package/src/resources/extensions/gsd/auto-worktree.ts +104 -0
- package/src/resources/extensions/gsd/auto.ts +136 -51
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +6 -1
- package/src/resources/extensions/gsd/bootstrap/subagent-input.ts +16 -6
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +19 -3
- package/src/resources/extensions/gsd/clean-root-preflight.ts +174 -8
- package/src/resources/extensions/gsd/commands/catalog.ts +4 -1
- package/src/resources/extensions/gsd/commands/handlers/core.ts +40 -0
- package/src/resources/extensions/gsd/commands-bootstrap.ts +10 -0
- package/src/resources/extensions/gsd/db/unit-dispatches.ts +3 -3
- package/src/resources/extensions/gsd/export-html.ts +27 -427
- package/src/resources/extensions/gsd/md-importer.ts +1 -1
- package/src/resources/extensions/gsd/migrate/command.ts +5 -0
- package/src/resources/extensions/gsd/migrate/preview.ts +10 -0
- package/src/resources/extensions/gsd/migrate/transformer.ts +58 -4
- package/src/resources/extensions/gsd/migrate/writer.ts +14 -1
- package/src/resources/extensions/gsd/milestone-actions.ts +10 -4
- package/src/resources/extensions/gsd/native-git-bridge.ts +8 -3
- package/src/resources/extensions/gsd/state-reconciliation/drift/merge-state.ts +8 -1
- package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +116 -0
- package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +408 -4
- package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +6 -5
- package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +12 -1
- package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +15 -1
- package/src/resources/extensions/gsd/tests/auto-runtime-state.test.ts +4 -4
- package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +69 -1
- package/src/resources/extensions/gsd/tests/brief-command.test.ts +89 -0
- package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +107 -2
- package/src/resources/extensions/gsd/tests/closeout-git-deferral.test.ts +16 -0
- package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +57 -2
- package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/evidence-cross-ref.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/export-html-enhancements.test.ts +8 -0
- package/src/resources/extensions/gsd/tests/integration/migrate-command.test.ts +48 -3
- package/src/resources/extensions/gsd/tests/migrate-transformer.test.ts +5 -1
- package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +6 -1
- package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +55 -1
- package/src/resources/extensions/gsd/tests/plan-slice.test.ts +25 -0
- package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
- package/src/resources/extensions/gsd/tests/post-unit-git-failure.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/remediation-completion-guard.test.ts +46 -2
- package/src/resources/extensions/gsd/tests/session-switch-abort-misclassification.test.ts +10 -0
- package/src/resources/extensions/gsd/tests/state-corruption-2945.test.ts +6 -0
- package/src/resources/extensions/gsd/tests/stuck-state-via-db.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +86 -7
- package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +78 -0
- package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +64 -12
- package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +25 -0
- package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +47 -0
- package/src/resources/extensions/gsd/tools/plan-slice.ts +2 -0
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +135 -0
- package/src/resources/extensions/gsd/unit-context-manifest.ts +47 -11
- package/src/resources/extensions/gsd/verification-verdict.ts +47 -0
- package/src/resources/extensions/gsd/worktree-lifecycle.ts +54 -9
- package/src/resources/extensions/shared/html-shell.ts +412 -0
- package/src/resources/extensions/subagent/index.ts +567 -103
- package/src/resources/extensions/subagent/launch.ts +131 -0
- package/src/resources/extensions/subagent/run-store.ts +218 -0
- package/src/resources/extensions/subagent/tests/launch.test.ts +115 -0
- package/src/resources/extensions/subagent/tests/run-store.test.ts +111 -0
- package/src/resources/extensions/visual-brief/artifact-policy.ts +41 -0
- package/src/resources/extensions/visual-brief/extension-manifest.json +8 -0
- package/src/resources/extensions/visual-brief/index.ts +8 -0
- package/src/resources/extensions/visual-brief/page-contract.ts +136 -0
- package/src/resources/extensions/visual-brief/prompts.ts +183 -0
- package/src/resources/extensions/visual-brief/tests/visual-brief.test.ts +212 -0
- package/dist/web/standalone/.next/server/chunks/5822.js +0 -2
- package/dist/web/standalone/.next/static/chunks/2556.0527fea66e123b7f.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/layout-a16c7a7ecdf0c2cf.js +0 -1
- package/dist/web/standalone/.next/static/css/54ec2745c1da488b.css +0 -1
- package/dist/web/standalone/.next/static/css/de70bee13400563f.css +0 -1
- package/dist/web/standalone/.next/static/media/4cf2300e9c8272f7-s.p.woff2 +0 -0
- package/dist/web/standalone/.next/static/media/747892c23ea88013-s.woff2 +0 -0
- package/dist/web/standalone/.next/static/media/8d697b304b401681-s.woff2 +0 -0
- package/dist/web/standalone/.next/static/media/93f479601ee12b01-s.p.woff2 +0 -0
- package/dist/web/standalone/.next/static/media/9610d9e46709d722-s.woff2 +0 -0
- package/dist/web/standalone/.next/static/media/ba015fad6dcf6784-s.woff2 +0 -0
- /package/dist/web/standalone/.next/static/{KDRTXR-22LPCsa80X9dey → BdZQhe8yKl6bdKLiXVEzh}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{KDRTXR-22LPCsa80X9dey → BdZQhe8yKl6bdKLiXVEzh}/_ssgManifest.js +0 -0
package/README.md
CHANGED
|
@@ -336,7 +336,7 @@ Plan (with integrated research) → Execute (per task) → Complete → Reassess
|
|
|
336
336
|
Validate Milestone → Complete Milestone
|
|
337
337
|
```
|
|
338
338
|
|
|
339
|
-
**Plan** scouts the codebase, researches relevant docs, and decomposes the slice into tasks with must-haves (mechanically verifiable outcomes). **Execute** runs each task in a fresh context window with only the relevant files pre-loaded
|
|
339
|
+
**Plan** scouts the codebase, researches relevant docs, and decomposes the slice into tasks with must-haves (mechanically verifiable outcomes). **Execute** runs each task in a fresh context window with only the relevant files pre-loaded, then runs configured verification commands (lint, test, etc.) with auto-fix retries before the task closeout commit or snapshot is published. Failed or incomplete verification blocks execute-task closeout. **Complete** writes the summary, UAT script, marks the roadmap, and commits with meaningful messages derived from task summaries. **Reassess** checks if the roadmap still makes sense given what was learned. **Validate Milestone** runs a reconciliation gate after all slices complete — comparing roadmap success criteria against actual results before sealing the milestone.
|
|
340
340
|
|
|
341
341
|
When progressive planning is enabled, the first slice is fully planned up front while later slices may appear in `M###-ROADMAP.md` with a `` `[sketch]` `` badge. A sketch slice has an approved title, dependency shape, demo line, and scope boundary, but it has not yet been expanded into task plans; auto mode runs `refine-slice` just before execution to turn the sketch into a full slice plan using the latest prior-slice summaries.
|
|
342
342
|
|
|
@@ -376,7 +376,7 @@ The database is authoritative for milestones, slices, tasks, requirements, summa
|
|
|
376
376
|
|
|
377
377
|
10. **Adaptive replanning** — After each slice completes, the roadmap is reassessed. If the work revealed new information that changes the plan, slices are reordered, added, or removed before continuing.
|
|
378
378
|
|
|
379
|
-
11. **Verification enforcement** — Configure shell commands (`npm run lint`, `npm run test`, etc.) that run automatically after task execution. Failures trigger auto-fix retries before advancing. Auto-discovered checks from `package.json` run in advisory mode — they log warnings but don't block on pre-existing errors. Configurable via `verification_commands`, `verification_auto_fix`, and `verification_max_retries` preferences.
|
|
379
|
+
11. **Verification enforcement** — Configure shell commands (`npm run lint`, `npm run test`, etc.) that run automatically after task execution. Failures trigger auto-fix retries before advancing. Execute-task commits and snapshots are deferred until verification passes; failed or incomplete verification blocks closeout instead of publishing changes. Auto-discovered checks from `package.json` run in advisory mode — they log warnings but don't block on pre-existing errors. Configurable via `verification_commands`, `verification_auto_fix`, and `verification_max_retries` preferences.
|
|
380
380
|
|
|
381
381
|
12. **Milestone validation** — After all slices complete, a `validate-milestone` gate compares roadmap success criteria against actual results before sealing the milestone.
|
|
382
382
|
|
|
@@ -502,6 +502,7 @@ On first run, GSD launches a branded setup wizard that walks you through LLM pro
|
|
|
502
502
|
| `/gsd rethink` | Conversational project reorganization |
|
|
503
503
|
| `/gsd mcp` | MCP server status and connectivity |
|
|
504
504
|
| `/gsd status` | Progress dashboard |
|
|
505
|
+
| `/gsd brief <mode>` | Generate a visual HTML brief (diagram, plan, diff, recap, table, slides) |
|
|
505
506
|
| `/gsd queue` | Queue future milestones (safe during auto mode) |
|
|
506
507
|
| `/gsd prefs` | Model selection, timeouts, budget ceiling |
|
|
507
508
|
| `/gsd migrate` | Migrate a v1 `.planning` directory to `.gsd` format |
|
|
@@ -1 +1 @@
|
|
|
1
|
-
|
|
1
|
+
ed49f911008c62ca
|
|
@@ -560,7 +560,7 @@ In all modes, slices and tasks commit sequentially on the active branch; there a
|
|
|
560
560
|
|
|
561
561
|
1. **Milestone starts** → capture the current integration branch.
|
|
562
562
|
2. **Optional isolation** → create `milestone/M001` only when `git.isolation` is `worktree` or `branch`.
|
|
563
|
-
3. **Per-task commits** — atomic, descriptive, bisectable.
|
|
563
|
+
3. **Per-task commits** — atomic, descriptive, bisectable, and published only after execute-task verification passes.
|
|
564
564
|
4. **Slice completes** → write slice summary, UAT script, roadmap checkbox, and milestone summary.
|
|
565
565
|
5. **Milestone completes** → if isolated, squash-merge the milestone branch back to the captured integration branch and clean up the worktree/branch.
|
|
566
566
|
|
|
@@ -574,6 +574,8 @@ fix: handle empty state rebuild
|
|
|
574
574
|
|
|
575
575
|
In `none` mode these commits land directly on the current branch. In isolated modes they land on `milestone/<MID>` and are squashed back at milestone completion.
|
|
576
576
|
|
|
577
|
+
Execute-task closeout is fail-closed: the system writes verification evidence first, defers the task commit or snapshot until verification passes, and pauses instead of publishing changes when verification fails or cannot complete.
|
|
578
|
+
|
|
577
579
|
### Commit Conventions
|
|
578
580
|
|
|
579
581
|
| When | Format | Example |
|
|
@@ -105,9 +105,10 @@ export function mapUsage(sdkUsage, totalCostUsd) {
|
|
|
105
105
|
output: sdkUsage.output_tokens,
|
|
106
106
|
cacheRead: sdkUsage.cache_read_input_tokens,
|
|
107
107
|
cacheWrite: sdkUsage.cache_creation_input_tokens,
|
|
108
|
+
// Claude Agent SDK result usage is cumulative across its internal loop;
|
|
109
|
+
// repeated cache reads do not represent additional live context.
|
|
108
110
|
totalTokens: sdkUsage.input_tokens +
|
|
109
111
|
sdkUsage.output_tokens +
|
|
110
|
-
sdkUsage.cache_read_input_tokens +
|
|
111
112
|
sdkUsage.cache_creation_input_tokens,
|
|
112
113
|
cost: {
|
|
113
114
|
input: 0,
|
|
@@ -319,6 +319,11 @@ export class CmuxClient {
|
|
|
319
319
|
const stdout = await this.runAsync(["send-surface", "--surface", surfaceId, payload]);
|
|
320
320
|
return stdout !== null;
|
|
321
321
|
}
|
|
322
|
+
// Send Ctrl-C (ETX) to a surface to interrupt the running command.
|
|
323
|
+
async sendInterrupt(surfaceId) {
|
|
324
|
+
const stdout = await this.runAsync(["send-surface", "--surface", surfaceId, "\x03"]);
|
|
325
|
+
return stdout !== null;
|
|
326
|
+
}
|
|
322
327
|
}
|
|
323
328
|
export function syncCmuxSidebar(preferences, state) {
|
|
324
329
|
const client = CmuxClient.fromPreferences(preferences);
|
|
@@ -3,6 +3,15 @@
|
|
|
3
3
|
function now() {
|
|
4
4
|
return Date.now();
|
|
5
5
|
}
|
|
6
|
+
/**
|
|
7
|
+
* Size of the dispatch-decision ring buffer used by the Auto Orchestration
|
|
8
|
+
* module's stuck-loop detector. When the same `${unitType}:${unitId}` key
|
|
9
|
+
* fills the window, advance() blocks with `action: "stop"`.
|
|
10
|
+
*
|
|
11
|
+
* Mirrors the legacy `STUCK_WINDOW_SIZE` in auto/phases.ts so behaviour is
|
|
12
|
+
* preserved across the eventual cutover (issue #5791).
|
|
13
|
+
*/
|
|
14
|
+
export const STUCK_WINDOW_SIZE = 6;
|
|
6
15
|
export class AutoOrchestrator {
|
|
7
16
|
status = {
|
|
8
17
|
phase: "idle",
|
|
@@ -10,11 +19,13 @@ export class AutoOrchestrator {
|
|
|
10
19
|
};
|
|
11
20
|
deps;
|
|
12
21
|
lastAdvanceKey = null;
|
|
22
|
+
dispatchKeyWindow = [];
|
|
13
23
|
constructor(deps) {
|
|
14
24
|
this.deps = deps;
|
|
15
25
|
}
|
|
16
26
|
async start(_sessionContext) {
|
|
17
27
|
this.lastAdvanceKey = null;
|
|
28
|
+
this.dispatchKeyWindow = [];
|
|
18
29
|
this.status.phase = "running";
|
|
19
30
|
this.bumpTransition();
|
|
20
31
|
await this.deps.runtime.journalTransition({ name: "start" });
|
|
@@ -24,18 +35,70 @@ export class AutoOrchestrator {
|
|
|
24
35
|
async advance() {
|
|
25
36
|
try {
|
|
26
37
|
await this.deps.runtime.ensureLockOwnership();
|
|
38
|
+
const staleMsg = this.deps.health.checkResourcesStale();
|
|
39
|
+
if (staleMsg) {
|
|
40
|
+
await this.deps.uokGate.emit({
|
|
41
|
+
gateId: "resource-version-guard",
|
|
42
|
+
gateType: "policy",
|
|
43
|
+
outcome: "fail",
|
|
44
|
+
failureClass: "policy",
|
|
45
|
+
rationale: "resource version guard blocked dispatch",
|
|
46
|
+
findings: staleMsg,
|
|
47
|
+
});
|
|
48
|
+
const blocked = { kind: "blocked", reason: staleMsg, action: "stop" };
|
|
49
|
+
await this.deps.runtime.journalTransition({ name: "advance-blocked", reason: blocked.reason });
|
|
50
|
+
await this.deps.health.postAdvanceRecord(blocked);
|
|
51
|
+
return blocked;
|
|
52
|
+
}
|
|
53
|
+
await this.deps.uokGate.emit({
|
|
54
|
+
gateId: "resource-version-guard",
|
|
55
|
+
gateType: "policy",
|
|
56
|
+
outcome: "pass",
|
|
57
|
+
failureClass: "none",
|
|
58
|
+
rationale: "resource version guard passed",
|
|
59
|
+
});
|
|
27
60
|
const gate = await this.deps.health.preAdvanceGate();
|
|
28
|
-
if (
|
|
29
|
-
|
|
61
|
+
if (gate.kind === "fail") {
|
|
62
|
+
await this.deps.uokGate.emit({
|
|
63
|
+
gateId: "pre-dispatch-health-gate",
|
|
64
|
+
gateType: "execution",
|
|
65
|
+
outcome: "manual-attention",
|
|
66
|
+
failureClass: "manual-attention",
|
|
67
|
+
rationale: "pre-dispatch health gate blocked dispatch",
|
|
68
|
+
findings: gate.reason,
|
|
69
|
+
});
|
|
70
|
+
const blocked = { kind: "blocked", reason: gate.reason, action: "pause" };
|
|
30
71
|
await this.deps.runtime.journalTransition({ name: "advance-blocked", reason: blocked.reason });
|
|
31
72
|
await this.deps.health.postAdvanceRecord(blocked);
|
|
32
73
|
return blocked;
|
|
33
74
|
}
|
|
75
|
+
if (gate.kind === "threw") {
|
|
76
|
+
await this.deps.uokGate.emit({
|
|
77
|
+
gateId: "pre-dispatch-health-gate",
|
|
78
|
+
gateType: "execution",
|
|
79
|
+
outcome: "manual-attention",
|
|
80
|
+
failureClass: "manual-attention",
|
|
81
|
+
rationale: "pre-dispatch health gate threw unexpectedly",
|
|
82
|
+
findings: String(gate.error),
|
|
83
|
+
});
|
|
84
|
+
// intentional fall-through: matches runPreDispatch behaviour
|
|
85
|
+
}
|
|
86
|
+
else {
|
|
87
|
+
await this.deps.uokGate.emit({
|
|
88
|
+
gateId: "pre-dispatch-health-gate",
|
|
89
|
+
gateType: "execution",
|
|
90
|
+
outcome: "pass",
|
|
91
|
+
failureClass: "none",
|
|
92
|
+
rationale: "pre-dispatch health gate passed",
|
|
93
|
+
findings: gate.fixesApplied?.join(", ") ?? "",
|
|
94
|
+
});
|
|
95
|
+
}
|
|
34
96
|
const reconciliation = await this.deps.stateReconciliation.reconcileBeforeDispatch();
|
|
35
97
|
if (!reconciliation.ok || !reconciliation.stateSnapshot) {
|
|
36
98
|
const blocked = {
|
|
37
99
|
kind: "blocked",
|
|
38
|
-
reason: reconciliation.reason,
|
|
100
|
+
reason: reconciliation.reason ?? "state reconciliation produced no snapshot",
|
|
101
|
+
action: "pause",
|
|
39
102
|
stateSnapshot: reconciliation.stateSnapshot,
|
|
40
103
|
};
|
|
41
104
|
await this.deps.runtime.journalTransition({ name: "advance-blocked", reason: blocked.reason });
|
|
@@ -48,14 +111,49 @@ export class AutoOrchestrator {
|
|
|
48
111
|
this.status.phase = "stopped";
|
|
49
112
|
this.status.activeUnit = undefined;
|
|
50
113
|
this.lastAdvanceKey = null;
|
|
114
|
+
this.dispatchKeyWindow = [];
|
|
51
115
|
this.bumpTransition();
|
|
52
116
|
await this.deps.runtime.journalTransition({ name: "advance-stopped", reason: stopped.reason });
|
|
53
117
|
await this.deps.health.postAdvanceRecord(stopped);
|
|
54
118
|
return stopped;
|
|
55
119
|
}
|
|
56
120
|
const nextKey = `${decision.unitType}:${decision.unitId}`;
|
|
57
|
-
|
|
58
|
-
|
|
121
|
+
// Record every dispatch decision in the ring buffer before pre-flight
|
|
122
|
+
// checks so the stuck-loop detector observes the full decision history
|
|
123
|
+
// (including decisions that idempotency would otherwise short-circuit).
|
|
124
|
+
// The ring is capped at STUCK_WINDOW_SIZE and evicts oldest-first.
|
|
125
|
+
this.dispatchKeyWindow.push(nextKey);
|
|
126
|
+
if (this.dispatchKeyWindow.length > STUCK_WINDOW_SIZE) {
|
|
127
|
+
this.dispatchKeyWindow.shift();
|
|
128
|
+
}
|
|
129
|
+
// Idempotency: same key as immediately previous successful advance.
|
|
130
|
+
// This is the soft, fast-path block kept from #5786. It only fires when
|
|
131
|
+
// the ring is NOT yet saturated for this key — once the ring is full of
|
|
132
|
+
// `nextKey`, the stuck-loop verdict takes precedence (see below). Both
|
|
133
|
+
// checks coexist: idempotency for the common immediate-repeat case,
|
|
134
|
+
// stuck-loop for the saturated-window case.
|
|
135
|
+
const matchingCount = this.dispatchKeyWindow.filter((k) => k === nextKey).length;
|
|
136
|
+
if (this.lastAdvanceKey === nextKey && matchingCount < STUCK_WINDOW_SIZE) {
|
|
137
|
+
const blocked = { kind: "blocked", reason: "idempotent advance: unit already active", action: "stop" };
|
|
138
|
+
await this.deps.runtime.journalTransition({
|
|
139
|
+
name: "advance-blocked",
|
|
140
|
+
reason: blocked.reason,
|
|
141
|
+
unitType: decision.unitType,
|
|
142
|
+
unitId: decision.unitId,
|
|
143
|
+
});
|
|
144
|
+
await this.deps.health.postAdvanceRecord(blocked);
|
|
145
|
+
return blocked;
|
|
146
|
+
}
|
|
147
|
+
// Stuck-loop detection: when the ring is saturated with copies of
|
|
148
|
+
// `nextKey` (count >= STUCK_WINDOW_SIZE), the orchestrator has been
|
|
149
|
+
// picking the same unit across the whole window and must hard-stop with
|
|
150
|
+
// a diagnosable reason.
|
|
151
|
+
if (matchingCount >= STUCK_WINDOW_SIZE) {
|
|
152
|
+
const blocked = {
|
|
153
|
+
kind: "blocked",
|
|
154
|
+
reason: `stuck-loop: ${nextKey} picked ${matchingCount} times`,
|
|
155
|
+
action: "stop",
|
|
156
|
+
};
|
|
59
157
|
await this.deps.runtime.journalTransition({
|
|
60
158
|
name: "advance-blocked",
|
|
61
159
|
reason: blocked.reason,
|
|
@@ -70,6 +168,7 @@ export class AutoOrchestrator {
|
|
|
70
168
|
const blocked = {
|
|
71
169
|
kind: "blocked",
|
|
72
170
|
reason: contract.reason,
|
|
171
|
+
action: "pause",
|
|
73
172
|
stateSnapshot: reconciliation.stateSnapshot,
|
|
74
173
|
};
|
|
75
174
|
await this.deps.runtime.journalTransition({
|
|
@@ -86,6 +185,7 @@ export class AutoOrchestrator {
|
|
|
86
185
|
const blocked = {
|
|
87
186
|
kind: "blocked",
|
|
88
187
|
reason: worktree.reason,
|
|
188
|
+
action: "pause",
|
|
89
189
|
stateSnapshot: reconciliation.stateSnapshot,
|
|
90
190
|
};
|
|
91
191
|
await this.deps.runtime.journalTransition({
|
|
@@ -108,7 +208,11 @@ export class AutoOrchestrator {
|
|
|
108
208
|
unitId: decision.unitId,
|
|
109
209
|
});
|
|
110
210
|
await this.deps.worktree.syncAfterUnit(decision.unitType, decision.unitId);
|
|
111
|
-
const advanced = {
|
|
211
|
+
const advanced = {
|
|
212
|
+
kind: "advanced",
|
|
213
|
+
unit: { unitType: decision.unitType, unitId: decision.unitId },
|
|
214
|
+
stateSnapshot: reconciliation.stateSnapshot,
|
|
215
|
+
};
|
|
112
216
|
await this.deps.health.postAdvanceRecord(advanced);
|
|
113
217
|
return advanced;
|
|
114
218
|
}
|
|
@@ -134,6 +238,7 @@ export class AutoOrchestrator {
|
|
|
134
238
|
}
|
|
135
239
|
if (result.kind === "stopped") {
|
|
136
240
|
this.lastAdvanceKey = null;
|
|
241
|
+
this.dispatchKeyWindow = [];
|
|
137
242
|
this.status.activeUnit = undefined;
|
|
138
243
|
}
|
|
139
244
|
this.bumpTransition();
|
|
@@ -158,6 +263,7 @@ export class AutoOrchestrator {
|
|
|
158
263
|
}
|
|
159
264
|
async resume() {
|
|
160
265
|
this.lastAdvanceKey = null;
|
|
266
|
+
this.dispatchKeyWindow = [];
|
|
161
267
|
this.status.phase = "running";
|
|
162
268
|
this.bumpTransition();
|
|
163
269
|
await this.deps.runtime.journalTransition({ name: "resume" });
|
|
@@ -172,6 +278,7 @@ export class AutoOrchestrator {
|
|
|
172
278
|
this.status.phase = "stopped";
|
|
173
279
|
this.status.activeUnit = undefined;
|
|
174
280
|
this.lastAdvanceKey = null;
|
|
281
|
+
this.dispatchKeyWindow = [];
|
|
175
282
|
this.bumpTransition();
|
|
176
283
|
await this.deps.runtime.journalTransition({ name: "stop", reason });
|
|
177
284
|
await this.deps.notifications.notifyLifecycle({ name: "stop", detail: reason });
|
|
@@ -102,6 +102,9 @@ function missingSliceStop(mid, phase) {
|
|
|
102
102
|
level: "error",
|
|
103
103
|
};
|
|
104
104
|
}
|
|
105
|
+
function isRegistryMilestoneComplete(state, mid) {
|
|
106
|
+
return state.registry.some((milestone) => milestone.id === mid && milestone.status === "complete");
|
|
107
|
+
}
|
|
105
108
|
/**
|
|
106
109
|
* Check for milestone slices missing SUMMARY files.
|
|
107
110
|
* Returns array of missing slice IDs, or empty array if all present or DB unavailable.
|
|
@@ -247,6 +250,8 @@ export const DISPATCH_RULES = [
|
|
|
247
250
|
return null;
|
|
248
251
|
if (!MILESTONE_ID_RE.test(mid))
|
|
249
252
|
return null;
|
|
253
|
+
if (isRegistryMilestoneComplete(state, mid))
|
|
254
|
+
return null;
|
|
250
255
|
// Align with the plan-v2 gate's lookup semantics: whitespace-only counts
|
|
251
256
|
// as missing, and an auto worktree may fall back to GSD_PROJECT_ROOT.
|
|
252
257
|
if (hasFinalizedMilestoneContext(basePath, mid))
|
|
@@ -557,6 +562,8 @@ export const DISPATCH_RULES = [
|
|
|
557
562
|
match: async ({ state, mid, midTitle, basePath, prefs, structuredQuestionsAvailable }) => {
|
|
558
563
|
if (state.phase !== "pre-planning")
|
|
559
564
|
return null;
|
|
565
|
+
if (isRegistryMilestoneComplete(state, mid))
|
|
566
|
+
return null;
|
|
560
567
|
const contextFile = resolveMilestoneFile(basePath, mid, "CONTEXT");
|
|
561
568
|
const hasContext = !!(contextFile && (await loadFile(contextFile)));
|
|
562
569
|
if (hasContext)
|
|
@@ -1091,19 +1098,19 @@ export const DISPATCH_RULES = [
|
|
|
1091
1098
|
return { action: "skip" };
|
|
1092
1099
|
}
|
|
1093
1100
|
}
|
|
1094
|
-
// Safety guard (#2675): block completion when VALIDATION
|
|
1095
|
-
//
|
|
1096
|
-
// terminal
|
|
1097
|
-
//
|
|
1101
|
+
// Safety guard (#2675, #5747): block completion when VALIDATION
|
|
1102
|
+
// verdict is non-passing. The state machine treats these verdicts as
|
|
1103
|
+
// terminal, but completing-milestone should NOT proceed — remediation
|
|
1104
|
+
// or human attention is needed.
|
|
1098
1105
|
const validationFile = resolveMilestoneFile(basePath, mid, "VALIDATION");
|
|
1099
1106
|
if (validationFile) {
|
|
1100
1107
|
const validationContent = await loadFile(validationFile);
|
|
1101
1108
|
if (validationContent) {
|
|
1102
1109
|
const verdict = extractVerdict(validationContent);
|
|
1103
|
-
if (verdict === "needs-remediation") {
|
|
1110
|
+
if (verdict === "needs-remediation" || verdict === "needs-attention") {
|
|
1104
1111
|
return {
|
|
1105
1112
|
action: "stop",
|
|
1106
|
-
reason: `Cannot complete milestone ${mid}: VALIDATION verdict is "
|
|
1113
|
+
reason: `Cannot complete milestone ${mid}: VALIDATION verdict is "${verdict}". Address the validation findings and re-run validation, or update the verdict manually.`,
|
|
1107
1114
|
level: "warning",
|
|
1108
1115
|
};
|
|
1109
1116
|
}
|