mirror of https://github.com/garrytan/gstack.git
171 lines
7.0 KiB
TypeScript
171 lines
7.0 KiB
TypeScript
/**
|
|
* AUQ behavioral matrix — drive each AUQ-heavy skill to its first
|
|
* AskUserQuestion and grade it to plan-ceo's bar (periodic, paid, SDK capture).
|
|
*
|
|
* Layer 0 (auq-format-always-loaded.test.ts) deterministically guarantees every
|
|
* skill SHIPS the format spec in its always-loaded skeleton. This test proves
|
|
* each skill's model OBEYS it: that the first real AUQ each skill fires is a
|
|
* compliant decision brief (all 7 format elements) with a substantive
|
|
* recommendation (>= 4). One parametrized case per skill so a single weak skill
|
|
* is an isolated failure, not a blocker for the rest.
|
|
*
|
|
* Capture is the SDK $OUT_FILE path (clean text, no TTY mangling), with the skill
|
|
* pinned to an absolute path and the agent restricted to Read/Write so it can't
|
|
* wander to the global install. See test/helpers/auq-sdk-capture.ts.
|
|
*
|
|
* Scope: skills whose first AUQ is reliably reachable from a text fixture. Skills
|
|
* that gate their first decision on external resources (a running browser for
|
|
* /qa, the design binary + comparison boards for /design-shotgun and
|
|
* /design-html — which by project policy use $D compare, not AUQ, for variant
|
|
* choices) are intentionally OUT of this matrix; Layer 0 covers their format
|
|
* spec, and a fixture can't fairly trigger their AUQ.
|
|
*
|
|
* Run a subset in the foreground with AUQ_MATRIX_ONLY="plan-eng-review,cso".
|
|
*/
|
|
import { describe, test } from 'bun:test';
|
|
import * as fs from 'node:fs';
|
|
import {
|
|
setupSkillDir,
|
|
captureFirstAuq,
|
|
scoreAuqFormat,
|
|
skillFromWorktree,
|
|
gradeAuqRecommendation,
|
|
} from './helpers/auq-sdk-capture';
|
|
|
|
const shouldRun = !!process.env.EVALS && process.env.EVALS_TIER === 'periodic';
|
|
const describeE2E = shouldRun ? describe : describe.skip;
|
|
const runId = `auq-matrix-${process.env.EVALS_RUN_ID ?? 'local'}`;
|
|
const ONLY = (process.env.AUQ_MATRIX_ONLY ?? '').split(',').map(s => s.trim()).filter(Boolean);
|
|
|
|
const FLAWED_PLAN = `# Plan: Launch a "developer-friendly" pricing tier
|
|
|
|
## Goal
|
|
Increase developer adoption.
|
|
|
|
## Success metric
|
|
More signups.
|
|
|
|
## Premise
|
|
We haven't talked to any developers about whether price is the barrier. The team
|
|
agreed it "feels like" it should be cheaper. We'll add a new Stripe tier, a React
|
|
pricing page, a Postgres entitlements table, and a Redis cache — no tests
|
|
mentioned, no rollout plan, no auth check on the upgrade endpoint.
|
|
`;
|
|
|
|
const VULN_CODE = `export function login(req, res) {
|
|
// builds SQL by string concat; sets a session cookie with no flags
|
|
const user = db.query("SELECT * FROM users WHERE name = '" + req.body.name + "'");
|
|
if (user && user.password === req.body.password) {
|
|
res.cookie('session', user.id); // no HttpOnly, Secure, SameSite, or expiry
|
|
return res.json({ ok: true });
|
|
}
|
|
return res.status(401).json({ ok: false });
|
|
}
|
|
`;
|
|
|
|
interface MatrixSkill {
|
|
skill: string;
|
|
fixtures: Record<string, string>;
|
|
scenario: string;
|
|
}
|
|
|
|
const MATRIX: MatrixSkill[] = [
|
|
{
|
|
skill: 'plan-eng-review',
|
|
fixtures: { 'plan.md': FLAWED_PLAN },
|
|
scenario: 'Read plan.md — that is the plan to review. It is a standalone plan document, not a codebase. Walk the review until the first AskUserQuestion (a per-issue finding or a scope decision).',
|
|
},
|
|
{
|
|
skill: 'plan-design-review',
|
|
fixtures: { 'plan.md': FLAWED_PLAN + '\n## UI\nA new pricing page with a comparison table, plan cards, and an upgrade modal.\n' },
|
|
scenario: 'Read plan.md — that is the plan to review (it has UI scope). Walk the review until the first AskUserQuestion.',
|
|
},
|
|
{
|
|
skill: 'plan-devex-review',
|
|
fixtures: { 'plan.md': FLAWED_PLAN + '\n## CLI\nShip a `mytool pricing` command and a setup wizard for the new tier.\n' },
|
|
scenario: 'Read plan.md — that is the plan to review (developer-experience scope). Walk the review until the first AskUserQuestion.',
|
|
},
|
|
{
|
|
skill: 'office-hours',
|
|
fixtures: {},
|
|
scenario: 'The founder says: "I am building an AI tool that auto-writes unit tests for any repo. I think it is a great idea but I have zero users. Should I build it, and how do I get my first users?" Run the office-hours diagnostic until the first AskUserQuestion.',
|
|
},
|
|
{
|
|
skill: 'cso',
|
|
fixtures: { 'server/auth.js': VULN_CODE },
|
|
scenario: 'Audit the code in this repo (server/auth.js) for security issues. Walk the audit until the first AskUserQuestion (scope/stack confirmation or first finding).',
|
|
},
|
|
{
|
|
skill: 'spec',
|
|
fixtures: {},
|
|
scenario: 'Turn this vague intent into a precise spec: "add email notifications when a task is assigned to someone." Walk the spec workflow until the first AskUserQuestion.',
|
|
},
|
|
{
|
|
skill: 'design-consultation',
|
|
fixtures: { 'product.md': '# Product\nA terminal-first task manager for developers. Audience: senior engineers. Stage: pre-launch.\n' },
|
|
scenario: 'Read product.md. Run the design consultation for this product until the first AskUserQuestion.',
|
|
},
|
|
];
|
|
|
|
const selected = ONLY.length ? MATRIX.filter(m => ONLY.includes(m.skill)) : MATRIX;
|
|
|
|
describeE2E('AUQ behavioral matrix (periodic)', () => {
|
|
for (const m of selected) {
|
|
test(
|
|
`${m.skill}: first AUQ is a compliant decision brief (7/7 format, substance >=4)`,
|
|
async () => {
|
|
const wt = skillFromWorktree(m.skill);
|
|
const dir = setupSkillDir({
|
|
skillName: m.skill,
|
|
skillMd: wt.skillMd,
|
|
sectionsFrom: wt.sectionsFrom,
|
|
fixtures: m.fixtures,
|
|
tmpPrefix: `auq-matrix-${m.skill}-`,
|
|
});
|
|
let text = '';
|
|
try {
|
|
text = await captureFirstAuq({
|
|
planDir: dir,
|
|
skillName: m.skill,
|
|
scenario: m.scenario,
|
|
testName: `auq-matrix-${m.skill}`,
|
|
runId,
|
|
});
|
|
} finally {
|
|
fs.rmSync(dir, { recursive: true, force: true });
|
|
}
|
|
|
|
const fmt = scoreAuqFormat(text);
|
|
let substance = 0;
|
|
let recPresent = false;
|
|
let hadBecause = false;
|
|
if (text.trim()) {
|
|
const g = await gradeAuqRecommendation(text);
|
|
substance = g.substance;
|
|
recPresent = g.present;
|
|
hadBecause = g.hadLiteralBecause;
|
|
}
|
|
// eslint-disable-next-line no-console
|
|
console.log(
|
|
`[AUQ-matrix ${m.skill}] captured=${text.length}B format=${fmt.present}/${fmt.total} ` +
|
|
`missing=[${fmt.missing.join(',')}] recPresent=${recPresent} substance=${substance} ` +
|
|
`literalBecause=${hadBecause}`,
|
|
);
|
|
|
|
if (!text.trim()) {
|
|
throw new Error(`${m.skill}: agent produced NO AUQ capture (never reached a question in budget).`);
|
|
}
|
|
const problems: string[] = [];
|
|
if (fmt.missing.length > 0) problems.push(`missing format element(s): ${fmt.missing.join(', ')}`);
|
|
if (substance < 4) problems.push(`recommendation substance ${substance} < 4 (boilerplate/weak)`);
|
|
if (problems.length > 0) {
|
|
throw new Error(
|
|
`${m.skill} AUQ not at plan-ceo bar:\n - ${problems.join('\n - ')}\n--- captured AUQ ---\n${text}`,
|
|
);
|
|
}
|
|
},
|
|
300_000,
|
|
);
|
|
}
|
|
});
|