gsd-pi 2.73.1-dev.6ddfa43 → 2.73.1-dev.9a4cd44
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli-web-branch.d.ts +4 -3
- package/dist/cli-web-branch.js +10 -7
- package/dist/cli.js +99 -206
- package/dist/logo.d.ts +1 -1
- package/dist/logo.js +1 -1
- package/dist/onboarding.js +59 -53
- package/dist/resource-loader.js +2 -2
- package/dist/resources/extensions/gsd/auto/phases.js +15 -9
- package/dist/resources/extensions/gsd/auto-dispatch.js +11 -3
- package/dist/resources/extensions/gsd/auto-post-unit.js +41 -1
- package/dist/resources/extensions/gsd/auto-start.js +3 -0
- package/dist/resources/extensions/gsd/auto-timeout-recovery.js +13 -0
- package/dist/resources/extensions/gsd/auto-verification.js +88 -3
- package/dist/resources/extensions/gsd/auto.js +29 -8
- package/dist/resources/extensions/gsd/commands-handlers.js +8 -2
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -1
- package/dist/resources/extensions/gsd/notification-widget.js +2 -2
- package/dist/resources/extensions/gsd/state.js +61 -14
- package/dist/update-check.d.ts +1 -0
- package/dist/update-check.js +13 -5
- package/dist/update-cmd.js +4 -3
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +14 -14
- package/dist/web/standalone/.next/build-manifest.json +2 -2
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/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 +14 -14
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/package.json +1 -2
- package/packages/pi-ai/dist/utils/overflow.d.ts.map +1 -1
- package/packages/pi-ai/dist/utils/overflow.js +12 -0
- package/packages/pi-ai/dist/utils/overflow.js.map +1 -1
- package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts +2 -0
- package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/utils/tests/overflow.test.js +50 -0
- package/packages/pi-ai/dist/utils/tests/overflow.test.js.map +1 -0
- package/packages/pi-ai/src/utils/overflow.ts +14 -1
- package/packages/pi-ai/src/utils/tests/overflow.test.ts +58 -0
- package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +138 -0
- package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction/utils.js +5 -5
- package/packages/pi-coding-agent/dist/core/compaction/utils.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/compaction-utils.test.js +45 -0
- package/packages/pi-coding-agent/dist/core/compaction-utils.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts +2 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js +9 -3
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +52 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +21 -4
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +11 -3
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +157 -0
- package/packages/pi-coding-agent/src/core/compaction/utils.ts +5 -5
- package/packages/pi-coding-agent/src/core/compaction-utils.test.ts +50 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +73 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.ts +9 -3
- package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +21 -4
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +11 -3
- package/packages/pi-tui/dist/__tests__/tui.test.js +60 -1
- package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
- package/packages/pi-tui/dist/tui.d.ts +8 -0
- package/packages/pi-tui/dist/tui.d.ts.map +1 -1
- package/packages/pi-tui/dist/tui.js +32 -3
- package/packages/pi-tui/dist/tui.js.map +1 -1
- package/packages/pi-tui/src/__tests__/tui.test.ts +76 -1
- package/packages/pi-tui/src/tui.ts +31 -3
- package/src/resources/extensions/gsd/auto/phases.ts +22 -9
- package/src/resources/extensions/gsd/auto-dispatch.ts +10 -4
- package/src/resources/extensions/gsd/auto-post-unit.ts +47 -1
- package/src/resources/extensions/gsd/auto-start.ts +3 -0
- package/src/resources/extensions/gsd/auto-timeout-recovery.ts +17 -0
- package/src/resources/extensions/gsd/auto-verification.ts +98 -3
- package/src/resources/extensions/gsd/auto.ts +31 -14
- package/src/resources/extensions/gsd/commands-handlers.ts +8 -2
- package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -1
- package/src/resources/extensions/gsd/notification-widget.ts +2 -2
- package/src/resources/extensions/gsd/state.ts +71 -15
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +53 -0
- package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +142 -0
- package/src/resources/extensions/gsd/tests/completed-at-reconcile.test.ts +42 -0
- package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/derive-state-helpers.test.ts +68 -8
- package/src/resources/extensions/gsd/tests/derive-state.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +4 -2
- package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +5 -7
- package/src/resources/extensions/gsd/tests/token-profile.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +179 -0
- /package/dist/web/standalone/.next/static/{r6AvNu-aMwn4nwqjHqAfw → ASJ2RGD7E1iiUYzA0xT2i}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{r6AvNu-aMwn4nwqjHqAfw → ASJ2RGD7E1iiUYzA0xT2i}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,142 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* complete-milestone-false-merge.test.ts — Regression test for #4175.
|
|
3
|
+
*
|
|
4
|
+
* Before the fix, a failed complete-milestone unit could leave a stub
|
|
5
|
+
* SUMMARY blocker placeholder on disk. stopAuto's SUMMARY-presence check
|
|
6
|
+
* then treated the milestone as complete and merged the worktree branch
|
|
7
|
+
* into main — emitting a misleading metadata-only merge warning for a
|
|
8
|
+
* milestone that was never legitimately finished.
|
|
9
|
+
*
|
|
10
|
+
* The fix has three cooperating parts:
|
|
11
|
+
* 1. stopAuto uses DB status (authoritative) instead of SUMMARY presence
|
|
12
|
+
* when the project DB is available.
|
|
13
|
+
* 2. postUnitPreVerification pauses auto-mode for complete-milestone
|
|
14
|
+
* after retries are exhausted instead of writing a blocker placeholder.
|
|
15
|
+
* 3. recoverTimedOutUnit pauses for complete-milestone instead of
|
|
16
|
+
* writing a blocker placeholder.
|
|
17
|
+
*
|
|
18
|
+
* This test guards all three via source inspection so a future refactor
|
|
19
|
+
* cannot silently reintroduce the false-merge path.
|
|
20
|
+
*/
|
|
21
|
+
|
|
22
|
+
import test from "node:test";
|
|
23
|
+
import assert from "node:assert/strict";
|
|
24
|
+
import { readFileSync } from "node:fs";
|
|
25
|
+
import { join } from "node:path";
|
|
26
|
+
|
|
27
|
+
const gsdDir = join(import.meta.dirname, "..");
|
|
28
|
+
const autoSrc = readFileSync(join(gsdDir, "auto.ts"), "utf-8");
|
|
29
|
+
const postUnitSrc = readFileSync(join(gsdDir, "auto-post-unit.ts"), "utf-8");
|
|
30
|
+
const timeoutSrc = readFileSync(join(gsdDir, "auto-timeout-recovery.ts"), "utf-8");
|
|
31
|
+
|
|
32
|
+
test("#4175: stopAuto uses DB status as the authoritative milestone-complete signal", () => {
|
|
33
|
+
const step4Idx = autoSrc.indexOf("Step 4: Auto-worktree exit");
|
|
34
|
+
assert.ok(step4Idx !== -1, "Step 4 comment exists in stopAuto");
|
|
35
|
+
const step5Idx = autoSrc.indexOf("Step 5:", step4Idx);
|
|
36
|
+
const step4Block = autoSrc.slice(step4Idx, step5Idx);
|
|
37
|
+
|
|
38
|
+
assert.ok(
|
|
39
|
+
step4Block.includes("isDbAvailable()"),
|
|
40
|
+
"Step 4 should branch on isDbAvailable() so DB is consulted when present",
|
|
41
|
+
);
|
|
42
|
+
assert.ok(
|
|
43
|
+
step4Block.includes("getMilestone(s.currentMilestoneId)"),
|
|
44
|
+
"Step 4 should read authoritative milestone status via getMilestone()",
|
|
45
|
+
);
|
|
46
|
+
assert.ok(
|
|
47
|
+
/status\s*===\s*"complete"/.test(step4Block),
|
|
48
|
+
'Step 4 should compare the DB row status to "complete"',
|
|
49
|
+
);
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
test("#4175: stopAuto imports getMilestone from gsd-db", () => {
|
|
53
|
+
assert.ok(
|
|
54
|
+
/import\s*\{[^}]*\bgetMilestone\b[^}]*\}\s*from\s*"\.\/gsd-db\.js"/.test(autoSrc),
|
|
55
|
+
"auto.ts should import getMilestone from ./gsd-db.js",
|
|
56
|
+
);
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
test("#4175: stopAuto still falls back to SUMMARY presence when DB is unavailable", () => {
|
|
60
|
+
const step4Idx = autoSrc.indexOf("Step 4: Auto-worktree exit");
|
|
61
|
+
const step5Idx = autoSrc.indexOf("Step 5:", step4Idx);
|
|
62
|
+
const step4Block = autoSrc.slice(step4Idx, step5Idx);
|
|
63
|
+
|
|
64
|
+
assert.ok(
|
|
65
|
+
step4Block.includes("resolveMilestoneFile"),
|
|
66
|
+
"Step 4 should keep SUMMARY-file resolution for DB-unavailable projects",
|
|
67
|
+
);
|
|
68
|
+
assert.ok(
|
|
69
|
+
step4Block.includes("preserveBranch"),
|
|
70
|
+
"Step 4 should still preserve branch for incomplete milestones (fallback path)",
|
|
71
|
+
);
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
test("#4175: postUnitPreVerification pauses complete-milestone after retries exhausted", () => {
|
|
75
|
+
// The pause branch must live inside the retries-exhausted block, above the
|
|
76
|
+
// writeBlockerPlaceholder call — otherwise the stub SUMMARY is still written.
|
|
77
|
+
const retriesExhaustedIdx = postUnitSrc.indexOf(
|
|
78
|
+
"if (attempt > MAX_VERIFICATION_RETRIES)",
|
|
79
|
+
);
|
|
80
|
+
assert.ok(
|
|
81
|
+
retriesExhaustedIdx !== -1,
|
|
82
|
+
"retries-exhausted guard exists in postUnitPreVerification",
|
|
83
|
+
);
|
|
84
|
+
|
|
85
|
+
const blockerCallIdx = postUnitSrc.indexOf("writeBlockerPlaceholder", retriesExhaustedIdx);
|
|
86
|
+
assert.ok(
|
|
87
|
+
blockerCallIdx !== -1,
|
|
88
|
+
"blocker placeholder call still exists for non-milestone units",
|
|
89
|
+
);
|
|
90
|
+
|
|
91
|
+
const exhaustedBlock = postUnitSrc.slice(retriesExhaustedIdx, blockerCallIdx);
|
|
92
|
+
|
|
93
|
+
assert.ok(
|
|
94
|
+
/s\.currentUnit\.type\s*===\s*"complete-milestone"/.test(exhaustedBlock),
|
|
95
|
+
"retries-exhausted block should specifically handle complete-milestone",
|
|
96
|
+
);
|
|
97
|
+
assert.ok(
|
|
98
|
+
/pauseAuto\s*\(\s*ctx\s*,\s*pi\s*\)/.test(exhaustedBlock),
|
|
99
|
+
"complete-milestone path should call pauseAuto instead of falling through",
|
|
100
|
+
);
|
|
101
|
+
// The pause branch must return so execution never reaches writeBlockerPlaceholder.
|
|
102
|
+
assert.ok(
|
|
103
|
+
/return\s+"dispatched"\s*;/.test(exhaustedBlock),
|
|
104
|
+
"complete-milestone pause branch should return before the placeholder call",
|
|
105
|
+
);
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
test("#4175: recoverTimedOutUnit pauses complete-milestone instead of writing a blocker placeholder", () => {
|
|
109
|
+
// The complete-milestone pause branch must sit immediately above the
|
|
110
|
+
// "retries exhausted" writeBlockerPlaceholder call so a failed
|
|
111
|
+
// complete-milestone never produces a stub SUMMARY. Anchor on the
|
|
112
|
+
// comment that precedes that specific placeholder call rather than the
|
|
113
|
+
// function's earlier writeBlockerPlaceholder use sites or its import.
|
|
114
|
+
// Use lastIndexOf so we find the final retries-exhausted block in
|
|
115
|
+
// recoverTimedOutUnit, not an earlier helper with the same comment.
|
|
116
|
+
const exhaustedAnchor = "Retries exhausted — write a blocker placeholder";
|
|
117
|
+
const exhaustedIdx = timeoutSrc.lastIndexOf(exhaustedAnchor);
|
|
118
|
+
assert.ok(
|
|
119
|
+
exhaustedIdx !== -1,
|
|
120
|
+
"retries-exhausted blocker-placeholder path still exists for non-milestone units",
|
|
121
|
+
);
|
|
122
|
+
|
|
123
|
+
const guardIdx = timeoutSrc.lastIndexOf(
|
|
124
|
+
'unitType === "complete-milestone"',
|
|
125
|
+
exhaustedIdx,
|
|
126
|
+
);
|
|
127
|
+
assert.ok(
|
|
128
|
+
guardIdx !== -1,
|
|
129
|
+
"complete-milestone guard should appear above the retries-exhausted placeholder call",
|
|
130
|
+
);
|
|
131
|
+
|
|
132
|
+
const guardBlock = timeoutSrc.slice(guardIdx, exhaustedIdx);
|
|
133
|
+
assert.ok(
|
|
134
|
+
/return\s+"paused"\s*;/.test(guardBlock),
|
|
135
|
+
"complete-milestone guard should return 'paused' before the placeholder call",
|
|
136
|
+
);
|
|
137
|
+
// The guard itself must not call writeBlockerPlaceholder.
|
|
138
|
+
assert.ok(
|
|
139
|
+
!guardBlock.includes("writeBlockerPlaceholder"),
|
|
140
|
+
"complete-milestone guard must not write a blocker placeholder",
|
|
141
|
+
);
|
|
142
|
+
});
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Regression test for #4129: tasks.completed_at stays NULL when status is
|
|
3
|
+
* reconciled to 'complete' via the file-existence path in state.ts.
|
|
4
|
+
*
|
|
5
|
+
* Root cause: reconcileSliceTasks called
|
|
6
|
+
* updateTaskStatus(milestoneId, sliceId, t.id, "complete")
|
|
7
|
+
* without a completedAt timestamp, so the column stays NULL.
|
|
8
|
+
*
|
|
9
|
+
* Fix: pass new Date().toISOString() as the 5th argument.
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import { describe, test } from "node:test";
|
|
13
|
+
import assert from "node:assert/strict";
|
|
14
|
+
import { readFileSync } from "node:fs";
|
|
15
|
+
import { join, dirname } from "node:path";
|
|
16
|
+
import { fileURLToPath } from "node:url";
|
|
17
|
+
|
|
18
|
+
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
19
|
+
const stateSource = readFileSync(join(__dirname, "..", "state.ts"), "utf-8");
|
|
20
|
+
|
|
21
|
+
describe("completed-at reconcile (#4129)", () => {
|
|
22
|
+
test("reconcileSliceTasks passes a completedAt timestamp when setting status to complete", () => {
|
|
23
|
+
// Before the fix, state.ts had:
|
|
24
|
+
// updateTaskStatus(milestoneId, sliceId, t.id, "complete")
|
|
25
|
+
// which leaves completed_at NULL in the DB.
|
|
26
|
+
// After the fix, a timestamp must be passed as the 5th argument.
|
|
27
|
+
assert.doesNotMatch(
|
|
28
|
+
stateSource,
|
|
29
|
+
/updateTaskStatus\(\s*milestoneId\s*,\s*sliceId\s*,\s*t\.id\s*,\s*["']complete["']\s*\)/,
|
|
30
|
+
"updateTaskStatus must not be called without a completedAt timestamp when reconciling tasks to 'complete' (#4129)",
|
|
31
|
+
);
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
test("reconcileSliceTasks passes new Date().toISOString() as the completedAt argument", () => {
|
|
35
|
+
// Positive assertion: the fixed call must include a timestamp.
|
|
36
|
+
assert.match(
|
|
37
|
+
stateSource,
|
|
38
|
+
/updateTaskStatus\(\s*milestoneId\s*,\s*sliceId\s*,\s*t\.id\s*,\s*["']complete["']\s*,\s*new Date\(\)\.toISOString\(\)\s*\)/,
|
|
39
|
+
"reconcileSliceTasks must pass new Date().toISOString() as completedAt when setting task status to 'complete' (#4129)",
|
|
40
|
+
);
|
|
41
|
+
});
|
|
42
|
+
});
|
|
@@ -351,8 +351,9 @@ skills_used: []
|
|
|
351
351
|
const dbState = await deriveStateFromDb(base);
|
|
352
352
|
|
|
353
353
|
assertStatesEqual(dbState, fileState, 'E-blocked');
|
|
354
|
-
|
|
355
|
-
assert.
|
|
354
|
+
// With partial-dep fallback, circular deps no longer block — fallback picks first eligible slice
|
|
355
|
+
assert.deepStrictEqual(dbState.phase, 'planning', 'E-blocked: phase is planning (fallback picks a slice)');
|
|
356
|
+
assert.ok(dbState.activeSlice !== null, 'E-blocked: activeSlice is set via fallback');
|
|
356
357
|
|
|
357
358
|
closeDatabase();
|
|
358
359
|
} finally {
|
|
@@ -616,9 +616,10 @@ describe('derive-state-db', async () => {
|
|
|
616
616
|
invalidateStateCache();
|
|
617
617
|
const dbState = await deriveStateFromDb(base);
|
|
618
618
|
|
|
619
|
-
|
|
619
|
+
// With partial-dep fallback, circular deps no longer block — fallback picks first eligible slice
|
|
620
|
+
assert.deepStrictEqual(dbState.phase, 'planning', 'blocked-db: phase is planning (fallback picks a slice)');
|
|
620
621
|
assert.deepStrictEqual(dbState.phase, fileState.phase, 'blocked-db: phase matches filesystem');
|
|
621
|
-
assert.ok(dbState.
|
|
622
|
+
assert.ok(dbState.activeSlice !== null, 'blocked-db: activeSlice is set via fallback');
|
|
622
623
|
|
|
623
624
|
closeDatabase();
|
|
624
625
|
} finally {
|
|
@@ -307,27 +307,87 @@ describe('derive-state-helpers', () => {
|
|
|
307
307
|
}
|
|
308
308
|
});
|
|
309
309
|
|
|
310
|
-
// ─── buildCompletenessSet:
|
|
311
|
-
test('buildCompletenessSet:
|
|
310
|
+
// ─── buildCompletenessSet: DB status is authoritative ──────────────
|
|
311
|
+
test('buildCompletenessSet: DB status=complete marks milestone complete', async () => {
|
|
312
312
|
const base = createFixtureBase();
|
|
313
313
|
try {
|
|
314
|
-
// M001 has summary on disk but DB status is still 'active'
|
|
315
314
|
writeFile(base, 'milestones/M001/M001-ROADMAP.md', ROADMAP_CONTENT);
|
|
316
315
|
writeFile(base, 'milestones/M001/M001-SUMMARY.md', '# M001 Summary\n\nDone.');
|
|
317
|
-
// M002 is the real active milestone
|
|
318
316
|
writeFile(base, 'milestones/M002/M002-CONTEXT.md', '# M002\n\nActive.');
|
|
319
317
|
|
|
320
318
|
openDatabase(':memory:');
|
|
321
|
-
insertMilestone({ id: 'M001', title: 'First', status: '
|
|
319
|
+
insertMilestone({ id: 'M001', title: 'First', status: 'complete' });
|
|
322
320
|
insertMilestone({ id: 'M002', title: 'Second', status: 'active' });
|
|
323
321
|
|
|
324
322
|
invalidateStateCache();
|
|
325
323
|
const state = await deriveStateFromDb(base);
|
|
326
324
|
|
|
327
|
-
// M001 should be complete (summary on disk), M002 should be active
|
|
328
325
|
const m1 = state.registry.find(e => e.id === 'M001');
|
|
329
|
-
assert.equal(m1?.status, 'complete', '
|
|
330
|
-
assert.equal(state.activeMilestone?.id, 'M002', '
|
|
326
|
+
assert.equal(m1?.status, 'complete', 'DB status=complete → registry entry complete');
|
|
327
|
+
assert.equal(state.activeMilestone?.id, 'M002', 'M002 is the active milestone');
|
|
328
|
+
} finally {
|
|
329
|
+
closeDatabase();
|
|
330
|
+
cleanup(base);
|
|
331
|
+
}
|
|
332
|
+
});
|
|
333
|
+
|
|
334
|
+
// ─── Regression #4179: orphan SUMMARY must NOT flip DB-active milestone ───
|
|
335
|
+
// A crashed complete-milestone turn (or stale/manual SUMMARY.md) can leave
|
|
336
|
+
// a milestone SUMMARY on disk while the DB row still reads 'active'. The
|
|
337
|
+
// read-side of state derivation must NOT treat the orphan SUMMARY as a
|
|
338
|
+
// completion signal, or the auto-loop advances and merges work that was
|
|
339
|
+
// never actually finished (same failure class as #4175, read-side twin).
|
|
340
|
+
test('buildCompletenessSet (#4179): orphan SUMMARY on disk does not mark DB-active milestone complete', async () => {
|
|
341
|
+
const base = createFixtureBase();
|
|
342
|
+
try {
|
|
343
|
+
writeFile(base, 'milestones/M001/M001-ROADMAP.md', ROADMAP_CONTENT);
|
|
344
|
+
writeFile(base, 'milestones/M001/M001-SUMMARY.md', '# M001 Orphan Summary\n\nLeft over from crashed turn.');
|
|
345
|
+
|
|
346
|
+
openDatabase(':memory:');
|
|
347
|
+
insertMilestone({ id: 'M001', title: 'First', status: 'active' });
|
|
348
|
+
// Slice still in-flight — auto should resume, not merge.
|
|
349
|
+
insertSlice({ id: 'S01', milestoneId: 'M001', title: 'First', status: 'active', risk: 'low', depends: [] });
|
|
350
|
+
insertSlice({ id: 'S02', milestoneId: 'M001', title: 'Second', status: 'pending', risk: 'low', depends: ['S01'] });
|
|
351
|
+
insertTask({ id: 'T01', sliceId: 'S01', milestoneId: 'M001', title: 'In-flight', status: 'pending' });
|
|
352
|
+
|
|
353
|
+
invalidateStateCache();
|
|
354
|
+
const state = await deriveStateFromDb(base);
|
|
355
|
+
|
|
356
|
+
const m1 = state.registry.find(e => e.id === 'M001');
|
|
357
|
+
assert.notEqual(m1?.status, 'complete', 'orphan SUMMARY must not mark milestone complete');
|
|
358
|
+
assert.equal(m1?.status, 'active', 'M001 remains active — DB is authoritative');
|
|
359
|
+
assert.equal(state.activeMilestone?.id, 'M001', 'M001 is still the active milestone');
|
|
360
|
+
assert.notEqual(state.phase, 'completing-milestone', 'must not short-circuit into completion');
|
|
361
|
+
} finally {
|
|
362
|
+
closeDatabase();
|
|
363
|
+
cleanup(base);
|
|
364
|
+
}
|
|
365
|
+
});
|
|
366
|
+
|
|
367
|
+
// Regression #4179 (companion): DB-active milestone with all slices done +
|
|
368
|
+
// validation terminal + orphan SUMMARY must still flow through completing-milestone
|
|
369
|
+
// (re-runs complete-milestone), not be reported as already-complete.
|
|
370
|
+
test('buildRegistryAndFindActive (#4179): orphan SUMMARY with validation-terminal falls through to completing-milestone', async () => {
|
|
371
|
+
const base = createFixtureBase();
|
|
372
|
+
try {
|
|
373
|
+
writeFile(base, 'milestones/M001/M001-ROADMAP.md', ROADMAP_CONTENT);
|
|
374
|
+
writeFile(base, 'milestones/M001/slices/S01/S01-PLAN.md', PLAN_CONTENT);
|
|
375
|
+
writeFile(base, 'milestones/M001/slices/S02/S02-PLAN.md', PLAN_CONTENT);
|
|
376
|
+
writeFile(base, 'milestones/M001/M001-VALIDATION.md', '---\nverdict: passed\n---\n# Validation\nAll good.');
|
|
377
|
+
writeFile(base, 'milestones/M001/M001-SUMMARY.md', '# M001 Orphan Summary\n\nLeft over.');
|
|
378
|
+
|
|
379
|
+
openDatabase(':memory:');
|
|
380
|
+
insertMilestone({ id: 'M001', title: 'First', status: 'active' });
|
|
381
|
+
insertSlice({ id: 'S01', milestoneId: 'M001', title: 'First', status: 'complete', risk: 'low', depends: [] });
|
|
382
|
+
insertSlice({ id: 'S02', milestoneId: 'M001', title: 'Second', status: 'complete', risk: 'low', depends: ['S01'] });
|
|
383
|
+
|
|
384
|
+
invalidateStateCache();
|
|
385
|
+
const state = await deriveStateFromDb(base);
|
|
386
|
+
|
|
387
|
+
const m1 = state.registry.find(e => e.id === 'M001');
|
|
388
|
+
assert.equal(m1?.status, 'active', 'M001 stays active despite orphan SUMMARY + validation-terminal');
|
|
389
|
+
assert.equal(state.activeMilestone?.id, 'M001', 'M001 is still the active milestone');
|
|
390
|
+
assert.equal(state.phase, 'completing-milestone', 'phase flows through completing-milestone (re-run)');
|
|
331
391
|
} finally {
|
|
332
392
|
closeDatabase();
|
|
333
393
|
cleanup(base);
|
|
@@ -446,9 +446,9 @@ Continue from step 2.
|
|
|
446
446
|
|
|
447
447
|
const state2 = await deriveState(base2);
|
|
448
448
|
|
|
449
|
-
|
|
450
|
-
assert.deepStrictEqual(state2.
|
|
451
|
-
assert.
|
|
449
|
+
// With partial-dep fallback, S01 is picked despite unmet dep on S99
|
|
450
|
+
assert.deepStrictEqual(state2.phase, 'planning', 'blocked-B: phase is planning (fallback picks S01)');
|
|
451
|
+
assert.deepStrictEqual(state2.activeSlice?.id, 'S01', 'blocked-B: activeSlice is S01 via fallback');
|
|
452
452
|
} finally {
|
|
453
453
|
cleanup(base2);
|
|
454
454
|
}
|
|
@@ -691,7 +691,7 @@ describe("transition boundary failures", () => {
|
|
|
691
691
|
);
|
|
692
692
|
});
|
|
693
693
|
|
|
694
|
-
test("blocked state: all slices have unmet deps →
|
|
694
|
+
test("blocked state: all slices have unmet deps → fallback picks slice", async () => {
|
|
695
695
|
base = makeTempDir();
|
|
696
696
|
const mDir = join(base, ".gsd", "milestones", "M001");
|
|
697
697
|
mkdirSync(join(mDir, "slices", "S01", "tasks"), { recursive: true });
|
|
@@ -736,7 +736,9 @@ describe("transition boundary failures", () => {
|
|
|
736
736
|
|
|
737
737
|
invalidateAllCaches();
|
|
738
738
|
const state = await deriveStateFromDb(base);
|
|
739
|
-
|
|
739
|
+
// With partial-dep fallback, circular deps no longer block — fallback picks first eligible slice
|
|
740
|
+
assert.equal(state.phase, "planning", "circular deps: fallback picks a slice instead of blocking");
|
|
741
|
+
assert.ok(state.activeSlice !== null, "activeSlice set via fallback");
|
|
740
742
|
});
|
|
741
743
|
});
|
|
742
744
|
|
|
@@ -811,9 +811,9 @@ describe("state-machine-full-walkthrough", () => {
|
|
|
811
811
|
assert.ok(state.blockers.length > 0, "should have blockers");
|
|
812
812
|
});
|
|
813
813
|
|
|
814
|
-
test("no eligible slice (all deps unmet) →
|
|
814
|
+
test("no eligible slice (all deps unmet) → fallback picks slice with most deps satisfied", async () => {
|
|
815
815
|
const base = createFixtureBase();
|
|
816
|
-
// S01 depends on S00 which doesn't exist
|
|
816
|
+
// S01 depends on S00 which doesn't exist — fallback picks S01 anyway
|
|
817
817
|
writeRoadmap(base, "M001", [
|
|
818
818
|
"# M001: Test Milestone",
|
|
819
819
|
"",
|
|
@@ -827,11 +827,9 @@ describe("state-machine-full-walkthrough", () => {
|
|
|
827
827
|
invalidateStateCache();
|
|
828
828
|
const state = await deriveState(base);
|
|
829
829
|
|
|
830
|
-
|
|
831
|
-
assert.
|
|
832
|
-
|
|
833
|
-
"blockers should mention dependency or eligibility",
|
|
834
|
-
);
|
|
830
|
+
// With partial-dep fallback, S01 is picked despite unmet dep on S00
|
|
831
|
+
assert.equal(state.phase, "planning");
|
|
832
|
+
assert.equal(state.activeSlice?.id, "S01");
|
|
835
833
|
});
|
|
836
834
|
});
|
|
837
835
|
|
|
@@ -263,6 +263,6 @@ test("dispatch: phase skip guards return null (not stop)", () => {
|
|
|
263
263
|
const researchGuard = dispatchSrc.match(/skip_research\).*?return null/s);
|
|
264
264
|
assert.ok(researchGuard, "skip_research guard should return null (fall-through)");
|
|
265
265
|
|
|
266
|
-
const reassessGuard = dispatchSrc.match(/reassess_after_slice
|
|
266
|
+
const reassessGuard = dispatchSrc.match(/reassess_after_slice.*?return null/s);
|
|
267
267
|
assert.ok(reassessGuard, "reassess_after_slice guard should return null (fall-through)");
|
|
268
268
|
});
|
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
// gsd-pi — Regression tests for the validate-milestone stuck-loop guard (#4094)
|
|
2
|
+
|
|
3
|
+
import { describe, test, mock, beforeEach, afterEach } from "node:test";
|
|
4
|
+
import assert from "node:assert/strict";
|
|
5
|
+
import { tmpdir } from "node:os";
|
|
6
|
+
import { mkdirSync, writeFileSync, rmSync } from "node:fs";
|
|
7
|
+
import { join } from "node:path";
|
|
8
|
+
|
|
9
|
+
import { runPostUnitVerification, type VerificationContext } from "../auto-verification.ts";
|
|
10
|
+
import { AutoSession } from "../auto/session.ts";
|
|
11
|
+
import {
|
|
12
|
+
openDatabase,
|
|
13
|
+
closeDatabase,
|
|
14
|
+
insertMilestone,
|
|
15
|
+
insertSlice,
|
|
16
|
+
} from "../gsd-db.ts";
|
|
17
|
+
import { invalidateAllCaches } from "../cache.ts";
|
|
18
|
+
import { _clearGsdRootCache } from "../paths.ts";
|
|
19
|
+
|
|
20
|
+
let tempDir: string;
|
|
21
|
+
let dbPath: string;
|
|
22
|
+
let originalCwd: string;
|
|
23
|
+
|
|
24
|
+
function makeMockCtx() {
|
|
25
|
+
return {
|
|
26
|
+
ui: {
|
|
27
|
+
notify: mock.fn(),
|
|
28
|
+
setStatus: () => {},
|
|
29
|
+
setWidget: () => {},
|
|
30
|
+
setFooter: () => {},
|
|
31
|
+
},
|
|
32
|
+
model: { id: "test-model" },
|
|
33
|
+
} as any;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
function makeMockPi() {
|
|
37
|
+
return {
|
|
38
|
+
sendMessage: mock.fn(),
|
|
39
|
+
setModel: mock.fn(async () => true),
|
|
40
|
+
} as any;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
function makeMockSession(basePath: string, unitType: string, unitId: string): AutoSession {
|
|
44
|
+
const s = new AutoSession();
|
|
45
|
+
s.basePath = basePath;
|
|
46
|
+
s.active = true;
|
|
47
|
+
s.pendingVerificationRetry = null;
|
|
48
|
+
s.currentUnit = { type: unitType, id: unitId, startedAt: Date.now() };
|
|
49
|
+
return s;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
function setupTestEnvironment(): void {
|
|
53
|
+
originalCwd = process.cwd();
|
|
54
|
+
tempDir = join(tmpdir(), `validate-milestone-guard-${Date.now()}-${Math.random().toString(36).slice(2)}`);
|
|
55
|
+
mkdirSync(tempDir, { recursive: true });
|
|
56
|
+
|
|
57
|
+
const milestoneDir = join(tempDir, ".gsd", "milestones", "M001");
|
|
58
|
+
mkdirSync(milestoneDir, { recursive: true });
|
|
59
|
+
|
|
60
|
+
process.chdir(tempDir);
|
|
61
|
+
_clearGsdRootCache();
|
|
62
|
+
|
|
63
|
+
dbPath = join(tempDir, ".gsd", "gsd.db");
|
|
64
|
+
openDatabase(dbPath);
|
|
65
|
+
invalidateAllCaches();
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
function cleanupTestEnvironment(): void {
|
|
69
|
+
try { process.chdir(originalCwd); } catch { /* ignore */ }
|
|
70
|
+
try { closeDatabase(); } catch { /* ignore */ }
|
|
71
|
+
try { rmSync(tempDir, { recursive: true, force: true }); } catch { /* ignore */ }
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
function writeValidationFile(verdict: string): void {
|
|
75
|
+
const path = join(tempDir, ".gsd", "milestones", "M001", "M001-VALIDATION.md");
|
|
76
|
+
const content = `---
|
|
77
|
+
verdict: ${verdict}
|
|
78
|
+
remediation_round: 1
|
|
79
|
+
---
|
|
80
|
+
|
|
81
|
+
# Milestone Validation: M001
|
|
82
|
+
|
|
83
|
+
## Verdict Rationale
|
|
84
|
+
Test fixture
|
|
85
|
+
`;
|
|
86
|
+
writeFileSync(path, content, "utf-8");
|
|
87
|
+
invalidateAllCaches();
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
describe("validate-milestone stuck-loop guard (#4094)", () => {
|
|
91
|
+
beforeEach(() => setupTestEnvironment());
|
|
92
|
+
afterEach(() => cleanupTestEnvironment());
|
|
93
|
+
|
|
94
|
+
test("pauses when verdict=needs-remediation and all slices are closed", async () => {
|
|
95
|
+
insertMilestone({ id: "M001" });
|
|
96
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Slice 1", status: "complete" });
|
|
97
|
+
insertSlice({ id: "S02", milestoneId: "M001", title: "Slice 2", status: "done" });
|
|
98
|
+
writeValidationFile("needs-remediation");
|
|
99
|
+
|
|
100
|
+
const ctx = makeMockCtx();
|
|
101
|
+
const pi = makeMockPi();
|
|
102
|
+
const pauseAutoMock = mock.fn(async () => {});
|
|
103
|
+
const s = makeMockSession(tempDir, "validate-milestone", "M001");
|
|
104
|
+
|
|
105
|
+
const result = await runPostUnitVerification({ s, ctx, pi } as VerificationContext, pauseAutoMock);
|
|
106
|
+
|
|
107
|
+
assert.equal(result, "pause");
|
|
108
|
+
assert.equal(pauseAutoMock.mock.callCount(), 1);
|
|
109
|
+
assert.equal(ctx.ui.notify.mock.callCount(), 1);
|
|
110
|
+
const notifyArgs = ctx.ui.notify.mock.calls[0].arguments;
|
|
111
|
+
assert.match(notifyArgs[0], /needs-remediation/);
|
|
112
|
+
assert.equal(notifyArgs[1], "error");
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
test("treats skipped slices as closed", async () => {
|
|
116
|
+
insertMilestone({ id: "M001" });
|
|
117
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Slice 1", status: "complete" });
|
|
118
|
+
insertSlice({ id: "S02", milestoneId: "M001", title: "Slice 2", status: "skipped" });
|
|
119
|
+
writeValidationFile("needs-remediation");
|
|
120
|
+
|
|
121
|
+
const ctx = makeMockCtx();
|
|
122
|
+
const pi = makeMockPi();
|
|
123
|
+
const pauseAutoMock = mock.fn(async () => {});
|
|
124
|
+
const s = makeMockSession(tempDir, "validate-milestone", "M001");
|
|
125
|
+
|
|
126
|
+
const result = await runPostUnitVerification({ s, ctx, pi } as VerificationContext, pauseAutoMock);
|
|
127
|
+
|
|
128
|
+
assert.equal(result, "pause");
|
|
129
|
+
assert.equal(pauseAutoMock.mock.callCount(), 1);
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
test("continues when verdict=needs-remediation but a queued remediation slice exists", async () => {
|
|
133
|
+
insertMilestone({ id: "M001" });
|
|
134
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Slice 1", status: "complete" });
|
|
135
|
+
insertSlice({ id: "S02", milestoneId: "M001", title: "Remediation", status: "queued" });
|
|
136
|
+
writeValidationFile("needs-remediation");
|
|
137
|
+
|
|
138
|
+
const ctx = makeMockCtx();
|
|
139
|
+
const pi = makeMockPi();
|
|
140
|
+
const pauseAutoMock = mock.fn(async () => {});
|
|
141
|
+
const s = makeMockSession(tempDir, "validate-milestone", "M001");
|
|
142
|
+
|
|
143
|
+
const result = await runPostUnitVerification({ s, ctx, pi } as VerificationContext, pauseAutoMock);
|
|
144
|
+
|
|
145
|
+
assert.equal(result, "continue");
|
|
146
|
+
assert.equal(pauseAutoMock.mock.callCount(), 0);
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
test("continues when verdict is pass", async () => {
|
|
150
|
+
insertMilestone({ id: "M001" });
|
|
151
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Slice 1", status: "complete" });
|
|
152
|
+
writeValidationFile("pass");
|
|
153
|
+
|
|
154
|
+
const ctx = makeMockCtx();
|
|
155
|
+
const pi = makeMockPi();
|
|
156
|
+
const pauseAutoMock = mock.fn(async () => {});
|
|
157
|
+
const s = makeMockSession(tempDir, "validate-milestone", "M001");
|
|
158
|
+
|
|
159
|
+
const result = await runPostUnitVerification({ s, ctx, pi } as VerificationContext, pauseAutoMock);
|
|
160
|
+
|
|
161
|
+
assert.equal(result, "continue");
|
|
162
|
+
assert.equal(pauseAutoMock.mock.callCount(), 0);
|
|
163
|
+
});
|
|
164
|
+
|
|
165
|
+
test("continues when no VALIDATION file exists yet", async () => {
|
|
166
|
+
insertMilestone({ id: "M001" });
|
|
167
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Slice 1", status: "complete" });
|
|
168
|
+
|
|
169
|
+
const ctx = makeMockCtx();
|
|
170
|
+
const pi = makeMockPi();
|
|
171
|
+
const pauseAutoMock = mock.fn(async () => {});
|
|
172
|
+
const s = makeMockSession(tempDir, "validate-milestone", "M001");
|
|
173
|
+
|
|
174
|
+
const result = await runPostUnitVerification({ s, ctx, pi } as VerificationContext, pauseAutoMock);
|
|
175
|
+
|
|
176
|
+
assert.equal(result, "continue");
|
|
177
|
+
assert.equal(pauseAutoMock.mock.callCount(), 0);
|
|
178
|
+
});
|
|
179
|
+
});
|
|
File without changes
|
|
File without changes
|