mirror of
https://github.com/garrytan/gstack.git
synced 2026-06-17 15:20:11 +02:00
3625f4c721
Stops the dev/Conductor workspace from dirtying tracked SKILL.md source. setup honors GSTACK_SKIP_GBRAIN_REGEN (passed inline by dev-setup, never exported) and skips the in-place :user regen; detection is still persisted (PID-unique tmp so concurrent workspaces can't clobber it). dev-setup instead renders the :user variant into .claude/gstack-rendered (gitignored, per-workspace) and repoints the workspace SKILL.md symlinks at it, so the workspace gets brain-aware blocks while the worktree stays canonical. dev-teardown removes the render. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
78 lines
2.9 KiB
TypeScript
78 lines
2.9 KiB
TypeScript
import { describe, test, expect } from 'bun:test';
|
|
import * as path from 'path';
|
|
import * as fs from 'fs';
|
|
|
|
// Static tripwires for the B2 render-isolation wiring. These fail CI if a
|
|
// refactor drops a load-bearing line, re-introducing the "dev-setup dirties
|
|
// tracked SKILL.md" drift (or worse, leaks the skip-guard into real installs).
|
|
const ROOT = path.resolve(import.meta.dir, '..');
|
|
const read = (rel: string) => fs.readFileSync(path.join(ROOT, rel), 'utf-8');
|
|
|
|
describe('dev-setup: worktree stays canonical', () => {
|
|
const devSetup = read('bin/dev-setup');
|
|
|
|
test('passes GSTACK_SKIP_GBRAIN_REGEN inline on the nested setup call', () => {
|
|
expect(devSetup).toContain('GSTACK_SKIP_GBRAIN_REGEN=1 "$GSTACK_LINK/setup"');
|
|
});
|
|
|
|
test('never exports GSTACK_SKIP_GBRAIN_REGEN (would leak into other setup paths)', () => {
|
|
expect(devSetup).not.toMatch(/export\s+GSTACK_SKIP_GBRAIN_REGEN/);
|
|
});
|
|
|
|
test('renders the :user variant into an out-dir, not in place', () => {
|
|
expect(devSetup).toContain('--out-dir');
|
|
expect(devSetup).toContain('.claude/gstack-rendered');
|
|
});
|
|
|
|
test('gates the render on gstack-gbrain-detect --is-ok', () => {
|
|
expect(devSetup).toContain('--is-ok');
|
|
});
|
|
});
|
|
|
|
describe('setup: honors GSTACK_SKIP_GBRAIN_REGEN', () => {
|
|
const setup = read('setup');
|
|
|
|
test('skips the in-place :user regen when the guard is set', () => {
|
|
expect(setup).toContain('${GSTACK_SKIP_GBRAIN_REGEN:-}');
|
|
// The guard must wrap the in-place render, not the detection persist.
|
|
const idx = setup.indexOf('GSTACK_SKIP_GBRAIN_REGEN');
|
|
const after = setup.slice(idx, idx + 600);
|
|
expect(after).toContain('leaving tracked SKILL.md canonical');
|
|
});
|
|
|
|
test('uses a PID-unique detection tmp (no concurrent clobber)', () => {
|
|
expect(setup).toContain('$DETECTION_FILE.$$.tmp');
|
|
});
|
|
|
|
test('gates detection on the shared --is-ok check', () => {
|
|
expect(setup).toContain('"$DETECT_BIN" --is-ok');
|
|
});
|
|
});
|
|
|
|
describe('gen-skill-docs: section rewrite is gated on --out-dir', () => {
|
|
const gen = read('scripts/gen-skill-docs.ts');
|
|
|
|
test('rewriteSectionBase is a no-op without --out-dir', () => {
|
|
expect(gen).toContain('function rewriteSectionBase');
|
|
const idx = gen.indexOf('function rewriteSectionBase');
|
|
const body = gen.slice(idx, idx + 400);
|
|
expect(body).toContain('if (!OUT_DIR) return content');
|
|
expect(body).toContain('sections'); // surgical: regex targets only /sections/ paths
|
|
});
|
|
});
|
|
|
|
describe('dev-teardown: removes the untracked render', () => {
|
|
const teardown = read('bin/dev-teardown');
|
|
|
|
test('rm -rf the gstack-rendered dir', () => {
|
|
expect(teardown).toContain('gstack-rendered');
|
|
expect(teardown).toMatch(/rm -rf .*RENDER_DIR/);
|
|
});
|
|
});
|
|
|
|
describe('.gitignore: render dir is declared untracked', () => {
|
|
test('.claude/gstack-rendered/ is ignored', () => {
|
|
expect(read('.gitignore')).toContain('.claude/gstack-rendered/');
|
|
});
|
|
});
|