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>
102 lines
3.3 KiB
TypeScript
102 lines
3.3 KiB
TypeScript
#!/usr/bin/env bun
|
|
/**
|
|
* dev:skill — Watch mode for SKILL.md template development.
|
|
*
|
|
* Watches .tmpl files, regenerates SKILL.md files on change,
|
|
* validates all $B commands immediately.
|
|
*/
|
|
|
|
import { validateSkill } from '../test/helpers/skill-parser';
|
|
import { discoverTemplates } from './discover-skills';
|
|
import { execSync } from 'child_process';
|
|
import * as fs from 'fs';
|
|
import * as path from 'path';
|
|
|
|
const ROOT = path.resolve(import.meta.dir, '..');
|
|
|
|
const TEMPLATES = discoverTemplates(ROOT).map(t => ({
|
|
tmpl: path.join(ROOT, t.tmpl),
|
|
output: t.output,
|
|
}));
|
|
|
|
function regenerateAndValidate() {
|
|
// Regenerate
|
|
try {
|
|
execSync('bun run scripts/gen-skill-docs.ts', { cwd: ROOT, stdio: 'pipe' });
|
|
} catch (err: any) {
|
|
console.log(` [gen] ERROR: ${err.stderr?.toString().trim() || err.message}`);
|
|
return;
|
|
}
|
|
|
|
// Validate each generated file
|
|
for (const { output } of TEMPLATES) {
|
|
const fullPath = path.join(ROOT, output);
|
|
if (!fs.existsSync(fullPath)) continue;
|
|
|
|
const result = validateSkill(fullPath);
|
|
const totalValid = result.valid.length;
|
|
const totalInvalid = result.invalid.length;
|
|
const totalSnapErrors = result.snapshotFlagErrors.length;
|
|
|
|
if (totalInvalid > 0 || totalSnapErrors > 0) {
|
|
console.log(` [check] \u274c ${output} (${totalValid} valid)`);
|
|
for (const inv of result.invalid) {
|
|
console.log(` Unknown command: '${inv.command}' at line ${inv.line}`);
|
|
}
|
|
for (const se of result.snapshotFlagErrors) {
|
|
console.log(` ${se.error} at line ${se.command.line}`);
|
|
}
|
|
} else {
|
|
console.log(` [check] \u2705 ${output} — ${totalValid} commands, all valid`);
|
|
}
|
|
}
|
|
|
|
// Dev workspace render isolation: the default in-place regen above keeps the
|
|
// worktree canonical. If bin/dev-setup set up an untracked brain-aware render
|
|
// (.claude/gstack-rendered), refresh it too so live template edits reflect at
|
|
// this workspace's runtime. Only runs when the render dir already exists — we
|
|
// never create it during plain template dev.
|
|
const RENDER_DIR = path.join(ROOT, '.claude', 'gstack-rendered');
|
|
if (fs.existsSync(RENDER_DIR)) {
|
|
try {
|
|
execSync(
|
|
`bun run scripts/gen-skill-docs.ts --respect-detection --host claude --out-dir ${JSON.stringify(RENDER_DIR)}`,
|
|
{ cwd: ROOT, stdio: 'pipe' },
|
|
);
|
|
console.log(' [render] refreshed .claude/gstack-rendered (brain-aware workspace copy)');
|
|
} catch (err: any) {
|
|
console.log(` [render] ERROR: ${err.stderr?.toString().trim() || err.message}`);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Initial run
|
|
console.log(' [watch] Watching *.md.tmpl files...');
|
|
regenerateAndValidate();
|
|
|
|
// Watch for changes
|
|
for (const { tmpl } of TEMPLATES) {
|
|
if (!fs.existsSync(tmpl)) continue;
|
|
fs.watch(tmpl, () => {
|
|
console.log(`\n [watch] ${path.relative(ROOT, tmpl)} changed`);
|
|
regenerateAndValidate();
|
|
});
|
|
}
|
|
|
|
// Also watch commands.ts and snapshot.ts (source of truth changes)
|
|
const SOURCE_FILES = [
|
|
path.join(ROOT, 'browse', 'src', 'commands.ts'),
|
|
path.join(ROOT, 'browse', 'src', 'snapshot.ts'),
|
|
];
|
|
|
|
for (const src of SOURCE_FILES) {
|
|
if (!fs.existsSync(src)) continue;
|
|
fs.watch(src, () => {
|
|
console.log(`\n [watch] ${path.relative(ROOT, src)} changed`);
|
|
regenerateAndValidate();
|
|
});
|
|
}
|
|
|
|
// Keep alive
|
|
console.log(' [watch] Press Ctrl+C to stop\n');
|