gsd-pi 2.80.0-dev.e146beb20 → 2.80.0-dev.e6c48c3af

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 (218) hide show
  1. package/README.md +4 -2
  2. package/dist/resources/.managed-resources-content-hash +1 -1
  3. package/dist/resources/extensions/gsd/auto/phases.js +59 -21
  4. package/dist/resources/extensions/gsd/auto/resolve.js +17 -0
  5. package/dist/resources/extensions/gsd/auto/run-unit.js +17 -2
  6. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +1 -1
  7. package/dist/resources/extensions/gsd/auto-prompts.js +13 -1
  8. package/dist/resources/extensions/gsd/auto-recovery.js +43 -1
  9. package/dist/resources/extensions/gsd/auto-supervisor.js +8 -1
  10. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +2 -2
  11. package/dist/resources/extensions/gsd/auto.js +84 -5
  12. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +21 -2
  13. package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +27 -20
  14. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +75 -4
  15. package/dist/resources/extensions/gsd/clean-root-preflight.js +24 -6
  16. package/dist/resources/extensions/gsd/context-budget.js +37 -2
  17. package/dist/resources/extensions/gsd/db/unit-dispatches.js +39 -0
  18. package/dist/resources/extensions/gsd/db-base-schema.js +4 -2
  19. package/dist/resources/extensions/gsd/db-migration-steps.js +6 -0
  20. package/dist/resources/extensions/gsd/git-service.js +36 -4
  21. package/dist/resources/extensions/gsd/gsd-db.js +46 -13
  22. package/dist/resources/extensions/gsd/guided-flow.js +33 -4
  23. package/dist/resources/extensions/gsd/memory-store.js +69 -12
  24. package/dist/resources/extensions/gsd/migrate/command.js +40 -1
  25. package/dist/resources/extensions/gsd/migration-auto-check.js +87 -0
  26. package/dist/resources/extensions/gsd/pre-execution-checks.js +7 -0
  27. package/dist/resources/extensions/gsd/prompt-loader.js +28 -2
  28. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +16 -13
  29. package/dist/resources/extensions/gsd/prompts/parallel-research-slices.md +1 -1
  30. package/dist/resources/extensions/gsd/prompts/quick-task.md +1 -5
  31. package/dist/resources/extensions/gsd/prompts/validate-milestone.md +2 -2
  32. package/dist/resources/extensions/gsd/quick.js +34 -2
  33. package/dist/resources/extensions/gsd/tools/context-mode-tool-result.js +15 -0
  34. package/dist/resources/extensions/gsd/tools/exec-search-tool.js +5 -0
  35. package/dist/resources/extensions/gsd/tools/exec-tool.js +3 -15
  36. package/dist/resources/extensions/gsd/tools/memory-tools.js +1 -0
  37. package/dist/resources/extensions/gsd/tools/resume-tool.js +5 -0
  38. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +1 -1
  39. package/dist/resources/extensions/gsd/unit-context-composer.js +12 -3
  40. package/dist/resources/extensions/gsd/unit-runtime.js +11 -0
  41. package/dist/resources/extensions/gsd/worktree-resolver.js +33 -17
  42. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  43. package/dist/web/standalone/.next/BUILD_ID +1 -1
  44. package/dist/web/standalone/.next/app-path-routes-manifest.json +16 -16
  45. package/dist/web/standalone/.next/build-manifest.json +2 -2
  46. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  47. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  48. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  49. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  50. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  51. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  52. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  53. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  54. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  55. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  56. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  57. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  58. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  59. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  60. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  61. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  62. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  63. package/dist/web/standalone/.next/server/app/index.html +1 -1
  64. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  65. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  66. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  67. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  68. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  69. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  70. package/dist/web/standalone/.next/server/app-paths-manifest.json +16 -16
  71. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  72. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  73. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  74. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  75. package/package.json +3 -3
  76. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  77. package/packages/mcp-server/dist/workflow-tools.js +22 -17
  78. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  79. package/packages/mcp-server/src/workflow-tools.test.ts +75 -2
  80. package/packages/mcp-server/src/workflow-tools.ts +30 -16
  81. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  82. package/packages/native/tsconfig.tsbuildinfo +1 -1
  83. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +32 -0
  84. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
  85. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js +15 -0
  86. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js.map +1 -1
  87. package/packages/pi-coding-agent/dist/core/agent-session.d.ts +2 -0
  88. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  89. package/packages/pi-coding-agent/dist/core/agent-session.js +12 -3
  90. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  91. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +3 -1
  92. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  93. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts +11 -0
  94. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts.map +1 -1
  95. package/packages/pi-coding-agent/dist/core/compaction/compaction.js +9 -0
  96. package/packages/pi-coding-agent/dist/core/compaction/compaction.js.map +1 -1
  97. package/packages/pi-coding-agent/dist/core/compaction-threshold.test.d.ts +2 -0
  98. package/packages/pi-coding-agent/dist/core/compaction-threshold.test.d.ts.map +1 -0
  99. package/packages/pi-coding-agent/dist/core/compaction-threshold.test.js +103 -0
  100. package/packages/pi-coding-agent/dist/core/compaction-threshold.test.js.map +1 -0
  101. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +3 -0
  102. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
  103. package/packages/pi-coding-agent/dist/core/extensions/runner.js +3 -0
  104. package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
  105. package/packages/pi-coding-agent/dist/core/extensions/runner.test.js +2 -0
  106. package/packages/pi-coding-agent/dist/core/extensions/runner.test.js.map +1 -1
  107. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +12 -0
  108. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
  109. package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
  110. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +20 -0
  111. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
  112. package/packages/pi-coding-agent/dist/core/settings-manager.js +25 -0
  113. package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
  114. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +1 -0
  115. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  116. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +3 -0
  117. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  118. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  119. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +13 -5
  120. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  121. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.test.js +53 -0
  122. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.test.js.map +1 -1
  123. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  124. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +3 -0
  125. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  126. package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +36 -0
  127. package/packages/pi-coding-agent/src/core/agent-session-tool-refresh.test.ts +18 -0
  128. package/packages/pi-coding-agent/src/core/agent-session.ts +14 -3
  129. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +3 -1
  130. package/packages/pi-coding-agent/src/core/compaction/compaction.ts +18 -0
  131. package/packages/pi-coding-agent/src/core/compaction-threshold.test.ts +121 -0
  132. package/packages/pi-coding-agent/src/core/extensions/runner.test.ts +2 -0
  133. package/packages/pi-coding-agent/src/core/extensions/runner.ts +5 -0
  134. package/packages/pi-coding-agent/src/core/extensions/types.ts +12 -0
  135. package/packages/pi-coding-agent/src/core/settings-manager.ts +39 -1
  136. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +4 -0
  137. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.test.ts +56 -0
  138. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +22 -7
  139. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +3 -0
  140. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  141. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  142. package/packages/pi-tui/dist/tui.js +18 -8
  143. package/packages/pi-tui/dist/tui.js.map +1 -1
  144. package/packages/pi-tui/src/tui.ts +20 -8
  145. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  146. package/src/resources/extensions/gsd/auto/loop-deps.ts +2 -2
  147. package/src/resources/extensions/gsd/auto/phases.ts +85 -35
  148. package/src/resources/extensions/gsd/auto/resolve.ts +23 -1
  149. package/src/resources/extensions/gsd/auto/run-unit.ts +22 -2
  150. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +1 -1
  151. package/src/resources/extensions/gsd/auto-prompts.ts +17 -1
  152. package/src/resources/extensions/gsd/auto-recovery.ts +54 -0
  153. package/src/resources/extensions/gsd/auto-supervisor.ts +7 -0
  154. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +2 -2
  155. package/src/resources/extensions/gsd/auto.ts +96 -4
  156. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +21 -1
  157. package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +27 -19
  158. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +88 -4
  159. package/src/resources/extensions/gsd/clean-root-preflight.ts +32 -7
  160. package/src/resources/extensions/gsd/context-budget.ts +44 -2
  161. package/src/resources/extensions/gsd/db/unit-dispatches.ts +41 -0
  162. package/src/resources/extensions/gsd/db-base-schema.ts +4 -2
  163. package/src/resources/extensions/gsd/db-migration-steps.ts +8 -0
  164. package/src/resources/extensions/gsd/git-service.ts +46 -8
  165. package/src/resources/extensions/gsd/gsd-db.ts +50 -13
  166. package/src/resources/extensions/gsd/guided-flow.ts +49 -4
  167. package/src/resources/extensions/gsd/memory-store.ts +77 -12
  168. package/src/resources/extensions/gsd/migrate/command.ts +47 -1
  169. package/src/resources/extensions/gsd/migration-auto-check.ts +129 -0
  170. package/src/resources/extensions/gsd/pre-execution-checks.ts +7 -0
  171. package/src/resources/extensions/gsd/preferences-types.ts +1 -1
  172. package/src/resources/extensions/gsd/prompt-loader.ts +27 -2
  173. package/src/resources/extensions/gsd/prompts/complete-milestone.md +16 -13
  174. package/src/resources/extensions/gsd/prompts/parallel-research-slices.md +1 -1
  175. package/src/resources/extensions/gsd/prompts/quick-task.md +1 -5
  176. package/src/resources/extensions/gsd/prompts/validate-milestone.md +2 -2
  177. package/src/resources/extensions/gsd/quick.ts +37 -2
  178. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +215 -1
  179. package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +56 -13
  180. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +14 -1
  181. package/src/resources/extensions/gsd/tests/auto-wrapup-inflight-guard.test.ts +166 -4
  182. package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +15 -6
  183. package/src/resources/extensions/gsd/tests/compaction-snapshot.test.ts +14 -1
  184. package/src/resources/extensions/gsd/tests/context-budget.test.ts +10 -1
  185. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +5 -1
  186. package/src/resources/extensions/gsd/tests/dispatch-rule-coverage.test.ts +313 -0
  187. package/src/resources/extensions/gsd/tests/exec-history.test.ts +15 -0
  188. package/src/resources/extensions/gsd/tests/exec-sandbox.test.ts +65 -0
  189. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +54 -0
  190. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +239 -1
  191. package/src/resources/extensions/gsd/tests/memory-decay-factor.test.ts +90 -0
  192. package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +48 -0
  193. package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +127 -0
  194. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +38 -0
  195. package/src/resources/extensions/gsd/tests/prompt-path-audit.test.ts +40 -0
  196. package/src/resources/extensions/gsd/tests/prompt-step-ordering.test.ts +19 -0
  197. package/src/resources/extensions/gsd/tests/quick-external-gsd.test.ts +40 -0
  198. package/src/resources/extensions/gsd/tests/schema-v27-v28-sequence.test.ts +156 -0
  199. package/src/resources/extensions/gsd/tests/signal-handlers.test.ts +27 -0
  200. package/src/resources/extensions/gsd/tests/stalled-tool-recovery.test.ts +49 -1
  201. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +55 -0
  202. package/src/resources/extensions/gsd/tests/status-db-open.test.ts +9 -0
  203. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +136 -4
  204. package/src/resources/extensions/gsd/tests/unit-dispatches.test.ts +30 -0
  205. package/src/resources/extensions/gsd/tests/unit-runtime.test.ts +30 -0
  206. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +3 -0
  207. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +63 -1
  208. package/src/resources/extensions/gsd/tools/context-mode-tool-result.ts +25 -0
  209. package/src/resources/extensions/gsd/tools/exec-search-tool.ts +7 -7
  210. package/src/resources/extensions/gsd/tools/exec-tool.ts +4 -23
  211. package/src/resources/extensions/gsd/tools/memory-tools.ts +1 -0
  212. package/src/resources/extensions/gsd/tools/resume-tool.ts +7 -7
  213. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +1 -1
  214. package/src/resources/extensions/gsd/unit-context-composer.ts +19 -4
  215. package/src/resources/extensions/gsd/unit-runtime.ts +11 -0
  216. package/src/resources/extensions/gsd/worktree-resolver.ts +36 -15
  217. /package/dist/web/standalone/.next/static/{y73quA-XdLo9n41nxphjW → 4dQ9NTZJ8pEvFwBgDUX93}/_buildManifest.js +0 -0
  218. /package/dist/web/standalone/.next/static/{y73quA-XdLo9n41nxphjW → 4dQ9NTZJ8pEvFwBgDUX93}/_ssgManifest.js +0 -0
