Files
get-shit-done/tests/ultraplan-phase.test.cjs
Tom Boucher 794f7e1b0b feat: /gsd-ultraplan-phase [BETA] — offload plan phase to Claude Code ultraplan (#2378)
* docs: add design spec for /gsd-ultraplan-phase beta command

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>

* feat: add /gsd-ultraplan-phase [BETA] command

Offloads GSD plan phase to Claude Code's ultraplan cloud infrastructure.
Plan drafts remotely while terminal stays free; browser UI for inline
comments and revisions; imports back via existing /gsd-import --from.

Intentionally isolated from /gsd-plan-phase so upstream ultraplan changes
cannot break the core planning pipeline.

Closes #2374

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>

* fix: resolve 5 pre-existing test failures before PR

- ARCHITECTURE.md: update command count 75→80 and workflow count 72→77
  (stale doc counts; also incremented by new ultraplan-phase files)
- sketch.md: add TEXT_MODE plain-text fallback for AskUserQuestion (#2012)
- read-guard.test.cjs: clear CLAUDECODE env var alongside CLAUDE_SESSION_ID
  so positive-path hook tests pass when run inside a Claude Code session

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>

* docs: add BETA.md with /gsd-ultraplan-phase user documentation

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>

* fix: address CodeRabbit review — MD040 fence labels and sketch.md TEXT_MODE duplicate

- Add language identifiers to all unlabeled fenced blocks in
  ultraplan-phase.md and design spec (resolves MD040)
- Remove duplicate TEXT_MODE explanation from sketch.md mood_intake step
  (was identical to the banner step definition)
- Make AskUserQuestion conditional explicit in mood_intake prose

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>

---------

Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-17 14:45:03 -04:00

190 lines
7.8 KiB
JavaScript

/**
* /gsd-ultraplan-phase [BETA] Tests
*
* Structural assertions for the ultraplan-phase command and workflow files.
* This command offloads GSD plan phase to Claude Code's ultraplan cloud infrastructure.
*/
'use strict';
const { describe, test } = require('node:test');
const assert = require('node:assert/strict');
const fs = require('fs');
const path = require('path');
const CMD_PATH = path.join(__dirname, '..', 'commands', 'gsd', 'ultraplan-phase.md');
const WF_PATH = path.join(__dirname, '..', 'get-shit-done', 'workflows', 'ultraplan-phase.md');
// ─── File Existence ────────────────────────────────────────────────────────────
describe('ultraplan-phase file existence', () => {
test('command file exists', () => {
assert.ok(fs.existsSync(CMD_PATH), 'commands/gsd/ultraplan-phase.md should exist');
});
test('workflow file exists', () => {
assert.ok(fs.existsSync(WF_PATH), 'get-shit-done/workflows/ultraplan-phase.md should exist');
});
});
// ─── Command Frontmatter ───────────────────────────────────────────────────────
describe('ultraplan-phase command frontmatter', () => {
const content = fs.readFileSync(CMD_PATH, 'utf-8');
test('has correct name field', () => {
assert.match(content, /^name:\s*gsd:ultraplan-phase$/m);
});
test('description marks feature as BETA', () => {
assert.match(content, /^description:.*\[BETA\]/m);
});
test('has argument-hint', () => {
assert.match(content, /^argument-hint:/m);
});
});
// ─── Command References ────────────────────────────────────────────────────────
describe('ultraplan-phase command references', () => {
const content = fs.readFileSync(CMD_PATH, 'utf-8');
test('references the ultraplan-phase workflow', () => {
assert.ok(
content.includes('@~/.claude/get-shit-done/workflows/ultraplan-phase.md'),
'command should reference ultraplan-phase workflow'
);
});
test('references ui-brand', () => {
assert.ok(content.includes('ui-brand'), 'command should reference ui-brand');
});
});
// ─── Workflow: Beta Marker ─────────────────────────────────────────────────────
describe('ultraplan-phase workflow beta marker', () => {
const content = fs.readFileSync(WF_PATH, 'utf-8');
test('workflow displays a BETA warning', () => {
assert.ok(content.includes('BETA') || content.includes('beta'), 'workflow should display a BETA warning');
});
test('workflow notes ultraplan is in research preview', () => {
assert.ok(content.includes('research preview') || content.includes('preview'), 'workflow should note ultraplan is in research preview');
});
});
// ─── Workflow: Runtime Gate ────────────────────────────────────────────────────
describe('ultraplan-phase workflow runtime gate', () => {
const content = fs.readFileSync(WF_PATH, 'utf-8');
test('checks CLAUDE_CODE_VERSION to detect Claude Code runtime', () => {
assert.ok(content.includes('CLAUDE_CODE_VERSION'), 'workflow must gate on CLAUDE_CODE_VERSION env var');
});
test('error message references /gsd-plan-phase as local alternative', () => {
assert.ok(
content.includes('gsd-plan-phase'),
'error message should direct users to /gsd-plan-phase as the local alternative'
);
});
});
// ─── Workflow: Initialization ──────────────────────────────────────────────────
describe('ultraplan-phase workflow initialization', () => {
const content = fs.readFileSync(WF_PATH, 'utf-8');
test('loads GSD phase context via gsd-sdk query init.plan-phase', () => {
assert.ok(content.includes('gsd-sdk query init.plan-phase'), 'workflow must load phase context via gsd-sdk query init.plan-phase');
});
test('handles missing .planning directory', () => {
assert.ok(
content.includes('gsd-new-project') || content.includes('/gsd-new-project'),
'workflow should direct user to /gsd-new-project when .planning is missing'
);
});
});
// ─── Workflow: Ultraplan Prompt ────────────────────────────────────────────────
describe('ultraplan-phase workflow prompt construction', () => {
const content = fs.readFileSync(WF_PATH, 'utf-8');
test('includes phase scope from ROADMAP in ultraplan prompt', () => {
assert.ok(
content.includes('ROADMAP') || content.includes('phase scope') || content.includes('phase_name'),
'workflow should include phase scope from ROADMAP.md in the ultraplan prompt'
);
});
test('includes REQUIREMENTS context in ultraplan prompt', () => {
assert.ok(content.includes('REQUIREMENTS'), 'workflow should include requirements context in the ultraplan prompt');
});
test('includes existing RESEARCH when available', () => {
assert.ok(
content.includes('RESEARCH') || content.includes('research_path'),
'workflow should include existing research in the ultraplan prompt'
);
});
});
// ─── Workflow: Ultraplan Trigger ───────────────────────────────────────────────
describe('ultraplan-phase workflow ultraplan trigger', () => {
const content = fs.readFileSync(WF_PATH, 'utf-8');
test('triggers /ultraplan command', () => {
assert.ok(content.includes('/ultraplan'), 'workflow must trigger /ultraplan');
});
});
// ─── Workflow: Return Path Instructions ───────────────────────────────────────
describe('ultraplan-phase workflow return path', () => {
const content = fs.readFileSync(WF_PATH, 'utf-8');
test('instructs user to choose Cancel to save plan to file', () => {
assert.ok(content.includes('Cancel'), 'workflow must instruct user to choose Cancel to save the plan to a file');
});
test('directs user to run /gsd-import --from after ultraplan completes', () => {
assert.ok(content.includes('gsd-import'), 'workflow must direct user to run /gsd-import --from with the saved file path');
});
test('mentions the --from flag for gsd-import', () => {
assert.ok(content.includes('--from'), 'workflow should reference /gsd-import --from <file-path>');
});
test('return-path instructions appear before the /ultraplan trigger', () => {
const ultraplanTriggerIndex = content.indexOf('/ultraplan');
const importIndex = content.indexOf('gsd-import');
assert.ok(
importIndex < ultraplanTriggerIndex,
'return-path instructions (gsd-import) must appear before /ultraplan trigger so they are visible in scroll-back'
);
});
});
// ─── Workflow: Isolation from Core Pipeline ────────────────────────────────────
describe('ultraplan-phase workflow isolation', () => {
const content = fs.readFileSync(WF_PATH, 'utf-8');
test('does NOT directly write PLAN.md files', () => {
assert.ok(
!content.includes('write PLAN.md') && !content.includes('Write(\'.planning'),
'workflow must NOT directly write PLAN.md — delegates to /gsd-import --from'
);
});
test('does NOT reference ultrareview', () => {
assert.ok(!content.includes('ultrareview'), 'workflow must not reference ultrareview');
});
});