mirror of
https://github.com/glittercowboy/get-shit-done
synced 2026-04-25 17:25:23 +02:00
* feat: Phase 2 caller migration — gsd-sdk query in workflows (#2122)
Cherry-picked orchestration rewrites from feat/sdk-foundation (#2008, 4018fee) onto current main, resolving conflicts to keep upstream worktree guards and post-merge test gate. SDK stub registry omitted (out of Phase 2 scope per #2122).
Refs: #2122 #2008
Made-with: Cursor
* docs: add gsd-sdk query migration blurb
Made-with: Cursor
* docs(workflows): extend Phase 2 gsd-sdk query caller migration
- Swap node gsd-tools.cjs for gsd-sdk query in review, plan-phase, execute-plan,
ship, extract_learnings, ai-integration-phase, eval-review, next, thread
- Document graphify CJS-only in gsd-planner; dual-path in CLI-TOOLS and ARCHITECTURE
- Update tests: workstreams gsd-sdk path, thread frontmatter.get, workspace init.*,
CRLF-safe autonomous frontmatter parse
- CHANGELOG: Phase 2 caller migration scope
Made-with: Cursor
* docs(phase2): USER-GUIDE + remaining gsd-sdk query call sites
- USER-GUIDE: dual-path CLI section; state validate/sync use full CJS path
- Commands: debug (config-get+tdd), quick (security note), intel Task prompt
- Agent: gsd-debug-session-manager resolve-model via jq
- Workflows: milestone-summary, forensics, next, complete-milestone/verify-work
(audit-open CJS notes), discuss-phase, progress, verify-phase, add/insert/remove
phase, transition, manager, quick workflow; remove-phase commit without --files
- Test: quick-session-management accepts frontmatter.get
- CHANGELOG: Phase 2 follow-up bullet
Made-with: Cursor
* docs(phase2): align gsd-sdk query examples in commands and agents
- init.* query names; frontmatter.get uses positional field name
- state.* handlers use positional args; commit uses positional paths
- CJS-only notes for from-gsd2 and graphify; learnings.query wording
- CHANGELOG: Phase 2 orchestration doc pass
Made-with: Cursor
* docs(phase2): normalize gsd-sdk query commit to positional file paths
- Strip --files from commit examples in workflows, references, commands
- Keep commit-to-subrepo ... --files (separate handler)
- git-planning-commit.md: document positional args
- Tests: new-project commit line, state.record-session, gates CRLF, roadmap.analyze
- CHANGELOG [Unreleased]
Made-with: Cursor
* feat(sdk): gsd-sdk query parity with gsd-tools and PR 2179 registry fixes
- Route query via longest-prefix match and dotted single-token expansion; fall back
to runGsdToolsQuery (same argv as node gsd-tools.cjs) for full CLI coverage.
- Parse gsd-sdk query permissively so gsd-tools flags (--json, --verify, etc.) are
not rejected by strict parseArgs.
- resolveGsdToolsPath: honor GSD_TOOLS_PATH; prefer bundled get-shit-done copy
over project .claude installs; export runGsdToolsQuery from the SDK.
- Fix gsd-tools audit-open (core.output; pass object for --json JSON).
- Register summary-extract as alias of summary.extract; fix audit-fix workflow to
call audit-uat instead of invalid init.audit-uat (PR review).
Updates QUERY-HANDLERS.md and CHANGELOG [Unreleased].
Made-with: Cursor
* fix(sdk): Phase 2 scope — Trek-e review (#2179, #2122)
- Remove gsd-sdk query passthrough to gsd-tools.cjs; drop GSD_TOOLS_PATH
- Consolidate argv routing in resolveQueryArgv(); update USAGE and QUERY-HANDLERS
- Surface @file: read failures in GSDTools.parseOutput
- execute-plan: defer Task Commit Protocol to gsd-executor
- stale-colon-refs: skip .planning/ and root CLAUDE.md (gitignored overlays)
- CHANGELOG [Unreleased]: maintainer review and routing notes
Made-with: Cursor
220 lines
8.9 KiB
JavaScript
220 lines
8.9 KiB
JavaScript
/**
|
|
* Tests for verifier deferred-items filtering (#1624)
|
|
*
|
|
* Verifies that the gsd-verifier agent filters gaps addressed in later
|
|
* milestone phases, preventing false-positive gap reports.
|
|
*/
|
|
|
|
const { test, describe } = require('node:test');
|
|
const assert = require('node:assert/strict');
|
|
const fs = require('fs');
|
|
const path = require('path');
|
|
|
|
const ROOT = path.resolve(__dirname, '..');
|
|
|
|
describe('verifier deferred-items filtering (#1624)', () => {
|
|
|
|
// ── gsd-verifier.md ────────────────────────────────────────────────────────
|
|
|
|
describe('agents/gsd-verifier.md', () => {
|
|
const verifierPath = path.join(ROOT, 'agents', 'gsd-verifier.md');
|
|
let verifierContent;
|
|
|
|
test('file exists', () => {
|
|
assert.ok(fs.existsSync(verifierPath), 'gsd-verifier.md should exist');
|
|
verifierContent = fs.readFileSync(verifierPath, 'utf-8');
|
|
});
|
|
|
|
test('contains Step 9b for filtering deferred items', () => {
|
|
verifierContent = verifierContent || fs.readFileSync(verifierPath, 'utf-8');
|
|
assert.ok(
|
|
verifierContent.includes('Step 9b') || verifierContent.includes('Filter Deferred'),
|
|
'gsd-verifier.md should contain Step 9b or "Filter Deferred" section'
|
|
);
|
|
});
|
|
|
|
test('Step 9b references roadmap analyze for cross-referencing', () => {
|
|
verifierContent = verifierContent || fs.readFileSync(verifierPath, 'utf-8');
|
|
assert.ok(
|
|
verifierContent.includes('roadmap analyze') || verifierContent.includes('roadmap.analyze'),
|
|
'Step 9b should reference roadmap analyze (CJS or gsd-sdk query) for loading full milestone data'
|
|
);
|
|
});
|
|
|
|
test('VERIFICATION.md frontmatter template includes deferred section', () => {
|
|
verifierContent = verifierContent || fs.readFileSync(verifierPath, 'utf-8');
|
|
assert.ok(
|
|
verifierContent.includes('deferred:'),
|
|
'VERIFICATION.md template should include a deferred: section in frontmatter'
|
|
);
|
|
});
|
|
|
|
test('deferred section includes addressed_in field', () => {
|
|
verifierContent = verifierContent || fs.readFileSync(verifierPath, 'utf-8');
|
|
assert.ok(
|
|
verifierContent.includes('addressed_in'),
|
|
'deferred items should include an addressed_in field referencing the later phase'
|
|
);
|
|
});
|
|
|
|
test('deferred section includes evidence field', () => {
|
|
verifierContent = verifierContent || fs.readFileSync(verifierPath, 'utf-8');
|
|
assert.ok(
|
|
verifierContent.includes('evidence'),
|
|
'deferred items should include an evidence field with matching goal/criteria'
|
|
);
|
|
});
|
|
|
|
test('deferred section is conditional (only if deferred items exist)', () => {
|
|
verifierContent = verifierContent || fs.readFileSync(verifierPath, 'utf-8');
|
|
assert.ok(
|
|
verifierContent.includes('if deferred items exist') ||
|
|
verifierContent.includes('If deferred items exist') ||
|
|
verifierContent.includes('Only if deferred'),
|
|
'deferred section should be conditional — only included when deferred items exist'
|
|
);
|
|
});
|
|
|
|
test('deferred items do not affect status determination', () => {
|
|
verifierContent = verifierContent || fs.readFileSync(verifierPath, 'utf-8');
|
|
assert.ok(
|
|
verifierContent.includes('do NOT affect the status') ||
|
|
verifierContent.includes('do not affect status') ||
|
|
verifierContent.includes('Deferred items do NOT affect'),
|
|
'should explicitly state that deferred items do not affect status'
|
|
);
|
|
});
|
|
|
|
test('includes conservative matching guidance', () => {
|
|
verifierContent = verifierContent || fs.readFileSync(verifierPath, 'utf-8');
|
|
assert.ok(
|
|
verifierContent.includes('conservative') || verifierContent.includes('when in doubt'),
|
|
'should include guidance to be conservative when matching gaps to later phases'
|
|
);
|
|
});
|
|
|
|
test('report body template includes Deferred Items table', () => {
|
|
verifierContent = verifierContent || fs.readFileSync(verifierPath, 'utf-8');
|
|
assert.ok(
|
|
verifierContent.includes('### Deferred Items'),
|
|
'report body template should include a Deferred Items section'
|
|
);
|
|
});
|
|
|
|
test('success criteria mentions deferred filtering', () => {
|
|
verifierContent = verifierContent || fs.readFileSync(verifierPath, 'utf-8');
|
|
assert.ok(
|
|
verifierContent.includes('Deferred items filtered') ||
|
|
verifierContent.includes('deferred items filtered') ||
|
|
verifierContent.includes('Deferred items structured'),
|
|
'success criteria should reference deferred item filtering'
|
|
);
|
|
});
|
|
});
|
|
|
|
// ── verify-phase.md (workflow) ─────────────────────────────────────────────
|
|
|
|
describe('get-shit-done/workflows/verify-phase.md', () => {
|
|
const workflowPath = path.join(ROOT, 'get-shit-done', 'workflows', 'verify-phase.md');
|
|
let workflowContent;
|
|
|
|
test('file exists', () => {
|
|
assert.ok(fs.existsSync(workflowPath), 'verify-phase.md should exist');
|
|
workflowContent = fs.readFileSync(workflowPath, 'utf-8');
|
|
});
|
|
|
|
test('loads roadmap analyze in context step', () => {
|
|
workflowContent = workflowContent || fs.readFileSync(workflowPath, 'utf-8');
|
|
assert.ok(
|
|
workflowContent.includes('roadmap analyze'),
|
|
'verify-phase.md should load roadmap analyze in its context step'
|
|
);
|
|
});
|
|
|
|
test('contains filter_deferred_items step', () => {
|
|
workflowContent = workflowContent || fs.readFileSync(workflowPath, 'utf-8');
|
|
assert.ok(
|
|
workflowContent.includes('filter_deferred_items') ||
|
|
workflowContent.includes('Filter Deferred'),
|
|
'verify-phase.md should contain a deferred-item filtering step'
|
|
);
|
|
});
|
|
|
|
test('success criteria mentions deferred filtering', () => {
|
|
workflowContent = workflowContent || fs.readFileSync(workflowPath, 'utf-8');
|
|
assert.ok(
|
|
workflowContent.includes('Deferred items filtered') ||
|
|
workflowContent.includes('deferred items filtered'),
|
|
'success criteria should mention deferred item filtering'
|
|
);
|
|
});
|
|
});
|
|
|
|
// ── verify-phase.md (SDK variant) ──────────────────────────────────────────
|
|
|
|
describe('sdk/prompts/workflows/verify-phase.md', () => {
|
|
const sdkPath = path.join(ROOT, 'sdk', 'prompts', 'workflows', 'verify-phase.md');
|
|
let sdkContent;
|
|
|
|
test('file exists', () => {
|
|
assert.ok(fs.existsSync(sdkPath), 'SDK verify-phase.md should exist');
|
|
sdkContent = fs.readFileSync(sdkPath, 'utf-8');
|
|
});
|
|
|
|
test('loads roadmap analyze in context step', () => {
|
|
sdkContent = sdkContent || fs.readFileSync(sdkPath, 'utf-8');
|
|
assert.ok(
|
|
sdkContent.includes('roadmap analyze'),
|
|
'SDK verify-phase.md should reference roadmap analyze for deferred-item filtering'
|
|
);
|
|
});
|
|
|
|
test('contains deferred-item filtering step', () => {
|
|
sdkContent = sdkContent || fs.readFileSync(sdkPath, 'utf-8');
|
|
assert.ok(
|
|
sdkContent.includes('filter_deferred_items') ||
|
|
sdkContent.includes('deferred'),
|
|
'SDK verify-phase.md should contain deferred-item filtering logic'
|
|
);
|
|
});
|
|
});
|
|
|
|
// ── planner-gap-closure.md ─────────────────────────────────────────────────
|
|
|
|
describe('get-shit-done/references/planner-gap-closure.md', () => {
|
|
const closurePath = path.join(ROOT, 'get-shit-done', 'references', 'planner-gap-closure.md');
|
|
let closureContent;
|
|
|
|
test('file exists', () => {
|
|
assert.ok(fs.existsSync(closurePath), 'planner-gap-closure.md should exist');
|
|
closureContent = fs.readFileSync(closurePath, 'utf-8');
|
|
});
|
|
|
|
test('mentions skipping deferred items', () => {
|
|
closureContent = closureContent || fs.readFileSync(closurePath, 'utf-8');
|
|
const lower = closureContent.toLowerCase();
|
|
assert.ok(
|
|
lower.includes('deferred') && lower.includes('skip'),
|
|
'planner-gap-closure.md should mention skipping deferred items'
|
|
);
|
|
});
|
|
|
|
test('distinguishes gaps from deferred sections', () => {
|
|
closureContent = closureContent || fs.readFileSync(closurePath, 'utf-8');
|
|
assert.ok(
|
|
closureContent.includes('gaps:') && closureContent.includes('deferred:'),
|
|
'should reference both gaps: and deferred: sections to distinguish them'
|
|
);
|
|
});
|
|
|
|
test('explains that deferred items are not actionable', () => {
|
|
closureContent = closureContent || fs.readFileSync(closurePath, 'utf-8');
|
|
assert.ok(
|
|
closureContent.includes('NOT gaps') || closureContent.includes('not gaps') ||
|
|
closureContent.includes('must be ignored'),
|
|
'should explain that deferred items are not actionable gaps'
|
|
);
|
|
});
|
|
});
|
|
});
|