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.
Files changed (126) hide show
  1. package/dist/cli-web-branch.d.ts +4 -3
  2. package/dist/cli-web-branch.js +10 -7
  3. package/dist/cli.js +99 -206
  4. package/dist/logo.d.ts +1 -1
  5. package/dist/logo.js +1 -1
  6. package/dist/onboarding.js +59 -53
  7. package/dist/resource-loader.js +2 -2
  8. package/dist/resources/extensions/gsd/auto/phases.js +15 -9
  9. package/dist/resources/extensions/gsd/auto-dispatch.js +11 -3
  10. package/dist/resources/extensions/gsd/auto-post-unit.js +41 -1
  11. package/dist/resources/extensions/gsd/auto-start.js +3 -0
  12. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +13 -0
  13. package/dist/resources/extensions/gsd/auto-verification.js +88 -3
  14. package/dist/resources/extensions/gsd/auto.js +29 -8
  15. package/dist/resources/extensions/gsd/commands-handlers.js +8 -2
  16. package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  17. package/dist/resources/extensions/gsd/notification-widget.js +2 -2
  18. package/dist/resources/extensions/gsd/state.js +61 -14
  19. package/dist/update-check.d.ts +1 -0
  20. package/dist/update-check.js +13 -5
  21. package/dist/update-cmd.js +4 -3
  22. package/dist/web/standalone/.next/BUILD_ID +1 -1
  23. package/dist/web/standalone/.next/app-path-routes-manifest.json +14 -14
  24. package/dist/web/standalone/.next/build-manifest.json +2 -2
  25. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  26. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  27. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  28. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  29. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  30. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  31. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  32. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  33. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  34. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  35. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  36. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  37. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  38. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  39. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  40. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  41. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  42. package/dist/web/standalone/.next/server/app/index.html +1 -1
  43. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  44. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  45. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  46. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  47. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  48. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  49. package/dist/web/standalone/.next/server/app-paths-manifest.json +14 -14
  50. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  51. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  52. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  53. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  54. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  55. package/package.json +1 -2
  56. package/packages/pi-ai/dist/utils/overflow.d.ts.map +1 -1
  57. package/packages/pi-ai/dist/utils/overflow.js +12 -0
  58. package/packages/pi-ai/dist/utils/overflow.js.map +1 -1
  59. package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts +2 -0
  60. package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts.map +1 -0
  61. package/packages/pi-ai/dist/utils/tests/overflow.test.js +50 -0
  62. package/packages/pi-ai/dist/utils/tests/overflow.test.js.map +1 -0
  63. package/packages/pi-ai/src/utils/overflow.ts +14 -1
  64. package/packages/pi-ai/src/utils/tests/overflow.test.ts +58 -0
  65. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +138 -0
  66. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  67. package/packages/pi-coding-agent/dist/core/compaction/utils.js +5 -5
  68. package/packages/pi-coding-agent/dist/core/compaction/utils.js.map +1 -1
  69. package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts +2 -0
  70. package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts.map +1 -0
  71. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js +45 -0
  72. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js.map +1 -0
  73. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts +2 -1
  74. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -1
  75. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js +9 -3
  76. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js.map +1 -1
  77. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts +2 -0
  78. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts.map +1 -0
  79. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +52 -0
  80. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -0
  81. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  82. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +21 -4
  83. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  84. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  85. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +11 -3
  86. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  87. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +157 -0
  88. package/packages/pi-coding-agent/src/core/compaction/utils.ts +5 -5
  89. package/packages/pi-coding-agent/src/core/compaction-utils.test.ts +50 -0
  90. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +73 -0
  91. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.ts +9 -3
  92. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +21 -4
  93. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +11 -3
  94. package/packages/pi-tui/dist/__tests__/tui.test.js +60 -1
  95. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  96. package/packages/pi-tui/dist/tui.d.ts +8 -0
  97. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  98. package/packages/pi-tui/dist/tui.js +32 -3
  99. package/packages/pi-tui/dist/tui.js.map +1 -1
  100. package/packages/pi-tui/src/__tests__/tui.test.ts +76 -1
  101. package/packages/pi-tui/src/tui.ts +31 -3
  102. package/src/resources/extensions/gsd/auto/phases.ts +22 -9
  103. package/src/resources/extensions/gsd/auto-dispatch.ts +10 -4
  104. package/src/resources/extensions/gsd/auto-post-unit.ts +47 -1
  105. package/src/resources/extensions/gsd/auto-start.ts +3 -0
  106. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +17 -0
  107. package/src/resources/extensions/gsd/auto-verification.ts +98 -3
  108. package/src/resources/extensions/gsd/auto.ts +31 -14
  109. package/src/resources/extensions/gsd/commands-handlers.ts +8 -2
  110. package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  111. package/src/resources/extensions/gsd/notification-widget.ts +2 -2
  112. package/src/resources/extensions/gsd/state.ts +71 -15
  113. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +2 -2
  114. package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +53 -0
  115. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +142 -0
  116. package/src/resources/extensions/gsd/tests/completed-at-reconcile.test.ts +42 -0
  117. package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -2
  118. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +3 -2
  119. package/src/resources/extensions/gsd/tests/derive-state-helpers.test.ts +68 -8
  120. package/src/resources/extensions/gsd/tests/derive-state.test.ts +3 -3
  121. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +4 -2
  122. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +5 -7
  123. package/src/resources/extensions/gsd/tests/token-profile.test.ts +1 -1
  124. package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +179 -0
  125. /package/dist/web/standalone/.next/static/{r6AvNu-aMwn4nwqjHqAfw → ASJ2RGD7E1iiUYzA0xT2i}/_buildManifest.js +0 -0
  126. /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
- assert.deepStrictEqual(dbState.phase, 'blocked', 'E-blocked: phase is blocked');
355
- assert.ok(dbState.blockers.length > 0, 'E-blocked: has blockers');
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
- assert.deepStrictEqual(dbState.phase, 'blocked', 'blocked-db: phase is blocked');
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.blockers.length > 0, 'blocked-db: has blockers');
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: SUMMARY-on-disk marks complete ───────────
311
- test('buildCompletenessSet: milestone with SUMMARY on disk treated as complete', async () => {
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: 'active' });
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', 'summary-disk: M001 marked complete via disk SUMMARY');
330
- assert.equal(state.activeMilestone?.id, 'M002', 'summary-disk: M002 is active');
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
- assert.deepStrictEqual(state2.phase, 'blocked', 'blocked-B: phase is blocked');
450
- assert.deepStrictEqual(state2.activeSlice, null, 'blocked-B: activeSlice is null');
451
- assert.ok(state2.blockers.length > 0, 'blocked-B: blockers array is non-empty');
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 → blocked phase", async () => {
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
- assert.equal(state.phase, "blocked", "circular deps should produce blocked phase");
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) → blocked at slice level", async () => {
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
- assert.equal(state.phase, "blocked");
831
- assert.ok(
832
- state.blockers.some(b => b.includes("dependency") || b.includes("eligible")),
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\).*?return null/s);
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
+ });