mirror of
https://github.com/garrytan/gstack.git
synced 2026-06-17 07:10:12 +02:00
8241949357
* feat(gbrain-detect): add --is-ok live-detection exit-code gate Single source of truth for 'is gbrain usable'. Runs live detection (never reads the possibly-stale gbrain-detection.json) and exits 0 iff status is ok, so setup, bin/dev-setup, and gstack-config can gate brain-aware rendering on one shared check instead of re-grepping the JSON. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com> * feat(gen-skill-docs): add --out-dir with surgical section-path rewrite --out-dir <abs-dir> mirrors the Claude skill tree (SKILL.md + sections) into a separate directory instead of writing in place, and rewrites the literal section-base path (~/.claude/skills/gstack/<skill>/sections/) in generated content to point at the out-dir. The rewrite is surgical: only /sections/ paths move; bin/, browse/, docs/ references stay pointed at the global install. Global extras (proactive-suggestions.json) are skipped in out-dir mode. Default (no flag) behavior is unchanged. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com> * feat(dev-setup): render gbrain :user variant to an untracked workspace dir 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> * feat(dev-skill): refresh the untracked brain-aware render on template change After the default in-place regen (which keeps the worktree canonical and runs validation), also re-render the :user variant into .claude/gstack-rendered when it exists, so live template edits reflect at the workspace's runtime. Never creates the render dir during plain template dev. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com> * feat(gstack-config): gbrain-refresh renders brain-aware blocks into the install Extends gbrain-refresh to render the :user variant into the global install (~/.claude/skills/gstack) so every project's Claude sessions get brain-aware blocks, not just the gstack dev workspace. Guarded against mutating the wrong directory: the target must exist, not be a symlink (a symlinked install points at a dev worktree), and look like a real gstack clone (VERSION + package.json). Idempotent and self-documenting. CLAUDE.md's deploy section now notes that 'git reset --hard' reverts the blocks and to re-run gbrain-refresh. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com> * test: cover gstack-gbrain-detect --is-ok + dev-skill render refresh Fills the two automated-coverage gaps from the eng review: --is-ok exit-code gate (no-cli -> nonzero, healthy -> 0, plus an agrees-with-JSON no-skew check reusing the deterministic fake-gbrain harness) and a static tripwire that dev-skill re-renders the :user variant into the workspace render dir only when it already exists. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com> * chore: bump version and changelog (v1.57.9.0) Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com> * docs: document brain-aware dev-setup render for v1.57.9.0 Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
85 lines
3.8 KiB
TypeScript
85 lines
3.8 KiB
TypeScript
import { describe, test, expect } from 'bun:test';
|
|
import { spawnSync } from 'child_process';
|
|
import { createHash } from 'crypto';
|
|
import * as path from 'path';
|
|
import * as fs from 'fs';
|
|
import * as os from 'os';
|
|
|
|
const ROOT = path.resolve(import.meta.dir, '..');
|
|
|
|
// Render the gbrain `:user` variant into a temp out-dir, forcing detection ON
|
|
// via a crafted GSTACK_HOME so the test is deterministic regardless of whether
|
|
// the dev machine actually has gbrain installed. Asserts the B2 contract:
|
|
// (a) the worktree SKILL.md is byte-unchanged (source stays canonical),
|
|
// (b) the out-dir SKILL.md gained the inline Brain Context Load block,
|
|
// (c) its section refs point at the out-dir, not ~/.claude/skills/gstack,
|
|
// (d) bin/ refs are left pointing at the global install,
|
|
// (e) the out-dir section file gained the Save Results to Brain block.
|
|
describe('gen-skill-docs --out-dir (B2 render isolation)', () => {
|
|
function hashFile(p: string): string {
|
|
return createHash('sha256').update(fs.readFileSync(p)).digest('hex');
|
|
}
|
|
|
|
test('renders :user to out-dir, rewrites section paths, leaves worktree canonical', () => {
|
|
const tmpHome = fs.mkdtempSync(path.join(os.tmpdir(), 'gstack-home-'));
|
|
const outDir = fs.mkdtempSync(path.join(os.tmpdir(), 'gstack-out-'));
|
|
const worktreeSkill = path.join(ROOT, 'ship', 'SKILL.md');
|
|
const beforeHash = hashFile(worktreeSkill);
|
|
try {
|
|
// Force gbrain detection ON for --respect-detection.
|
|
fs.writeFileSync(
|
|
path.join(tmpHome, 'gbrain-detection.json'),
|
|
JSON.stringify({ gbrain_local_status: 'ok', gbrain_version: '9.9.9' }),
|
|
);
|
|
|
|
const res = spawnSync(
|
|
'bun',
|
|
['run', 'scripts/gen-skill-docs.ts', '--respect-detection', '--host', 'claude', '--out-dir', outDir],
|
|
{ cwd: ROOT, encoding: 'utf-8', timeout: 120_000, env: { ...process.env, GSTACK_HOME: tmpHome } },
|
|
);
|
|
expect(res.status).toBe(0);
|
|
|
|
const outSkill = path.join(outDir, 'ship', 'SKILL.md');
|
|
const outSection = path.join(outDir, 'ship', 'sections', 'adversarial.md');
|
|
expect(fs.existsSync(outSkill)).toBe(true);
|
|
const skillContent = fs.readFileSync(outSkill, 'utf-8');
|
|
|
|
// (a) worktree byte-unchanged
|
|
expect(hashFile(worktreeSkill)).toBe(beforeHash);
|
|
|
|
// (b) inline block present in the rendered SKILL.md
|
|
expect(skillContent).toContain('Brain Context Load');
|
|
|
|
// (c) section refs repointed to the out-dir; none left pointing at the install
|
|
expect(skillContent).toContain(`${outDir}/ship/sections/`);
|
|
expect(skillContent).not.toContain('~/.claude/skills/gstack/ship/sections/');
|
|
|
|
// (d) bin refs are NOT rewritten — they still resolve to the global install
|
|
expect(skillContent).toContain('~/.claude/skills/gstack/bin/');
|
|
|
|
// (e) the SAVE block landed in the rendered section file
|
|
expect(fs.existsSync(outSection)).toBe(true);
|
|
expect(fs.readFileSync(outSection, 'utf-8')).toContain('Save Results to Brain');
|
|
} finally {
|
|
fs.rmSync(tmpHome, { recursive: true, force: true });
|
|
fs.rmSync(outDir, { recursive: true, force: true });
|
|
}
|
|
});
|
|
|
|
test('global extras (proactive-suggestions.json) are NOT written in out-dir mode', () => {
|
|
const outDir = fs.mkdtempSync(path.join(os.tmpdir(), 'gstack-out-'));
|
|
try {
|
|
const res = spawnSync(
|
|
'bun',
|
|
['run', 'scripts/gen-skill-docs.ts', '--host', 'claude', '--out-dir', outDir],
|
|
{ cwd: ROOT, encoding: 'utf-8', timeout: 120_000 },
|
|
);
|
|
expect(res.status).toBe(0);
|
|
// proactive-suggestions.json lives at a repo path; out-dir mode must skip it.
|
|
expect(fs.existsSync(path.join(outDir, 'scripts', 'proactive-suggestions.json'))).toBe(false);
|
|
} finally {
|
|
fs.rmSync(outDir, { recursive: true, force: true });
|
|
}
|
|
});
|
|
});
|