mirror of
https://github.com/glittercowboy/get-shit-done
synced 2026-04-25 17:25:23 +02:00
* refactor(tests): standardize to node:assert/strict and t.after() per CONTRIBUTING.md
- Replace require('node:assert') with require('node:assert/strict') across
all 73 test files to enforce strict equality (no type coercion)
- Replace try/finally cleanup blocks with t.after() hooks in core.test.cjs
and hooks-opt-in.test.cjs per the test lifecycle standards
- Utility functions in codex-config and security-scan retain try/finally
as that is appropriate for per-function resource guards, not lifecycle hooks
Closes #1674
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* perf(tests): add --test-concurrency=4 to test runner for parallel file execution
Node.js --test-concurrency controls how many test files run as parallel child
processes. Set to 4 by default, configurable via TEST_CONCURRENCY env var.
Fixes tests at a known level rather than inheriting os.availableParallelism()
which varies across CI environments.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* fix(security): allowlist verify.test.cjs in prompt-injection scanner
tests/verify.test.cjs uses <human>...</human> as GSD phase task-type
XML (meaning "a human should verify this step"), which matches the
scanner's fake-message-boundary pattern for LLM APIs. This is a
false positive — add it to the allowlist alongside the other test files
that legitimately contain injection-adjacent patterns.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
---------
Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com>
284 lines
13 KiB
JavaScript
284 lines
13 KiB
JavaScript
/**
|
|
* GSD Tools Tests - Dispatcher
|
|
*
|
|
* Tests for gsd-tools.cjs dispatch routing and error paths.
|
|
* Covers: no-command, unknown command, unknown subcommands for every command group,
|
|
* --cwd parsing, and previously untouched routing branches.
|
|
*
|
|
* Requirements: DISP-01, DISP-02
|
|
*/
|
|
|
|
const { test, describe, 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');
|
|
|
|
// ─── Dispatcher Error Paths ──────────────────────────────────────────────────
|
|
|
|
describe('dispatcher error paths', () => {
|
|
let tmpDir;
|
|
|
|
beforeEach(() => {
|
|
tmpDir = createTempProject();
|
|
});
|
|
|
|
afterEach(() => {
|
|
cleanup(tmpDir);
|
|
});
|
|
|
|
// No command
|
|
test('no-command invocation prints usage and exits non-zero', () => {
|
|
const result = runGsdTools('', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Usage:'), `Expected "Usage:" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown command
|
|
test('unknown command produces clear error and exits non-zero', () => {
|
|
const result = runGsdTools('nonexistent-cmd', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown command'), `Expected "Unknown command" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// --cwd= form with valid directory
|
|
test('--cwd= form overrides working directory', () => {
|
|
// Create STATE.md in tmpDir so state load can find it
|
|
fs.writeFileSync(
|
|
path.join(tmpDir, '.planning', 'STATE.md'),
|
|
'# Project State\n\n## Current Position\n\nPhase: 1 of 1 (Test)\n'
|
|
);
|
|
const result = runGsdTools(`--cwd=${tmpDir} state load`, process.cwd());
|
|
assert.strictEqual(result.success, true, `Should succeed with --cwd=, got: ${result.error}`);
|
|
});
|
|
|
|
// --cwd= with empty value
|
|
test('--cwd= with empty value produces error', () => {
|
|
const result = runGsdTools('--cwd= state load', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Missing value for --cwd'), `Expected "Missing value for --cwd" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// --cwd with nonexistent path
|
|
test('--cwd with invalid path produces error', () => {
|
|
const result = runGsdTools('--cwd /nonexistent/path/xyz state load', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Invalid --cwd'), `Expected "Invalid --cwd" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown subcommand: template
|
|
test('template unknown subcommand errors', () => {
|
|
const result = runGsdTools('template bogus', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown template subcommand'), `Expected "Unknown template subcommand" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown subcommand: frontmatter
|
|
test('frontmatter unknown subcommand errors', () => {
|
|
const result = runGsdTools('frontmatter bogus file.md', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown frontmatter subcommand'), `Expected "Unknown frontmatter subcommand" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown subcommand: verify
|
|
test('verify unknown subcommand errors', () => {
|
|
const result = runGsdTools('verify bogus', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown verify subcommand'), `Expected "Unknown verify subcommand" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown subcommand: phases
|
|
test('phases unknown subcommand errors', () => {
|
|
const result = runGsdTools('phases bogus', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown phases subcommand'), `Expected "Unknown phases subcommand" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown subcommand: roadmap
|
|
test('roadmap unknown subcommand errors', () => {
|
|
const result = runGsdTools('roadmap bogus', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown roadmap subcommand'), `Expected "Unknown roadmap subcommand" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown subcommand: requirements
|
|
test('requirements unknown subcommand errors', () => {
|
|
const result = runGsdTools('requirements bogus', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown requirements subcommand'), `Expected "Unknown requirements subcommand" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown subcommand: phase
|
|
test('phase unknown subcommand errors', () => {
|
|
const result = runGsdTools('phase bogus', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown phase subcommand'), `Expected "Unknown phase subcommand" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown subcommand: milestone
|
|
test('milestone unknown subcommand errors', () => {
|
|
const result = runGsdTools('milestone bogus', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown milestone subcommand'), `Expected "Unknown milestone subcommand" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown subcommand: validate
|
|
test('validate unknown subcommand errors', () => {
|
|
const result = runGsdTools('validate bogus', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown validate subcommand'), `Expected "Unknown validate subcommand" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown subcommand: todo
|
|
test('todo unknown subcommand errors', () => {
|
|
const result = runGsdTools('todo bogus', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown todo subcommand'), `Expected "Unknown todo subcommand" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
test('uat unknown subcommand errors', () => {
|
|
const result = runGsdTools('uat bogus', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown uat subcommand'), `Expected "Unknown uat subcommand" in stderr, got: ${result.error}`);
|
|
});
|
|
|
|
// Unknown subcommand: init
|
|
test('init unknown workflow errors', () => {
|
|
const result = runGsdTools('init bogus', tmpDir);
|
|
assert.strictEqual(result.success, false, 'Should exit non-zero');
|
|
assert.ok(result.error.includes('Unknown init workflow'), `Expected "Unknown init workflow" in stderr, got: ${result.error}`);
|
|
});
|
|
});
|
|
|
|
// ─── Dispatcher Routing Branches ─────────────────────────────────────────────
|
|
|
|
describe('dispatcher routing branches', () => {
|
|
let tmpDir;
|
|
|
|
beforeEach(() => {
|
|
tmpDir = createTempProject();
|
|
});
|
|
|
|
afterEach(() => {
|
|
cleanup(tmpDir);
|
|
});
|
|
|
|
// find-phase
|
|
test('find-phase locates phase directory by number', () => {
|
|
const phaseDir = path.join(tmpDir, '.planning', 'phases', '01-test-phase');
|
|
fs.mkdirSync(phaseDir, { recursive: true });
|
|
|
|
const result = runGsdTools('find-phase 01', tmpDir);
|
|
assert.strictEqual(result.success, true, `find-phase failed: ${result.error}`);
|
|
assert.ok(result.output.includes('01-test-phase'), `Expected output to contain "01-test-phase", got: ${result.output}`);
|
|
});
|
|
|
|
// init resume
|
|
test('init resume returns valid JSON', () => {
|
|
fs.writeFileSync(
|
|
path.join(tmpDir, '.planning', 'STATE.md'),
|
|
'# Project State\n\n## Current Position\n\nPhase: 1 of 1 (Test)\nPlan: 01-01 complete\nStatus: Ready\nLast activity: 2026-01-01\n\nProgress: [##########] 100%\n\n## Session Continuity\n\nLast session: 2026-01-01\nStopped at: Test\nResume file: None\n'
|
|
);
|
|
|
|
const result = runGsdTools('init resume', tmpDir);
|
|
assert.strictEqual(result.success, true, `init resume failed: ${result.error}`);
|
|
const parsed = JSON.parse(result.output);
|
|
assert.ok(typeof parsed === 'object', 'Output should be valid JSON object');
|
|
});
|
|
|
|
// init verify-work
|
|
test('init verify-work returns valid JSON', () => {
|
|
// Create STATE.md
|
|
fs.writeFileSync(
|
|
path.join(tmpDir, '.planning', 'STATE.md'),
|
|
'# Project State\n\n## Current Position\n\nPhase: 1 of 1 (Test)\nPlan: 01-01 complete\nStatus: Ready\nLast activity: 2026-01-01\n\nProgress: [##########] 100%\n\n## Session Continuity\n\nLast session: 2026-01-01\nStopped at: Test\nResume file: None\n'
|
|
);
|
|
|
|
// Create ROADMAP.md with phase section
|
|
fs.writeFileSync(
|
|
path.join(tmpDir, '.planning', 'ROADMAP.md'),
|
|
'# Roadmap\n\n## Milestone: v1.0 Test\n\n### Phase 1: Test Phase\n**Goal**: Test goal\n**Depends on**: None\n**Requirements**: TEST-01\n**Success Criteria**:\n 1. Tests pass\n**Plans**: 1 plan\nPlans:\n- [x] 01-01-PLAN.md\n\n## Progress\n\n| Phase | Plans | Status | Date |\n|-------|-------|--------|------|\n| 1 | 1/1 | Complete | 2026-01-01 |\n'
|
|
);
|
|
|
|
// Create phase dir
|
|
const phaseDir = path.join(tmpDir, '.planning', 'phases', '01-test');
|
|
fs.mkdirSync(phaseDir, { recursive: true });
|
|
|
|
const result = runGsdTools('init verify-work 01', tmpDir);
|
|
assert.strictEqual(result.success, true, `init verify-work failed: ${result.error}`);
|
|
const parsed = JSON.parse(result.output);
|
|
assert.ok(typeof parsed === 'object', 'Output should be valid JSON object');
|
|
});
|
|
|
|
// roadmap update-plan-progress
|
|
test('roadmap update-plan-progress updates phase progress', () => {
|
|
// Create ROADMAP.md with progress table
|
|
fs.writeFileSync(
|
|
path.join(tmpDir, '.planning', 'ROADMAP.md'),
|
|
'# Roadmap\n\n## Milestone: v1.0 Test\n\n### Phase 1: Test Phase\n**Goal**: Test goal\n**Depends on**: None\n**Requirements**: TEST-01\n**Success Criteria**:\n 1. Tests pass\n**Plans**: 1 plan\nPlans:\n- [ ] 01-01-PLAN.md\n\n## Progress\n\n| Phase | Plans | Status | Date |\n|-------|-------|--------|------|\n| 1 | 0/1 | Not Started | - |\n'
|
|
);
|
|
|
|
// Create phase dir with PLAN and SUMMARY
|
|
const phaseDir = path.join(tmpDir, '.planning', 'phases', '01-test-phase');
|
|
fs.mkdirSync(phaseDir, { recursive: true });
|
|
fs.writeFileSync(
|
|
path.join(phaseDir, '01-01-PLAN.md'),
|
|
'---\nphase: 01-test-phase\nplan: "01"\n---\n\n# Plan\n'
|
|
);
|
|
fs.writeFileSync(
|
|
path.join(phaseDir, '01-01-SUMMARY.md'),
|
|
'---\nphase: 01-test-phase\nplan: "01"\n---\n\n# Summary\n'
|
|
);
|
|
|
|
const result = runGsdTools('roadmap update-plan-progress 1', tmpDir);
|
|
assert.strictEqual(result.success, true, `roadmap update-plan-progress failed: ${result.error}`);
|
|
});
|
|
|
|
// state (no subcommand) — default load
|
|
test('state with no subcommand calls cmdStateLoad', () => {
|
|
fs.writeFileSync(
|
|
path.join(tmpDir, '.planning', 'STATE.md'),
|
|
'# Project State\n\n## Current Position\n\nPhase: 1 of 1 (Test)\nPlan: 01-01 complete\nStatus: Ready\nLast activity: 2026-01-01\n\nProgress: [##########] 100%\n\n## Session Continuity\n\nLast session: 2026-01-01\nStopped at: Test\nResume file: None\n'
|
|
);
|
|
|
|
const result = runGsdTools('state', tmpDir);
|
|
assert.strictEqual(result.success, true, `state load failed: ${result.error}`);
|
|
const parsed = JSON.parse(result.output);
|
|
assert.ok(typeof parsed === 'object', 'Output should be valid JSON object');
|
|
});
|
|
|
|
// summary-extract
|
|
test('summary-extract parses SUMMARY.md frontmatter', () => {
|
|
const phaseDir = path.join(tmpDir, '.planning', 'phases', '01-test');
|
|
fs.mkdirSync(phaseDir, { recursive: true });
|
|
|
|
const summaryContent = `---
|
|
phase: 01-test
|
|
plan: "01"
|
|
subsystem: testing
|
|
tags: [node, test]
|
|
duration: 5min
|
|
completed: "2026-01-01"
|
|
key-decisions:
|
|
- "Used node:test"
|
|
requirements-completed: [TEST-01]
|
|
---
|
|
|
|
# Phase 1 Plan 01: Test Summary
|
|
|
|
**Tests added for core module**
|
|
`;
|
|
|
|
const summaryPath = path.join(phaseDir, '01-01-SUMMARY.md');
|
|
fs.writeFileSync(summaryPath, summaryContent);
|
|
|
|
// Use relative path from tmpDir
|
|
const result = runGsdTools(`summary-extract .planning/phases/01-test/01-01-SUMMARY.md`, tmpDir);
|
|
assert.strictEqual(result.success, true, `summary-extract failed: ${result.error}`);
|
|
const parsed = JSON.parse(result.output);
|
|
assert.ok(typeof parsed === 'object', 'Output should be valid JSON object');
|
|
assert.strictEqual(parsed.path, '.planning/phases/01-test/01-01-SUMMARY.md', 'Path should match input');
|
|
assert.deepStrictEqual(parsed.requirements_completed, ['TEST-01'], 'requirements_completed should contain TEST-01');
|
|
});
|
|
});
|