/** * /plan-design-review per-finding AskUserQuestion count (periodic, paid, real-PTY). * * Same shape as skill-e2e-plan-ceo-finding-count: drives /plan-design-review * against a 5-finding seeded plan and asserts review-phase AUQ count ∈ [N-1, N+2]. * Plus D19: review report at bottom of produced plan file. * * Tier: periodic (~25 min, ~$5/run). Sequential by default per plan §D15. */ import { describe, test } from 'bun:test'; import * as fs from 'node:fs'; import { runPlanSkillCounting, designStep0Boundary, assertReviewReportAtBottom, } from './helpers/claude-pty-runner'; const shouldRun = !!process.env.EVALS && process.env.EVALS_TIER === 'periodic'; const describeE2E = shouldRun ? describe : describe.skip; const N = 5; const FLOOR = N - 1; const CEILING = N + 2; const PLAN_DESIGN_5_FINDINGS = [ 'Please review this plan thoroughly. As you go, write your plan-mode plan to /tmp/gstack-test-plan-design.md (use Edit/Write to that exact path).', '', '# Plan: Settings Page UI redesign', '', '## Visual Hierarchy', 'The "Save" button is rendered with the same size, weight, and color as', 'three other buttons in the page header (Reset, Cancel, Export). Nothing', 'tells the user which is the primary action.', '', '## Spacing', 'Between sections we have 24px in some places, 32px in others, and 16px', 'in a third — no consistent vertical rhythm.', '', '## Color', 'The error message uses red text on a light pink background. Contrast', 'ratio is approximately 3:1 (below WCAG AA).', '', '## Typography', 'We use 14px, 16px, and 18px font sizes across the form labels. Two', 'sizes would suffice and create stronger hierarchy.', '', '## Motion', 'The "Save" action takes 2-5 seconds with no loading indicator. Users', 'see a frozen page; we should add a spinner or skeleton state.', ].join('\n'); const PLAN_DESIGN_PATH = '/tmp/gstack-test-plan-design.md'; describeE2E('/plan-design-review per-finding AskUserQuestion count (periodic)', () => { test( `5-finding plan emits ${FLOOR}-${CEILING} review-phase AskUserQuestions`, async () => { try { fs.rmSync(PLAN_DESIGN_PATH, { force: true }); } catch { /* best-effort */ } const obs = await runPlanSkillCounting({ skillName: 'plan-design-review', slashCommand: '/plan-design-review', followUpPrompt: PLAN_DESIGN_5_FINDINGS, isLastStep0AUQ: designStep0Boundary, reviewCountCeiling: CEILING + 1, cwd: process.cwd(), timeoutMs: 1_500_000, env: { QUESTION_TUNING: 'false', EXPLAIN_LEVEL: 'default' }, }); try { if (!['plan_ready', 'completion_summary', 'ceiling_reached'].includes(obs.outcome)) { throw new Error( `plan-design-review finding-count FAILED: outcome=${obs.outcome}\n` + `step0=${obs.step0Count} review=${obs.reviewCount} elapsed=${obs.elapsedMs}ms\n` + `fingerprints (last 8):\n` + obs.fingerprints .slice(-8) .map( (f, i) => ` ${i}. preReview=${f.preReview} sig=${f.signature.slice(0, 12)} prompt="${f.promptSnippet.slice(0, 60)}"`, ) .join('\n') + `\n--- evidence (last 3KB) ---\n${obs.evidence}`, ); } if (obs.reviewCount < FLOOR) { throw new Error( `BAND FAIL (below floor): reviewCount=${obs.reviewCount} < FLOOR=${FLOOR}.\n` + `Likely batching regression. Review-phase fingerprints:\n` + obs.fingerprints .filter((f) => !f.preReview) .map((f) => ` - "${f.promptSnippet.slice(0, 80)}"`) .join('\n'), ); } if (obs.reviewCount > CEILING) { throw new Error( `BAND FAIL (above ceiling): reviewCount=${obs.reviewCount} > CEILING=${CEILING}.`, ); } if (!fs.existsSync(PLAN_DESIGN_PATH)) { throw new Error( `D19 FAIL: agent did not produce expected plan file at ${PLAN_DESIGN_PATH}. ` + `outcome=${obs.outcome} review=${obs.reviewCount}`, ); } const planContent = fs.readFileSync(PLAN_DESIGN_PATH, 'utf-8'); const verdict = assertReviewReportAtBottom(planContent); if (!verdict.ok) { throw new Error( `D19 FAIL: plan file at ${PLAN_DESIGN_PATH} ${verdict.reason}\n` + (verdict.trailingHeadings ? `Trailing headings: ${verdict.trailingHeadings.join(' | ')}\n` : '') + `--- plan content (last 1KB) ---\n${planContent.slice(-1024)}`, ); } } finally { try { fs.rmSync(PLAN_DESIGN_PATH, { force: true }); } catch { /* best-effort */ } } }, 1_700_000, ); });