@@ -85,7 +85,11 @@ function makeMockDeps(
85
85
  resolveMilestoneFile: () => null,
86
86
  reconcileMergeState: () => "clean",
87
87
  preflightCleanRoot: () => ({ stashPushed: false, summary: "" }),
88
- postflightPopStash: () => {},
88
+ postflightPopStash: () => ({
89
+ restored: true,
90
+ needsManualRecovery: false,
91
+ message: "restored",
92
+ }),
89
93
  getLedger: () => ({ units: [] }),
90
94
  getProjectTotals: () => ({ cost: 0 }),
91
95
  formatCost: (c: number) => `$${c.toFixed(2)}`,
@@ -399,6 +403,240 @@ test("runDispatch pauses when complete-milestone summary exists on disk but the
399
403
  assert.equal(stopCalls, 0, "mismatch pause should not hard-stop the loop");
400
404
  });
401
405
 
406
+ test("runDispatch pauses when execute-task artifacts exist but DB status is still open", async (t) => {
407
+ const capture = createEventCapture();
408
+ let pauseCalls = 0;
409
+ let stopCalls = 0;
410
+ let invalidateCalls = 0;
411
+ const base = join(tmpdir(), `gsd-stuck-execute-task-${randomUUID()}`);
412
+ t.after(() => {
413
+ closeDatabase();
414
+ rmSync(base, { recursive: true, force: true });
415
+ });
416
+
417
+ const sliceDir = join(base, ".gsd", "milestones", "M001", "slices", "S01");
418
+ const tasksDir = join(sliceDir, "tasks");
419
+ mkdirSync(tasksDir, { recursive: true });
420
+ openDatabase(join(base, ".gsd", "gsd.db"));
421
+ insertMilestone({ id: "M001", title: "Test", status: "active" });
422
+ insertSlice({ id: "S01", milestoneId: "M001", title: "Slice", status: "in_progress" });
423
+ insertTask({ id: "T01", milestoneId: "M001", sliceId: "S01", title: "First task", status: "pending" });
424
+ writeFileSync(
425
+ join(sliceDir, "S01-PLAN.md"),
426
+ [
427
+ "# S01",
428
+ "",
429
+ "## Tasks",
430
+ "",
431
+ "- [x] **T01: First task** `est:1h`",
432
+ "",
433
+ ].join("\n"),
434
+ );
435
+ writeFileSync(join(tasksDir, "T01-PLAN.md"), "# T01 Plan\n");
436
+ writeFileSync(join(tasksDir, "T01-SUMMARY.md"), "# T01 Summary\n\nDone on disk.\n");
437
+
438
+ const deps = makeMockDeps(capture, {
439
+ pauseAuto: async () => { pauseCalls++; },
440
+ stopAuto: async () => { stopCalls++; },
441
+ invalidateAllCaches: () => { invalidateCalls++; },
442
+ resolveDispatch: async () => ({
443
+ action: "dispatch" as const,
444
+ unitType: "execute-task",
445
+ unitId: "M001/S01/T01",
446
+ prompt: "execute the task",
447
+ matchedRule: "executing → execute-task",
448
+ }),
449
+ });
450
+ const ic = makeIC(deps, {
451
+ s: {
452
+ ...makeSession(),
453
+ basePath: base,
454
+ originalBasePath: base,
455
+ } as any,
456
+ });
457
+ const preData: PreDispatchData = {
458
+ state: {
459
+ phase: "executing",
460
+ activeMilestone: { id: "M001", title: "Test", status: "active" },
461
+ activeSlice: { id: "S01", title: "Slice" },
462
+ activeTask: { id: "T01", title: "First task" },
463
+ registry: [{ id: "M001", status: "active" }],
464
+ blockers: [],
465
+ } as any,
466
+ mid: "M001",
467
+ midTitle: "Test Milestone",
468
+ };
469
+ const loopState: LoopState = {
470
+ recentUnits: [
471
+ { key: "execute-task/M001/S01/T01" },
472
+ { key: "execute-task/M001/S01/T01" },
473
+ ],
474
+ stuckRecoveryAttempts: 0,
475
+ consecutiveFinalizeTimeouts: 0,
476
+ };
477
+
478
+ const result = await runDispatch(ic, preData, loopState);
479
+
480
+ assert.equal(result.action, "break");
481
+ assert.equal((result as any).reason, "execute-task-artifact-db-mismatch");
482
+ assert.equal(pauseCalls, 1, "execute-task disk/db mismatch should pause auto-mode");
483
+ assert.equal(stopCalls, 0, "execute-task disk/db mismatch should not hard-stop the loop");
484
+ assert.equal(invalidateCalls, 0, "mismatch should not clear caches and continue toward redispatch");
485
+ assert.equal(loopState.recentUnits.length, 3, "mismatch should keep the stuck window intact");
486
+ assert.equal(loopState.stuckRecoveryAttempts, 1, "mismatch should not reset the recovery counter");
487
+ });
488
+
489
+ test("runDispatch pauses at Level 2 when execute-task artifacts exist but DB status is still open", async (t) => {
490
+ const capture = createEventCapture();
491
+ let pauseCalls = 0;
492
+ let stopCalls = 0;
493
+ let invalidateCalls = 0;
494
+ const base = join(tmpdir(), `gsd-stuck-execute-task-l2-${randomUUID()}`);
495
+ t.after(() => {
496
+ closeDatabase();
497
+ rmSync(base, { recursive: true, force: true });
498
+ });
499
+
500
+ const sliceDir = join(base, ".gsd", "milestones", "M001", "slices", "S01");
501
+ const tasksDir = join(sliceDir, "tasks");
502
+ mkdirSync(tasksDir, { recursive: true });
503
+ openDatabase(join(base, ".gsd", "gsd.db"));
504
+ insertMilestone({ id: "M001", title: "Test", status: "active" });
505
+ insertSlice({ id: "S01", milestoneId: "M001", title: "Slice", status: "in_progress" });
506
+ insertTask({ id: "T01", milestoneId: "M001", sliceId: "S01", title: "First task", status: "pending" });
507
+ writeFileSync(
508
+ join(sliceDir, "S01-PLAN.md"),
509
+ "# S01\n\n## Tasks\n\n- [x] **T01: First task** `est:1h`\n",
510
+ );
511
+ writeFileSync(join(tasksDir, "T01-PLAN.md"), "# T01 Plan\n");
512
+ writeFileSync(join(tasksDir, "T01-SUMMARY.md"), "# T01 Summary\n\nDone on disk.\n");
513
+
514
+ const deps = makeMockDeps(capture, {
515
+ pauseAuto: async () => { pauseCalls++; },
516
+ stopAuto: async () => { stopCalls++; },
517
+ invalidateAllCaches: () => { invalidateCalls++; },
518
+ resolveDispatch: async () => ({
519
+ action: "dispatch" as const,
520
+ unitType: "execute-task",
521
+ unitId: "M001/S01/T01",
522
+ prompt: "execute the task",
523
+ matchedRule: "executing execute-task",
524
+ }),
525
+ });
526
+ const ic = makeIC(deps, {
527
+ s: {
528
+ ...makeSession(),
529
+ basePath: base,
530
+ originalBasePath: base,
531
+ } as any,
532
+ });
533
+ const preData: PreDispatchData = {
534
+ state: {
535
+ phase: "executing",
536
+ activeMilestone: { id: "M001", title: "Test", status: "active" },
537
+ activeSlice: { id: "S01", title: "Slice" },
538
+ activeTask: { id: "T01", title: "First task" },
539
+ registry: [{ id: "M001", status: "active" }],
540
+ blockers: [],
541
+ } as any,
542
+ mid: "M001",
543
+ midTitle: "Test Milestone",
544
+ };
545
+ const loopState: LoopState = {
546
+ recentUnits: [
547
+ { key: "execute-task/M001/S01/T01" },
548
+ { key: "execute-task/M001/S01/T01" },
549
+ ],
550
+ stuckRecoveryAttempts: 1,
551
+ consecutiveFinalizeTimeouts: 0,
552
+ };
553
+
554
+ const result = await runDispatch(ic, preData, loopState);
555
+
556
+ assert.equal(result.action, "break");
557
+ assert.equal((result as any).reason, "execute-task-artifact-db-mismatch");
558
+ assert.equal(pauseCalls, 1, "Level 2 execute-task disk/db mismatch should pause auto-mode");
559
+ assert.equal(stopCalls, 0, "Level 2 execute-task disk/db mismatch should not hard-stop the loop");
560
+ assert.equal(invalidateCalls, 1, "Level 2 should invalidate caches before the final artifact recheck");
561
+ assert.equal(loopState.recentUnits.length, 3, "Level 2 mismatch should keep the stuck window intact");
562
+ assert.equal(loopState.stuckRecoveryAttempts, 1, "Level 2 mismatch should not reset the recovery counter");
563
+ });
564
+
565
+ test("runDispatch clears execute-task stuck state when artifacts and DB status are complete", async (t) => {
566
+ const capture = createEventCapture();
567
+ let pauseCalls = 0;
568
+ let stopCalls = 0;
569
+ let invalidateCalls = 0;
570
+ const base = join(tmpdir(), `gsd-stuck-execute-task-complete-${randomUUID()}`);
571
+ t.after(() => {
572
+ closeDatabase();
573
+ rmSync(base, { recursive: true, force: true });
574
+ });
575
+
576
+ const sliceDir = join(base, ".gsd", "milestones", "M001", "slices", "S01");
577
+ const tasksDir = join(sliceDir, "tasks");
578
+ mkdirSync(tasksDir, { recursive: true });
579
+ openDatabase(join(base, ".gsd", "gsd.db"));
580
+ insertMilestone({ id: "M001", title: "Test", status: "active" });
581
+ insertSlice({ id: "S01", milestoneId: "M001", title: "Slice", status: "in_progress" });
582
+ insertTask({ id: "T01", milestoneId: "M001", sliceId: "S01", title: "First task", status: "complete" });
583
+ writeFileSync(
584
+ join(sliceDir, "S01-PLAN.md"),
585
+ "# S01\n\n## Tasks\n\n- [x] **T01: First task** `est:1h`\n",
586
+ );
587
+ writeFileSync(join(tasksDir, "T01-PLAN.md"), "# T01 Plan\n");
588
+ writeFileSync(join(tasksDir, "T01-SUMMARY.md"), "# T01 Summary\n\nDone on disk.\n");
589
+
590
+ const deps = makeMockDeps(capture, {
591
+ pauseAuto: async () => { pauseCalls++; },
592
+ stopAuto: async () => { stopCalls++; },
593
+ invalidateAllCaches: () => { invalidateCalls++; },
594
+ resolveDispatch: async () => ({
595
+ action: "dispatch" as const,
596
+ unitType: "execute-task",
597
+ unitId: "M001/S01/T01",
598
+ prompt: "execute the task",
599
+ matchedRule: "executing execute-task",
600
+ }),
601
+ });
602
+ const ic = makeIC(deps, {
603
+ s: {
604
+ ...makeSession(),
605
+ basePath: base,
606
+ originalBasePath: base,
607
+ } as any,
608
+ });
609
+ const preData: PreDispatchData = {
610
+ state: {
611
+ phase: "executing",
612
+ activeMilestone: { id: "M001", title: "Test", status: "active" },
613
+ activeSlice: { id: "S01", title: "Slice" },
614
+ activeTask: { id: "T01", title: "First task" },
615
+ registry: [{ id: "M001", status: "active" }],
616
+ blockers: [],
617
+ } as any,
618
+ mid: "M001",
619
+ midTitle: "Test Milestone",
620
+ };
621
+ const loopState: LoopState = {
622
+ recentUnits: [
623
+ { key: "execute-task/M001/S01/T01" },
624
+ { key: "execute-task/M001/S01/T01" },
625
+ ],
626
+ stuckRecoveryAttempts: 0,
627
+ consecutiveFinalizeTimeouts: 0,
628
+ };
629
+
630
+ const result = await runDispatch(ic, preData, loopState);
631
+
632
+ assert.equal(result.action, "continue");
633
+ assert.equal(pauseCalls, 0, "closed DB task should not pause auto-mode");
634
+ assert.equal(stopCalls, 0, "closed DB task should not hard-stop the loop");
635
+ assert.equal(invalidateCalls, 1, "closed DB task recovery should invalidate caches once");
636
+ assert.deepEqual(loopState.recentUnits, [], "closed DB task recovery should clear the stuck window");
637
+ assert.equal(loopState.stuckRecoveryAttempts, 0, "closed DB task recovery should reset the recovery counter");
638
+ });
639
+
402
640
  test("runDispatch clears stuck state after Level 1 artifact recovery", async (t) => {
403
641
  const capture = createEventCapture();
404
642
  let invalidateCalls = 0;
@@ -0,0 +1,90 @@
1
+ // gsd-2 / memoryDecayFactor unit tests
2
+ //
3
+ // Pure-function boundary tests for the V28 time-decay scoring helper.
4
+ // The function maps last_hit_at → multiplier in [0.7, 1.0] used by
5
+ // queryMemoriesRanked to down-weight stale memories without fully suppressing
6
+ // them. These tests pin the contract:
7
+ //
8
+ // - null / invalid / future timestamps → 1.0 (no decay penalty)
9
+ // - 0 days ago → 1.0
10
+ // - linear decay between 0 and 90 days
11
+ // - 90+ days ago → 0.7 floor
12
+
13
+ import test from "node:test";
14
+ import assert from "node:assert/strict";
15
+
16
+ import { memoryDecayFactor } from "../memory-store.ts";
17
+
18
+ const DAY_MS = 86_400_000;
19
+
20
+ function isoDaysAgo(days: number): string {
21
+ return new Date(Date.now() - days * DAY_MS).toISOString();
22
+ }
23
+
24
+ test("memoryDecayFactor: null lastHitAt returns 1.0 (never-hit = no decay)", () => {
25
+ assert.equal(memoryDecayFactor(null), 1.0);
26
+ });
27
+
28
+ test("memoryDecayFactor: invalid timestamp string returns 1.0 (defensive)", () => {
29
+ assert.equal(memoryDecayFactor("not-a-date"), 1.0);
30
+ assert.equal(memoryDecayFactor(""), 1.0);
31
+ });
32
+
33
+ test("memoryDecayFactor: future timestamp clamps to daysAgo=0 → 1.0", () => {
34
+ // Clock skew or manual DB edits can yield future last_hit_at values.
35
+ // The factor must stay within [0.7, 1.0] regardless.
36
+ const future = new Date(Date.now() + 30 * DAY_MS).toISOString();
37
+ const factor = memoryDecayFactor(future);
38
+ assert.ok(factor <= 1.0, `factor must not exceed 1.0, got ${factor}`);
39
+ assert.ok(factor >= 0.7, `factor must not fall below 0.7, got ${factor}`);
40
+ assert.equal(factor, 1.0);
41
+ });
42
+
43
+ test("memoryDecayFactor: 0 days ago returns 1.0", () => {
44
+ const factor = memoryDecayFactor(new Date().toISOString());
45
+ // Tiny clock drift between now-string and Date.now() inside the function;
46
+ // assert it's effectively 1.0 within float tolerance.
47
+ assert.ok(Math.abs(factor - 1.0) < 1e-6, `expected ≈1.0, got ${factor}`);
48
+ });
49
+
50
+ test("memoryDecayFactor: 30 days ago returns ~0.90 (linear midpoint)", () => {
51
+ // Formula: 1.0 - 0.3 * (30/90) = 1.0 - 0.1 = 0.90
52
+ const factor = memoryDecayFactor(isoDaysAgo(30));
53
+ assert.ok(Math.abs(factor - 0.90) < 1e-3, `expected ≈0.90, got ${factor}`);
54
+ });
55
+
56
+ test("memoryDecayFactor: 60 days ago returns ~0.80", () => {
57
+ // Formula: 1.0 - 0.3 * (60/90) = 1.0 - 0.2 = 0.80
58
+ const factor = memoryDecayFactor(isoDaysAgo(60));
59
+ assert.ok(Math.abs(factor - 0.80) < 1e-3, `expected ≈0.80, got ${factor}`);
60
+ });
61
+
62
+ test("memoryDecayFactor: 90 days ago returns 0.70 (floor)", () => {
63
+ const factor = memoryDecayFactor(isoDaysAgo(90));
64
+ assert.ok(Math.abs(factor - 0.70) < 1e-3, `expected ≈0.70, got ${factor}`);
65
+ });
66
+
67
+ test("memoryDecayFactor: 180 days ago stays at 0.70 floor", () => {
68
+ const factor = memoryDecayFactor(isoDaysAgo(180));
69
+ assert.equal(factor, 0.70);
70
+ });
71
+
72
+ test("memoryDecayFactor: result always in [0.7, 1.0] for any input", () => {
73
+ const samples: (string | null)[] = [
74
+ null,
75
+ "",
76
+ "garbage",
77
+ new Date(0).toISOString(),
78
+ isoDaysAgo(0),
79
+ isoDaysAgo(15),
80
+ isoDaysAgo(45),
81
+ isoDaysAgo(89),
82
+ isoDaysAgo(91),
83
+ isoDaysAgo(365),
84
+ new Date(Date.now() + 365 * DAY_MS).toISOString(),
85
+ ];
86
+ for (const s of samples) {
87
+ const f = memoryDecayFactor(s);
88
+ assert.ok(f >= 0.7 && f <= 1.0, `factor out of [0.7, 1.0] for ${s}: ${f}`);
89
+ }
90
+ });
@@ -13,6 +13,9 @@ import { parseRoadmap, parsePlan } from '../parsers-legacy.ts';
13
13
  import { parseSummary } from '../files.ts';
14
14
  import { deriveState } from '../state.ts';
15
15
  import { invalidateAllCaches } from '../cache.ts';
16
+ import { ensureDbOpen } from '../bootstrap/dynamic-tools.ts';
17
+ import { closeDatabase, getAllMilestones } from '../gsd-db.ts';
18
+ import { importWrittenMigrationToDb } from '../migrate/command.ts';
16
19
  import type {
17
20
  GSDProject,
18
21
  GSDMilestone,
@@ -250,6 +253,51 @@ test('Scenario 1: Incomplete project — write, parse, deriveState', async () =>
250
253
  }
251
254
  });
252
255
 
256
+ test('Scenario 1b: written migration imports into authoritative DB state', async () => {
257
+ const base = mkdtempSync(join(tmpdir(), 'gsd-writer-db-import-'));
258
+ try {
259
+ const project = buildIncompleteProject();
260
+ await writeGSDDirectory(project, base);
261
+
262
+ assert.equal(await ensureDbOpen(base), true, 'db import: ensureDbOpen creates authoritative DB');
263
+
264
+ invalidateAllCaches();
265
+ const before = await deriveState(base);
266
+ assert.equal(before.activeMilestone, null, 'db import: markdown-only migration is invisible before DB import');
267
+
268
+ const imported = await importWrittenMigrationToDb(base);
269
+ assert.deepStrictEqual(imported.hierarchy, { milestones: 1, slices: 2, tasks: 3 }, 'db import: hierarchy counts');
270
+
271
+ invalidateAllCaches();
272
+ const after = await deriveState(base);
273
+ assert.deepStrictEqual(after.phase, 'executing', 'db import: deriveState sees imported DB hierarchy');
274
+ assert.deepStrictEqual(after.activeMilestone?.id, 'M001', 'db import: active milestone');
275
+ assert.deepStrictEqual(after.activeSlice?.id, 'S02', 'db import: active slice');
276
+ assert.deepStrictEqual(after.activeTask?.id, 'T03', 'db import: active task');
277
+ } finally {
278
+ closeDatabase();
279
+ rmSync(base, { recursive: true, force: true });
280
+ }
281
+ });
282
+
283
+ test('Scenario 1c: DB import verification fails when preview counts do not match', async () => {
284
+ const base = mkdtempSync(join(tmpdir(), 'gsd-writer-db-check-'));
285
+ try {
286
+ const project = buildIncompleteProject();
287
+ await writeGSDDirectory(project, base);
288
+
289
+ const preview = generatePreview(project);
290
+ await assert.rejects(
291
+ () => importWrittenMigrationToDb(base, { ...preview, totalTasks: preview.totalTasks + 1 }),
292
+ /migration DB import verification failed: tasks 3\/4/,
293
+ );
294
+ assert.deepStrictEqual(getAllMilestones(), [], 'db import: failed verification rolls back hierarchy rewrite');
295
+ } finally {
296
+ closeDatabase();
297
+ rmSync(base, { recursive: true, force: true });
298
+ }
299
+ });
300
+
253
301
  // ─── Scenario 2: Fully complete project ────────────────────────────────
254
302
 
255
303
  test('Scenario 2: Fully complete project — deriveState phase', async () => {
@@ -0,0 +1,127 @@
1
+ import assert from "node:assert/strict";
2
+ import { mkdtempSync, rmSync } from "node:fs";
3
+ import { join } from "node:path";
4
+ import { tmpdir } from "node:os";
5
+ import test from "node:test";
6
+
7
+ import { ensureDbOpen } from "../bootstrap/dynamic-tools.ts";
8
+ import {
9
+ _getAdapter,
10
+ closeDatabase,
11
+ getAllMilestones,
12
+ getSliceTasks,
13
+ } from "../gsd-db.ts";
14
+ import {
15
+ autoImportMarkdownHierarchyIfDbMismatch,
16
+ countMarkdownHierarchy,
17
+ } from "../migration-auto-check.ts";
18
+ import { writeGSDDirectory } from "../migrate/writer.ts";
19
+ import type { GSDProject } from "../migrate/types.ts";
20
+
21
+ function makeBase(): string {
22
+ return mkdtempSync(join(tmpdir(), "gsd-migration-auto-check-"));
23
+ }
24
+
25
+ function cleanup(base: string): void {
26
+ closeDatabase();
27
+ rmSync(base, { recursive: true, force: true });
28
+ }
29
+
30
+ function projectFixture(): GSDProject {
31
+ return {
32
+ projectContent: "# Legacy Project\n",
33
+ decisionsContent: "",
34
+ requirements: [],
35
+ milestones: [
36
+ {
37
+ id: "M001",
38
+ title: "Legacy Milestone",
39
+ vision: "Carry forward previous work",
40
+ successCriteria: ["Existing task is visible"],
41
+ research: null,
42
+ boundaryMap: [],
43
+ slices: [
44
+ {
45
+ id: "S01",
46
+ title: "Legacy Slice",
47
+ risk: "medium",
48
+ depends: [],
49
+ done: false,
50
+ demo: "Legacy slice demo",
51
+ goal: "Legacy slice demo",
52
+ research: null,
53
+ summary: null,
54
+ tasks: [
55
+ {
56
+ id: "T01",
57
+ title: "Legacy Task",
58
+ description: "Task carried from markdown",
59
+ done: false,
60
+ estimate: "",
61
+ files: ["src/index.ts"],
62
+ mustHaves: [],
63
+ summary: null,
64
+ },
65
+ ],
66
+ },
67
+ ],
68
+ },
69
+ ],
70
+ };
71
+ }
72
+
73
+ test("migration auto-check imports markdown hierarchy when DB is empty", async () => {
74
+ const base = makeBase();
75
+ try {
76
+ await writeGSDDirectory(projectFixture(), base);
77
+ assert.deepEqual(countMarkdownHierarchy(base), { milestones: 1, slices: 1, tasks: 1 });
78
+
79
+ assert.equal(await ensureDbOpen(base), true);
80
+ assert.equal(getAllMilestones().length, 0, "fresh authoritative DB starts empty");
81
+
82
+ const result = await autoImportMarkdownHierarchyIfDbMismatch(base);
83
+ assert.equal(result.action, "imported");
84
+ assert.equal(result.reason, "db-empty");
85
+ assert.deepEqual(result.afterDb, { milestones: 1, slices: 1, tasks: 1 });
86
+ assert.equal(getAllMilestones().length, 1);
87
+ assert.equal(getSliceTasks("M001", "S01").length, 1);
88
+ } finally {
89
+ cleanup(base);
90
+ }
91
+ });
92
+
93
+ test("migration auto-check repairs DB hierarchy count mismatch", async () => {
94
+ const base = makeBase();
95
+ try {
96
+ await writeGSDDirectory(projectFixture(), base);
97
+ await autoImportMarkdownHierarchyIfDbMismatch(base);
98
+
99
+ _getAdapter()!.prepare("DELETE FROM tasks WHERE milestone_id = ? AND slice_id = ? AND id = ?").run("M001", "S01", "T01");
100
+ assert.equal(getSliceTasks("M001", "S01").length, 0, "test fixture simulates stale DB task count");
101
+
102
+ const result = await autoImportMarkdownHierarchyIfDbMismatch(base);
103
+ assert.equal(result.action, "imported");
104
+ assert.equal(result.reason, "count-mismatch");
105
+ assert.deepEqual(result.beforeDb, { milestones: 1, slices: 1, tasks: 0 });
106
+ assert.deepEqual(result.afterDb, { milestones: 1, slices: 1, tasks: 1 });
107
+ assert.equal(getSliceTasks("M001", "S01").length, 1);
108
+ } finally {
109
+ cleanup(base);
110
+ }
111
+ });
112
+
113
+ test("migration auto-check leaves matching DB hierarchy alone", async () => {
114
+ const base = makeBase();
115
+ try {
116
+ await writeGSDDirectory(projectFixture(), base);
117
+ await autoImportMarkdownHierarchyIfDbMismatch(base);
118
+
119
+ const result = await autoImportMarkdownHierarchyIfDbMismatch(base);
120
+ assert.equal(result.action, "none");
121
+ assert.equal(result.reason, "in-sync");
122
+ assert.deepEqual(result.markdown, { milestones: 1, slices: 1, tasks: 1 });
123
+ assert.deepEqual(result.afterDb, { milestones: 1, slices: 1, tasks: 1 });
124
+ } finally {
125
+ cleanup(base);
126
+ }
127
+ });
@@ -1,3 +1,6 @@
1
+ // Project/App: GSD-2
2
+ // File Purpose: Unit tests for pre-execution validation checks.
3
+
1
4
  /**
2
5
  * pre-execution-checks.test.ts — Unit tests for pre-execution validation checks.
3
6
  *
@@ -1542,6 +1545,27 @@ describe("checkFilePathConsistency directory inputs (#4446)", () => {
1542
1545
  assert.equal(results[0].blocking, true);
1543
1546
  });
1544
1547
 
1548
+ test("runtime directory annotation is skipped as a pre-execution file dependency", (t) => {
1549
+ const tempDir = join(tmpdir(), `pre-exec-dir-runtime-${Date.now()}`);
1550
+ mkdirSync(tempDir, { recursive: true });
1551
+ t.after(() => rmSync(tempDir, { recursive: true, force: true }));
1552
+
1553
+ const tasks = [
1554
+ createTask({
1555
+ id: "T02",
1556
+ inputs: ["entries/ directory (runtime)"],
1557
+ expected_output: ["src/commands/delete.ts", "src/index.ts"],
1558
+ }),
1559
+ ];
1560
+
1561
+ const results = checkFilePathConsistency(tasks, tempDir);
1562
+ assert.deepEqual(
1563
+ results,
1564
+ [],
1565
+ "Runtime-only directory inputs are created during command execution, not required before the task starts",
1566
+ );
1567
+ });
1568
+
1545
1569
  test("tilde-prefixed input is matched against $HOME, not the project basePath", (t) => {
1546
1570
  const fakeHome = join(tmpdir(), `pre-exec-tilde-home-${Date.now()}`);
1547
1571
  const projectDir = join(tmpdir(), `pre-exec-tilde-proj-${Date.now()}`);
@@ -1597,6 +1621,20 @@ describe("checkTaskOrdering directory inputs (#4446)", () => {
1597
1621
  "Directory reference should not be treated as reading a file created later",
1598
1622
  );
1599
1623
  });
1624
+
1625
+ test("runtime directory annotation does not produce an ordering violation", () => {
1626
+ const tasks = [
1627
+ createTask({
1628
+ id: "T02",
1629
+ sequence: 0,
1630
+ inputs: ["entries/ directory (runtime)"],
1631
+ expected_output: [],
1632
+ }),
1633
+ ];
1634
+
1635
+ const results = checkTaskOrdering(tasks, "/tmp");
1636
+ assert.deepEqual(results, []);
1637
+ });
1600
1638
  });
1601
1639
 
1602
1640
  // ─── Regression Tests: checkTaskOrdering false positive for pre-execution refs (#4071) ──
@@ -0,0 +1,40 @@
1
+ import test from "node:test";
2
+ import assert from "node:assert/strict";
3
+ import { readFileSync, readdirSync } from "node:fs";
4
+ import { join } from "node:path";
5
+ import { fileURLToPath } from "node:url";
6
+ import { dirname } from "node:path";
7
+
8
+ const __filename = fileURLToPath(import.meta.url);
9
+ const __dirname = dirname(__filename);
10
+ const promptsDir = join(__dirname, "..", "prompts");
11
+
12
+ test("prompt templates do not reference legacy milestone-root .gsd paths", () => {
13
+ const offenders: string[] = [];
14
+ for (const file of readdirSync(promptsDir)) {
15
+ if (!file.endsWith(".md")) continue;
16
+ const content = readFileSync(join(promptsDir, file), "utf-8");
17
+ const legacyPatterns = [
18
+ /\.gsd\/\{\{(?:milestoneId|mid)\}\}\//g,
19
+ /\.gsd\/<milestone-id>\//g,
20
+ /\.gsd\/<ID>\//g,
21
+ ];
22
+ for (const pattern of legacyPatterns) {
23
+ if (pattern.test(content)) {
24
+ offenders.push(`${file}: ${pattern.source}`);
25
+ }
26
+ }
27
+ }
28
+
29
+ assert.deepEqual(
30
+ offenders,
31
+ [],
32
+ "Milestone artifacts must use .gsd/milestones/<MID>/..., not legacy .gsd/<MID>/...",
33
+ );
34
+ });
35
+
36
+ test("quick task prompt delegates commit policy to quick.ts", () => {
37
+ const content = readFileSync(join(promptsDir, "quick-task.md"), "utf-8");
38
+ assert.match(content, /\{\{commitInstruction\}\}/);
39
+ assert.doesNotMatch(content, /Stage only relevant files/);
40
+ });
@@ -65,6 +65,25 @@ describe('prompt step ordering (#3696)', () => {
65
65
  assert.ok(learningsIdx < completeMilestoneIdx, 'learnings extraction must happen before gsd_complete_milestone');
66
66
  });
67
67
 
68
+ test('complete-milestone duplicate guard checks milestone status before durable writes', () => {
69
+ const guardMatch = completeMilestoneMd.match(/^\d+\.\s.*gsd_milestone_status/m);
70
+ const reqUpdateMatch = completeMilestoneMd.match(/^\d+\.\s.*gsd_requirement_update/m);
71
+ assert.ok(guardMatch, 'complete-milestone must start with a gsd_milestone_status duplicate guard');
72
+ assert.ok(reqUpdateMatch, 'gsd_requirement_update should appear in a numbered step');
73
+
74
+ const guardIdx = completeMilestoneMd.indexOf(guardMatch![0]);
75
+ const reqUpdateIdx = completeMilestoneMd.indexOf(reqUpdateMatch![0]);
76
+ assert.ok(
77
+ guardIdx < reqUpdateIdx,
78
+ 'duplicate guard must run before requirement/project/learnings writes',
79
+ );
80
+ assert.match(
81
+ completeMilestoneMd,
82
+ /status(?:`|\*\*)?\s+(?:is\s+)?(?:`complete`|"complete")/i,
83
+ 'duplicate guard must tell the agent to stop when status is complete',
84
+ );
85
+ });
86
+
68
87
  test('complete-slice.md uses gsd_requirement_update', () => {
69
88
  assert.match(completeSliceMd, /gsd_requirement_update/,
70
89
  'complete-slice.md should reference gsd_requirement_update');