Per-skill SKILL.md symlinks don't expose gstack's shared infrastructure. Multiple skills hardcode ~/.claude/skills/gstack/bin/ (gstack-config, gstack-update-check, gstack-paths) and gstack/browse/dist/ (browse binary). Create targeted symlinks in link.sh: - ~/.claude/skills/gstack/bin/ → skills-external/gstack/bin/ - ~/.claude/skills/gstack/browse/dist/ → skills-external/gstack/browse/dist/ Fixes: browse binary not found, gstack-config failures, freeze gstack-paths resolution. Safe with profile toggles (profiles move per-skill dirs, not the gstack/ infra dir). Safe with stale link cleanup (only removes -L symlinks, not real dirs created by mkdir -p). Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
115 lines
3.8 KiB
Bash
115 lines
3.8 KiB
Bash
#!/usr/bin/env bash
|
|
# Symlink this repo into ~/.claude/
|
|
set -euo pipefail
|
|
|
|
REPO="$(cd "$(dirname "$0")" && pwd)"
|
|
CLAUDE="$HOME/.claude"
|
|
CHANGED=0
|
|
|
|
mkdir -p "$CLAUDE"
|
|
|
|
link_file() {
|
|
local src="$1" dst="$2"
|
|
if [ -L "$dst" ] && [ "$(readlink "$dst")" = "$src" ]; then
|
|
return # already correct
|
|
fi
|
|
ln -sf "$src" "$dst"
|
|
CHANGED=$((CHANGED + 1))
|
|
}
|
|
|
|
link_file "$REPO/CLAUDE.md" "$CLAUDE/CLAUDE.md"
|
|
link_file "$REPO/settings.json" "$CLAUDE/settings.json"
|
|
|
|
for item in hooks agents skills lib templates; do
|
|
target="$CLAUDE/$item"
|
|
if [ -L "$target" ]; then
|
|
if [ "$(readlink "$target")" = "$REPO/$item" ]; then
|
|
continue # already correct
|
|
fi
|
|
rm -f "$target"
|
|
elif [ -d "$target" ]; then
|
|
echo "⚠️ ~/.claude/$item is a real directory. Rename or remove it, then re-run link.sh."
|
|
continue
|
|
fi
|
|
ln -sf "$REPO/$item" "$target"
|
|
CHANGED=$((CHANGED + 1))
|
|
done
|
|
|
|
# GStack is exposed via per-skill symlinks under skills/ (browse,
|
|
# canary, autoplan, design-review, …) created by gstack's own
|
|
# `./setup`. A global `skills/gstack -> skills-external/gstack/`
|
|
# symlink duplicated the top-level gstack SKILL.md alongside those
|
|
# individual skills, producing two entries with the same description
|
|
# ("Fast headless browser for QA testing…"). Remove any stale global
|
|
# link — only per-skill entries remain.
|
|
if [ -L "$REPO/skills/gstack" ] || [ -L "$CLAUDE/skills/gstack" ]; then
|
|
rm -f "$REPO/skills/gstack" "$CLAUDE/skills/gstack"
|
|
CHANGED=$((CHANGED + 1))
|
|
fi
|
|
if [ ! -d "$REPO/skills-external/gstack" ]; then
|
|
echo "⚠️ GStack submodule not found — run: git submodule update --init"
|
|
fi
|
|
|
|
# GStack shared infrastructure: bin/ (CLI tools, config, analytics) and
|
|
# browse/dist/ (compiled browse binary). Per-skill SKILL.md symlinks don't
|
|
# expose these, but multiple skills hardcode ~/.claude/skills/gstack/bin/
|
|
# and ~/.claude/skills/gstack/browse/dist/. Create targeted symlinks.
|
|
GSTACK_SRC="$REPO/skills-external/gstack"
|
|
GSTACK_DST="$CLAUDE/skills/gstack"
|
|
if [ -d "$GSTACK_SRC/bin" ]; then
|
|
mkdir -p "$GSTACK_DST"
|
|
if [ ! -L "$GSTACK_DST/bin" ]; then
|
|
ln -sf "$GSTACK_SRC/bin" "$GSTACK_DST/bin"
|
|
CHANGED=$((CHANGED + 1))
|
|
fi
|
|
fi
|
|
if [ -d "$GSTACK_SRC/browse/dist" ]; then
|
|
mkdir -p "$GSTACK_DST/browse"
|
|
if [ ! -L "$GSTACK_DST/browse/dist" ]; then
|
|
ln -sf "$GSTACK_SRC/browse/dist" "$GSTACK_DST/browse/dist"
|
|
CHANGED=$((CHANGED + 1))
|
|
fi
|
|
fi
|
|
|
|
EXTERNAL_SKILLS=(emil-design-eng frontend-design design-motion-principles)
|
|
for _ext_skill in "${EXTERNAL_SKILLS[@]}"; do
|
|
if [ -d "$REPO/skills-external/$_ext_skill" ]; then
|
|
if [ -L "$CLAUDE/skills/$_ext_skill" ] && [ "$(readlink "$CLAUDE/skills/$_ext_skill")" = "$REPO/skills-external/$_ext_skill" ]; then
|
|
: # already correct
|
|
else
|
|
ln -sf "$REPO/skills-external/$_ext_skill" "$CLAUDE/skills/$_ext_skill"
|
|
CHANGED=$((CHANGED + 1))
|
|
fi
|
|
else
|
|
echo "⚠️ $_ext_skill not found — run: make plugin"
|
|
fi
|
|
done
|
|
|
|
# External skills installed via `npx skills add` live under
|
|
# $HOME/.agents/skills/. We symlink them into $REPO/skills/ with
|
|
# absolute paths so the link stays valid regardless of where the
|
|
# repo is cloned (relative ../../ paths broke on repos deeper than
|
|
# one level below $HOME).
|
|
NPX_EXTERNAL_SKILLS=(darwin-skill find-skills)
|
|
for _ext in "${NPX_EXTERNAL_SKILLS[@]}"; do
|
|
_target="$HOME/.agents/skills/$_ext"
|
|
_link="$REPO/skills/$_ext"
|
|
if [ ! -d "$_target" ]; then
|
|
echo "⚠️ $_ext not installed at $_target — run: make plugin"
|
|
continue
|
|
fi
|
|
if [ -L "$_link" ] && [ "$(readlink "$_link")" = "$_target" ]; then
|
|
continue
|
|
fi
|
|
rm -f "$_link"
|
|
ln -sf "$_target" "$_link"
|
|
CHANGED=$((CHANGED + 1))
|
|
done
|
|
|
|
if [ "$CHANGED" -eq 0 ]; then
|
|
echo "✅ All symlinks already up to date."
|
|
else
|
|
echo "✅ $CHANGED symlink(s) updated in ~/.claude/"
|
|
fi
|
|
echo " Next: bash install-plugins.sh"
|