feat: dual-host setup + find-browse for Codex/Gemini/Cursor

- setup: add --host codex|claude|auto flag, install to ~/.codex/skills/
  when targeting Codex, auto-detect installed agents
- find-browse: priority chain .codex > .agents > .claude (both
  workspace-local and global)
- dev-setup/teardown: create .agents/skills/gstack symlinks for dev mode

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
Garry Tan
2026-03-19 00:57:43 -07:00
parent 0e76c81b01
commit 10e6d39f27
5 changed files with 179 additions and 51 deletions
+13 -9
View File
@@ -5,13 +5,17 @@ DIR="$(cd "$(dirname "$0")/.." && pwd)/dist"
if test -x "$DIR/find-browse"; then
exec "$DIR/find-browse" "$@"
fi
# Fallback: basic discovery
# Fallback: basic discovery with priority chain
ROOT=$(git rev-parse --show-toplevel 2>/dev/null)
if [ -n "$ROOT" ] && test -x "$ROOT/.claude/skills/gstack/browse/dist/browse"; then
echo "$ROOT/.claude/skills/gstack/browse/dist/browse"
elif test -x "$HOME/.claude/skills/gstack/browse/dist/browse"; then
echo "$HOME/.claude/skills/gstack/browse/dist/browse"
else
echo "ERROR: browse binary not found. Run: cd <skill-dir> && ./setup" >&2
exit 1
fi
for MARKER in .codex .agents .claude; do
if [ -n "$ROOT" ] && test -x "$ROOT/$MARKER/skills/gstack/browse/dist/browse"; then
echo "$ROOT/$MARKER/skills/gstack/browse/dist/browse"
exit 0
fi
if test -x "$HOME/$MARKER/skills/gstack/browse/dist/browse"; then
echo "$HOME/$MARKER/skills/gstack/browse/dist/browse"
exit 0
fi
done
echo "ERROR: browse binary not found. Run: cd <skill-dir> && ./setup" >&2
exit 1
+9 -4
View File
@@ -27,16 +27,21 @@ function getGitRoot(): string | null {
export function locateBinary(): string | null {
const root = getGitRoot();
const home = homedir();
const markers = ['.codex', '.agents', '.claude'];
// Workspace-local takes priority (for development)
if (root) {
const local = join(root, '.claude', 'skills', 'gstack', 'browse', 'dist', 'browse');
if (existsSync(local)) return local;
for (const m of markers) {
const local = join(root, m, 'skills', 'gstack', 'browse', 'dist', 'browse');
if (existsSync(local)) return local;
}
}
// Global fallback
const global = join(home, '.claude', 'skills', 'gstack', 'browse', 'dist', 'browse');
if (existsSync(global)) return global;
for (const m of markers) {
const global = join(home, m, 'skills', 'gstack', 'browse', 'dist', 'browse');
if (existsSync(global)) return global;
}
return null;
}