#!/usr/bin/env bash set -euo pipefail MOSAIC_HOME="${MOSAIC_HOME:-$HOME/.config/mosaic}" backup_stamp="$(date +%Y%m%d%H%M%S)" copy_file_managed() { local src="$1" local dst="$2" mkdir -p "$(dirname "$dst")" if [[ -L "$dst" ]]; then rm -f "$dst" fi if [[ -f "$dst" ]]; then if cmp -s "$src" "$dst"; then return fi mv "$dst" "${dst}.mosaic-bak-${backup_stamp}" fi cp "$src" "$dst" } remove_legacy_path() { local p="$1" if [[ -L "$p" ]]; then rm -f "$p" return fi if [[ -d "$p" ]]; then find "$p" -depth -type l -delete 2>/dev/null || true find "$p" -depth -type d -empty -delete 2>/dev/null || true return fi # Remove stale symlinked files if present. if [[ -e "$p" && -L "$p" ]]; then rm -f "$p" fi } # Remove compatibility symlink surfaces for migrated content. legacy_paths=( "$HOME/.claude/agent-guides" "$HOME/.claude/scripts/git" "$HOME/.claude/scripts/codex" "$HOME/.claude/scripts/bootstrap" "$HOME/.claude/scripts/cicd" "$HOME/.claude/scripts/portainer" "$HOME/.claude/scripts/debug-hook.sh" "$HOME/.claude/scripts/qa-hook-handler.sh" "$HOME/.claude/scripts/qa-hook-stdin.sh" "$HOME/.claude/scripts/qa-hook-wrapper.sh" "$HOME/.claude/scripts/qa-queue-monitor.sh" "$HOME/.claude/scripts/remediation-hook-handler.sh" "$HOME/.claude/templates" "$HOME/.claude/presets/domains" "$HOME/.claude/presets/tech-stacks" "$HOME/.claude/presets/workflows" "$HOME/.claude/presets/jarvis-loop.json" ) for p in "${legacy_paths[@]}"; do remove_legacy_path "$p" done # Claude-specific runtime files (settings, hooks — NOT CLAUDE.md which is now a thin pointer) # When MOSAIC_SKIP_CLAUDE_HOOKS=1 is set (user declined hooks in the wizard # preview stage), skip hooks-config.json but still copy the other runtime # files so Claude still gets CLAUDE.md/settings.json/context7 guidance. for runtime_file in \ CLAUDE.md \ settings.json \ hooks-config.json \ context7-integration.md; do if [[ "$runtime_file" == "hooks-config.json" ]] && [[ "${MOSAIC_SKIP_CLAUDE_HOOKS:-0}" == "1" ]]; then echo "[mosaic-link] Skipping hooks-config.json (user declined in wizard)" # An existing ~/.claude/hooks-config.json that we previously installed # is identified by one of: # 1. It's a symlink (legacy symlink-mode install) # 2. It contains the `mosaic-managed` marker string we embed in the # template (survives template updates unlike byte-equality) # 3. It is byte-identical to the current Mosaic template (fallback # for templates that pre-date the marker) # Anything else is user-owned and we must leave it alone. existing_hooks="$HOME/.claude/hooks-config.json" mosaic_hooks_src="$MOSAIC_HOME/runtime/claude/hooks-config.json" if [[ -L "$existing_hooks" ]]; then rm -f "$existing_hooks" echo "[mosaic-link] Removed previously-linked Mosaic hooks-config.json (was symlink)" elif [[ -f "$existing_hooks" ]]; then is_mosaic_managed=0 if grep -q 'mosaic-managed' "$existing_hooks" 2>/dev/null; then is_mosaic_managed=1 elif [[ -f "$mosaic_hooks_src" ]] && cmp -s "$existing_hooks" "$mosaic_hooks_src"; then is_mosaic_managed=1 fi if [[ "$is_mosaic_managed" == "1" ]]; then mv "$existing_hooks" "${existing_hooks}.mosaic-bak-${backup_stamp}" echo "[mosaic-link] Removed previously-linked Mosaic hooks-config.json (backup at ${existing_hooks}.mosaic-bak-${backup_stamp})" else echo "[mosaic-link] Leaving existing non-Mosaic hooks-config.json in place" fi fi continue fi src="$MOSAIC_HOME/runtime/claude/$runtime_file" [[ -f "$src" ]] || continue copy_file_managed "$src" "$HOME/.claude/$runtime_file" done # OpenCode runtime adapter (thin pointer to AGENTS.md) opencode_adapter="$MOSAIC_HOME/runtime/opencode/AGENTS.md" if [[ -f "$opencode_adapter" ]]; then copy_file_managed "$opencode_adapter" "$HOME/.config/opencode/AGENTS.md" fi # Codex runtime adapter (thin pointer to AGENTS.md) codex_adapter="$MOSAIC_HOME/runtime/codex/instructions.md" if [[ -f "$codex_adapter" ]]; then mkdir -p "$HOME/.codex" copy_file_managed "$codex_adapter" "$HOME/.codex/instructions.md" fi # Pi runtime settings (MCP + skills paths) pi_settings_dir="$HOME/.pi/agent" pi_settings_file="$pi_settings_dir/settings.json" mkdir -p "$pi_settings_dir" if [[ ! -f "$pi_settings_file" ]]; then echo '{}' > "$pi_settings_file" fi # Ensure Pi settings.json has Mosaic skills paths mosaic_skills_path="$MOSAIC_HOME/skills" mosaic_local_path="$MOSAIC_HOME/skills-local" if ! grep -q "$mosaic_skills_path" "$pi_settings_file" 2>/dev/null; then if command -v python3 >/dev/null 2>&1; then python3 -c " import json with open('$pi_settings_file', 'r') as f: data = json.load(f) skills = data.get('skills', []) if not isinstance(skills, list): skills = [] for p in ['$mosaic_skills_path', '$mosaic_local_path']: if p not in skills: skills.append(p) data['skills'] = skills with open('$pi_settings_file', 'w') as f: json.dump(data, f, indent=2) f.write('\\n') " 2>/dev/null fi fi # Pi extension is loaded via --extension flag in the mosaic launcher. # Do NOT copy into ~/.pi/agent/extensions/ — that causes duplicate loading. if [[ -x "$MOSAIC_HOME/tools/_scripts/mosaic-ensure-sequential-thinking" ]]; then "$MOSAIC_HOME/tools/_scripts/mosaic-ensure-sequential-thinking" fi echo "[mosaic-link] Runtime assets synced (non-symlink mode)" echo "[mosaic-link] Canonical source: $MOSAIC_HOME"