Ship lib/profile.sh + 9 profiles in lib/profiles/. A profile is a
plain-text file listing items + types (gstack | personal | external |
plugin@<marketplace> | mcp | cli). `profile set <name>` enables the
listed items and disables the rest:
- gstack/personal/external skills: symlink toggle skills/ ↔
skills-disabled/ (gstack__<name> prefix to avoid collisions; no
prefix for personal/external).
- plugins typed `plugin@<marketplace>`: actually toggled via
`claude plugin enable|disable <name>@<marketplace>`. Allowlist:
MANAGED_PLUGINS = ui-ux-pro-max, plugin-dev, pr-review-toolkit.
Denylist: PROTECTED_PLUGINS = caveman, security-guidance,
superpowers (always-on, never disabled even if absent from a
profile).
- mcp magic: delegated to lib/toggle-external.sh which already
handles the MAGIC_API_KEY env lookup. Other MCPs stay advisory.
- cli (rtk, gsd, ctx7, graphify): status-only, never auto-installed.
Profiles shipped:
web public website work — frontend + content + light dev
seo SEO + GEO + W3C audit (search/AI indexability + a11y)
web-full production website end-to-end (web ∪ seo ∪ qa-only/canary)
backend backend / API / system dev — no design, no SEO
design visual QA, design systems, mockups, polish
dev daily code work — features, fixes, refactor, ship
qa site testing, perf, canary, validation
audit comprehensive audit — security + SEO + perf + health
minimal strip all gstack skills (quiet session)
Commands:
profile list / show <name> / current / apply <name> / set <name> /
reset / diff <a> <b>
`current` heuristic returns "full" when nothing is disabled, otherwise
picks the profile with the highest available-ratio (counts both
"enabled" and "installed" — the latter for CLIs). Tiebreaker: larger
profile total wins, so web-full beats web at a 100% tie.
`reset` re-enables every gstack skill but does NOT touch plugins —
the user re-enables a managed plugin manually or via `apply <profile>`.
This is documented in the trailing info line.
Integration:
- skills/profile/SKILL.md — `/profile` slash command, lists profiles,
documents the per-type mechanism, points at lib/profile.sh.
- agents/plugin-advisor.md — DETECT phase calls `profile current`,
OUTPUT adds a PROFILE line, and TOGGLING EXTERNAL TOOLS gains a
"Skill profiles" section with a signal → profile recommendation
table.
- lib/toggle-external.sh — header pointer to profile.sh for fine-
grained activation (toggle-external still owns whole-gstack and
magic-MCP toggles).
- Makefile — `make profile cmd="set <name>"`, profile-list,
profile-current, profile-reset.
Tested end-to-end: `set web` enables ui-ux-pro-max + magic; `set seo`
disables ui-ux-pro-max; `set minimal` disables ui-ux-pro-max but
spares always-on plugins; `reset` restores all 64 skills; shellcheck
clean.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
218 lines
6.8 KiB
Bash
Executable File
218 lines
6.8 KiB
Bash
Executable File
#!/usr/bin/env bash
|
|
# ============================================================
|
|
# lib/toggle-external.sh — enable/disable non-plugin tools
|
|
#
|
|
# Marketplace plugins are toggled by `claude plugin enable|disable`.
|
|
# Tools distributed outside the marketplace (gstack submodule, emil
|
|
# curl install, npx-installed skills) have no such lever — they live
|
|
# as symlinks inside skills/. This script moves those symlinks
|
|
# to/from skills-disabled/ so Claude Code stops/starts scanning them.
|
|
#
|
|
# MCP servers are toggled via `claude mcp add|remove` (not symlinks).
|
|
#
|
|
# Usage:
|
|
# toggle-external.sh list
|
|
# toggle-external.sh status <tool>
|
|
# toggle-external.sh enable <tool>
|
|
# toggle-external.sh disable <tool>
|
|
#
|
|
# Managed tools:
|
|
# gstack — per-skill symlinks populated by gstack's own setup
|
|
# emil-design-eng — single symlink → skills-external/emil-design-eng
|
|
# darwin-skill — single symlink → ~/.agents/skills/darwin-skill
|
|
# find-skills — single symlink → ~/.agents/skills/find-skills
|
|
# magic — 21st-dev Magic MCP server (API key in .env)
|
|
#
|
|
# For fine-grained activation (only design skills, only qa skills, only
|
|
# audit skills, etc.) instead of all-or-nothing gstack toggling, use:
|
|
# bash lib/profile.sh list
|
|
# bash lib/profile.sh set <design|dev|qa|audit|minimal>
|
|
# bash lib/profile.sh reset
|
|
# ============================================================
|
|
set -euo pipefail
|
|
|
|
REPO="$(cd "$(dirname "$0")/.." && pwd)"
|
|
SKILLS_DIR="$REPO/skills"
|
|
DISABLED_DIR="$REPO/skills-disabled"
|
|
|
|
GREEN='\033[0;32m'; YELLOW='\033[1;33m'; RED='\033[0;31m'; NC='\033[0m'
|
|
ok() { echo -e "${GREEN}✓${NC} $1"; }
|
|
warn() { echo -e "${YELLOW}⚠${NC} $1"; }
|
|
err() { echo -e "${RED}✗${NC} $1"; }
|
|
|
|
# All non-plugin tools this script can toggle.
|
|
MANAGED_TOOLS=(gstack emil-design-eng darwin-skill find-skills magic)
|
|
|
|
# Load MAGIC_API_KEY (and any other secrets) from $REPO/.env if present.
|
|
# Called only by the magic branch — other tools don't need env vars.
|
|
load_env() {
|
|
if [ -z "${MAGIC_API_KEY:-}" ] && [ -f "$REPO/.env" ]; then
|
|
set -a
|
|
# shellcheck source=/dev/null
|
|
source "$REPO/.env"
|
|
set +a
|
|
fi
|
|
}
|
|
|
|
# Prints the names (directory basenames) that belong to "gstack".
|
|
# Source of truth: skills-external/gstack/*/SKILL.md. The repo's
|
|
# skills/<name> symlinks are generated from these by gstack ./setup.
|
|
gstack_skills() {
|
|
local gstack_src="$REPO/skills-external/gstack"
|
|
[ -d "$gstack_src" ] || return 0
|
|
for d in "$gstack_src"/*/; do
|
|
[ -f "${d}SKILL.md" ] || continue
|
|
basename "$d"
|
|
done
|
|
}
|
|
|
|
# Prints "enabled" / "disabled" / "missing" for a tool.
|
|
status_tool() {
|
|
local tool="$1"
|
|
case "$tool" in
|
|
gstack)
|
|
[ -d "$REPO/skills-external/gstack" ] || { echo "missing"; return; }
|
|
while read -r name; do
|
|
[ -e "$SKILLS_DIR/$name" ] && { echo "enabled"; return; }
|
|
done < <(gstack_skills)
|
|
echo "disabled"
|
|
;;
|
|
emil-design-eng)
|
|
[ -d "$REPO/skills-external/emil-design-eng" ] || { echo "missing"; return; }
|
|
[ -e "$SKILLS_DIR/emil-design-eng" ] && echo "enabled" || echo "disabled"
|
|
;;
|
|
darwin-skill|find-skills)
|
|
[ -d "$HOME/.agents/skills/$tool" ] || { echo "missing"; return; }
|
|
[ -e "$SKILLS_DIR/$tool" ] && echo "enabled" || echo "disabled"
|
|
;;
|
|
magic)
|
|
command -v claude >/dev/null || { echo "missing"; return; }
|
|
if claude mcp list 2>/dev/null | grep -q '^magic:'; then
|
|
echo "enabled"
|
|
else
|
|
echo "disabled"
|
|
fi
|
|
;;
|
|
*)
|
|
echo "unknown"; return 1 ;;
|
|
esac
|
|
}
|
|
|
|
disable_tool() {
|
|
local tool="$1"
|
|
mkdir -p "$DISABLED_DIR"
|
|
case "$tool" in
|
|
gstack)
|
|
local moved=0
|
|
while read -r name; do
|
|
[ -e "$SKILLS_DIR/$name" ] || continue
|
|
# Clobber any stale destination. gstack ./setup now creates
|
|
# skills/<name>/ as directories, so mv onto an existing dir
|
|
# would nest it (gstack__<name>/<name>/) instead of renaming.
|
|
# Content is symlinks to the submodule — `gstack ./setup` regenerates.
|
|
rm -rf "$DISABLED_DIR/gstack__$name"
|
|
mv "$SKILLS_DIR/$name" "$DISABLED_DIR/gstack__$name"
|
|
moved=$((moved + 1))
|
|
done < <(gstack_skills)
|
|
ok "gstack disabled ($moved symlinks moved)"
|
|
;;
|
|
emil-design-eng|darwin-skill|find-skills)
|
|
if [ -e "$SKILLS_DIR/$tool" ]; then
|
|
rm -rf "${DISABLED_DIR:?}/${tool:?}"
|
|
mv "$SKILLS_DIR/$tool" "$DISABLED_DIR/$tool"
|
|
ok "$tool disabled"
|
|
else
|
|
warn "$tool already disabled"
|
|
fi
|
|
;;
|
|
magic)
|
|
if [ "$(status_tool magic)" = "enabled" ]; then
|
|
claude mcp remove magic -s user >/dev/null
|
|
ok "magic disabled"
|
|
else
|
|
warn "magic already disabled"
|
|
fi
|
|
;;
|
|
*) err "Unknown tool: $tool"; return 1 ;;
|
|
esac
|
|
}
|
|
|
|
enable_tool() {
|
|
local tool="$1"
|
|
case "$tool" in
|
|
gstack)
|
|
local moved=0
|
|
if [ -d "$DISABLED_DIR" ]; then
|
|
for entry in "$DISABLED_DIR"/gstack__*; do
|
|
[ -e "$entry" ] || continue
|
|
local name
|
|
name="$(basename "$entry" | sed 's/^gstack__//')"
|
|
rm -rf "${SKILLS_DIR:?}/${name:?}"
|
|
mv "$entry" "$SKILLS_DIR/$name"
|
|
moved=$((moved + 1))
|
|
done
|
|
fi
|
|
if [ "$moved" -eq 0 ]; then
|
|
warn "gstack was not disabled — re-run gstack setup to (re)create symlinks"
|
|
else
|
|
ok "gstack enabled ($moved symlinks restored)"
|
|
fi
|
|
;;
|
|
emil-design-eng|darwin-skill|find-skills)
|
|
if [ -e "$DISABLED_DIR/$tool" ]; then
|
|
rm -rf "${SKILLS_DIR:?}/${tool:?}"
|
|
mv "$DISABLED_DIR/$tool" "$SKILLS_DIR/$tool"
|
|
ok "$tool enabled"
|
|
elif [ -e "$SKILLS_DIR/$tool" ]; then
|
|
warn "$tool already enabled"
|
|
else
|
|
err "$tool not installed — run: make plugin"
|
|
return 1
|
|
fi
|
|
;;
|
|
magic)
|
|
load_env
|
|
if [ -z "${MAGIC_API_KEY:-}" ]; then
|
|
err "MAGIC_API_KEY not set — add it to $REPO/.env (template: .env.example)"
|
|
return 1
|
|
fi
|
|
if [ "$(status_tool magic)" = "enabled" ]; then
|
|
warn "magic already enabled"
|
|
return 0
|
|
fi
|
|
claude mcp add magic --scope user \
|
|
--env API_KEY="$MAGIC_API_KEY" \
|
|
-- npx -y @21st-dev/magic@latest
|
|
ok "magic enabled (user scope)"
|
|
;;
|
|
*) err "Unknown tool: $tool"; return 1 ;;
|
|
esac
|
|
}
|
|
|
|
list_all() {
|
|
printf "%-20s %s\n" "TOOL" "STATUS"
|
|
printf "%-20s %s\n" "----" "------"
|
|
for t in "${MANAGED_TOOLS[@]}"; do
|
|
printf "%-20s %s\n" "$t" "$(status_tool "$t")"
|
|
done
|
|
}
|
|
|
|
usage() {
|
|
sed -n '3,23p' "$0" | sed 's/^# \?//'
|
|
exit "${1:-0}"
|
|
}
|
|
|
|
main() {
|
|
local cmd="${1:-}"
|
|
case "$cmd" in
|
|
list) list_all ;;
|
|
status) [ $# -ge 2 ] || usage 1; status_tool "$2" ;;
|
|
enable) [ $# -ge 2 ] || usage 1; enable_tool "$2" ;;
|
|
disable) [ $# -ge 2 ] || usage 1; disable_tool "$2" ;;
|
|
""|-h|--help|help) usage 0 ;;
|
|
*) err "Unknown command: $cmd"; usage 1 ;;
|
|
esac
|
|
}
|
|
|
|
main "$@"
|