Files
get-shit-done/tests/init-manager-deps.test.cjs
Tom Boucher 2f28c99db4 fix(init): include shipped-milestone phases in deps_satisfied check (#2269)
- Build completedNums from current milestone phases as before
- Also scan full rawContent for [x]-checked Phase lines across all
  milestone sections (including <details>-wrapped shipped milestones)
- Phases from prior milestones are complete by definition, so any
  dep on them should always resolve to deps_satisfied: true
- Add regression tests in tests/init-manager-deps.test.cjs

Closes #2267
2026-04-15 14:59:07 -04:00

121 lines
3.7 KiB
JavaScript

/**
* Tests for bug #2267: deps_satisfied should include phases from shipped milestones.
*
* Root cause: completedNums was built only from the current milestone's phases,
* so a dependency on a phase from a previously shipped milestone was never
* satisfied — even though all prior-milestone phases are complete by definition.
*/
const { describe, test, beforeEach, afterEach } = require('node:test');
const assert = require('node:assert/strict');
const fs = require('fs');
const path = require('path');
const { runGsdTools, createTempProject, cleanup } = require('./helpers.cjs');
describe('init manager — cross-milestone dependency satisfaction', () => {
let tmpDir;
beforeEach(() => {
tmpDir = createTempProject();
});
afterEach(() => {
cleanup(tmpDir);
});
/**
* Write a ROADMAP.md that has:
* - A shipped previous milestone (v1.0) inside a <details> block containing
* Phase 5 marked [x] complete.
* - A current active milestone (v2.0) containing Phase 6 that depends on
* Phase 5.
*/
function writeRoadmapWithShippedMilestone(dir) {
const content = [
'# Roadmap',
'',
'<details>',
'<summary>v1.0 — Initial Release (Shipped)</summary>',
'',
'## Roadmap v1.0: Initial Release',
'',
'- [x] **Phase 5: Auth**',
'',
'### Phase 5: Auth',
'**Goal:** Add authentication',
'',
'</details>',
'',
'## Roadmap v2.0: Dashboard',
'',
'- [ ] **Phase 6: Dashboard**',
'',
'### Phase 6: Dashboard',
'**Goal:** Build dashboard',
'**Depends on:** Phase 5',
'',
].join('\n');
fs.writeFileSync(path.join(dir, '.planning', 'ROADMAP.md'), content);
}
function writeStateWithMilestone(dir, version) {
fs.writeFileSync(
path.join(dir, '.planning', 'STATE.md'),
`---\nmilestone: ${version}\n---\n# State\n`
);
}
test('phase depending on a shipped-milestone phase has deps_satisfied: true', () => {
writeRoadmapWithShippedMilestone(tmpDir);
writeStateWithMilestone(tmpDir, 'v2.0');
const result = runGsdTools('init manager', tmpDir);
assert.ok(result.success, `Command failed: ${result.error}`);
const output = JSON.parse(result.output);
// Only the current milestone's phases should appear in the phases array
assert.strictEqual(output.phases.length, 1, 'Should have exactly one phase from the current milestone');
const phase6 = output.phases[0];
assert.strictEqual(phase6.number, '6', 'Should be Phase 6');
// Phase 6 depends on Phase 5 from the prior milestone — must be satisfied
assert.strictEqual(
phase6.deps_satisfied,
true,
'Phase 6 dep on shipped Phase 5 should be satisfied'
);
});
test('phase depending on a non-existent phase has deps_satisfied: false', () => {
writeRoadmapWithShippedMilestone(tmpDir);
writeStateWithMilestone(tmpDir, 'v2.0');
// Add a second phase in the current milestone that depends on a phantom phase
const roadmapPath = path.join(tmpDir, '.planning', 'ROADMAP.md');
const existing = fs.readFileSync(roadmapPath, 'utf-8');
const withExtra = existing + [
'### Phase 7: Extra',
'**Goal:** Extra work',
'**Depends on:** Phase 99',
'',
].join('\n');
fs.writeFileSync(roadmapPath, withExtra);
const result = runGsdTools('init manager', tmpDir);
assert.ok(result.success, `Command failed: ${result.error}`);
const output = JSON.parse(result.output);
const phase7 = output.phases.find(p => p.number === '7');
assert.ok(phase7, 'Phase 7 should be in the output');
assert.strictEqual(
phase7.deps_satisfied,
false,
'Phase 7 dep on non-existent Phase 99 should not be satisfied'
);
});
});