mirror of https://github.com/garrytan/gstack.git
199 lines
7.2 KiB
TypeScript
199 lines
7.2 KiB
TypeScript
/**
|
|
* Regression pin for the setup-time gbrain detection → gen-skill-docs
|
|
* override (T2 / v1.50.0.0).
|
|
*
|
|
* The override mechanism lives in scripts/gen-skill-docs.ts: when invoked
|
|
* with --respect-detection, it reads ~/.gstack/gbrain-detection.json and
|
|
* un-suppresses GBRAIN_CONTEXT_LOAD + GBRAIN_SAVE_RESULTS for hosts that
|
|
* statically list them in suppressedResolvers (claude, codex, slate,
|
|
* factory, opencode, openclaw, cursor, kiro).
|
|
*
|
|
* Tests drive gen-skill-docs as a subprocess against a temp GSTACK_HOME
|
|
* with each detection state, then assert what landed in the generated
|
|
* Claude-host SKILL.md. This is end-to-end through the actual override
|
|
* pipeline — no mocking — so it catches regressions in either the loader
|
|
* or the suppressedResolvers filter.
|
|
*
|
|
* Gate-tier, free, ~3-5s per test (gen-skill-docs runs the full skill
|
|
* generation against the real repo; --host claude scopes to one host).
|
|
*/
|
|
|
|
import { describe, test, expect, beforeAll, afterAll } from 'bun:test';
|
|
import { execFileSync } from 'child_process';
|
|
import { mkdtempSync, mkdirSync, readFileSync, rmSync, writeFileSync } from 'fs';
|
|
import { tmpdir } from 'os';
|
|
import { join } from 'path';
|
|
|
|
const REPO_ROOT = join(import.meta.dir, '..');
|
|
|
|
interface FixtureEnv {
|
|
tmpHome: string;
|
|
cleanup: () => void;
|
|
}
|
|
|
|
function makeFixture(detectionJson: string | null): FixtureEnv {
|
|
const tmpHome = mkdtempSync(join(tmpdir(), 'gbrain-detect-test-'));
|
|
if (detectionJson !== null) {
|
|
writeFileSync(join(tmpHome, 'gbrain-detection.json'), detectionJson);
|
|
}
|
|
return {
|
|
tmpHome,
|
|
cleanup: () => {
|
|
try {
|
|
rmSync(tmpHome, { recursive: true, force: true });
|
|
} catch {
|
|
// best effort
|
|
}
|
|
},
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Run gen-skill-docs with --respect-detection and an isolated GSTACK_HOME.
|
|
* Returns the regenerated office-hours/SKILL.md content WITHOUT writing
|
|
* over the committed file: we use --dry-run to keep the working tree
|
|
* clean, then parse the output via re-reading the committed file... no,
|
|
* that doesn't work for dry-run since dry-run doesn't write.
|
|
*
|
|
* Approach: generate to a temp output dir by running gen-skill-docs in a
|
|
* temp checkout. Simpler alternative: actually regenerate, snapshot the
|
|
* file content, then git-checkout the committed version back. We use this
|
|
* since gen-skill-docs doesn't expose an output-path arg.
|
|
*/
|
|
function regenAndSnapshot(opts: {
|
|
respectDetection: boolean;
|
|
tmpHome: string;
|
|
files: string[];
|
|
}): Map<string, string> {
|
|
// Save committed content so we can restore after snapshotting.
|
|
const original = new Map<string, string>();
|
|
for (const f of opts.files) {
|
|
original.set(f, readFileSync(join(REPO_ROOT, f), 'utf-8'));
|
|
}
|
|
|
|
const args = [
|
|
'run',
|
|
'scripts/gen-skill-docs.ts',
|
|
'--host',
|
|
'claude',
|
|
];
|
|
if (opts.respectDetection) args.push('--respect-detection');
|
|
|
|
try {
|
|
execFileSync('bun', args, {
|
|
cwd: REPO_ROOT,
|
|
env: { ...process.env, GSTACK_HOME: opts.tmpHome },
|
|
stdio: ['ignore', 'pipe', 'pipe'],
|
|
timeout: 30_000,
|
|
});
|
|
|
|
// Snapshot the regenerated content.
|
|
const snapshot = new Map<string, string>();
|
|
for (const f of opts.files) {
|
|
snapshot.set(f, readFileSync(join(REPO_ROOT, f), 'utf-8'));
|
|
}
|
|
return snapshot;
|
|
} finally {
|
|
// Always restore so the test leaves the working tree clean.
|
|
for (const [f, content] of original) {
|
|
writeFileSync(join(REPO_ROOT, f), content);
|
|
}
|
|
}
|
|
}
|
|
|
|
describe('gbrain detection override → gen-skill-docs', () => {
|
|
// Single skill probe is enough to assert the override pipeline. The
|
|
// resolver unit test (test/resolvers-gbrain-save-results.test.ts) covers
|
|
// per-skill metadata correctness already.
|
|
// office-hours is carved (v2 plan T9): GBRAIN_CONTEXT_LOAD stays in the
|
|
// skeleton, GBRAIN_SAVE_RESULTS moved into sections/design-and-handoff.md.
|
|
// Probe the union so the detection override is asserted wherever the blocks land.
|
|
const PROBE_FILES = ['office-hours/SKILL.md', 'office-hours/sections/design-and-handoff.md'];
|
|
const probeUnion = (snap: Map<string, string>): string =>
|
|
(snap.get('office-hours/SKILL.md') ?? '') + '\n' + (snap.get('office-hours/sections/design-and-handoff.md') ?? '');
|
|
|
|
test('with detected:true, Claude-host SKILL.md gains brain-aware blocks', () => {
|
|
const { tmpHome, cleanup } = makeFixture(
|
|
JSON.stringify({ gbrain_local_status: 'ok', gbrain_on_path: true, gbrain_version: 'test-0.41.0' }),
|
|
);
|
|
try {
|
|
const snap = regenAndSnapshot({
|
|
respectDetection: true,
|
|
tmpHome,
|
|
files: PROBE_FILES,
|
|
});
|
|
const content = probeUnion(snap);
|
|
|
|
// GBRAIN_SAVE_RESULTS un-suppressed → resolver output rendered.
|
|
expect(content).toContain('## Save Results to Brain');
|
|
expect(content).toContain('gbrain put "office-hours/');
|
|
expect(content).toContain('Skip this entire section if `gbrain` is not on PATH');
|
|
|
|
// GBRAIN_CONTEXT_LOAD also un-suppressed (D6 bundling).
|
|
expect(content).toContain('## Brain Context Load');
|
|
} finally {
|
|
cleanup();
|
|
}
|
|
});
|
|
|
|
test('with detected:false (status != "ok"), brain blocks stay suppressed', () => {
|
|
const { tmpHome, cleanup } = makeFixture(
|
|
JSON.stringify({ gbrain_local_status: 'no-cli', gbrain_on_path: false, gbrain_version: null }),
|
|
);
|
|
try {
|
|
const snap = regenAndSnapshot({
|
|
respectDetection: true,
|
|
tmpHome,
|
|
files: PROBE_FILES,
|
|
});
|
|
const content = probeUnion(snap);
|
|
|
|
// GBRAIN_SAVE_RESULTS suppressed → no rendered block, no gbrain put line.
|
|
expect(content).not.toContain('gbrain put "office-hours/');
|
|
// Section header from the resolver also absent (resolver returns "").
|
|
// BUT — the BRAIN_CACHE_REFRESH and BRAIN_WRITE_BACK resolvers are NOT
|
|
// gated by detection (host-agnostic), so other "Brain ..." sections may
|
|
// still appear. We only assert the SAVE_RESULTS-specific marker is gone.
|
|
} finally {
|
|
cleanup();
|
|
}
|
|
});
|
|
|
|
test('with NO detection file, brain blocks stay suppressed (same as detected:false)', () => {
|
|
const { tmpHome, cleanup } = makeFixture(null);
|
|
try {
|
|
const snap = regenAndSnapshot({
|
|
respectDetection: true,
|
|
tmpHome,
|
|
files: PROBE_FILES,
|
|
});
|
|
const content = probeUnion(snap);
|
|
expect(content).not.toContain('gbrain put "office-hours/');
|
|
} finally {
|
|
cleanup();
|
|
}
|
|
});
|
|
|
|
test('without --respect-detection flag, detection file is IGNORED (CI canonical path)', () => {
|
|
// Even if a detection file exists with detected:true, the default
|
|
// `bun run gen:skill-docs` (CI) must produce no-gbrain output so the
|
|
// committed SKILL.md stays reproducible regardless of any developer's
|
|
// local gbrain install state.
|
|
const { tmpHome, cleanup } = makeFixture(
|
|
JSON.stringify({ gbrain_local_status: 'ok', gbrain_on_path: true, gbrain_version: 'test-0.41.0' }),
|
|
);
|
|
try {
|
|
const snap = regenAndSnapshot({
|
|
respectDetection: false,
|
|
tmpHome,
|
|
files: PROBE_FILES,
|
|
});
|
|
const content = probeUnion(snap);
|
|
expect(content).not.toContain('gbrain put "office-hours/');
|
|
expect(content).not.toContain('## Save Results to Brain');
|
|
} finally {
|
|
cleanup();
|
|
}
|
|
});
|
|
});
|