@howlil/ez-agents 5.0.0 → 5.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -11,7 +11,7 @@ import { fileURLToPath } from "url";
11
11
  // package.json
12
12
  var package_default = {
13
13
  name: "@howlil/ez-agents",
14
- version: "5.0.0",
14
+ version: "5.0.2",
15
15
  description: "EZ Agents \u2014 Meta-prompting with multi-model support (Qwen, Kimi, OpenAI)",
16
16
  type: "module",
17
17
  sideEffects: false,
@@ -2079,7 +2079,16 @@ function install(isGlobal, runtime = "claude") {
2079
2079
  const isQwen = runtime === "qwen";
2080
2080
  const isKimi = runtime === "kimi";
2081
2081
  const dirName = getDirName(runtime);
2082
- const src = path.join(__dirname, "..");
2082
+ let pkgRoot = path.join(__dirname, "..");
2083
+ while (!fs.existsSync(path.join(pkgRoot, "package.json"))) {
2084
+ const parent = path.dirname(pkgRoot);
2085
+ if (parent === pkgRoot) {
2086
+ pkgRoot = path.join(__dirname, "..");
2087
+ break;
2088
+ }
2089
+ pkgRoot = parent;
2090
+ }
2091
+ const src = pkgRoot;
2083
2092
  const targetDir = isGlobal ? getGlobalDir(runtime, explicitConfigDir) : path.join(process.cwd(), dirName);
2084
2093
  const locationLabel = isGlobal ? targetDir.replace(os.homedir(), "~") : targetDir.replace(process.cwd(), ".");
2085
2094
  const pathPrefix = isGlobal ? `${targetDir.replace(/\\/g, "/").replace(os.homedir().replace(/\\/g, "/"), "~")}/` : `./${dirName}/`;
@@ -2694,4 +2703,4 @@ if (hasGlobal && hasLocal) {
2694
2703
  export {
2695
2704
  testExports
2696
2705
  };
2697
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../bin/install.ts", "../../package.json"],
  "sourcesContent": ["#!/usr/bin/env node\r\n\r\nimport * as fs from 'fs';\r\nimport * as path from 'path';\r\nimport * as os from 'os';\r\nimport * as readline from 'readline';\r\nimport * as crypto from 'crypto';\r\nimport { fileURLToPath } from 'url';\r\n\r\nconst __filename = fileURLToPath(import.meta.url);\r\nconst __dirname = path.dirname(__filename);\r\n\r\n// Colors\r\nconst cyan = '\\x1b[36m';\r\nconst green = '\\x1b[32m';\r\nconst yellow = '\\x1b[33m';\r\nconst dim = '\\x1b[2m';\r\nconst reset = '\\x1b[0m';\r\n\r\n// Codex config.toml constants\r\nconst EZ_CODEX_MARKER = '# EZ_Agents Agent Configuration \\u2014 managed by ez-agents installer';\r\n\r\n// Copilot instructions marker constants\r\nconst EZ_COPILOT_INSTRUCTIONS_MARKER = '<!-- EZ_AGENTS Configuration \\u2014 managed by ez-agents installer -->';\r\nconst EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER = '<!-- /EZ_AGENTS Configuration -->';\r\n// Legacy markers kept for backward compatibility with older installs.\r\nconst LEGACY_EZ_COPILOT_INSTRUCTIONS_MARKER = '<!-- EZ_Agents Configuration \\u2014 managed by ez-agents installer -->';\r\nconst LEGACY_EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER = '<!-- /EZ_Agents Configuration -->';\r\n\r\nconst CODEX_AGENT_SANDBOX = {\r\n  'ez-executor': 'workspace-write',\r\n  'ez-planner': 'workspace-write',\r\n  'ez-phase-researcher': 'workspace-write',\r\n  'ez-project-researcher': 'workspace-write',\r\n  'ez-research-synthesizer': 'workspace-write',\r\n  'ez-verifier': 'workspace-write',\r\n  'ez-codebase-mapper': 'workspace-write',\r\n  'ez-roadmapper': 'workspace-write',\r\n  'ez-debugger': 'workspace-write',\r\n  'ez-plan-checker': 'read-only',\r\n  'ez-integration-checker': 'read-only',\r\n};\r\n\r\n// Copilot tool name mapping \u2014 Claude Code tools to GitHub Copilot tools\r\n// Tool mapping applies ONLY to agents, NOT to skills (per CONTEXT.md decision)\r\nconst claudeToCopilotTools = {\r\n  Read: 'read',\r\n  Write: 'edit',\r\n  Edit: 'edit',\r\n  Bash: 'execute',\r\n  Grep: 'search',\r\n  Glob: 'search',\r\n  Task: 'agent',\r\n  WebSearch: 'web',\r\n  WebFetch: 'web',\r\n  TodoWrite: 'todo',\r\n  AskUserQuestion: 'ask_user',\r\n  SlashCommand: 'skill',\r\n};\r\n\r\n// Get version from package.json\r\nimport pkg from '../package.json' with { type: 'json' };\r\n\r\n// Parse args\r\nconst args = process.argv.slice(2);\r\nconst hasGlobal = args.includes('--global') || args.includes('-g');\r\nconst hasLocal = args.includes('--local') || args.includes('-l');\r\nconst hasOpencode = args.includes('--opencode');\r\nconst hasClaude = args.includes('--claude');\r\nconst hasGemini = args.includes('--gemini');\r\nconst hasCodex = args.includes('--codex');\r\nconst hasCopilot = args.includes('--copilot');\r\nconst hasQwen = args.includes('--qwen');\r\nconst hasKimi = args.includes('--kimi');\r\nconst hasBoth = args.includes('--both'); // Legacy flag, keeps working\r\nconst hasAll = args.includes('--all');\r\nconst hasUninstall = args.includes('--uninstall') || args.includes('-u');\r\n\r\n// Runtime selection - can be set by flags or interactive prompt\r\nlet selectedRuntimes: string[] = [];\r\nif (hasAll) {\r\n  selectedRuntimes = ['claude', 'opencode', 'gemini', 'codex', 'copilot', 'qwen', 'kimi'];\r\n} else if (hasBoth) {\r\n  selectedRuntimes = ['claude', 'opencode'];\r\n} else {\r\n  if (hasOpencode) selectedRuntimes.push('opencode');\r\n  if (hasClaude) selectedRuntimes.push('claude');\r\n  if (hasGemini) selectedRuntimes.push('gemini');\r\n  if (hasCodex) selectedRuntimes.push('codex');\r\n  if (hasCopilot) selectedRuntimes.push('copilot');\r\n  if (hasQwen) selectedRuntimes.push('qwen');\r\n  if (hasKimi) selectedRuntimes.push('kimi');\r\n}\r\n\r\n// WSL + Windows Node.js detection\r\n// When Windows-native Node runs on WSL, os.homedir() and path.join() produce\r\n// backslash paths that don't resolve correctly on the Linux filesystem.\r\nif (process.platform === 'win32') {\r\n  let isWSL = false;\r\n  try {\r\n    if (process.env.WSL_DISTRO_NAME) {\r\n      isWSL = true;\r\n    } else if (fs.existsSync('/proc/version')) {\r\n      const procVersion = fs.readFileSync('/proc/version', 'utf8').toLowerCase();\r\n      if (procVersion.includes('microsoft') || procVersion.includes('wsl')) {\r\n        isWSL = true;\r\n      }\r\n    }\r\n  } catch {\r\n    // Ignore read errors \u2014 not WSL\r\n  }\r\n\r\n  if (isWSL) {\r\n    console.error(`\r\n${yellow}\u26A0 Detected WSL with Windows-native Node.js.${reset}\r\n\r\nThis causes path resolution issues that prevent correct installation.\r\nPlease install a Linux-native Node.js inside WSL:\r\n\r\n  curl -fsSL https://fnm.vercel.app/install | bash\r\n  fnm install --lts\r\n\r\nThen re-run: npx ez-agents\r\n`);\r\n    process.exit(1);\r\n  }\r\n}\r\n\r\n/**\r\n * Convert a pathPrefix (which uses absolute paths for global installs) to a\r\n * $HOME-relative form for replacing $HOME/.claude/ references in bash code blocks.\r\n * Preserves $HOME as a shell variable so paths remain portable across machines.\r\n */\r\nfunction toHomePrefix(pathPrefix: string): string {\r\n  const home = os.homedir().replace(/\\\\/g, '/');\r\n  const normalized = pathPrefix.replace(/\\\\/g, '/');\r\n  if (normalized.startsWith(home)) {\r\n    return '$HOME' + normalized.slice(home.length);\r\n  }\r\n  // Convert tilde-based paths to $HOME-based paths for bash code blocks\r\n  if (normalized.startsWith('~/')) {\r\n    return '$HOME' + normalized.slice(1);\r\n  }\r\n  // For relative paths or paths not under $HOME, return as-is\r\n  return normalized;\r\n}\r\n\r\n// Helper to get directory name for a runtime (used for local/project installs)\r\nfunction getDirName(runtime: string): string {\r\n  if (runtime === 'copilot') return '.github';\r\n  if (runtime === 'opencode') return '.opencode';\r\n  if (runtime === 'gemini') return '.gemini';\r\n  if (runtime === 'codex') return '.codex';\r\n  if (runtime === 'qwen') return '.qwen';\r\n  if (runtime === 'kimi') return '.kimi';\r\n  return '.claude';\r\n}\r\n\r\n/**\r\n * Get the config directory path relative to home directory for a runtime\r\n * Used for templating hooks that use path.join(homeDir, '<configDir>', ...)\r\n * @param {string} runtime - 'claude', 'opencode', 'gemini', 'codex', 'copilot', or 'qwen'\r\n * @param {boolean} isGlobal - Whether this is a global install\r\n */\r\nfunction getConfigDirFromHome(runtime: string, isGlobal: boolean): string {\r\n  if (!isGlobal) {\r\n    // Local installs use the same dir name pattern\r\n    return `'${getDirName(runtime)}'`;\r\n  }\r\n  // Global installs - OpenCode uses XDG path structure\r\n  if (runtime === 'copilot') return \"'.copilot'\";\r\n  if (runtime === 'opencode') {\r\n    // OpenCode: ~/.config/opencode -> '.config', 'opencode'\r\n    // Return as comma-separated for path.join() replacement\r\n    return \"'.config', 'opencode'\";\r\n  }\r\n  if (runtime === 'gemini') return \"'.gemini'\";\r\n  if (runtime === 'codex') return \"'.codex'\";\r\n  if (runtime === 'qwen') return \"'.qwen'\";\r\n  if (runtime === 'kimi') return \"'.kimi'\";\r\n  return \"'.claude'\";\r\n}\r\n\r\n/**\r\n * Get the global config directory for OpenCode\r\n * OpenCode follows XDG Base Directory spec and uses ~/.config/opencode/\r\n * Priority: OPENCODE_CONFIG_DIR > dirname(OPENCODE_CONFIG) > XDG_CONFIG_HOME/opencode > ~/.config/opencode\r\n */\r\nfunction getOpencodeGlobalDir() {\r\n  // 1. Explicit OPENCODE_CONFIG_DIR env var\r\n  if (process.env.OPENCODE_CONFIG_DIR) {\r\n    return expandTilde(process.env.OPENCODE_CONFIG_DIR);\r\n  }\r\n  \r\n  // 2. OPENCODE_CONFIG env var (use its directory)\r\n  if (process.env.OPENCODE_CONFIG) {\r\n    return path.dirname(expandTilde(process.env.OPENCODE_CONFIG));\r\n  }\r\n  \r\n  // 3. XDG_CONFIG_HOME/opencode\r\n  if (process.env.XDG_CONFIG_HOME) {\r\n    return path.join(expandTilde(process.env.XDG_CONFIG_HOME), 'opencode');\r\n  }\r\n  \r\n  // 4. Default: ~/.config/opencode (XDG default)\r\n  return path.join(os.homedir(), '.config', 'opencode');\r\n}\r\n\r\n/**\r\n * Get the global config directory for a runtime\r\n * @param {string} runtime - 'claude', 'opencode', 'gemini', 'codex', or 'copilot'\r\n * @param {string|null} explicitDir - Explicit directory from --config-dir flag\r\n */\r\nfunction getGlobalDir(runtime: string, explicitDir: string | null = null): string {\r\n  if (runtime === 'opencode') {\r\n    // For OpenCode, --config-dir overrides env vars\r\n    if (explicitDir) {\r\n      return expandTilde(explicitDir);\r\n    }\r\n    return getOpencodeGlobalDir();\r\n  }\r\n  \r\n  if (runtime === 'gemini') {\r\n    // Gemini: --config-dir > GEMINI_CONFIG_DIR > ~/.gemini\r\n    if (explicitDir) {\r\n      return expandTilde(explicitDir);\r\n    }\r\n    if (process.env.GEMINI_CONFIG_DIR) {\r\n      return expandTilde(process.env.GEMINI_CONFIG_DIR);\r\n    }\r\n    return path.join(os.homedir(), '.gemini');\r\n  }\r\n\r\n  if (runtime === 'codex') {\r\n    // Codex: --config-dir > CODEX_HOME > ~/.codex\r\n    if (explicitDir) {\r\n      return expandTilde(explicitDir);\r\n    }\r\n    if (process.env.CODEX_HOME) {\r\n      return expandTilde(process.env.CODEX_HOME);\r\n    }\r\n    return path.join(os.homedir(), '.codex');\r\n  }\r\n\r\n  if (runtime === 'copilot') {\r\n    // Copilot: --config-dir > COPILOT_CONFIG_DIR > ~/.copilot\r\n    if (explicitDir) {\r\n      return expandTilde(explicitDir);\r\n    }\r\n    if (process.env.COPILOT_CONFIG_DIR) {\r\n      return expandTilde(process.env.COPILOT_CONFIG_DIR);\r\n    }\r\n    return path.join(os.homedir(), '.copilot');\r\n  }\r\n\r\n  if (runtime === 'qwen') {\r\n    // Qwen Code: --config-dir > QWEN_CONFIG_DIR > ~/.qwen\r\n    if (explicitDir) {\r\n      return expandTilde(explicitDir);\r\n    }\r\n    if (process.env.QWEN_CONFIG_DIR) {\r\n      return expandTilde(process.env.QWEN_CONFIG_DIR);\r\n    }\r\n    return path.join(os.homedir(), '.qwen');\r\n  }\r\n\r\n  if (runtime === 'kimi') {\r\n    // Kimi Code: --config-dir > KIMI_CONFIG_DIR > ~/.kimi\r\n    if (explicitDir) {\r\n      return expandTilde(explicitDir);\r\n    }\r\n    if (process.env.KIMI_CONFIG_DIR) {\r\n      return expandTilde(process.env.KIMI_CONFIG_DIR);\r\n    }\r\n    return path.join(os.homedir(), '.kimi');\r\n  }\r\n\r\n  // Claude Code: --config-dir > CLAUDE_CONFIG_DIR > ~/.claude\r\n  if (explicitDir) {\r\n    return expandTilde(explicitDir);\r\n  }\r\n  if (process.env.CLAUDE_CONFIG_DIR) {\r\n    return expandTilde(process.env.CLAUDE_CONFIG_DIR);\r\n  }\r\n  return path.join(os.homedir(), '.claude');\r\n}\r\n\r\nconst banner = '\\n' +\r\n  cyan + '  \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557\u2588\u2588\u2557  \u2588\u2588\u2557\\n' +\r\n  '  \u2588\u2588\u2554\u2550\u2550\u2550\u2550\u255D\u255A\u2588\u2588\u2557\u2588\u2588\u2554\u255D\\n' +\r\n  '  \u2588\u2588\u2588\u2588\u2588\u2557   \u255A\u2588\u2588\u2588\u2554\u255D \\n' +\r\n  '  \u2588\u2588\u2554\u2550\u2550\u255D   \u2588\u2588\u2554\u2588\u2588\u2557 \\n' +\r\n  '  \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557\u2588\u2588\u2554\u255D\u255A\u2588\u2588\u2557\\n' +\r\n  '  \u255A\u2550\u2550\u2550\u2550\u2550\u2550\u255D\u255A\u2550\u255D \u255A\u2550\u255D' + reset + '\\n' +\r\n  '\\n' +\r\n  '  EZ_Agents ' + dim + 'v' + pkg.version + reset + '\\n' +\r\n  '  Multi-Model Edition' + reset + '\\n' +\r\n  '  A meta-prompting, context engineering and spec-driven\\n' +\r\n  '  development system for Claude Code, OpenCode, Gemini, Codex, Copilot,\\n' +\r\n  '  Qwen Code, Kimi Code, with multi-model support.\\n' +\r\n  '  Original by T\u00C2CHES | Multi-Model Fork by @howlil.\\n';\r\n\r\n\r\n// Parse --config-dir argument\r\nfunction parseConfigDirArg() {\r\n  const configDirIndex = args.findIndex(arg => arg === '--config-dir' || arg === '-c');\r\n  if (configDirIndex !== -1) {\r\n    const nextArg = args[configDirIndex + 1];\r\n    // Error if --config-dir is provided without a value or next arg is another flag\r\n    if (!nextArg || nextArg.startsWith('-')) {\r\n      console.error(`  ${yellow}--config-dir requires a path argument${reset}`);\r\n      process.exit(1);\r\n    }\r\n    return nextArg;\r\n  }\r\n  // Also handle --config-dir=value format\r\n  const configDirArg = args.find(arg => arg.startsWith('--config-dir=') || arg.startsWith('-c='));\r\n  if (configDirArg) {\r\n    const value = configDirArg.split('=')[1];\r\n    if (!value) {\r\n      console.error(`  ${yellow}--config-dir requires a non-empty path${reset}`);\r\n      process.exit(1);\r\n    }\r\n    return value;\r\n  }\r\n  return null;\r\n}\r\nconst explicitConfigDir = parseConfigDirArg();\r\nconst hasHelp = args.includes('--help') || args.includes('-h');\r\nconst hasVersion = args.includes('--version') || args.includes('-v');\r\nconst forceStatusline = args.includes('--force-statusline');\r\n\r\n// Show version if requested (skip banner, just output version)\r\nif (hasVersion) {\r\n  console.log(pkg.version);\r\n  process.exit(0);\r\n}\r\n\r\nconsole.log(banner);\r\n\r\nif (hasUninstall) {\r\n  console.log('  Mode: Uninstall\\n');\r\n}\r\n\r\n// Show help if requested\r\nif (hasHelp) {\r\n  console.log(`  ${yellow}Usage:${reset} npx ez-agents [options]\\n\\n  ${yellow}Options:${reset}\\n    ${cyan}-g, --global${reset}              Install globally (to config directory)\\n    ${cyan}-l, --local${reset}               Install locally (to current directory)\\n    ${cyan}--claude${reset}                  Install for Claude Code only\\n    ${cyan}--opencode${reset}                Install for OpenCode only\\n    ${cyan}--gemini${reset}                  Install for Gemini only\\n    ${cyan}--codex${reset}                   Install for Codex only\\n    ${cyan}--copilot${reset}                 Install for Copilot only\\n    ${cyan}--qwen${reset}                    Install for Qwen Code only\\n    ${cyan}--kimi${reset}                    Install for Kimi Code only\\n    ${cyan}--all${reset}                     Install for all runtimes\\n    ${cyan}-u, --uninstall${reset}           Uninstall EZ_Agents (remove all EZ_Agents files)\\n    ${cyan}-c, --config-dir <path>${reset}   Specify custom config directory\\n    ${cyan}-v, --version${reset}             Show version number\\n    ${cyan}-h, --help${reset}                Show this help message\\n    ${cyan}--force-statusline${reset}        Replace existing statusline config\\n\\n  ${yellow}Examples:${reset}\\n    ${dim}# Interactive install (prompts for runtime and location)${reset}\\n    npx ez-agents\\n\\n    ${dim}# Install for Claude Code globally${reset}\\n    npx ez-agents --claude --global\\n\\n    ${dim}# Install for Qwen Code globally${reset}\\n    npx ez-agents --qwen --global\\n\\n    ${dim}# Install for Kimi Code globally${reset}\\n    npx ez-agents --kimi --global\\n\\n    ${dim}# Install for all runtimes globally${reset}\\n    npx ez-agents --all --global\\n\\n    ${dim}# Uninstall EZ_Agents globally${reset}\\n    npx ez-agents --all --global --uninstall\\n\\n    ${dim}# Show version${reset}\\n    npx ez-agents --version\\n\\n  ${yellow}Notes:${reset}\\n    The --config-dir option is useful when you have multiple configurations.\\n    It takes priority over CLAUDE_CONFIG_DIR / GEMINI_CONFIG_DIR / CODEX_HOME / COPILOT_CONFIG_DIR / QWEN_CONFIG_DIR / KIMI_CONFIG_DIR environment variables.\\n\\n  ${yellow}Model Providers:${reset}\\n    OpenAI and Anthropic are model providers configured in settings.json,\\n    not separate CLI runtimes. See README.md for model configuration.\\n`);\r\n  process.exit(0);\r\n}\r\n\r\n/**\r\n * Expand ~ to home directory (shell doesn't expand in env vars passed to node)\r\n */\r\nfunction expandTilde(filePath) {\r\n  if (filePath && filePath.startsWith('~/')) {\r\n    return path.join(os.homedir(), filePath.slice(2));\r\n  }\r\n  return filePath;\r\n}\r\n\r\n/**\r\n * Build a hook command path using forward slashes for cross-platform compatibility.\r\n * On Windows, $HOME is not expanded by cmd.exe/PowerShell, so we use the actual path.\r\n */\r\nfunction buildHookCommand(configDir, hookName) {\r\n  // Use forward slashes for Node.js compatibility on all platforms\r\n  const hooksPath = configDir.replace(/\\\\/g, '/') + '/hooks/' + hookName;\r\n  return `node \"${hooksPath}\"`;\r\n}\r\n\r\n/**\r\n * Read and parse settings.json, returning empty object if it doesn't exist\r\n */\r\nfunction readSettings(settingsPath) {\r\n  if (fs.existsSync(settingsPath)) {\r\n    try {\r\n      return JSON.parse(fs.readFileSync(settingsPath, 'utf8'));\r\n    } catch (e) {\r\n      return {};\r\n    }\r\n  }\r\n  return {};\r\n}\r\n\r\n/**\r\n * Write settings.json with proper formatting\r\n */\r\nfunction writeSettings(settingsPath, settings) {\r\n  fs.writeFileSync(settingsPath, JSON.stringify(settings, null, 2) + '\\n');\r\n}\r\n\r\n// Cache for attribution settings (populated once per runtime during install)\r\nconst attributionCache = new Map<string, null | undefined | string>();\r\n\r\n/**\r\n * Get commit attribution setting for a runtime\r\n * @param {string} runtime - 'claude', 'opencode', 'gemini', 'codex', or 'copilot'\r\n * @returns {null|undefined|string} null = remove, undefined = keep default, string = custom\r\n */\r\nfunction getCommitAttribution(runtime: string): null | undefined | string {\r\n  // Return cached value if available\r\n  if (attributionCache.has(runtime)) {\r\n    return attributionCache.get(runtime);\r\n  }\r\n\r\n  let result: null | undefined | string;\r\n\r\n  if (runtime === 'opencode') {\r\n    const config = readSettings(path.join(getGlobalDir('opencode', null), 'opencode.json'));\r\n    result = config.disable_ai_attribution === true ? null : undefined;\r\n  } else if (runtime === 'gemini') {\r\n    // Gemini: check gemini settings.json for attribution config\r\n    const settings = readSettings(path.join(getGlobalDir('gemini', explicitConfigDir), 'settings.json'));\r\n    if (!settings.attribution || settings.attribution.commit === undefined) {\r\n      result = undefined;\r\n    } else if (settings.attribution.commit === '') {\r\n      result = null;\r\n    } else {\r\n      result = settings.attribution.commit;\r\n    }\r\n  } else if (runtime === 'claude') {\r\n    // Claude Code\r\n    const settings = readSettings(path.join(getGlobalDir('claude', explicitConfigDir), 'settings.json'));\r\n    if (!settings.attribution || settings.attribution.commit === undefined) {\r\n      result = undefined;\r\n    } else if (settings.attribution.commit === '') {\r\n      result = null;\r\n    } else {\r\n      result = settings.attribution.commit;\r\n    }\r\n  } else {\r\n    // Codex and Copilot currently have no attribution setting equivalent\r\n    result = undefined;\r\n  }\r\n\r\n  // Cache and return\r\n  attributionCache.set(runtime, result);\r\n  return result;\r\n}\r\n\r\n/**\r\n * Process Co-Authored-By lines based on attribution setting\r\n * @param {string} content - File content to process\r\n * @param {null|undefined|string} attribution - null=remove, undefined=keep, string=replace\r\n * @returns {string} Processed content\r\n */\r\nfunction processAttribution(content: string, attribution: null | undefined | string): string {\r\n  if (attribution === null) {\r\n    // Remove Co-Authored-By lines and the preceding blank line\r\n    return content.replace(/(\\r?\\n){2}Co-Authored-By:.*$/gim, '');\r\n  }\r\n  if (attribution === undefined) {\r\n    return content;\r\n  }\r\n  // Replace with custom attribution (escape $ to prevent backreference injection)\r\n  const safeAttribution = attribution.replace(/\\$/g, '$$$$');\r\n  return content.replace(/Co-Authored-By:.*$/gim, `Co-Authored-By: ${safeAttribution}`);\r\n}\r\n\r\n/**\r\n * Convert Claude Code frontmatter to opencode format\r\n * - Converts 'allowed-tools:' array to 'permission:' object\r\n * @param {string} content - Markdown file content with YAML frontmatter\r\n * @returns {string} - Content with converted frontmatter\r\n */\r\n// Color name to hex mapping for opencode compatibility\r\nconst colorNameToHex = {\r\n  cyan: '#00FFFF',\r\n  red: '#FF0000',\r\n  green: '#00FF00',\r\n  blue: '#0000FF',\r\n  yellow: '#FFFF00',\r\n  magenta: '#FF00FF',\r\n  orange: '#FFA500',\r\n  purple: '#800080',\r\n  pink: '#FFC0CB',\r\n  white: '#FFFFFF',\r\n  black: '#000000',\r\n  gray: '#808080',\r\n  grey: '#808080',\r\n};\r\n\r\n// Tool name mapping from Claude Code to OpenCode\r\n// OpenCode uses lowercase tool names; special mappings for renamed tools\r\nconst claudeToOpencodeTools = {\r\n  AskUserQuestion: 'question',\r\n  SlashCommand: 'skill',\r\n  TodoWrite: 'todowrite',\r\n  WebFetch: 'webfetch',\r\n  WebSearch: 'websearch',  // Plugin/MCP - keep for compatibility\r\n};\r\n\r\n// Tool name mapping from Claude Code to Gemini CLI\r\n// Gemini CLI uses snake_case built-in tool names\r\nconst claudeToGeminiTools = {\r\n  Read: 'read_file',\r\n  Write: 'write_file',\r\n  Edit: 'replace',\r\n  Bash: 'run_shell_command',\r\n  Glob: 'glob',\r\n  Grep: 'search_file_content',\r\n  WebSearch: 'google_web_search',\r\n  WebFetch: 'web_fetch',\r\n  TodoWrite: 'write_todos',\r\n  AskUserQuestion: 'ask_user',\r\n};\r\n\r\n/**\r\n * Convert a Claude Code tool name to OpenCode format\r\n * - Applies special mappings (AskUserQuestion -> question, etc.)\r\n * - Converts to lowercase (except MCP tools which keep their format)\r\n */\r\nfunction convertToolName(claudeTool) {\r\n  // Check for special mapping first\r\n  if (claudeToOpencodeTools[claudeTool]) {\r\n    return claudeToOpencodeTools[claudeTool];\r\n  }\r\n  // MCP tools (mcp__*) keep their format\r\n  if (claudeTool.startsWith('mcp__')) {\r\n    return claudeTool;\r\n  }\r\n  // Default: convert to lowercase\r\n  return claudeTool.toLowerCase();\r\n}\r\n\r\n/**\r\n * Convert a Claude Code tool name to Gemini CLI format\r\n * - Applies Claude\u2192Gemini mapping (Read\u2192read_file, Bash\u2192run_shell_command, etc.)\r\n * - Filters out MCP tools (mcp__*) \u2014 they are auto-discovered at runtime in Gemini\r\n * - Filters out Task \u2014 agents are auto-registered as tools in Gemini\r\n * @returns {string|null} Gemini tool name, or null if tool should be excluded\r\n */\r\nfunction convertGeminiToolName(claudeTool) {\r\n  // MCP tools: exclude \u2014 auto-discovered from mcpServers config at runtime\r\n  if (claudeTool.startsWith('mcp__')) {\r\n    return null;\r\n  }\r\n  // Task: exclude \u2014 agents are auto-registered as callable tools\r\n  if (claudeTool === 'Task') {\r\n    return null;\r\n  }\r\n  // Check for explicit mapping\r\n  if (claudeToGeminiTools[claudeTool]) {\r\n    return claudeToGeminiTools[claudeTool];\r\n  }\r\n  // Default: lowercase\r\n  return claudeTool.toLowerCase();\r\n}\r\n\r\n/**\r\n * Convert a Claude Code tool name to GitHub Copilot format.\r\n * - Applies explicit mapping from claudeToCopilotTools\r\n * - Handles mcp__context7__* prefix \u2192 io.github.upstash/context7/*\r\n * - Falls back to lowercase for unknown tools\r\n */\r\nfunction convertCopilotToolName(claudeTool) {\r\n  // mcp__context7__* wildcard \u2192 io.github.upstash/context7/*\r\n  if (claudeTool.startsWith('mcp__context7__')) {\r\n    return 'io.github.upstash/context7/' + claudeTool.slice('mcp__context7__'.length);\r\n  }\r\n  // Check explicit mapping\r\n  if (claudeToCopilotTools[claudeTool]) {\r\n    return claudeToCopilotTools[claudeTool];\r\n  }\r\n  // Default: lowercase\r\n  return claudeTool.toLowerCase();\r\n}\r\n\r\n/**\r\n * Apply Copilot-specific content conversion \u2014 CONV-06 (paths) + CONV-07 (command names).\r\n * Path mappings depend on install mode:\r\n *   Global: ~/.claude/ \u2192 ~/.copilot/, ./.claude/ \u2192 ./.github/\r\n *   Local:  ~/.claude/ \u2192 ./.github/, ./.claude/ \u2192 ./.github/\r\n * Applied to ALL Copilot content (skills, agents, engine files).\r\n * @param {string} content - Source content to convert\r\n * @param {boolean} [isGlobal=false] - Whether this is a global install\r\n */\r\nfunction convertClaudeToCopilotContent(content, isGlobal = false) {\r\n  let c = content;\r\n  // CONV-06: Path replacement \u2014 most specific first to avoid substring matches\r\n  if (isGlobal) {\r\n    c = c.replace(/\\$HOME\\/\\.claude\\//g, '$HOME/.copilot/');\r\n    c = c.replace(/~\\/\\.claude\\//g, '~/.copilot/');\r\n  } else {\r\n    c = c.replace(/\\$HOME\\/\\.claude\\//g, '.github/');\r\n    c = c.replace(/~\\/\\.claude\\//g, '.github/');\r\n  }\r\n  c = c.replace(/\\.\\/\\.claude\\//g, './.github/');\r\n  c = c.replace(/\\.claude\\//g, '.github/');\r\n  // CONV-07: Command name conversion (all ez: references \u2192 ez-)\r\n  c = c.replace(/ez:/g, 'ez-');\r\n  return c;\r\n}\r\n\r\n/**\r\n * Convert a Claude command (.md) to a Copilot skill (SKILL.md).\r\n * Transforms frontmatter only \u2014 body passes through with CONV-06/07 applied.\r\n * Skills keep original tool names (no mapping) per CONTEXT.md decision.\r\n */\r\nfunction convertClaudeCommandToCopilotSkill(content, skillName, isGlobal = false) {\r\n  const converted = convertClaudeToCopilotContent(content, isGlobal);\r\n  const { frontmatter, body } = extractFrontmatterAndBody(converted);\r\n  if (!frontmatter) return converted;\r\n\r\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\r\n  const argumentHint = extractFrontmatterField(frontmatter, 'argument-hint');\r\n  const agent = extractFrontmatterField(frontmatter, 'agent');\r\n\r\n  // CONV-02: Extract allowed-tools YAML multiline list \u2192 comma-separated string\r\n  const toolsMatch = frontmatter.match(/^allowed-tools:\\s*\\n((?:\\s+-\\s+.+\\n?)*)/m);\r\n  let toolsLine = '';\r\n  if (toolsMatch) {\r\n    const tools = toolsMatch[1].match(/^\\s+-\\s+(.+)/gm);\r\n    if (tools) {\r\n      toolsLine = tools.map(t => t.replace(/^\\s+-\\s+/, '').trim()).join(', ');\r\n    }\r\n  }\r\n\r\n  // Reconstruct frontmatter in Copilot format\r\n  let fm = `---\\nname: ${skillName}\\ndescription: ${description}\\n`;\r\n  if (argumentHint) fm += `argument-hint: ${yamlQuote(argumentHint)}\\n`;\r\n  if (agent) fm += `agent: ${agent}\\n`;\r\n  if (toolsLine) fm += `allowed-tools: ${toolsLine}\\n`;\r\n  fm += '---';\r\n\r\n  return `${fm}\\n${body}`;\r\n}\r\n\r\n/**\r\n * Convert a Claude agent (.md) to a Copilot agent (.agent.md).\r\n * Applies tool mapping + deduplication, formats tools as JSON array.\r\n * CONV-04: JSON array format. CONV-05: Tool name mapping.\r\n */\r\nfunction convertClaudeAgentToCopilotAgent(content, isGlobal = false) {\r\n  const converted = convertClaudeToCopilotContent(content, isGlobal);\r\n  const { frontmatter, body } = extractFrontmatterAndBody(converted);\r\n  if (!frontmatter) return converted;\r\n\r\n  const name = extractFrontmatterField(frontmatter, 'name') || 'unknown';\r\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\r\n  const color = extractFrontmatterField(frontmatter, 'color');\r\n  const toolsRaw = extractFrontmatterField(frontmatter, 'tools') || '';\r\n\r\n  // CONV-04 + CONV-05: Map tools, deduplicate, format as JSON array\r\n  const claudeTools = toolsRaw.split(',').map(t => t.trim()).filter(Boolean);\r\n  const mappedTools = claudeTools.map(t => convertCopilotToolName(t));\r\n  const uniqueTools = [...new Set(mappedTools)];\r\n  const toolsArray = uniqueTools.length > 0\r\n    ? \"['\" + uniqueTools.join(\"', '\") + \"']\"\r\n    : '[]';\r\n\r\n  // Reconstruct frontmatter in Copilot format\r\n  let fm = `---\\nname: ${name}\\ndescription: ${description}\\ntools: ${toolsArray}\\n`;\r\n  if (color) fm += `color: ${color}\\n`;\r\n  fm += '---';\r\n\r\n  return `${fm}\\n${body}`;\r\n}\r\n\r\nfunction toSingleLine(value) {\r\n  return value.replace(/\\s+/g, ' ').trim();\r\n}\r\n\r\nfunction yamlQuote(value) {\r\n  return JSON.stringify(value);\r\n}\r\n\r\nfunction extractFrontmatterAndBody(content) {\r\n  if (!content.startsWith('---')) {\r\n    return { frontmatter: null, body: content };\r\n  }\r\n\r\n  const endIndex = content.indexOf('---', 3);\r\n  if (endIndex === -1) {\r\n    return { frontmatter: null, body: content };\r\n  }\r\n\r\n  return {\r\n    frontmatter: content.substring(3, endIndex).trim(),\r\n    body: content.substring(endIndex + 3),\r\n  };\r\n}\r\n\r\nfunction extractFrontmatterField(frontmatter, fieldName) {\r\n  const regex = new RegExp(`^${fieldName}:\\\\s*(.+)$`, 'm');\r\n  const match = frontmatter.match(regex);\r\n  if (!match) return null;\r\n  return match[1].trim().replace(/^['\"]|['\"]$/g, '');\r\n}\r\n\r\nfunction convertSlashCommandsToCodexSkillMentions(content) {\r\n  let converted = content.replace(/\\/ez:([a-z0-9-]+)/gi, (_, commandName) => {\r\n    return `$ez-${String(commandName).toLowerCase()}`;\r\n  });\r\n  converted = converted.replace(/\\/ez-help\\b/g, '$ez-help');\r\n  return converted;\r\n}\r\n\r\nfunction convertClaudeToCodexMarkdown(content) {\r\n  let converted = convertSlashCommandsToCodexSkillMentions(content);\r\n  converted = converted.replace(/\\$ARGUMENTS\\b/g, '{{EZ_ARGS}}');\r\n  return converted;\r\n}\r\n\r\nfunction getCodexSkillAdapterHeader(skillName) {\r\n  const invocation = `$${skillName}`;\r\n  return `<codex_skill_adapter>\r\n## A. Skill Invocation\r\n- This skill is invoked by mentioning \\`${invocation}\\`.\r\n- Treat all user text after \\`${invocation}\\` as \\`{{EZ_ARGS}}\\`.\r\n- If no arguments are present, treat \\`{{EZ_ARGS}}\\` as empty.\r\n\r\n## B. AskUserQuestion \u2192 request_user_input Mapping\r\nEZ_Agents workflows use \\`AskUserQuestion\\` (Claude Code syntax). Translate to Codex \\`request_user_input\\`:\r\n\r\nParameter mapping:\r\n- \\`header\\` \u2192 \\`header\\`\r\n- \\`question\\` \u2192 \\`question\\`\r\n- Options formatted as \\`\"Label\" \u2014 description\\` \u2192 \\`{label: \"Label\", description: \"description\"}\\`\r\n- Generate \\`id\\` from header: lowercase, replace spaces with underscores\r\n\r\nBatched calls:\r\n- \\`AskUserQuestion([q1, q2])\\` \u2192 single \\`request_user_input\\` with multiple entries in \\`questions[]\\`\r\n\r\nMulti-select workaround:\r\n- Codex has no \\`multiSelect\\`. Use sequential single-selects, or present a numbered freeform list asking the user to enter comma-separated numbers.\r\n\r\nExecute mode fallback:\r\n- When \\`request_user_input\\` is rejected (Execute mode), present a plain-text numbered list and pick a reasonable default.\r\n\r\n## C. Task() \u2192 spawn_agent Mapping\r\nEZ_Agents workflows use \\`Task(...)\\` (Claude Code syntax). Translate to Codex collaboration tools:\r\n\r\nDirect mapping:\r\n- \\`Task(subagent_type=\"X\", prompt=\"Y\")\\` \u2192 \\`spawn_agent(agent_type=\"X\", message=\"Y\")\\`\r\n- \\`Task(model=\"...\")\\` \u2192 omit (Codex uses per-role config, not inline model selection)\r\n- \\`fork_context: false\\` by default \u2014 EZ_Agents agents load their own context via \\`<files_to_read>\\` blocks\r\n\r\nParallel fan-out:\r\n- Spawn multiple agents \u2192 collect agent IDs \u2192 \\`wait(ids)\\` for all to complete\r\n\r\nResult parsing:\r\n- Look for structured markers in agent output: \\`CHECKPOINT\\`, \\`PLAN COMPLETE\\`, \\`SUMMARY\\`, etc.\r\n- \\`close_agent(id)\\` after collecting results from each agent\r\n</codex_skill_adapter>`;\r\n}\r\n\r\nfunction convertClaudeCommandToCodexSkill(content, skillName) {\r\n  const converted = convertClaudeToCodexMarkdown(content);\r\n  const { frontmatter, body } = extractFrontmatterAndBody(converted);\r\n  let description = `Run EZ_Agents workflow ${skillName}.`;\r\n  if (frontmatter) {\r\n    const maybeDescription = extractFrontmatterField(frontmatter, 'description');\r\n    if (maybeDescription) {\r\n      description = maybeDescription;\r\n    }\r\n  }\r\n  description = toSingleLine(description);\r\n  const shortDescription = description.length > 180 ? `${description.slice(0, 177)}...` : description;\r\n  const adapter = getCodexSkillAdapterHeader(skillName);\r\n\r\n  return `---\\nname: ${yamlQuote(skillName)}\\ndescription: ${yamlQuote(description)}\\nmetadata:\\n  short-description: ${yamlQuote(shortDescription)}\\n---\\n\\n${adapter}\\n\\n${body.trimStart()}`;\r\n}\r\n\r\nfunction convertClaudeToQwenAgent(content) {\r\n  const { frontmatter, body } = extractFrontmatterAndBody(content);\r\n  if (!frontmatter) return content;\r\n\r\n  const name = extractFrontmatterField(frontmatter, 'name') || 'unknown';\r\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\r\n\r\n  // Qwen Code uses simple markdown. We'll put name and description in the body.\r\n  return `# Agent: ${name}\\n\\n${description}\\n\\n${body.trimStart()}`;\r\n}\r\n\r\nfunction convertClaudeToKimiSkill(content, skillName) {\r\n  const { frontmatter, body } = extractFrontmatterAndBody(content);\r\n  if (!frontmatter) return content;\r\n\r\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\r\n\r\n  // Reconstruct as simple Markdown\r\n  return `# Skill: ${skillName}\\n\\n${description}\\n\\n${body.trimStart()}`;\r\n}\r\n\r\nfunction convertClaudeToKimiAgent(content) {\r\n  const { frontmatter, body } = extractFrontmatterAndBody(content);\r\n  if (!frontmatter) return content;\r\n\r\n  const name = extractFrontmatterField(frontmatter, 'name') || 'unknown';\r\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\r\n\r\n  // Kimi Code uses simple markdown\r\n  return `# Agent: ${name}\\n\\n${description}\\n\\n${body.trimStart()}`;\r\n}\r\n\r\n/**\r\n * Convert a Claude command (.md) to a Qwen skill (SKILL.md).\r\n * Strips frontmatter and formats as simple Markdown.\r\n */\r\nfunction convertClaudeToQwenSkill(content, skillName) {\r\n  const { frontmatter, body } = extractFrontmatterAndBody(content);\r\n  if (!frontmatter) return content;\r\n\r\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\r\n\r\n  // Reconstruct as simple Markdown\r\n  return `# Skill: ${skillName}\\n\\n${description}\\n\\n${body.trimStart()}`;\r\n}\r\n\r\n/**\r\n * Convert Claude Code agent markdown to Codex agent format.\r\n * Applies base markdown conversions, then adds a <codex_agent_role> header\r\n * and cleans up frontmatter (removes tools/color fields).\r\n */\r\nfunction convertClaudeAgentToCodexAgent(content) {\r\n  let converted = convertClaudeToCodexMarkdown(content);\r\n\r\n  const { frontmatter, body } = extractFrontmatterAndBody(converted);\r\n  if (!frontmatter) return converted;\r\n\r\n  const name = extractFrontmatterField(frontmatter, 'name') || 'unknown';\r\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\r\n  const tools = extractFrontmatterField(frontmatter, 'tools') || '';\r\n\r\n  const roleHeader = `<codex_agent_role>\r\nrole: ${name}\r\ntools: ${tools}\r\npurpose: ${toSingleLine(description)}\r\n</codex_agent_role>`;\r\n\r\n  const cleanFrontmatter = `---\\nname: ${yamlQuote(name)}\\ndescription: ${yamlQuote(toSingleLine(description))}\\n---`;\r\n\r\n  return `${cleanFrontmatter}\\n\\n${roleHeader}\\n${body}`;\r\n}\r\n\r\n/**\r\n * Generate a per-agent .toml config file for Codex.\r\n * Sets sandbox_mode and developer_instructions from the agent markdown body.\r\n */\r\nfunction generateCodexAgentToml(agentName, agentContent) {\r\n  const sandboxMode = CODEX_AGENT_SANDBOX[agentName] || 'read-only';\r\n  const { body } = extractFrontmatterAndBody(agentContent);\r\n  const instructions = body.trim();\r\n\r\n  const lines = [\r\n    `sandbox_mode = \"${sandboxMode}\"`,\r\n    // Agent prompts contain raw backslashes in regexes and shell snippets.\r\n    // TOML literal multiline strings preserve them without escape parsing.\r\n    `developer_instructions = '''`,\r\n    instructions,\r\n    `'''`,\r\n  ];\r\n  return lines.join('\\n') + '\\n';\r\n}\r\n\r\n/**\r\n * Generate the EZ_Agents config block for Codex config.toml.\r\n * @param {Array<{name: string, description: string}>} agents\r\n */\r\nfunction generateCodexConfigBlock(agents) {\r\n  const lines = [\r\n    EZ_CODEX_MARKER,\r\n    '',\r\n  ];\r\n\r\n  for (const { name, description } of agents) {\r\n    lines.push(`[agents.${name}]`);\r\n    lines.push(`description = ${JSON.stringify(description)}`);\r\n    lines.push(`config_file = \"agents/${name}.toml\"`);\r\n    lines.push('');\r\n  }\r\n\r\n  return lines.join('\\n');\r\n}\r\n\r\n/**\r\n * Strip EZ_Agents sections from Codex config.toml content.\r\n * Returns cleaned content, or null if file would be empty.\r\n */\r\nfunction stripEzAgentsFromCodexConfig(content) {\r\n  const markerIndex = content.indexOf(EZ_CODEX_MARKER);\r\n\r\n  if (markerIndex !== -1) {\r\n    // Has EZ_Agents marker \u2014 remove everything from marker to EOF\r\n    let before = content.substring(0, markerIndex).trimEnd();\r\n    // Also strip EZ-injected feature keys above the marker (Case 3 inject)\r\n    before = before.replace(/^multi_agent\\s*=\\s*true\\s*\\n?/m, '');\r\n    before = before.replace(/^default_mode_request_user_input\\s*=\\s*true\\s*\\n?/m, '');\r\n    before = before.replace(/^\\[features\\]\\s*\\n(?=\\[|$)/m, '');\r\n    before = before.replace(/\\n{3,}/g, '\\n\\n').trim();\r\n    if (!before) return null;\r\n    return before + '\\n';\r\n  }\r\n\r\n  // No marker but may have EZ-injected feature keys\r\n  let cleaned = content;\r\n  cleaned = cleaned.replace(/^multi_agent\\s*=\\s*true\\s*\\n?/m, '');\r\n  cleaned = cleaned.replace(/^default_mode_request_user_input\\s*=\\s*true\\s*\\n?/m, '');\r\n\r\n  // Remove [agents.ez-*] sections (from header to next section or EOF)\r\n  cleaned = cleaned.replace(/^\\[agents\\.ez-[^\\]]+\\]\\n(?:(?!\\[)[^\\n]*\\n?)*/gm, '');\r\n\r\n  // Remove [features] section if now empty (only header, no keys before next section)\r\n  cleaned = cleaned.replace(/^\\[features\\]\\s*\\n(?=\\[|$)/m, '');\r\n\r\n  // Remove [agents] section if now empty\r\n  cleaned = cleaned.replace(/^\\[agents\\]\\s*\\n(?=\\[|$)/m, '');\r\n\r\n  // Clean up excessive blank lines\r\n  cleaned = cleaned.replace(/\\n{3,}/g, '\\n\\n').trim();\r\n\r\n  if (!cleaned) return null;\r\n  return cleaned + '\\n';\r\n}\r\n\r\n/**\r\n * Merge EZ_Agents config block into an existing or new config.toml.\r\n * Three cases: new file, existing with EZ_Agents marker, existing without marker.\r\n */\r\nfunction mergeCodexConfig(configPath, EZ_AgentsBlock) {\r\n  // Case 1: No config.toml \u2014 create fresh\r\n  if (!fs.existsSync(configPath)) {\r\n    fs.writeFileSync(configPath, EZ_AgentsBlock + '\\n');\r\n    return;\r\n  }\r\n\r\n  const existing = fs.readFileSync(configPath, 'utf8');\r\n  const markerIndex = existing.indexOf(EZ_CODEX_MARKER);\r\n\r\n  // Case 2: Has EZ_Agents marker \u2014 truncate and re-append\r\n  if (markerIndex !== -1) {\r\n    let before = existing.substring(0, markerIndex).trimEnd();\r\n    if (before) {\r\n      // Strip any EZ-managed sections that leaked above the marker from previous installs\r\n      before = before.replace(/^\\[agents\\.ez-[^\\]]+\\]\\n(?:(?!\\[)[^\\n]*\\n?)*/gm, '');\r\n      before = before.replace(/^\\[agents\\]\\n(?:(?!\\[)[^\\n]*\\n?)*/m, '');\r\n      before = before.replace(/\\n{3,}/g, '\\n\\n').trimEnd();\r\n\r\n      fs.writeFileSync(configPath, before + '\\n\\n' + EZ_AgentsBlock + '\\n');\r\n    } else {\r\n      fs.writeFileSync(configPath, EZ_AgentsBlock + '\\n');\r\n    }\r\n    return;\r\n  }\r\n\r\n  // Case 3: No marker \u2014 append EZ_Agents block\r\n  let content = existing;\r\n  content = content.trimEnd() + '\\n\\n' + EZ_AgentsBlock + '\\n';\r\n\r\n  fs.writeFileSync(configPath, content);\r\n}\r\n\r\nfunction findCopilotInstructionsMarkerPair(content: string): { openIndex: number; closeIndex: number; closeMarker: string } | null {\r\n  const markerPairs: [string, string][] = [\r\n    [EZ_COPILOT_INSTRUCTIONS_MARKER, EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER],\r\n    [LEGACY_EZ_COPILOT_INSTRUCTIONS_MARKER, LEGACY_EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER],\r\n  ];\r\n\r\n  for (const [openMarker, closeMarker] of markerPairs) {\r\n    const openIndex = content.indexOf(openMarker);\r\n    const closeIndex = content.indexOf(closeMarker);\r\n    if (openIndex !== -1 && closeIndex !== -1 && closeIndex >= openIndex) {\r\n      return { openIndex, closeIndex, closeMarker };\r\n    }\r\n  }\r\n\r\n  return null;\r\n}\r\n\r\n/**\r\n * Merge EZ_Agents instructions into copilot-instructions.md.\r\n * Three cases: new file, existing with markers, existing without markers.\r\n * @param {string} filePath - Full path to copilot-instructions.md\r\n * @param {string} EZ_AgentsContent - Template content (without markers)\r\n */\r\nfunction mergeCopilotInstructions(filePath, EZ_AgentsContent) {\r\n  const EZ_AgentsBlock = EZ_COPILOT_INSTRUCTIONS_MARKER + '\\n' +\r\n    EZ_AgentsContent.trim() + '\\n' +\r\n    EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER;\r\n\r\n  // Case 1: No file \u2014 create fresh\r\n  if (!fs.existsSync(filePath)) {\r\n    fs.writeFileSync(filePath, EZ_AgentsBlock + '\\n');\r\n    return;\r\n  }\r\n\r\n  const existing = fs.readFileSync(filePath, 'utf8');\r\n  const markerMatch = findCopilotInstructionsMarkerPair(existing);\r\n\r\n  // Case 2: Has EZ_Agents markers \u2014 replace between markers\r\n  if (markerMatch) {\r\n    const before = existing.substring(0, markerMatch.openIndex).trimEnd();\r\n    const after = existing.substring(markerMatch.closeIndex + markerMatch.closeMarker.length).trimStart();\r\n    let newContent = '';\r\n    if (before) newContent += before + '\\n\\n';\r\n    newContent += EZ_AgentsBlock;\r\n    if (after) newContent += '\\n\\n' + after;\r\n    newContent += '\\n';\r\n    fs.writeFileSync(filePath, newContent);\r\n    return;\r\n  }\r\n\r\n  // Case 3: No markers \u2014 append at end\r\n  const content = existing.trimEnd() + '\\n\\n' + EZ_AgentsBlock + '\\n';\r\n  fs.writeFileSync(filePath, content);\r\n}\r\n\r\n/**\r\n * Strip EZ_Agents section from copilot-instructions.md content.\r\n * Returns cleaned content, or null if file should be deleted (was EZ-only).\r\n * @param {string} content - File content\r\n * @returns {string|null} - Cleaned content or null if empty\r\n */\r\nfunction stripEzAgentsFromCopilotInstructions(content) {\r\n  const markerMatch = findCopilotInstructionsMarkerPair(content);\r\n\r\n  if (markerMatch) {\r\n    const before = content.substring(0, markerMatch.openIndex).trimEnd();\r\n    const after = content.substring(markerMatch.closeIndex + markerMatch.closeMarker.length).trimStart();\r\n    const cleaned = (before + (before && after ? '\\n\\n' : '') + after).trim();\r\n    if (!cleaned) return null;\r\n    return cleaned + '\\n';\r\n  }\r\n\r\n  // No markers found \u2014 nothing to strip\r\n  return content;\r\n}\r\n\r\n/**\r\n * Generate config.toml and per-agent .toml files for Codex.\r\n * Reads agent .md files from source, extracts metadata, writes .toml configs.\r\n */\r\nfunction installCodexConfig(targetDir: string, agentsSrc: string): number {\r\n  const configPath = path.join(targetDir, 'config.toml');\r\n  const agentsTomlDir = path.join(targetDir, 'agents');\r\n  fs.mkdirSync(agentsTomlDir, { recursive: true });\r\n\r\n  const agentEntries = fs.readdirSync(agentsSrc).filter(f => f.startsWith('ez-') && f.endsWith('.md'));\r\n  const agents: Array<{ name: string; description: string }> = [];\r\n\r\n  // Compute the Codex pathPrefix for replacing .claude paths\r\n  // Use tilde-based path to avoid baking absolute paths into templates\r\n  const codexPathPrefix = `${targetDir.replace(/\\\\/g, '/').replace(os.homedir().replace(/\\\\/g, '/'), '~')}/`;\r\n\r\n  for (const file of agentEntries) {\r\n    let content = fs.readFileSync(path.join(agentsSrc, file), 'utf8');\r\n    // Replace .claude paths before generating TOML (source files use ~/.claude and $HOME/.claude)\r\n    content = content.replace(/~\\/\\.claude\\//g, codexPathPrefix);\r\n    content = content.replace(/\\$HOME\\/\\.claude\\//g, toHomePrefix(codexPathPrefix));\r\n    const { frontmatter } = extractFrontmatterAndBody(content);\r\n    const name = extractFrontmatterField(frontmatter, 'name') || file.replace('.md', '');\r\n    const description = extractFrontmatterField(frontmatter, 'description') || '';\r\n\r\n    agents.push({ name, description: toSingleLine(description) });\r\n\r\n    const tomlContent = generateCodexAgentToml(name, content);\r\n    fs.writeFileSync(path.join(agentsTomlDir, `${name}.toml`), tomlContent);\r\n  }\r\n\r\n  const EZ_AgentsBlock = generateCodexConfigBlock(agents);\r\n  mergeCodexConfig(configPath, EZ_AgentsBlock);\r\n\r\n  return agents.length;\r\n}\r\n\r\n/**\r\n * Strip HTML <sub> tags for Gemini CLI output\r\n * Terminals don't support subscript \u2014 Gemini renders these as raw HTML.\r\n * Converts <sub>text</sub> to italic *(text)* for readable terminal output.\r\n */\r\nfunction stripSubTags(content) {\r\n  return content.replace(/<sub>(.*?)<\\/sub>/g, '*($1)*');\r\n}\r\n\r\n/**\r\n * Convert Claude Code agent frontmatter to Gemini CLI format\r\n * Gemini agents use .md files with YAML frontmatter, same as Claude,\r\n * but with different field names and formats:\r\n * - tools: must be a YAML array (not comma-separated string)\r\n * - tool names: must use Gemini built-in names (read_file, not Read)\r\n * - color: must be removed (causes validation error)\r\n * - skills: must be removed (causes validation error)\r\n * - mcp__* tools: must be excluded (auto-discovered at runtime)\r\n */\r\nfunction convertClaudeToGeminiAgent(content: string): string {\r\n  if (!content.startsWith('---')) return content;\r\n\r\n  const endIndex = content.indexOf('---', 3);\r\n  if (endIndex === -1) return content;\r\n\r\n  const frontmatter = content.substring(3, endIndex).trim();\r\n  const body = content.substring(endIndex + 3);\r\n\r\n  const lines = frontmatter.split('\\n');\r\n  const newLines: string[] = [];\r\n  let inAllowedTools = false;\r\n  let inSkippedArrayField = false;\r\n  const tools: string[] = [];\r\n\r\n  for (const line of lines) {\r\n    const trimmed = line.trim();\r\n\r\n    if (inSkippedArrayField) {\r\n      if (!trimmed || trimmed.startsWith('- ')) {\r\n        continue;\r\n      }\r\n      inSkippedArrayField = false;\r\n    }\r\n\r\n    // Convert allowed-tools YAML array to tools list\r\n    if (trimmed.startsWith('allowed-tools:')) {\r\n      inAllowedTools = true;\r\n      continue;\r\n    }\r\n\r\n    // Handle inline tools: field (comma-separated string)\r\n    if (trimmed.startsWith('tools:')) {\r\n      const toolsValue = trimmed.substring(6).trim();\r\n      if (toolsValue) {\r\n        const parsed = toolsValue.split(',').map(t => t.trim()).filter(t => t);\r\n        for (const t of parsed) {\r\n          const mapped = convertGeminiToolName(t);\r\n          if (mapped) tools.push(mapped);\r\n        }\r\n      } else {\r\n        // tools: with no value means YAML array follows\r\n        inAllowedTools = true;\r\n      }\r\n      continue;\r\n    }\r\n\r\n    // Strip color field (not supported by Gemini CLI, causes validation error)\r\n    if (trimmed.startsWith('color:')) continue;\r\n\r\n    // Strip skills field (not supported by Gemini CLI, causes validation error)\r\n    if (trimmed.startsWith('skills:')) {\r\n      inSkippedArrayField = true;\r\n      continue;\r\n    }\r\n\r\n    // Collect allowed-tools/tools array items\r\n    if (inAllowedTools) {\r\n      if (trimmed.startsWith('- ')) {\r\n        const mapped = convertGeminiToolName(trimmed.substring(2).trim());\r\n        if (mapped) tools.push(mapped);\r\n        continue;\r\n      } else if (trimmed && !trimmed.startsWith('-')) {\r\n        inAllowedTools = false;\r\n      }\r\n    }\r\n\r\n    if (!inAllowedTools) {\r\n      newLines.push(line);\r\n    }\r\n  }\r\n\r\n  // Add tools as YAML array (Gemini requires array format)\r\n  if (tools.length > 0) {\r\n    newLines.push('tools:');\r\n    for (const tool of tools) {\r\n      newLines.push(`  - ${tool}`);\r\n    }\r\n  }\r\n\r\n  const newFrontmatter = newLines.join('\\n').trim();\r\n\r\n  // Escape ${VAR} patterns in agent body for Gemini CLI compatibility.\r\n  // Gemini's templateString() treats all ${word} patterns as template variables\r\n  // and throws \"Template validation failed: Missing required input parameters\"\r\n  // when they can't be resolved. EZ_Agents agents use ${PHASE}, ${PLAN}, etc. as\r\n  // shell variables in bash code blocks \u2014 convert to $VAR (no braces) which\r\n  // is equivalent bash and invisible to Gemini's /\\$\\{(\\w+)\\}/g regex.\r\n  const escapedBody = body.replace(/\\$\\{(\\w+)\\}/g, '$$$1');\r\n\r\n  return `---\\n${newFrontmatter}\\n---${stripSubTags(escapedBody)}`;\r\n}\r\n\r\nfunction convertClaudeToOpencodeFrontmatter(content: string): string {\r\n  // Replace tool name references in content (applies to all files)\r\n  let convertedContent = content;\r\n  convertedContent = convertedContent.replace(/\\bAskUserQuestion\\b/g, 'question');\r\n  convertedContent = convertedContent.replace(/\\bSlashCommand\\b/g, 'skill');\r\n  convertedContent = convertedContent.replace(/\\bTodoWrite\\b/g, 'todowrite');\r\n  // Replace /ez:command with /ez-command for opencode (flat command structure)\r\n  convertedContent = convertedContent.replace(/\\/ez:/g, '/ez-');\r\n  // Replace ~/.claude and $HOME/.claude with OpenCode's config location\r\n  convertedContent = convertedContent.replace(/~\\/\\.claude\\b/g, '~/.config/opencode');\r\n  convertedContent = convertedContent.replace(/\\$HOME\\/\\.claude\\b/g, '$HOME/.config/opencode');\r\n  // Replace general-purpose subagent type with OpenCode's equivalent \"general\"\r\n  convertedContent = convertedContent.replace(/subagent_type=\"general-purpose\"/g, 'subagent_type=\"general\"');\r\n\r\n  // Check if content has frontmatter\r\n  if (!convertedContent.startsWith('---')) {\r\n    return convertedContent;\r\n  }\r\n\r\n  // Find the end of frontmatter\r\n  const endIndex = convertedContent.indexOf('---', 3);\r\n  if (endIndex === -1) {\r\n    return convertedContent;\r\n  }\r\n\r\n  const frontmatter = convertedContent.substring(3, endIndex).trim();\r\n  const body = convertedContent.substring(endIndex + 3);\r\n\r\n  // Parse frontmatter line by line (simple YAML parsing)\r\n  const lines = frontmatter.split('\\n');\r\n  const newLines: string[] = [];\r\n  let inAllowedTools = false;\r\n  const allowedTools: string[] = [];\r\n\r\n  for (const line of lines) {\r\n    const trimmed = line.trim();\r\n\r\n    // Detect start of allowed-tools array\r\n    if (trimmed.startsWith('allowed-tools:')) {\r\n      inAllowedTools = true;\r\n      continue;\r\n    }\r\n\r\n    // Detect inline tools: field (comma-separated string)\r\n    if (trimmed.startsWith('tools:')) {\r\n      const toolsValue = trimmed.substring(6).trim();\r\n      if (toolsValue) {\r\n        // Parse comma-separated tools\r\n        const tools = toolsValue.split(',').map(t => t.trim()).filter(t => t);\r\n        allowedTools.push(...tools);\r\n      }\r\n      continue;\r\n    }\r\n\r\n    // Remove name: field - opencode uses filename for command name\r\n    if (trimmed.startsWith('name:')) {\r\n      continue;\r\n    }\r\n\r\n    // Convert color names to hex for opencode\r\n    if (trimmed.startsWith('color:')) {\r\n      const colorValue = trimmed.substring(6).trim().toLowerCase();\r\n      const hexColor = colorNameToHex[colorValue];\r\n      if (hexColor) {\r\n        newLines.push(`color: \"${hexColor}\"`);\r\n      } else if (colorValue.startsWith('#')) {\r\n        // Validate hex color format (#RGB or #RRGGBB)\r\n        if (/^#[0-9a-f]{3}$|^#[0-9a-f]{6}$/i.test(colorValue)) {\r\n          // Already hex and valid, keep as is\r\n          newLines.push(line);\r\n        }\r\n        // Skip invalid hex colors\r\n      }\r\n      // Skip unknown color names\r\n      continue;\r\n    }\r\n\r\n    // Collect allowed-tools items\r\n    if (inAllowedTools) {\r\n      if (trimmed.startsWith('- ')) {\r\n        allowedTools.push(trimmed.substring(2).trim());\r\n        continue;\r\n      } else if (trimmed && !trimmed.startsWith('-')) {\r\n        // End of array, new field started\r\n        inAllowedTools = false;\r\n      }\r\n    }\r\n\r\n    // Keep other fields (including name: which opencode ignores)\r\n    if (!inAllowedTools) {\r\n      newLines.push(line);\r\n    }\r\n  }\r\n\r\n  // Add tools object if we had allowed-tools or tools\r\n  if (allowedTools.length > 0) {\r\n    newLines.push('tools:');\r\n    for (const tool of allowedTools) {\r\n      newLines.push(`  ${convertToolName(tool)}: true`);\r\n    }\r\n  }\r\n\r\n  // Rebuild frontmatter (body already has tool names converted)\r\n  const newFrontmatter = newLines.join('\\n').trim();\r\n  return `---\\n${newFrontmatter}\\n---${body}`;\r\n}\r\n\r\n/**\r\n * Convert Claude Code markdown command to Gemini TOML format\r\n * @param {string} content - Markdown file content with YAML frontmatter\r\n * @returns {string} - TOML content\r\n */\r\nfunction convertClaudeToGeminiToml(content) {\r\n  // Check if content has frontmatter\r\n  if (!content.startsWith('---')) {\r\n    return `prompt = ${JSON.stringify(content)}\\n`;\r\n  }\r\n\r\n  const endIndex = content.indexOf('---', 3);\r\n  if (endIndex === -1) {\r\n    return `prompt = ${JSON.stringify(content)}\\n`;\r\n  }\r\n\r\n  const frontmatter = content.substring(3, endIndex).trim();\r\n  const body = content.substring(endIndex + 3).trim();\r\n  \r\n  // Extract description from frontmatter\r\n  let description = '';\r\n  const lines = frontmatter.split('\\n');\r\n  for (const line of lines) {\r\n    const trimmed = line.trim();\r\n    if (trimmed.startsWith('description:')) {\r\n      description = trimmed.substring(12).trim();\r\n      break;\r\n    }\r\n  }\r\n\r\n  // Construct TOML\r\n  let toml = '';\r\n  if (description) {\r\n    toml += `description = ${JSON.stringify(description)}\\n`;\r\n  }\r\n  \r\n  toml += `prompt = ${JSON.stringify(body)}\\n`;\r\n  \r\n  return toml;\r\n}\r\n\r\n/**\r\n * Copy commands to a flat structure for OpenCode\r\n * OpenCode expects: command/ez-help.md (invoked as /ez-help)\r\n * Source structure: commands/ez/help.md\r\n *\r\n * @param {string} srcDir - Source directory (e.g., commands/ez/)\r\n * @param {string} destDir - Destination directory (e.g., command/)\r\n * @param {string} prefix - Prefix for filenames (e.g., 'ez')\r\n * @param {string} pathPrefix - Path prefix for file references\r\n * @param {string} runtime - Target runtime ('claude' or 'opencode')\r\n */\r\nfunction copyFlattenedCommands(srcDir, destDir, prefix, pathPrefix, runtime) {\r\n  if (!fs.existsSync(srcDir)) {\r\n    return;\r\n  }\r\n\r\n  // Remove old ez-*.md files before copying new ones\r\n  if (fs.existsSync(destDir)) {\r\n    for (const file of fs.readdirSync(destDir)) {\r\n      if (file.startsWith(`${prefix}-`) && file.endsWith('.md')) {\r\n        fs.unlinkSync(path.join(destDir, file));\r\n      }\r\n    }\r\n  } else {\r\n    fs.mkdirSync(destDir, { recursive: true });\r\n  }\r\n\r\n  const entries = fs.readdirSync(srcDir, { withFileTypes: true });\r\n\r\n  for (const entry of entries) {\r\n    const srcPath = path.join(srcDir, entry.name);\r\n\r\n    if (entry.isDirectory()) {\r\n      // Recurse into subdirectories, adding to prefix\r\n      // e.g., commands/ez/debug/start.md -> command/ez-debug-start.md\r\n      copyFlattenedCommands(srcPath, destDir, `${prefix}-${entry.name}`, pathPrefix, runtime);\r\n    } else if (entry.name.endsWith('.md')) {\r\n      // Flatten: help.md -> ez-help.md\r\n      const baseName = entry.name.replace('.md', '');\r\n      const destName = `${prefix}-${baseName}.md`;\r\n      const destPath = path.join(destDir, destName);\r\n\r\n      let content = fs.readFileSync(srcPath, 'utf8');\r\n      const globalClaudeRegex = /~\\/\\.claude\\//g;\r\n      const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\r\n      const localClaudeRegex = /\\.\\/\\.claude\\//g;\r\n      const opencodeDirRegex = /~\\/\\.opencode\\//g;\r\n      content = content.replace(globalClaudeRegex, pathPrefix);\r\n      content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\r\n      content = content.replace(localClaudeRegex, `./${getDirName(runtime)}/`);\r\n      content = content.replace(opencodeDirRegex, pathPrefix);\r\n      content = processAttribution(content, getCommitAttribution(runtime));\r\n      content = convertClaudeToOpencodeFrontmatter(content);\r\n\r\n      fs.writeFileSync(destPath, content);\r\n    }\r\n  }\r\n}\r\n\r\n/**\r\n * Copy EZ Agents skills to global skills directory\r\n * Skills are installed to $HOME/.skills/ez-agents/ for universal access\r\n * @param {string} srcDir - Source directory (ez-agents root)\r\n * @param {boolean} isGlobal - Whether this is a global install\r\n * @param {string} runtime - Runtime identifier\r\n */\r\nfunction copySkills(srcDir: string, isGlobal: boolean, runtime: string): void {\r\n  // Only install skills for global installs\r\n  if (!isGlobal) {\r\n    console.log(`  ${yellow}\u26A0${reset} Skipping skills (local install)`);\r\n    return;\r\n  }\r\n\r\n  const homeDir = os.homedir().replace(/\\\\/g, '/');\r\n  const skillsDestDir = path.posix.join(homeDir, '.skills', 'ez-agents');\r\n  const srcSkillsDir = path.join(srcDir, 'skills');\r\n\r\n  console.log(`  Installing skills to ${cyan}${skillsDestDir}${reset}...`);\r\n\r\n  // Create skills directory\r\n  fs.mkdirSync(skillsDestDir, { recursive: true });\r\n\r\n  // Copy all skills from source\r\n  if (fs.existsSync(srcSkillsDir)) {\r\n    fs.cpSync(srcSkillsDir, skillsDestDir, {\r\n      recursive: true,\r\n      force: true\r\n    });\r\n\r\n    // Count installed skills\r\n    let skillCount = 0;\r\n    const categories = ['stack', 'architecture', 'domain', 'operational', 'governance', 'testing', 'observability', 'devops', 'security', 'ai', 'data', 'product'];\r\n\r\n    for (const category of categories) {\r\n      const categoryPath = path.join(skillsDestDir, category);\r\n      if (fs.existsSync(categoryPath)) {\r\n        const skills = fs.readdirSync(categoryPath, { withFileTypes: true })\r\n          .filter(d => d.isDirectory()).length;\r\n        skillCount += skills;\r\n      }\r\n    }\r\n\r\n    console.log(`  ${green}\u2713${reset} Installed ${skillCount} skills to ${skillsDestDir}`);\r\n  } else {\r\n    console.log(`  ${yellow}\u26A0${reset} Skills directory not found: ${srcSkillsDir}`);\r\n  }\r\n}\r\n\r\nfunction listCodexSkillNames(skillsDir, prefix = 'ez-') {\r\n  if (!fs.existsSync(skillsDir)) return [];\r\n  const entries = fs.readdirSync(skillsDir, { withFileTypes: true });\r\n  return entries\r\n    .filter(entry => entry.isDirectory() && entry.name.startsWith(prefix))\r\n    .filter(entry => fs.existsSync(path.join(skillsDir, entry.name, 'SKILL.md')))\r\n    .map(entry => entry.name)\r\n    .sort();\r\n}\r\n\r\nfunction copyCommandsAsCodexSkills(srcDir, skillsDir, prefix, pathPrefix, runtime) {\r\n  if (!fs.existsSync(srcDir)) {\r\n    return;\r\n  }\r\n\r\n  fs.mkdirSync(skillsDir, { recursive: true });\r\n\r\n  // Remove previous EZ Codex skills to avoid stale command skills.\r\n  const existing = fs.readdirSync(skillsDir, { withFileTypes: true });\r\n  for (const entry of existing) {\r\n    if (entry.isDirectory() && entry.name.startsWith(`${prefix}-`)) {\r\n      fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\r\n    }\r\n  }\r\n\r\n  function recurse(currentSrcDir, currentPrefix) {\r\n    const entries = fs.readdirSync(currentSrcDir, { withFileTypes: true });\r\n\r\n    for (const entry of entries) {\r\n      const srcPath = path.join(currentSrcDir, entry.name);\r\n      if (entry.isDirectory()) {\r\n        recurse(srcPath, `${currentPrefix}-${entry.name}`);\r\n        continue;\r\n      }\r\n\r\n      if (!entry.name.endsWith('.md')) {\r\n        continue;\r\n      }\r\n\r\n      const baseName = entry.name.replace('.md', '');\r\n      const skillName = `${currentPrefix}-${baseName}`;\r\n      const skillDir = path.join(skillsDir, skillName);\r\n      fs.mkdirSync(skillDir, { recursive: true });\r\n\r\n      let content = fs.readFileSync(srcPath, 'utf8');\r\n      const globalClaudeRegex = /~\\/\\.claude\\//g;\r\n      const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\r\n      const localClaudeRegex = /\\.\\/\\.claude\\//g;\r\n      const codexDirRegex = /~\\/\\.codex\\//g;\r\n      content = content.replace(globalClaudeRegex, pathPrefix);\r\n      content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\r\n      content = content.replace(localClaudeRegex, `./${getDirName(runtime)}/`);\r\n      content = content.replace(codexDirRegex, pathPrefix);\r\n      content = processAttribution(content, getCommitAttribution(runtime));\r\n      content = convertClaudeCommandToCodexSkill(content, skillName);\r\n\r\n      fs.writeFileSync(path.join(skillDir, 'SKILL.md'), content);\r\n    }\r\n  }\r\n\r\n  recurse(srcDir, prefix);\r\n}\r\n\r\n/**\r\n * Copy Claude commands as Copilot skills \u2014 one folder per skill with SKILL.md.\r\n * Applies CONV-01 (structure), CONV-02 (allowed-tools), CONV-06 (paths), CONV-07 (command names).\r\n */\r\nfunction copyCommandsAsCopilotSkills(srcDir, skillsDir, prefix, isGlobal = false) {\r\n  if (!fs.existsSync(srcDir)) {\r\n    return;\r\n  }\r\n\r\n  fs.mkdirSync(skillsDir, { recursive: true });\r\n\r\n  // Remove previous EZ Copilot skills\r\n  const existing = fs.readdirSync(skillsDir, { withFileTypes: true });\r\n  for (const entry of existing) {\r\n    if (entry.isDirectory() && entry.name.startsWith(`${prefix}-`)) {\r\n      fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\r\n    }\r\n  }\r\n\r\n  function recurse(currentSrcDir, currentPrefix) {\r\n    const entries = fs.readdirSync(currentSrcDir, { withFileTypes: true });\r\n\r\n    for (const entry of entries) {\r\n      const srcPath = path.join(currentSrcDir, entry.name);\r\n      if (entry.isDirectory()) {\r\n        recurse(srcPath, `${currentPrefix}-${entry.name}`);\r\n        continue;\r\n      }\r\n\r\n      if (!entry.name.endsWith('.md')) {\r\n        continue;\r\n      }\r\n\r\n      const baseName = entry.name.replace('.md', '');\r\n      const skillName = `${currentPrefix}-${baseName}`;\r\n      const skillDir = path.join(skillsDir, skillName);\r\n      fs.mkdirSync(skillDir, { recursive: true });\r\n\r\n      let content = fs.readFileSync(srcPath, 'utf8');\r\n      content = convertClaudeCommandToCopilotSkill(content, skillName, isGlobal);\r\n      content = processAttribution(content, getCommitAttribution('copilot'));\r\n\r\n      fs.writeFileSync(path.join(skillDir, 'SKILL.md'), content);\r\n    }\r\n  }\r\n\r\n  recurse(srcDir, prefix);\r\n}\r\n\r\n/**\r\n * Copy Claude commands as Kimi Code skills \u2014 one folder per skill with SKILL.md.\r\n */\r\nfunction copyCommandsAsKimiSkills(srcDir, skillsDir, prefix, pathPrefix, runtime) {\r\n  if (!fs.existsSync(srcDir)) {\r\n    return;\r\n  }\r\n\r\n  fs.mkdirSync(skillsDir, { recursive: true });\r\n\r\n  // Remove previous EZ Kimi skills\r\n  const existing = fs.readdirSync(skillsDir, { withFileTypes: true });\r\n  for (const entry of existing) {\r\n    if (entry.isDirectory() && entry.name.startsWith(`${prefix}-`)) {\r\n      fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\r\n    }\r\n  }\r\n\r\n  function recurse(currentSrcDir, currentPrefix) {\r\n    const entries = fs.readdirSync(currentSrcDir, { withFileTypes: true });\r\n\r\n    for (const entry of entries) {\r\n      const srcPath = path.join(currentSrcDir, entry.name);\r\n      if (entry.isDirectory()) {\r\n        recurse(srcPath, `${currentPrefix}-${entry.name}`);\r\n        continue;\r\n      }\r\n\r\n      if (!entry.name.endsWith('.md')) {\r\n        continue;\r\n      }\r\n\r\n      const baseName = entry.name.replace('.md', '');\r\n      const skillName = `${currentPrefix}-${baseName}`;\r\n      const skillDir = path.join(skillsDir, skillName);\r\n      fs.mkdirSync(skillDir, { recursive: true });\r\n\r\n      let content = fs.readFileSync(srcPath, 'utf8');\r\n      const globalClaudeRegex = /~\\/\\.claude\\//g;\r\n      const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\r\n      const localClaudeRegex = /\\.\\/\\.claude\\//g;\r\n      const kimiDirRegex = /~\\/\\.kimi\\//g;\r\n      content = content.replace(globalClaudeRegex, pathPrefix);\r\n      content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\r\n      content = content.replace(localClaudeRegex, `./${getDirName(runtime)}/`);\r\n      content = content.replace(kimiDirRegex, pathPrefix);\r\n      content = processAttribution(content, getCommitAttribution(runtime));\r\n      content = convertClaudeToKimiSkill(content, skillName);\r\n      fs.writeFileSync(path.join(skillDir, 'SKILL.md'), content);\r\n    }\r\n  }\r\n\r\n  recurse(srcDir, prefix);\r\n}\r\n\r\n/**\r\n * Copy Claude commands as Qwen Code commands \u2014 flat structure in commands/ez/.\r\n * Qwen Code uses ~/.qwen/commands/ez/*.md (like Claude Code ~/.claude/commands/ez/)\r\n */\r\nfunction copyCommandsAsQwenCommands(srcDir, commandsDir, prefix, pathPrefix, runtime) {\r\n  if (!fs.existsSync(srcDir)) {\r\n    return;\r\n  }\r\n\r\n  fs.mkdirSync(commandsDir, { recursive: true });\r\n\r\n  // Remove previous EZ Qwen commands\r\n  const existing = fs.readdirSync(commandsDir, { withFileTypes: true });\r\n  for (const entry of existing) {\r\n    if (entry.isFile() && entry.name.startsWith(`${prefix}-`) && entry.name.endsWith('.md')) {\r\n      fs.rmSync(path.join(commandsDir, entry.name));\r\n    }\r\n  }\r\n\r\n  function recurse(currentSrcDir, currentPrefix) {\r\n    const entries = fs.readdirSync(currentSrcDir, { withFileTypes: true });\r\n\r\n    for (const entry of entries) {\r\n      const srcPath = path.join(currentSrcDir, entry.name);\r\n      if (entry.isDirectory()) {\r\n        recurse(srcPath, `${currentPrefix}-${entry.name}`);\r\n        continue;\r\n      }\r\n\r\n      if (!entry.name.endsWith('.md')) {\r\n        continue;\r\n      }\r\n\r\n      const baseName = entry.name.replace('.md', '');\r\n      const commandName = `${currentPrefix}-${baseName}`;\r\n      const commandFile = path.join(commandsDir, `${commandName}.md`);\r\n\r\n      let content = fs.readFileSync(srcPath, 'utf8');\r\n      const globalClaudeRegex = /~\\/\\.claude\\//g;\r\n      const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\r\n      const localClaudeRegex = /\\.\\/\\.claude\\//g;\r\n      const qwenDirRegex = /~\\/\\.qwen\\//g;\r\n      content = content.replace(globalClaudeRegex, pathPrefix);\r\n      content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\r\n      content = content.replace(localClaudeRegex, `./${getDirName(runtime)}/`);\r\n      content = content.replace(qwenDirRegex, pathPrefix);\r\n      content = processAttribution(content, getCommitAttribution(runtime));\r\n      \r\n      // Qwen Code uses simple markdown commands (no SKILL.md wrapper)\r\n      // Just copy the content as-is with path replacements\r\n      fs.writeFileSync(commandFile, content);\r\n    }\r\n  }\r\n\r\n  recurse(srcDir, prefix);\r\n}\r\n\r\n/**\r\n * Copy Claude commands as Qwen Code skills \u2014 one folder per skill with SKILL.md.\r\n * Qwen Code uses the same skills/ structure as Codex but with simpler format (no adapters needed).\r\n * @deprecated Use copyCommandsAsQwenCommands instead - Qwen Code uses commands/ez/ not skills/\r\n */\r\nfunction copyCommandsAsQwenSkills(srcDir, skillsDir, prefix, pathPrefix, runtime) {\r\n  if (!fs.existsSync(srcDir)) {\r\n    return;\r\n  }\r\n\r\n  fs.mkdirSync(skillsDir, { recursive: true });\r\n\r\n  // Remove previous EZ Qwen skills\r\n  const existing = fs.readdirSync(skillsDir, { withFileTypes: true });\r\n  for (const entry of existing) {\r\n    if (entry.isDirectory() && entry.name.startsWith(`${prefix}-`)) {\r\n      fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\r\n    }\r\n  }\r\n\r\n  function recurse(currentSrcDir, currentPrefix) {\r\n    const entries = fs.readdirSync(currentSrcDir, { withFileTypes: true });\r\n\r\n    for (const entry of entries) {\r\n      const srcPath = path.join(currentSrcDir, entry.name);\r\n      if (entry.isDirectory()) {\r\n        recurse(srcPath, `${currentPrefix}-${entry.name}`);\r\n        continue;\r\n      }\r\n\r\n      if (!entry.name.endsWith('.md')) {\r\n        continue;\r\n      }\r\n\r\n      const baseName = entry.name.replace('.md', '');\r\n      const skillName = `${currentPrefix}-${baseName}`;\r\n      const skillDir = path.join(skillsDir, skillName);\r\n      fs.mkdirSync(skillDir, { recursive: true });\r\n\r\n      let content = fs.readFileSync(srcPath, 'utf8');\r\n      const globalClaudeRegex = /~\\/\\.claude\\//g;\r\n      const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\r\n      const localClaudeRegex = /\\.\\/\\.claude\\//g;\r\n      const qwenDirRegex = /~\\/\\.qwen\\//g;\r\n      content = content.replace(globalClaudeRegex, pathPrefix);\r\n      content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\r\n      content = content.replace(localClaudeRegex, `./${getDirName(runtime)}/`);\r\n      content = content.replace(qwenDirRegex, pathPrefix);\r\n      content = processAttribution(content, getCommitAttribution(runtime));\r\n      // Qwen Code uses simple markdown skills without frontmatter adapters\r\n      content = convertClaudeToQwenSkill(content, skillName);\r\n      fs.writeFileSync(path.join(skillDir, 'SKILL.md'), content);\r\n    }\r\n  }\r\n\r\n  recurse(srcDir, prefix);\r\n}\r\n\r\n/**\r\n * Recursively copy directory, replacing paths in .md files\r\n * Deletes existing destDir first to remove orphaned files from previous versions\r\n * @param {string} srcDir - Source directory\r\n * @param {string} destDir - Destination directory\r\n * @param {string} pathPrefix - Path prefix for file references\r\n * @param {string} runtime - Target runtime ('claude', 'opencode', 'gemini', 'codex')\r\n */\r\nfunction copyWithPathReplacement(srcDir, destDir, pathPrefix, runtime, isCommand = false, isGlobal = false) {\r\n  const isOpencode = runtime === 'opencode';\r\n  const isCodex = runtime === 'codex';\r\n  const isCopilot = runtime === 'copilot';\r\n  const dirName = getDirName(runtime);\r\n\r\n  // Clean install: remove existing destination to prevent orphaned files\r\n  if (fs.existsSync(destDir)) {\r\n    fs.rmSync(destDir, { recursive: true });\r\n  }\r\n  fs.mkdirSync(destDir, { recursive: true });\r\n\r\n  const entries = fs.readdirSync(srcDir, { withFileTypes: true });\r\n\r\n  for (const entry of entries) {\r\n    const srcPath = path.join(srcDir, entry.name);\r\n    const destPath = path.join(destDir, entry.name);\r\n\r\n    if (entry.isDirectory()) {\r\n      copyWithPathReplacement(srcPath, destPath, pathPrefix, runtime, isCommand, isGlobal);\r\n    } else if (entry.name.endsWith('.md')) {\r\n      // Replace ~/.claude/ and $HOME/.claude/ and ./.claude/ with runtime-appropriate paths\r\n      // Skip generic replacement for Copilot \u2014 convertClaudeToCopilotContent handles all paths\r\n      let content = fs.readFileSync(srcPath, 'utf8');\r\n      if (!isCopilot) {\r\n        const globalClaudeRegex = /~\\/\\.claude\\//g;\r\n        const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\r\n        const localClaudeRegex = /\\.\\/\\.claude\\//g;\r\n        content = content.replace(globalClaudeRegex, pathPrefix);\r\n        content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\r\n        content = content.replace(localClaudeRegex, `./${dirName}/`);\r\n      }\r\n      content = processAttribution(content, getCommitAttribution(runtime));\r\n\r\n      // Convert frontmatter for opencode compatibility\r\n      if (isOpencode) {\r\n        content = convertClaudeToOpencodeFrontmatter(content);\r\n        fs.writeFileSync(destPath, content);\r\n      } else if (runtime === 'gemini') {\r\n        if (isCommand) {\r\n          // Convert to TOML for Gemini (strip <sub> tags \u2014 terminals can't render subscript)\r\n          content = stripSubTags(content);\r\n          const tomlContent = convertClaudeToGeminiToml(content);\r\n          // Replace extension with .toml\r\n          const tomlPath = destPath.replace(/\\.md$/, '.toml');\r\n          fs.writeFileSync(tomlPath, tomlContent);\r\n        } else {\r\n          fs.writeFileSync(destPath, content);\r\n        }\r\n      } else if (isCodex) {\r\n        content = convertClaudeToCodexMarkdown(content);\r\n        fs.writeFileSync(destPath, content);\r\n      } else if (isCopilot) {\r\n        content = convertClaudeToCopilotContent(content, isGlobal);\r\n        content = processAttribution(content, getCommitAttribution(runtime));\r\n        fs.writeFileSync(destPath, content);\r\n      } else {\r\n        fs.writeFileSync(destPath, content);\r\n      }\r\n    } else if (isCopilot && (entry.name.endsWith('.cjs') || entry.name.endsWith('.js'))) {\r\n      // Copilot: also transform .cjs/.js files for CONV-06 and CONV-07\r\n      let content = fs.readFileSync(srcPath, 'utf8');\r\n      content = convertClaudeToCopilotContent(content, isGlobal);\r\n      fs.writeFileSync(destPath, content);\r\n    } else {\r\n      fs.copyFileSync(srcPath, destPath);\r\n    }\r\n  }\r\n}\r\n\r\n/**\r\n * Clean up orphaned files from previous versions\r\n */\r\nfunction cleanupOrphanedFiles(configDir) {\r\n  const orphanedFiles = [\r\n    'hooks/ez-notify.sh',  // Removed in v1.6.x\r\n    'hooks/statusline.js',  // Renamed to ez-statusline.js in v1.9.0\r\n  ];\r\n\r\n  for (const relPath of orphanedFiles) {\r\n    const fullPath = path.join(configDir, relPath);\r\n    if (fs.existsSync(fullPath)) {\r\n      fs.unlinkSync(fullPath);\r\n      console.log(`  ${green}\u2713${reset} Removed orphaned ${relPath}`);\r\n    }\r\n  }\r\n}\r\n\r\n/**\r\n * Clean up orphaned hook registrations from settings.json\r\n */\r\nfunction cleanupOrphanedHooks(settings) {\r\n  const orphanedHookPatterns = [\r\n    'ez-notify.sh',  // Removed in v1.6.x\r\n    'hooks/statusline.js',  // Renamed to ez-statusline.js in v1.9.0\r\n    'ez-intel-index.js',  // Removed in v1.9.2\r\n    'ez-intel-session.js',  // Removed in v1.9.2\r\n    'ez-intel-prune.js',  // Removed in v1.9.2\r\n  ];\r\n\r\n  let cleanedHooks = false;\r\n\r\n  // Check all hook event types (Stop, SessionStart, etc.)\r\n  if (settings.hooks) {\r\n    for (const eventType of Object.keys(settings.hooks)) {\r\n      const hookEntries = settings.hooks[eventType];\r\n      if (Array.isArray(hookEntries)) {\r\n        // Filter out entries that contain orphaned hooks\r\n        const filtered = hookEntries.filter(entry => {\r\n          if (entry.hooks && Array.isArray(entry.hooks)) {\r\n            // Check if any hook in this entry matches orphaned patterns\r\n            const hasOrphaned = entry.hooks.some(h =>\r\n              h.command && orphanedHookPatterns.some(pattern => h.command.includes(pattern))\r\n            );\r\n            if (hasOrphaned) {\r\n              cleanedHooks = true;\r\n              return false;  // Remove this entry\r\n            }\r\n          }\r\n          return true;  // Keep this entry\r\n        });\r\n        settings.hooks[eventType] = filtered;\r\n      }\r\n    }\r\n  }\r\n\r\n  if (cleanedHooks) {\r\n    console.log(`  ${green}\u2713${reset} Removed orphaned hook registrations`);\r\n  }\r\n\r\n  // Fix #330: Update statusLine if it points to old statusline.js path\r\n  // Only match the specific old path pattern (hooks/statusline.js),\r\n  // not third-party statusline scripts that happen to contain 'statusline.js'\r\n  if (settings.statusLine && settings.statusLine.command &&\r\n      /hooks[\\/\\\\]statusline\\.js/.test(settings.statusLine.command)) {\r\n    settings.statusLine.command = settings.statusLine.command.replace(\r\n      /hooks([\\/\\\\])statusline\\.js/,\r\n      'hooks$1ez-statusline.js'\r\n    );\r\n    console.log(`  ${green}\u2713${reset} Updated statusline path (hooks/statusline.js \u2192 hooks/ez-statusline.js)`);\r\n  }\r\n\r\n  return settings;\r\n}\r\n\r\n/**\r\n * Uninstall EZ_Agents from the specified directory for a specific runtime\r\n * Removes only EZ-specific files/directories, preserves user content\r\n * @param {boolean} isGlobal - Whether to uninstall from global or local\r\n * @param {string} runtime - Target runtime ('claude', 'opencode', 'gemini', 'codex', 'copilot')\r\n */\r\nfunction uninstall(isGlobal, runtime = 'claude') {\r\n  const isOpencode = runtime === 'opencode';\r\n  const isCodex = runtime === 'codex';\r\n  const isCopilot = runtime === 'copilot';\r\n  const isQwen = runtime === 'qwen';\r\n  const isKimi = runtime === 'kimi';\r\n  const dirName = getDirName(runtime);\r\n\r\n  // Get the target directory based on runtime and install type\r\n  const targetDir = isGlobal\r\n    ? getGlobalDir(runtime, explicitConfigDir)\r\n    : path.join(process.cwd(), dirName);\r\n\r\n  const locationLabel = isGlobal\r\n    ? targetDir.replace(os.homedir(), '~')\r\n    : targetDir.replace(process.cwd(), '.');\r\n\r\n  let runtimeLabel = 'Claude Code';\r\n  if (runtime === 'opencode') runtimeLabel = 'OpenCode';\r\n  if (runtime === 'gemini') runtimeLabel = 'Gemini';\r\n  if (runtime === 'codex') runtimeLabel = 'Codex';\r\n  if (runtime === 'copilot') runtimeLabel = 'Copilot';\r\n  if (runtime === 'qwen') runtimeLabel = 'Qwen Code';\r\n\r\n  console.log(`  Uninstalling EZ_Agents from ${cyan}${runtimeLabel}${reset} at ${cyan}${locationLabel}${reset}\\n`);\r\n\r\n  // Check if target directory exists\r\n  if (!fs.existsSync(targetDir)) {\r\n    console.log(`  ${yellow}\u26A0${reset} Directory does not exist: ${locationLabel}`);\r\n    console.log(`  Nothing to uninstall.\\n`);\r\n    return;\r\n  }\r\n\r\n  let removedCount = 0;\r\n\r\n  // 1. Remove EZ commands/skills\r\n  if (isOpencode) {\r\n    // OpenCode: remove command/ez-*.md files\r\n    const commandDir = path.join(targetDir, 'command');\r\n    if (fs.existsSync(commandDir)) {\r\n      const files = fs.readdirSync(commandDir);\r\n      for (const file of files) {\r\n        if (file.startsWith('ez-') && file.endsWith('.md')) {\r\n          fs.unlinkSync(path.join(commandDir, file));\r\n          removedCount++;\r\n        }\r\n      }\r\n      console.log(`  ${green}\u2713${reset} Removed EZ_Agents commands from command/`);\r\n    }\r\n  } else if (isCodex || isQwen || isKimi) {\r\n    // Codex/Qwen/Kimi: remove skills/ez-*/SKILL.md skill directories\r\n    const skillsDir = path.join(targetDir, 'skills');\r\n    if (fs.existsSync(skillsDir)) {\r\n      let skillCount = 0;\r\n      const entries = fs.readdirSync(skillsDir, { withFileTypes: true });\r\n      for (const entry of entries) {\r\n        if (entry.isDirectory() && entry.name.startsWith('ez-')) {\r\n          fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\r\n          skillCount++;\r\n        }\r\n      }\r\n      if (skillCount > 0) {\r\n        removedCount++;\r\n        let runtimeLabel = 'Codex';\r\n        if (isQwen) runtimeLabel = 'Qwen Code';\r\n        if (isKimi) runtimeLabel = 'Kimi Code';\r\n        console.log(`  ${green}\u2713${reset} Removed ${skillCount} ${runtimeLabel} skills`);\r\n      }\r\n    }\r\n\r\n    if (isCodex) {\r\n      // Codex: remove EZ agent .toml config files\r\n      const codexAgentsDir = path.join(targetDir, 'agents');\r\n      if (fs.existsSync(codexAgentsDir)) {\r\n        const tomlFiles = fs.readdirSync(codexAgentsDir);\r\n        let tomlCount = 0;\r\n        for (const file of tomlFiles) {\r\n          if (file.startsWith('ez-') && file.endsWith('.toml')) {\r\n            fs.unlinkSync(path.join(codexAgentsDir, file));\r\n            tomlCount++;\r\n          }\r\n        }\r\n        if (tomlCount > 0) {\r\n          removedCount++;\r\n          console.log(`  ${green}\u2713${reset} Removed ${tomlCount} agent .toml configs`);\r\n        }\r\n      }\r\n\r\n      // Codex: clean EZ sections from config.toml\r\n      const configPath = path.join(targetDir, 'config.toml');\r\n      if (fs.existsSync(configPath)) {\r\n        const content = fs.readFileSync(configPath, 'utf8');\r\n        const cleaned = stripEzAgentsFromCodexConfig(content);\r\n        if (cleaned === null) {\r\n          // File is empty after stripping \u2014 delete it\r\n          fs.unlinkSync(configPath);\r\n          removedCount++;\r\n          console.log(`  ${green}\u2713${reset} Removed config.toml (was EZ-only)`);\r\n        } else if (cleaned !== content) {\r\n          fs.writeFileSync(configPath, cleaned);\r\n          removedCount++;\r\n          console.log(`  ${green}\u2713${reset} Cleaned EZ_Agents sections from config.toml`);\r\n        }\r\n      }\r\n    }\r\n  } else if (isCopilot) {\r\n    // Copilot: remove skills/ez-*/ directories (same layout as Codex skills)\r\n    const skillsDir = path.join(targetDir, 'skills');\r\n    if (fs.existsSync(skillsDir)) {\r\n      let skillCount = 0;\r\n      const entries = fs.readdirSync(skillsDir, { withFileTypes: true });\r\n      for (const entry of entries) {\r\n        if (entry.isDirectory() && entry.name.startsWith('ez-')) {\r\n          fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\r\n          skillCount++;\r\n        }\r\n      }\r\n      if (skillCount > 0) {\r\n        removedCount++;\r\n        console.log(`  ${green}\u2713${reset} Removed ${skillCount} Copilot skills`);\r\n      }\r\n    }\r\n\r\n    // Copilot: clean EZ section from copilot-instructions.md\r\n    const instructionsPath = path.join(targetDir, 'copilot-instructions.md');\r\n    if (fs.existsSync(instructionsPath)) {\r\n      const content = fs.readFileSync(instructionsPath, 'utf8');\r\n      const cleaned = stripEzAgentsFromCopilotInstructions(content);\r\n      if (cleaned === null) {\r\n        fs.unlinkSync(instructionsPath);\r\n        removedCount++;\r\n        console.log(`  ${green}\u2713${reset} Removed copilot-instructions.md (was EZ-only)`);\r\n      } else if (cleaned !== content) {\r\n        fs.writeFileSync(instructionsPath, cleaned);\r\n        removedCount++;\r\n        console.log(`  ${green}\u2713${reset} Cleaned EZ_Agents section from copilot-instructions.md`);\r\n      }\r\n    }\r\n  } else {\r\n    const ezCommandsDir = path.join(targetDir, 'commands', 'ez');\r\n    if (fs.existsSync(ezCommandsDir)) {\r\n      fs.rmSync(ezCommandsDir, { recursive: true });\r\n      removedCount++;\r\n      console.log(`  ${green}\u2713${reset} Removed commands/ez/`);\r\n    }\r\n  }\r\n\r\n  // 2. Remove ez-agents directory\r\n  const ezDir = path.join(targetDir, 'ez-agents');\r\n  if (fs.existsSync(ezDir)) {\r\n    fs.rmSync(ezDir, { recursive: true });\r\n    removedCount++;\r\n    console.log(`  ${green}\u2713${reset} Removed ez-agents/`);\r\n  }\r\n\r\n  // 3. Remove EZ_Agents agents (ez-*.md files only)\r\n  const agentsDir = path.join(targetDir, 'agents');\r\n  if (fs.existsSync(agentsDir)) {\r\n    const files = fs.readdirSync(agentsDir);\r\n    let agentCount = 0;\r\n    for (const file of files) {\r\n      if (file.startsWith('ez-') && file.endsWith('.md')) {\r\n        fs.unlinkSync(path.join(agentsDir, file));\r\n        agentCount++;\r\n      }\r\n    }\r\n    if (agentCount > 0) {\r\n      removedCount++;\r\n      console.log(`  ${green}\u2713${reset} Removed ${agentCount} EZ_Agents agents`);\r\n    }\r\n  }\r\n\r\n  // 4. Remove EZ hooks\r\n  const hooksDir = path.join(targetDir, 'hooks');\r\n  if (fs.existsSync(hooksDir)) {\r\n    const ezHooks = ['ez-statusline.js', 'ez-check-update.js', 'ez-check-update.sh', 'ez-context-monitor.js'];\r\n    let hookCount = 0;\r\n    for (const hook of ezHooks) {\r\n      const hookPath = path.join(hooksDir, hook);\r\n      if (fs.existsSync(hookPath)) {\r\n        fs.unlinkSync(hookPath);\r\n        hookCount++;\r\n      }\r\n    }\r\n    if (hookCount > 0) {\r\n      removedCount++;\r\n      console.log(`  ${green}\u2713${reset} Removed ${hookCount} EZ hooks`);\r\n    }\r\n  }\r\n\r\n  // 5. Remove EZ package.json (CommonJS mode marker)\r\n  const pkgJsonPath = path.join(targetDir, 'package.json');\r\n  if (fs.existsSync(pkgJsonPath)) {\r\n    try {\r\n      const content = fs.readFileSync(pkgJsonPath, 'utf8').trim();\r\n      // Only remove if it's our minimal CommonJS marker\r\n      if (content === '{\"type\":\"commonjs\"}') {\r\n        fs.unlinkSync(pkgJsonPath);\r\n        removedCount++;\r\n        console.log(`  ${green}\u2713${reset} Removed EZ package.json`);\r\n      }\r\n    } catch (e) {\r\n      // Ignore read errors\r\n    }\r\n  }\r\n\r\n  // 6. Clean up settings.json (remove EZ hooks and statusline)\r\n  const settingsPath = path.join(targetDir, 'settings.json');\r\n  if (fs.existsSync(settingsPath)) {\r\n    let settings = readSettings(settingsPath);\r\n    let settingsModified = false;\r\n\r\n    // Remove EZ statusline if it references our hook\r\n    if (settings.statusLine && settings.statusLine.command &&\r\n        settings.statusLine.command.includes('ez-statusline')) {\r\n      delete settings.statusLine;\r\n      settingsModified = true;\r\n      console.log(`  ${green}\u2713${reset} Removed EZ statusline from settings`);\r\n    }\r\n\r\n    // Remove EZ hooks from SessionStart\r\n    if (settings.hooks && settings.hooks.SessionStart) {\r\n      const before = settings.hooks.SessionStart.length;\r\n      settings.hooks.SessionStart = settings.hooks.SessionStart.filter(entry => {\r\n        if (entry.hooks && Array.isArray(entry.hooks)) {\r\n          // Filter out EZ hooks\r\n          const hasEzHook = entry.hooks.some(h =>\r\n            h.command && (h.command.includes('ez-check-update') || h.command.includes('ez-statusline'))\r\n          );\r\n          return !hasEzHook;\r\n        }\r\n        return true;\r\n      });\r\n      if (settings.hooks.SessionStart.length < before) {\r\n        settingsModified = true;\r\n        console.log(`  ${green}\u2713${reset} Removed EZ hooks from settings`);\r\n      }\r\n      // Clean up empty array\r\n      if (settings.hooks.SessionStart.length === 0) {\r\n        delete settings.hooks.SessionStart;\r\n      }\r\n    }\r\n\r\n    // Remove EZ hooks from PostToolUse and AfterTool (Gemini uses AfterTool)\r\n    for (const eventName of ['PostToolUse', 'AfterTool']) {\r\n      if (settings.hooks && settings.hooks[eventName]) {\r\n        const before = settings.hooks[eventName].length;\r\n        settings.hooks[eventName] = settings.hooks[eventName].filter(entry => {\r\n          if (entry.hooks && Array.isArray(entry.hooks)) {\r\n            const hasEzHook = entry.hooks.some(h =>\r\n              h.command && h.command.includes('ez-context-monitor')\r\n            );\r\n            return !hasEzHook;\r\n          }\r\n          return true;\r\n        });\r\n        if (settings.hooks[eventName].length < before) {\r\n          settingsModified = true;\r\n          console.log(`  ${green}\u2713${reset} Removed context monitor hook from settings`);\r\n        }\r\n        if (settings.hooks[eventName].length === 0) {\r\n          delete settings.hooks[eventName];\r\n        }\r\n      }\r\n    }\r\n\r\n    // Clean up empty hooks object\r\n    if (settings.hooks && Object.keys(settings.hooks).length === 0) {\r\n      delete settings.hooks;\r\n    }\r\n\r\n    if (settingsModified) {\r\n      writeSettings(settingsPath, settings);\r\n      removedCount++;\r\n    }\r\n  }\r\n\r\n  // 6. For OpenCode, clean up permissions from opencode.json\r\n  if (isOpencode) {\r\n    // For local uninstalls, clean up ./.opencode/opencode.json\r\n    // For global uninstalls, clean up ~/.config/opencode/opencode.json\r\n    const opencodeConfigDir = isGlobal\r\n      ? getOpencodeGlobalDir()\r\n      : path.join(process.cwd(), '.opencode');\r\n    const configPath = path.join(opencodeConfigDir, 'opencode.json');\r\n    if (fs.existsSync(configPath)) {\r\n      try {\r\n        const config = JSON.parse(fs.readFileSync(configPath, 'utf8'));\r\n        let modified = false;\r\n\r\n        // Remove EZ permission entries\r\n        if (config.permission) {\r\n          for (const permType of ['read', 'external_directory']) {\r\n            if (config.permission[permType]) {\r\n              const keys = Object.keys(config.permission[permType]);\r\n              for (const key of keys) {\r\n                if (key.includes('ez-agents')) {\r\n                  delete config.permission[permType][key];\r\n                  modified = true;\r\n                }\r\n              }\r\n              // Clean up empty objects\r\n              if (Object.keys(config.permission[permType]).length === 0) {\r\n                delete config.permission[permType];\r\n              }\r\n            }\r\n          }\r\n          if (Object.keys(config.permission).length === 0) {\r\n            delete config.permission;\r\n          }\r\n        }\r\n\r\n        if (modified) {\r\n          fs.writeFileSync(configPath, JSON.stringify(config, null, 2) + '\\n');\r\n          removedCount++;\r\n          console.log(`  ${green}\u2713${reset} Removed EZ permissions from opencode.json`);\r\n        }\r\n      } catch (e) {\r\n        // Ignore JSON parse errors\r\n      }\r\n    }\r\n  }\r\n\r\n  if (removedCount === 0) {\r\n    console.log(`  ${yellow}\u26A0${reset} No EZ_Agents files found to remove.`);\r\n  }\r\n\r\n  console.log(`\r\n  ${green}Done!${reset} EZ_Agents has been uninstalled from ${runtimeLabel}.\r\n  Your other files and settings have been preserved.\r\n`);\r\n}\r\n\r\n/**\r\n * Parse JSONC (JSON with Comments) by stripping comments and trailing commas.\r\n * OpenCode supports JSONC format via jsonc-parser, so users may have comments.\r\n * This is a lightweight inline parser to avoid adding dependencies.\r\n */\r\nfunction parseJsonc(content) {\r\n  // Strip BOM if present\r\n  if (content.charCodeAt(0) === 0xFEFF) {\r\n    content = content.slice(1);\r\n  }\r\n\r\n  // Remove single-line and block comments while preserving strings\r\n  let result = '';\r\n  let inString = false;\r\n  let i = 0;\r\n  while (i < content.length) {\r\n    const char = content[i];\r\n    const next = content[i + 1];\r\n\r\n    if (inString) {\r\n      result += char;\r\n      // Handle escape sequences\r\n      if (char === '\\\\' && i + 1 < content.length) {\r\n        result += next;\r\n        i += 2;\r\n        continue;\r\n      }\r\n      if (char === '\"') {\r\n        inString = false;\r\n      }\r\n      i++;\r\n    } else {\r\n      if (char === '\"') {\r\n        inString = true;\r\n        result += char;\r\n        i++;\r\n      } else if (char === '/' && next === '/') {\r\n        // Skip single-line comment until end of line\r\n        while (i < content.length && content[i] !== '\\n') {\r\n          i++;\r\n        }\r\n      } else if (char === '/' && next === '*') {\r\n        // Skip block comment\r\n        i += 2;\r\n        while (i < content.length - 1 && !(content[i] === '*' && content[i + 1] === '/')) {\r\n          i++;\r\n        }\r\n        i += 2; // Skip closing */\r\n      } else {\r\n        result += char;\r\n        i++;\r\n      }\r\n    }\r\n  }\r\n\r\n  // Remove trailing commas before } or ]\r\n  result = result.replace(/,(\\s*[}\\]])/g, '$1');\r\n\r\n  return JSON.parse(result);\r\n}\r\n\r\n/**\r\n * Configure OpenCode permissions to allow reading EZ reference docs\r\n * This prevents permission prompts when EZ accesses the ez-agents directory\r\n * @param {boolean} isGlobal - Whether this is a global or local install\r\n */\r\nfunction configureOpencodePermissions(isGlobal = true) {\r\n  // For local installs, use ./.opencode/opencode.json\r\n  // For global installs, use ~/.config/opencode/opencode.json\r\n  const opencodeConfigDir = isGlobal\r\n    ? getOpencodeGlobalDir()\r\n    : path.join(process.cwd(), '.opencode');\r\n  const configPath = path.join(opencodeConfigDir, 'opencode.json');\r\n\r\n  // Ensure config directory exists\r\n  fs.mkdirSync(opencodeConfigDir, { recursive: true });\r\n\r\n  // Read existing config or create empty object\r\n  let config: Record<string, unknown> = {};\r\n  if (fs.existsSync(configPath)) {\r\n    try {\r\n      const content = fs.readFileSync(configPath, 'utf8');\r\n      config = parseJsonc(content);\r\n    } catch (e) {\r\n      // Cannot parse - DO NOT overwrite user's config\r\n      console.log(`  ${yellow}\u26A0${reset} Could not parse opencode.json - skipping permission config`);\r\n      console.log(`    ${dim}Reason: ${(e as Error).message}${reset}`);\r\n      console.log(`    ${dim}Your config was NOT modified. Fix the syntax manually if needed.${reset}`);\r\n      return;\r\n    }\r\n  }\r\n\r\n  // Ensure permission structure exists\r\n  if (!config.permission) {\r\n    config.permission = {};\r\n  }\r\n\r\n  // Build the EZ path using the actual config directory\r\n  // Use ~ shorthand if it's in the default location, otherwise use full path\r\n  const defaultConfigDir = path.join(os.homedir(), '.config', 'opencode');\r\n  const ezPath = opencodeConfigDir === defaultConfigDir\r\n    ? '~/.config/opencode/ez-agents/*'\r\n    : `${opencodeConfigDir.replace(/\\\\/g, '/')}/ez-agents/*`;\r\n\r\n  let modified = false;\r\n\r\n  // Configure read permission\r\n  const permission = config.permission as Record<string, unknown> || {};\r\n  config.permission = permission;\r\n  if (!permission.read || typeof permission.read !== 'object') {\r\n    permission.read = {};\r\n  }\r\n  const readPerm = permission.read as Record<string, unknown>;\r\n  if (readPerm[ezPath] !== 'allow') {\r\n    readPerm[ezPath] = 'allow';\r\n    modified = true;\r\n  }\r\n\r\n  // Configure external_directory permission (the safety guard for paths outside project)\r\n  if (!permission.external_directory || typeof permission.external_directory !== 'object') {\r\n    permission.external_directory = {};\r\n  }\r\n  const extDirPerm = permission.external_directory as Record<string, unknown>;\r\n  if (extDirPerm[ezPath] !== 'allow') {\r\n    extDirPerm[ezPath] = 'allow';\r\n    modified = true;\r\n  }\r\n\r\n  if (!modified) {\r\n    return; // Already configured\r\n  }\r\n\r\n  // Write config back\r\n  fs.writeFileSync(configPath, JSON.stringify(config, null, 2) + '\\n');\r\n  console.log(`  ${green}\u2713${reset} Configured read permission for EZ_Agents docs`);\r\n}\r\n\r\n/**\r\n * Verify a directory exists and contains files\r\n */\r\nfunction verifyInstalled(dirPath, description) {\r\n  if (!fs.existsSync(dirPath)) {\r\n    console.error(`  ${yellow}\u2717${reset} Failed to install ${description}: directory not created`);\r\n    return false;\r\n  }\r\n  try {\r\n    const entries = fs.readdirSync(dirPath);\r\n    if (entries.length === 0) {\r\n      console.error(`  ${yellow}\u2717${reset} Failed to install ${description}: directory is empty`);\r\n      return false;\r\n    }\r\n  } catch (e) {\r\n    console.error(`  ${yellow}\u2717${reset} Failed to install ${description}: ${(e as Error).message}`);\r\n    return false;\r\n  }\r\n  return true;\r\n}\r\n\r\n/**\r\n * Verify a file exists\r\n */\r\nfunction verifyFileInstalled(filePath, description) {\r\n  if (!fs.existsSync(filePath)) {\r\n    console.error(`  ${yellow}\u2717${reset} Failed to install ${description}: file not created`);\r\n    return false;\r\n  }\r\n  return true;\r\n}\r\n\r\n/**\r\n * Install to the specified directory for a specific runtime\r\n * @param {boolean} isGlobal - Whether to install globally or locally\r\n * @param {string} runtime - Target runtime ('claude', 'opencode', 'gemini', 'codex')\r\n */\r\n\r\n// \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n// Local Patch Persistence\r\n// \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\nconst PATCHES_DIR_NAME = 'ez-local-patches';\r\nconst MANIFEST_NAME = 'ez-file-manifest.json';\r\n\r\n/**\r\n * Compute SHA256 hash of file contents\r\n */\r\nfunction fileHash(filePath: string): string {\r\n  const content = fs.readFileSync(filePath);\r\n  // Convert to Uint8Array to satisfy BinaryLike type requirement\r\n  return crypto.createHash('sha256').update(new Uint8Array(content)).digest('hex');\r\n}\r\n\r\n/**\r\n * Recursively collect all files in dir with their hashes\r\n */\r\nfunction generateManifest(dir: string, baseDir?: string): Record<string, string> {\r\n  if (!baseDir) baseDir = dir;\r\n  const manifest: Record<string, string> = {};\r\n  if (!fs.existsSync(dir)) return manifest;\r\n  const entries = fs.readdirSync(dir, { withFileTypes: true });\r\n  for (const entry of entries) {\r\n    const fullPath = path.join(dir, entry.name);\r\n    const relPath = path.relative(baseDir, fullPath).replace(/\\\\/g, '/');\r\n    if (entry.isDirectory()) {\r\n      Object.assign(manifest, generateManifest(fullPath, baseDir));\r\n    } else {\r\n      manifest[relPath] = fileHash(fullPath);\r\n    }\r\n  }\r\n  return manifest;\r\n}\r\n\r\n/**\r\n * Write file manifest after installation for future modification detection\r\n */\r\nfunction writeManifest(configDir: string, runtime = 'claude') {\r\n  const isOpencode = runtime === 'opencode';\r\n  const isCodex = runtime === 'codex';\r\n  const isCopilot = runtime === 'copilot';\r\n  const ezDir = path.join(configDir, 'ez-agents');\r\n  const commandsDir = path.join(configDir, 'commands', 'ez');\r\n  const opencodeCommandDir = path.join(configDir, 'command');\r\n  const codexSkillsDir = path.join(configDir, 'skills');\r\n  const agentsDir = path.join(configDir, 'agents');\r\n  const manifest = { version: pkg.version, timestamp: new Date().toISOString(), files: {} as Record<string, string> };\r\n\r\n  const ezHashes = generateManifest(ezDir);\r\n  for (const [rel, hash] of Object.entries(ezHashes)) {\r\n    manifest.files['ez-agents/' + rel] = hash;\r\n  }\r\n  if (!isOpencode && !isCodex && !isCopilot && fs.existsSync(commandsDir)) {\r\n    const cmdHashes = generateManifest(commandsDir);\r\n    for (const [rel, hash] of Object.entries(cmdHashes)) {\r\n      manifest.files['commands/ez/' + rel] = hash;\r\n    }\r\n  }\r\n  if (isOpencode && fs.existsSync(opencodeCommandDir)) {\r\n    for (const file of fs.readdirSync(opencodeCommandDir)) {\r\n      if (file.startsWith('ez-') && file.endsWith('.md')) {\r\n        manifest.files['command/' + file] = fileHash(path.join(opencodeCommandDir, file));\r\n      }\r\n    }\r\n  }\r\n  if ((isCodex || isCopilot) && fs.existsSync(codexSkillsDir)) {\r\n    for (const skillName of listCodexSkillNames(codexSkillsDir)) {\r\n      const skillRoot = path.join(codexSkillsDir, skillName);\r\n      const skillHashes = generateManifest(skillRoot);\r\n      for (const [rel, hash] of Object.entries(skillHashes)) {\r\n        manifest.files[`skills/${skillName}/${rel}`] = hash;\r\n      }\r\n    }\r\n  }\r\n  if (fs.existsSync(agentsDir)) {\r\n    for (const file of fs.readdirSync(agentsDir)) {\r\n      if (file.startsWith('ez-') && file.endsWith('.md')) {\r\n        manifest.files['agents/' + file] = fileHash(path.join(agentsDir, file));\r\n      }\r\n    }\r\n  }\r\n\r\n  fs.writeFileSync(path.join(configDir, MANIFEST_NAME), JSON.stringify(manifest, null, 2));\r\n  return manifest;\r\n}\r\n\r\n/**\r\n * Detect user-modified EZ_Agents files by comparing against install manifest.\r\n * Backs up modified files to ez-local-patches/ for reapply after update.\r\n */\r\nfunction saveLocalPatches(configDir) {\r\n  const manifestPath = path.join(configDir, MANIFEST_NAME);\r\n  if (!fs.existsSync(manifestPath)) return [];\r\n\r\n  let manifest;\r\n  try { manifest = JSON.parse(fs.readFileSync(manifestPath, 'utf8')); } catch { return []; }\r\n\r\n  const patchesDir = path.join(configDir, PATCHES_DIR_NAME);\r\n  const modified: string[] = [];\r\n\r\n  for (const [relPath, originalHash] of Object.entries(manifest.files || {})) {\r\n    const fullPath = path.join(configDir, relPath);\r\n    if (!fs.existsSync(fullPath)) continue;\r\n    const currentHash = fileHash(fullPath);\r\n    if (currentHash !== originalHash) {\r\n      const backupPath = path.join(patchesDir, relPath);\r\n      fs.mkdirSync(path.dirname(backupPath), { recursive: true });\r\n      fs.copyFileSync(fullPath, backupPath);\r\n      modified.push(relPath);\r\n    }\r\n  }\r\n\r\n  if (modified.length > 0) {\r\n    const meta = {\r\n      backed_up_at: new Date().toISOString(),\r\n      from_version: manifest.version,\r\n      files: modified\r\n    };\r\n    fs.writeFileSync(path.join(patchesDir, 'backup-meta.json'), JSON.stringify(meta, null, 2));\r\n    console.log('  ' + yellow + 'i' + reset + '  Found ' + modified.length + ' locally modified EZ file(s) \u2014 backed up to ' + PATCHES_DIR_NAME + '/');\r\n    for (const f of modified) {\r\n      console.log('     ' + dim + f + reset);\r\n    }\r\n  }\r\n  return modified;\r\n}\r\n\r\n/**\r\n * After install, report backed-up patches for user to reapply.\r\n */\r\nfunction reportLocalPatches(configDir, runtime = 'claude') {\r\n  const patchesDir = path.join(configDir, PATCHES_DIR_NAME);\r\n  const metaPath = path.join(patchesDir, 'backup-meta.json');\r\n  if (!fs.existsSync(metaPath)) return [];\r\n\r\n  let meta;\r\n  try { meta = JSON.parse(fs.readFileSync(metaPath, 'utf8')); } catch { return []; }\r\n\r\n  if (meta.files && meta.files.length > 0) {\r\n    const reapplyCommand = (runtime === 'opencode' || runtime === 'copilot')\r\n      ? '/ez-reapply-patches'\r\n      : runtime === 'codex'\r\n        ? '$ez-reapply-patches'\r\n        : '/ez:reapply-patches';\r\n    console.log('');\r\n    console.log('  ' + yellow + 'Local patches detected' + reset + ' (from v' + meta.from_version + '):');\r\n    for (const f of meta.files) {\r\n      console.log('     ' + cyan + f + reset);\r\n    }\r\n    console.log('');\r\n    console.log('  Your modifications are saved in ' + cyan + PATCHES_DIR_NAME + '/' + reset);\r\n    console.log('  Run ' + cyan + reapplyCommand + reset + ' to merge them into the new version.');\r\n    console.log('  Or manually compare and merge the files.');\r\n    console.log('');\r\n  }\r\n  return meta.files || [];\r\n}\r\n\r\nfunction install(isGlobal: boolean, runtime = 'claude'): { settingsPath: string | null; settings: Record<string, unknown> | null; statuslineCommand: string | null; runtime: string } {\r\n  const isOpencode = runtime === 'opencode';\r\n  const isGemini = runtime === 'gemini';\r\n  const isCodex = runtime === 'codex';\r\n  const isCopilot = runtime === 'copilot';\r\n  const isQwen = runtime === 'qwen';\r\n  const isKimi = runtime === 'kimi';\r\n  const dirName = getDirName(runtime);\r\n  const src = path.join(__dirname, '..');\r\n\r\n  // Get the target directory based on runtime and install type\r\n  const targetDir = isGlobal\r\n    ? getGlobalDir(runtime, explicitConfigDir)\r\n    : path.join(process.cwd(), dirName);\r\n\r\n  const locationLabel = isGlobal\r\n    ? targetDir.replace(os.homedir(), '~')\r\n    : targetDir.replace(process.cwd(), '.');\r\n\r\n  // Path prefix for file references in markdown content\r\n  // For global installs: use tilde-based path (~/.claude/) to avoid baking\r\n  // absolute paths (containing OS username) into templates\r\n  // For local installs: use relative\r\n  const pathPrefix = isGlobal\r\n    ? `${targetDir.replace(/\\\\/g, '/').replace(os.homedir().replace(/\\\\/g, '/'), '~')}/`\r\n    : `./${dirName}/`;\r\n\r\n  let runtimeLabel = 'Claude Code';\r\n  if (isOpencode) runtimeLabel = 'OpenCode';\r\n  if (isGemini) runtimeLabel = 'Gemini';\r\n  if (isCodex) runtimeLabel = 'Codex';\r\n  if (isCopilot) runtimeLabel = 'Copilot';\r\n  if (isQwen) runtimeLabel = 'Qwen Code';\r\n  if (isKimi) runtimeLabel = 'Kimi Code';\r\n\r\n  console.log(`  Installing for ${cyan}${runtimeLabel}${reset} to ${cyan}${locationLabel}${reset}\\n`);\r\n\r\n  // Track installation failures\r\n  const failures: string[] = [];\r\n\r\n  // Save any locally modified EZ_Agents files before they get wiped\r\n  saveLocalPatches(targetDir);\r\n\r\n  // Clean up orphaned files from previous versions\r\n  cleanupOrphanedFiles(targetDir);\r\n\r\n  // OpenCode uses command/ (flat), Codex/Copilot/Qwen use skills/, Claude/Gemini use commands/ez/\r\n  if (isOpencode) {\r\n    // OpenCode: flat structure in command/ directory\r\n    const commandDir = path.join(targetDir, 'command');\r\n    fs.mkdirSync(commandDir, { recursive: true });\r\n\r\n    // Copy commands/ez/*.md as command/ez-*.md (flatten structure)\r\n    const ezSrc = path.join(src, 'commands', 'ez');\r\n    copyFlattenedCommands(ezSrc, commandDir, 'ez', pathPrefix, runtime);\r\n    if (verifyInstalled(commandDir, 'command/ez-*')) {\r\n      const count = fs.readdirSync(commandDir).filter(f => f.startsWith('ez-')).length;\r\n      console.log(`  ${green}\u2713${reset} Installed ${count} commands to command/`);\r\n    } else {\r\n      failures.push('command/ez-*');\r\n    }\r\n  } else if (isCodex || isKimi) {\r\n    // Codex, Kimi: skills structure in skills/ directory\r\n    const skillsDir = path.join(targetDir, 'skills');\r\n    const ezSrc = path.join(src, 'commands', 'ez');\r\n    if (isKimi) {\r\n      copyCommandsAsKimiSkills(ezSrc, skillsDir, 'ez', pathPrefix, runtime);\r\n    } else {\r\n      copyCommandsAsCodexSkills(ezSrc, skillsDir, 'ez', pathPrefix, runtime);\r\n    }\r\n    const installedSkillNames = listCodexSkillNames(skillsDir);\r\n    if (installedSkillNames.length > 0) {\r\n      console.log(`  ${green}\u2713${reset} Installed ${installedSkillNames.length} skills to skills/`);\r\n    } else {\r\n      failures.push('skills/ez-*');\r\n    }\r\n  } else if (isQwen) {\r\n    // Qwen Code: commands structure in commands/ez/ directory (like Claude Code/Gemini)\r\n    // Qwen Code uses ~/.qwen/commands/ez/*.md, NOT ~/.qwen/skills/\r\n    const commandsDir = path.join(targetDir, 'commands');\r\n    const ezCommandsDir = path.join(commandsDir, 'ez');\r\n    fs.mkdirSync(ezCommandsDir, { recursive: true });\r\n    \r\n    const ezSrc = path.join(src, 'commands', 'ez');\r\n    copyCommandsAsQwenCommands(ezSrc, ezCommandsDir, 'ez', pathPrefix, runtime);\r\n    \r\n    if (fs.existsSync(ezCommandsDir)) {\r\n      const count = fs.readdirSync(ezCommandsDir).filter(f => f.endsWith('.md')).length;\r\n      if (count > 0) {\r\n        console.log(`  ${green}\u2713${reset} Installed ${count} commands to commands/ez/`);\r\n      } else {\r\n        failures.push('commands/ez/*.md');\r\n      }\r\n    } else {\r\n      failures.push('commands/ez/*.md');\r\n    }\r\n  } else if (isCopilot) {\r\n    const skillsDir = path.join(targetDir, 'skills');\r\n    const ezSrc = path.join(src, 'commands', 'ez');\r\n    copyCommandsAsCopilotSkills(ezSrc, skillsDir, 'ez', isGlobal);\r\n    if (fs.existsSync(skillsDir)) {\r\n      const count = fs.readdirSync(skillsDir, { withFileTypes: true })\r\n        .filter(e => e.isDirectory() && e.name.startsWith('ez-')).length;\r\n      if (count > 0) {\r\n        console.log(`  ${green}\u2713${reset} Installed ${count} skills to skills/`);\r\n      } else {\r\n        failures.push('skills/ez-*');\r\n      }\r\n    } else {\r\n      failures.push('skills/ez-*');\r\n    }\r\n  } else {\r\n    // Claude Code & Gemini: nested structure in commands/ directory\r\n    const commandsDir = path.join(targetDir, 'commands');\r\n    fs.mkdirSync(commandsDir, { recursive: true });\r\n\r\n    const ezSrc = path.join(src, 'commands', 'ez');\r\n    const ezDest = path.join(commandsDir, 'ez');\r\n    copyWithPathReplacement(ezSrc, ezDest, pathPrefix, runtime, true, isGlobal);\r\n    if (verifyInstalled(ezDest, 'commands/ez')) {\r\n      console.log(`  ${green}\u2713${reset} Installed commands/ez`);\r\n    } else {\r\n      failures.push('commands/ez');\r\n    }\r\n  }\r\n\r\n  // Copy ez-agents skill with path replacement\r\n  const skillSrc = path.join(src, 'ez-agents');\r\n  const skillDest = path.join(targetDir, 'ez-agents');\r\n  copyWithPathReplacement(skillSrc, skillDest, pathPrefix, runtime, false, isGlobal);\r\n  if (verifyInstalled(skillDest, 'ez-agents')) {\r\n    console.log(`  ${green}\u2713${reset} Installed ez-agents`);\r\n  } else {\r\n    failures.push('ez-agents');\r\n  }\r\n\r\n  // Copy EZ Agents skills to global directory ($HOME/.skills/ez-agents/)\r\n  copySkills(src, isGlobal, runtime);\r\n\r\n  // Copy agents to agents directory\r\n  const agentsSrc = path.join(src, 'agents');\r\n  if (fs.existsSync(agentsSrc)) {\r\n    const agentsDest = path.join(targetDir, 'agents');\r\n    fs.mkdirSync(agentsDest, { recursive: true });\r\n\r\n    // Remove old EZ_Agents agents (ez-*.md) before copying new ones\r\n    if (fs.existsSync(agentsDest)) {\r\n      for (const file of fs.readdirSync(agentsDest)) {\r\n        if (file.startsWith('ez-') && file.endsWith('.md')) {\r\n          fs.unlinkSync(path.join(agentsDest, file));\r\n        }\r\n      }\r\n    }\r\n\r\n    // Copy new agents\r\n    const agentEntries = fs.readdirSync(agentsSrc, { withFileTypes: true });\r\n    for (const entry of agentEntries) {\r\n      if (entry.isFile() && entry.name.endsWith('.md')) {\r\n        let content = fs.readFileSync(path.join(agentsSrc, entry.name), 'utf8');\r\n        // Replace ~/.claude/ and $HOME/.claude/ as they are the source of truth in the repo\r\n        const dirRegex = /~\\/\\.claude\\//g;\r\n        const homeDirRegex = /\\$HOME\\/\\.claude\\//g;\r\n        if (!isCopilot) {\r\n          content = content.replace(dirRegex, pathPrefix);\r\n          content = content.replace(homeDirRegex, toHomePrefix(pathPrefix));\r\n        }\r\n        content = processAttribution(content, getCommitAttribution(runtime));\r\n        // Convert frontmatter for runtime compatibility\r\n        if (isOpencode) {\r\n          content = convertClaudeToOpencodeFrontmatter(content);\r\n        } else if (isGemini) {\r\n          content = convertClaudeToGeminiAgent(content);\r\n        } else if (isCodex) {\r\n          content = convertClaudeAgentToCodexAgent(content);\r\n        } else if (isCopilot) {\r\n          content = convertClaudeAgentToCopilotAgent(content, isGlobal);\r\n        } else if (isQwen) {\r\n          content = convertClaudeToQwenAgent(content);\r\n        } else if (isKimi) {\r\n          content = convertClaudeToKimiAgent(content);\r\n        }\r\n        const destName = isCopilot ? entry.name.replace('.md', '.agent.md') : entry.name;\r\n        fs.writeFileSync(path.join(agentsDest, destName), content);\r\n      }\r\n    }\r\n    if (verifyInstalled(agentsDest, 'agents')) {\r\n      console.log(`  ${green}\u2713${reset} Installed agents`);\r\n    } else {\r\n      failures.push('agents');\r\n    }\r\n  }\r\n\r\n  // Copy CHANGELOG.md\r\n  const changelogSrc = path.join(src, 'CHANGELOG.md');\r\n  const changelogDest = path.join(targetDir, 'ez-agents', 'CHANGELOG.md');\r\n  if (fs.existsSync(changelogSrc)) {\r\n    fs.copyFileSync(changelogSrc, changelogDest);\r\n    if (verifyFileInstalled(changelogDest, 'CHANGELOG.md')) {\r\n      console.log(`  ${green}\u2713${reset} Installed CHANGELOG.md`);\r\n    } else {\r\n      failures.push('CHANGELOG.md');\r\n    }\r\n  }\r\n\r\n  // Write VERSION file\r\n  const versionDest = path.join(targetDir, 'ez-agents', 'VERSION');\r\n  fs.writeFileSync(versionDest, pkg.version);\r\n  if (verifyFileInstalled(versionDest, 'VERSION')) {\r\n    console.log(`  ${green}\u2713${reset} Wrote VERSION (${pkg.version})`);\r\n  } else {\r\n    failures.push('VERSION');\r\n  }\r\n\r\n  if (!isCodex && !isCopilot && !isQwen && !isKimi) {\r\n    // Write package.json to force CommonJS mode for EZ scripts\r\n    // Prevents \"require is not defined\" errors when project has \"type\": \"module\"\r\n    // Node.js walks up looking for package.json - this stops inheritance from project\r\n    const pkgJsonDest = path.join(targetDir, 'package.json');\r\n    fs.writeFileSync(pkgJsonDest, '{\"type\":\"commonjs\"}\\n');\r\n    console.log(`  ${green}\u2713${reset} Wrote package.json (CommonJS mode)`);\r\n\r\n    // Copy hooks from dist/ (bundled with dependencies)\r\n    // Template paths for the target runtime (replaces '.claude' with correct config dir)\r\n    const hooksSrc = path.join(src, 'hooks', 'dist');\r\n    if (fs.existsSync(hooksSrc)) {\r\n      const hooksDest = path.join(targetDir, 'hooks');\r\n      fs.mkdirSync(hooksDest, { recursive: true });\r\n      const hookEntries = fs.readdirSync(hooksSrc);\r\n      const configDirReplacement = getConfigDirFromHome(runtime, isGlobal);\r\n      for (const entry of hookEntries) {\r\n        const srcFile = path.join(hooksSrc, entry);\r\n        if (fs.statSync(srcFile).isFile()) {\r\n          const destFile = path.join(hooksDest, entry);\r\n          // Template .js files to replace '.claude' with runtime-specific config dir\r\n          if (entry.endsWith('.js')) {\r\n            let content = fs.readFileSync(srcFile, 'utf8');\r\n            content = content.replace(/'\\.claude'/g, configDirReplacement);\r\n            fs.writeFileSync(destFile, content);\r\n          } else {\r\n            fs.copyFileSync(srcFile, destFile);\r\n          }\r\n        }\r\n      }\r\n      if (verifyInstalled(hooksDest, 'hooks')) {\r\n        console.log(`  ${green}\u2713${reset} Installed hooks (bundled)`);\r\n      } else {\r\n        failures.push('hooks');\r\n      }\r\n    }\r\n  }\r\n\r\n  if (failures.length > 0) {\r\n    console.error(`\\n  ${yellow}Installation incomplete!${reset} Failed: ${failures.join(', ')}`);\r\n    process.exit(1);\r\n  }\r\n\r\n  // Write file manifest for future modification detection\r\n  writeManifest(targetDir, runtime);\r\n  console.log(`  ${green}\u2713${reset} Wrote file manifest (${MANIFEST_NAME})`);\r\n\r\n  // Report any backed-up local patches\r\n  reportLocalPatches(targetDir, runtime);\r\n\r\n  // Verify no leaked .claude paths in non-Claude runtimes\r\n  if (runtime !== 'claude') {\r\n    const leakedPaths: Array<{ file: string; count: number }> = [];\r\n    function scanForLeakedPaths(dir: string) {\r\n      if (!fs.existsSync(dir)) return;\r\n      const entries = fs.readdirSync(dir, { withFileTypes: true });\r\n      for (const entry of entries) {\r\n        const fullPath = path.join(dir, entry.name);\r\n        if (entry.isDirectory()) {\r\n          scanForLeakedPaths(fullPath);\r\n        } else if ((entry.name.endsWith('.md') || entry.name.endsWith('.toml')) && entry.name !== 'CHANGELOG.md') {\r\n          const content = fs.readFileSync(fullPath, 'utf8');\r\n          const matches = content.match(/(?:~|\\$HOME)\\/\\.claude\\b/g);\r\n          if (matches) {\r\n            leakedPaths.push({ file: fullPath.replace(targetDir + '/', ''), count: matches.length });\r\n          }\r\n        }\r\n      }\r\n    }\r\n    scanForLeakedPaths(targetDir);\r\n    if (leakedPaths.length > 0) {\r\n      const totalLeaks = leakedPaths.reduce((sum, l) => sum + l.count, 0);\r\n      console.warn(`\\n  ${yellow}\u26A0${reset}  Found ${totalLeaks} unreplaced .claude path reference(s) in ${leakedPaths.length} file(s):`);\r\n      for (const leak of leakedPaths.slice(0, 5)) {\r\n        console.warn(`     ${dim}${leak.file}${reset} (${leak.count})`);\r\n      }\r\n      if (leakedPaths.length > 5) {\r\n        console.warn(`     ${dim}... and ${leakedPaths.length - 5} more file(s)${reset}`);\r\n      }\r\n      console.warn(`  ${dim}These paths may not resolve correctly for ${runtimeLabel}.${reset}`);\r\n    }\r\n  }\r\n\r\n  if (isCodex) {\r\n    // Generate Codex config.toml and per-agent .toml files\r\n    const agentCount = installCodexConfig(targetDir, agentsSrc);\r\n    console.log(`  ${green}\u2713${reset} Generated config.toml with ${agentCount} agent roles`);\r\n    console.log(`  ${green}\u2713${reset} Generated ${agentCount} agent .toml config files`);\r\n    return { settingsPath: null, settings: null, statuslineCommand: null, runtime };\r\n  }\r\n\r\n  if (isCopilot) {\r\n    // Generate copilot-instructions.md\r\n    const templatePath = path.join(targetDir, 'ez-agents', 'templates', 'copilot-instructions.md');\r\n    const instructionsPath = path.join(targetDir, 'copilot-instructions.md');\r\n    if (fs.existsSync(templatePath)) {\r\n      const template = fs.readFileSync(templatePath, 'utf8');\r\n      mergeCopilotInstructions(instructionsPath, template);\r\n      console.log(`  ${green}\u2713${reset} Generated copilot-instructions.md`);\r\n    }\r\n    // Copilot: no settings.json, no hooks, no statusline (like Codex)\r\n    return { settingsPath: null, settings: null, statuslineCommand: null, runtime };\r\n  }\r\n\r\n  if (isQwen || isKimi) {\r\n    // Qwen/Kimi: no settings.json hooks, no statusline (like Codex/Copilot)\r\n    // Skills are installed directly in skills/ez-*/SKILL.md\r\n    return { settingsPath: null, settings: null, statuslineCommand: null, runtime };\r\n  }\r\n\r\n  // Configure statusline and hooks in settings.json\r\n  // Gemini uses AfterTool instead of PostToolUse for post-tool hooks\r\n  const postToolEvent = runtime === 'gemini' ? 'AfterTool' : 'PostToolUse';\r\n  const settingsPath = path.join(targetDir, 'settings.json');\r\n  const settings = cleanupOrphanedHooks(readSettings(settingsPath));\r\n  const statuslineCommand = isGlobal\r\n    ? buildHookCommand(targetDir, 'ez-statusline.js')\r\n    : 'node ' + dirName + '/hooks/ez-statusline.js';\r\n  const updateCheckCommand = isGlobal\r\n    ? buildHookCommand(targetDir, 'ez-check-update.js')\r\n    : 'node ' + dirName + '/hooks/ez-check-update.js';\r\n  const contextMonitorCommand = isGlobal\r\n    ? buildHookCommand(targetDir, 'ez-context-monitor.js')\r\n    : 'node ' + dirName + '/hooks/ez-context-monitor.js';\r\n\r\n  // Enable experimental agents for Gemini CLI (required for custom sub-agents)\r\n  if (isGemini) {\r\n    if (!settings.experimental) {\r\n      settings.experimental = {};\r\n    }\r\n    if (!settings.experimental.enableAgents) {\r\n      settings.experimental.enableAgents = true;\r\n      console.log(`  ${green}\u2713${reset} Enabled experimental agents`);\r\n    }\r\n  }\r\n\r\n  // Configure SessionStart hook for update checking (skip for opencode)\r\n  if (!isOpencode) {\r\n    if (!settings.hooks) {\r\n      settings.hooks = {};\r\n    }\r\n    if (!settings.hooks.SessionStart) {\r\n      settings.hooks.SessionStart = [];\r\n    }\r\n\r\n    const hasEzUpdateHook = settings.hooks.SessionStart.some(entry =>\r\n      entry.hooks && entry.hooks.some(h => h.command && h.command.includes('ez-check-update'))\r\n    );\r\n\r\n    if (!hasEzUpdateHook) {\r\n      settings.hooks.SessionStart.push({\r\n        hooks: [\r\n          {\r\n            type: 'command',\r\n            command: updateCheckCommand\r\n          }\r\n        ]\r\n      });\r\n      console.log(`  ${green}\u2713${reset} Configured update check hook`);\r\n    }\r\n\r\n    // Configure post-tool hook for context window monitoring\r\n    if (!settings.hooks[postToolEvent]) {\r\n      settings.hooks[postToolEvent] = [];\r\n    }\r\n\r\n    const hasContextMonitorHook = settings.hooks[postToolEvent].some(entry =>\r\n      entry.hooks && entry.hooks.some(h => h.command && h.command.includes('ez-context-monitor'))\r\n    );\r\n\r\n    if (!hasContextMonitorHook) {\r\n      settings.hooks[postToolEvent].push({\r\n        hooks: [\r\n          {\r\n            type: 'command',\r\n            command: contextMonitorCommand\r\n          }\r\n        ]\r\n      });\r\n      console.log(`  ${green}\u2713${reset} Configured context window monitor hook`);\r\n    }\r\n  }\r\n\r\n  return { settingsPath, settings, statuslineCommand, runtime };\r\n}\r\n\r\n/**\r\n * Apply statusline config, then print completion message\r\n */\r\nfunction finishInstall(settingsPath, settings, statuslineCommand, shouldInstallStatusline, runtime = 'claude', isGlobal = true) {\r\n  const isOpencode = runtime === 'opencode';\r\n  const isCodex = runtime === 'codex';\r\n  const isCopilot = runtime === 'copilot';\r\n  const isQwen = runtime === 'qwen';\r\n  const isKimi = runtime === 'kimi';\r\n\r\n  if (shouldInstallStatusline && !isOpencode && !isCodex && !isCopilot && !isQwen && !isKimi) {\r\n    settings.statusLine = {\r\n      type: 'command',\r\n      command: statuslineCommand\r\n    };\r\n    console.log(`  ${green}\u2713${reset} Configured statusline`);\r\n  }\r\n\r\n  // Write settings when runtime supports settings.json\r\n  if (!isCodex && !isCopilot && !isQwen && !isKimi) {\r\n    writeSettings(settingsPath, settings);\r\n  }\r\n\r\n  // Configure OpenCode permissions\r\n  if (isOpencode) {\r\n    configureOpencodePermissions(isGlobal);\r\n  }\r\n\r\n  let program = 'Claude Code';\r\n  if (runtime === 'opencode') program = 'OpenCode';\r\n  if (runtime === 'gemini') program = 'Gemini';\r\n  if (runtime === 'codex') program = 'Codex';\r\n  if (runtime === 'copilot') program = 'Copilot';\r\n  if (runtime === 'qwen') program = 'Qwen Code';\r\n  if (runtime === 'kimi') program = 'Kimi Code';\r\n\r\n  let command = '/ez:new-project';\r\n  if (runtime === 'opencode' || runtime === 'gemini' || runtime === 'copilot' || runtime === 'qwen' || runtime === 'kimi') command = '/ez-new-project';\r\n  if (runtime === 'codex') command = '$ez-new-project';\r\n  console.log(`\r\n  ${green}Done!${reset} Open a blank directory in ${program} and run ${cyan}${command}${reset}.\r\n\r\n  ${cyan}Join the community:${reset} https://discord.gg/ez-agents\r\n`);\r\n}\r\n\r\n/**\r\n * Handle statusline configuration with optional prompt\r\n */\r\nfunction handleStatusline(settings, isInteractive, callback) {\r\n  const hasExisting = settings.statusLine != null;\r\n\r\n  if (!hasExisting) {\r\n    callback(true);\r\n    return;\r\n  }\r\n\r\n  if (forceStatusline) {\r\n    callback(true);\r\n    return;\r\n  }\r\n\r\n  if (!isInteractive) {\r\n    console.log(`  ${yellow}\u26A0${reset} Skipping statusline (already configured)`);\r\n    console.log(`    Use ${cyan}--force-statusline${reset} to replace\\n`);\r\n    callback(false);\r\n    return;\r\n  }\r\n\r\n  const existingCmd = settings.statusLine.command || settings.statusLine.url || '(custom)';\r\n\r\n  const rl = readline.createInterface({\r\n    input: process.stdin,\r\n    output: process.stdout\r\n  });\r\n\r\n  console.log(`\r\n  ${yellow}\u26A0${reset} Existing statusline detected\\n\r\n  Your current statusline:\r\n    ${dim}command: ${existingCmd}${reset}\r\n\r\n  EZ_Agents includes a statusline showing:\r\n    \u2022 Model name\r\n    \u2022 Current task (from todo list)\r\n    \u2022 Context window usage (color-coded)\r\n\r\n  ${cyan}1${reset}) Keep existing\r\n  ${cyan}2${reset}) Replace with EZ_Agents statusline\r\n`);\r\n\r\n  rl.question(`  Choice ${dim}[1]${reset}: `, (answer) => {\r\n    rl.close();\r\n    const choice = answer.trim() || '1';\r\n    callback(choice === '2');\r\n  });\r\n}\r\n\r\n/**\r\n * Prompt for runtime selection\r\n */\r\nfunction promptRuntime(callback) {\r\n  const rl = readline.createInterface({\r\n    input: process.stdin,\r\n    output: process.stdout\r\n  });\r\n\r\n  let answered = false;\r\n\r\n  rl.on('close', () => {\r\n    if (!answered) {\r\n      answered = true;\r\n      console.log(`\\n  ${yellow}Installation cancelled${reset}\\n`);\r\n      process.exit(0);\r\n    }\r\n  });\r\n\r\n  console.log(`  ${yellow}Which runtime(s) would you like to install for?${reset}\\n\\n  ${cyan}1${reset}) Claude Code ${dim}(~/.claude)${reset}\r\n  ${cyan}2${reset}) OpenCode    ${dim}(~/.config/opencode)${reset} - open source, free models\r\n  ${cyan}3${reset}) Gemini      ${dim}(~/.gemini)${reset}\r\n  ${cyan}4${reset}) Codex       ${dim}(~/.codex)${reset}\r\n  ${cyan}5${reset}) Copilot     ${dim}(~/.copilot)${reset}\r\n  ${cyan}6${reset}) Qwen Code   ${dim}(~/.qwen)${reset} - Alibaba Qwen official CLI\r\n  ${cyan}7${reset}) Kimi Code   ${dim}(~/.kimi)${reset} - Moonshot Kimi official CLI\r\n  ${cyan}8${reset}) All ${dim}(all 7 runtimes above)${reset}\r\n\r\n  ${dim}Note: OpenAI and Anthropic are model providers configured within each runtime's settings,\r\n  not separate CLI runtimes. See README.md for model configuration.${reset}\r\n`);\r\n\r\n  rl.question(`  Choice ${dim}[1]${reset}: `, (answer) => {\r\n    answered = true;\r\n    rl.close();\r\n    const choice = answer.trim() || '1';\r\n    if (choice === '8') {\r\n      callback(['claude', 'opencode', 'gemini', 'codex', 'copilot', 'qwen', 'kimi']);\r\n    } else if (choice === '7') {\r\n      callback(['kimi']);\r\n    } else if (choice === '6') {\r\n      callback(['qwen']);\r\n    } else if (choice === '5') {\r\n      callback(['copilot']);\r\n    } else if (choice === '4') {\r\n      callback(['codex']);\r\n    } else if (choice === '3') {\r\n      callback(['gemini']);\r\n    } else if (choice === '2') {\r\n      callback(['opencode']);\r\n    } else {\r\n      callback(['claude']);\r\n    }\r\n  });\r\n}\r\n\r\n/**\r\n * Prompt for install location\r\n */\r\nfunction promptLocation(runtimes) {\r\n  if (!process.stdin.isTTY) {\r\n    console.log(`  ${yellow}Non-interactive terminal detected, defaulting to global install${reset}\\n`);\r\n    installAllRuntimes(runtimes, true, false);\r\n    return;\r\n  }\r\n\r\n  const rl = readline.createInterface({\r\n    input: process.stdin,\r\n    output: process.stdout\r\n  });\r\n\r\n  let answered = false;\r\n\r\n  rl.on('close', () => {\r\n    if (!answered) {\r\n      answered = true;\r\n      console.log(`\\n  ${yellow}Installation cancelled${reset}\\n`);\r\n      process.exit(0);\r\n    }\r\n  });\r\n\r\n  const pathExamples = runtimes.map(r => {\r\n    const globalPath = getGlobalDir(r, explicitConfigDir);\r\n    return globalPath.replace(os.homedir(), '~');\r\n  }).join(', ');\r\n\r\n  const localExamples = runtimes.map(r => `./${getDirName(r)}`).join(', ');\r\n\r\n  console.log(`  ${yellow}Where would you like to install?${reset}\\n\\n  ${cyan}1${reset}) Global ${dim}(${pathExamples})${reset} - available in all projects\r\n  ${cyan}2${reset}) Local  ${dim}(${localExamples})${reset} - this project only\r\n`);\r\n\r\n  rl.question(`  Choice ${dim}[1]${reset}: `, (answer) => {\r\n    answered = true;\r\n    rl.close();\r\n    const choice = answer.trim() || '1';\r\n    const isGlobal = choice !== '2';\r\n    installAllRuntimes(runtimes, isGlobal, true);\r\n  });\r\n}\r\n\r\n/**\r\n * Install EZ_Agents for all selected runtimes\r\n */\r\nfunction installAllRuntimes(runtimes: string[], isGlobal: boolean, isInteractive: boolean) {\r\n  const results: Array<{ settingsPath: string | null; settings: Record<string, unknown> | null; statuslineCommand: string | null; runtime: string }> = [];\r\n\r\n  for (const runtime of runtimes) {\r\n    const result = install(isGlobal, runtime);\r\n    results.push(result);\r\n  }\r\n\r\n  const statuslineRuntimes = ['claude', 'gemini'];\r\n  const statuslineResults = results.filter(r => statuslineRuntimes.includes(r.runtime));\r\n\r\n  // Handle statusline per-runtime to avoid conflicts\r\n  // Each runtime with settings.json gets its own statusline prompt\r\n  const finalizePerRuntime = (runtimeResult: { settingsPath: string | null; settings: Record<string, unknown> | null; statuslineCommand: string | null; runtime: string }, shouldInstallStatusline: boolean) => {\r\n    const useStatusline = statuslineRuntimes.includes(runtimeResult.runtime) && shouldInstallStatusline;\r\n    finishInstall(\r\n      runtimeResult.settingsPath,\r\n      runtimeResult.settings,\r\n      runtimeResult.statuslineCommand,\r\n      useStatusline,\r\n      runtimeResult.runtime,\r\n      isGlobal\r\n    );\r\n  };\r\n\r\n  if (statuslineResults.length === 0) {\r\n    // No runtimes support statusline - finalize all without statusline\r\n    for (const result of results) {\r\n      finalizePerRuntime(result, false);\r\n    }\r\n  } else if (statuslineResults.length === 1) {\r\n    // Single runtime with statusline support - prompt once\r\n    const singleResult = statuslineResults[0];\r\n    if (singleResult) {\r\n      handleStatusline(singleResult.settings, isInteractive, (shouldInstall) => {\r\n        finalizePerRuntime(singleResult, shouldInstall);\r\n        // Finalize other runtimes without statusline\r\n        for (const result of results) {\r\n          if (!statuslineRuntimes.includes(result.runtime)) {\r\n            finalizePerRuntime(result, false);\r\n          }\r\n        }\r\n      });\r\n    }\r\n  } else {\r\n    // Multiple runtimes with statusline support - prompt for each\r\n    let currentIndex = 0;\r\n    const statuslineChoices = new Array(statuslineResults.length).fill(false);\r\n\r\n    const promptNextStatusline = () => {\r\n      if (currentIndex >= statuslineResults.length) {\r\n        // All statusline prompts done - finalize all runtimes\r\n        for (let i = 0; i < results.length; i++) {\r\n          const result = results[i];\r\n          if (!result) continue;\r\n          const statuslineIndex = statuslineResults.findIndex(r => r.runtime === result.runtime);\r\n          const shouldInstall = statuslineIndex !== -1 ? statuslineChoices[statuslineIndex] : false;\r\n          finalizePerRuntime(result, shouldInstall);\r\n        }\r\n        return;\r\n      }\r\n\r\n      const currentResult = statuslineResults[currentIndex];\r\n      if (currentResult) {\r\n        const currentRuntimeIndex = currentIndex;\r\n        handleStatusline(currentResult.settings, isInteractive, (shouldInstall) => {\r\n          statuslineChoices[currentRuntimeIndex] = shouldInstall;\r\n          currentIndex++;\r\n          promptNextStatusline();\r\n        });\r\n      }\r\n    };\r\n\r\n    promptNextStatusline();\r\n  }\r\n}\r\n\r\n// Test-only exports \u2014 skip main logic when loaded as a module for testing\r\nconst testExports = {\r\n    getCodexSkillAdapterHeader,\r\n    convertClaudeToGeminiAgent,\r\n    convertClaudeAgentToCodexAgent,\r\n    generateCodexAgentToml,\r\n    generateCodexConfigBlock,\r\n    stripEzAgentsFromCodexConfig,\r\n    mergeCodexConfig,\r\n    installCodexConfig,\r\n    convertClaudeCommandToCodexSkill,\r\n    EZ_CODEX_MARKER,\r\n    CODEX_AGENT_SANDBOX,\r\n    getDirName,\r\n    getGlobalDir,\r\n    getConfigDirFromHome,\r\n    claudeToCopilotTools,\r\n    convertCopilotToolName,\r\n    convertClaudeToCopilotContent,\r\n    convertClaudeCommandToCopilotSkill,\r\n    convertClaudeAgentToCopilotAgent,\r\n    copyCommandsAsCopilotSkills,\r\n    EZ_COPILOT_INSTRUCTIONS_MARKER,\r\n    EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER,\r\n    mergeCopilotInstructions,\r\n    stripEzAgentsFromCopilotInstructions,\r\n    writeManifest,\r\n    reportLocalPatches,\r\n    convertClaudeToQwenSkill,\r\n    convertClaudeToQwenAgent,\r\n    convertClaudeToKimiSkill,\r\n    convertClaudeToKimiAgent,\r\n  };\r\n\r\n// Main logic\r\nif (hasGlobal && hasLocal) {\r\n  console.error(`  ${yellow}Cannot specify both --global and --local${reset}`);\r\n  process.exit(1);\r\n} else if (explicitConfigDir && hasLocal) {\r\n  console.error(`  ${yellow}Cannot use --config-dir with --local${reset}`);\r\n  process.exit(1);\r\n} else if (hasUninstall) {\r\n  if (!hasGlobal && !hasLocal) {\r\n    console.error(`  ${yellow}--uninstall requires --global or --local${reset}`);\r\n    process.exit(1);\r\n  }\r\n  const runtimes = selectedRuntimes.length > 0 ? selectedRuntimes : ['claude'];\r\n  for (const runtime of runtimes) {\r\n    uninstall(hasGlobal, runtime);\r\n  }\r\n} else if (selectedRuntimes.length > 0) {\r\n  if (!hasGlobal && !hasLocal) {\r\n    promptLocation(selectedRuntimes);\r\n  } else {\r\n    installAllRuntimes(selectedRuntimes, hasGlobal, false);\r\n  }\r\n} else if (hasGlobal || hasLocal) {\r\n  // Default to Claude if no runtime specified but location is\r\n  installAllRuntimes(['claude'], hasGlobal, false);\r\n} else {\r\n  // Interactive\r\n  if (!process.stdin.isTTY) {\r\n    console.log(`  ${yellow}Non-interactive terminal detected, defaulting to Claude Code global install${reset}\\n`);\r\n    installAllRuntimes(['claude'], true, false);\r\n  } else {\r\n    promptRuntime((runtimes) => {\r\n      promptLocation(runtimes);\r\n    });\r\n  }\r\n}\r\n\r\n// Export for testing\r\nexport { testExports };\r\n", "{\n  \"name\": \"@howlil/ez-agents\",\n  \"version\": \"5.0.0\",\n  \"description\": \"EZ Agents \u2014 Meta-prompting with multi-model support (Qwen, Kimi, OpenAI)\",\n  \"type\": \"module\",\n  \"sideEffects\": false,\n  \"bin\": {\n    \"ez-agents\": \"dist/bin/install.js\",\n    \"ez-agents-update\": \"dist/bin/update.js\"\n  },\n  \"exports\": {\n    \".\": {\n      \"types\": \"./dist/index.d.ts\",\n      \"import\": \"./dist/index.js\"\n    },\n    \"./factories\": {\n      \"types\": \"./dist/lib/factories/index.d.ts\",\n      \"import\": \"./dist/lib/factories/index.js\"\n    },\n    \"./strategies\": {\n      \"types\": \"./dist/lib/strategies/index.d.ts\",\n      \"import\": \"./dist/lib/strategies/index.js\"\n    },\n    \"./adapters\": {\n      \"types\": \"./dist/lib/adapters/index.d.ts\",\n      \"import\": \"./dist/lib/adapters/index.js\"\n    },\n    \"./observer\": {\n      \"types\": \"./dist/lib/observer/index.d.ts\",\n      \"import\": \"./dist/lib/observer/index.js\"\n    },\n    \"./decorators\": {\n      \"types\": \"./dist/lib/decorators/index.d.ts\",\n      \"import\": \"./dist/lib/decorators/index.js\"\n    },\n    \"./facades\": {\n      \"types\": \"./dist/lib/facades/index.d.ts\",\n      \"import\": \"./dist/lib/facades/index.js\"\n    },\n    \"./commands\": {\n      \"types\": \"./dist/lib/commands/index.d.ts\",\n      \"import\": \"./dist/lib/commands/index.js\"\n    },\n    \"./services\": {\n      \"types\": \"./dist/lib/services/index.d.ts\",\n      \"import\": \"./dist/lib/services/index.js\"\n    },\n    \"./repositories\": {\n      \"types\": \"./dist/lib/repositories/index.d.ts\",\n      \"import\": \"./dist/lib/repositories/index.js\"\n    },\n    \"./fp\": {\n      \"types\": \"./dist/lib/fp/index.d.ts\",\n      \"import\": \"./dist/lib/fp/index.js\"\n    }\n  },\n  \"main\": \"./dist/index.js\",\n  \"types\": \"./dist/index.d.ts\",\n  \"files\": [\n    \"dist\",\n    \"commands\",\n    \"ez-agents\",\n    \"agents\",\n    \"hooks/dist\",\n    \"scripts\"\n  ],\n  \"keywords\": [\n    \"claude\",\n    \"claude-code\",\n    \"ai\",\n    \"meta-prompting\",\n    \"context-engineering\",\n    \"spec-driven-development\",\n    \"gemini\",\n    \"gemini-cli\",\n    \"codex\",\n    \"codex-cli\",\n    \"ez-agents\",\n    \"agent-orchestration\"\n  ],\n  \"author\": \"T\u00C2CHES\",\n  \"license\": \"MIT\",\n  \"repository\": {\n    \"type\": \"git\",\n    \"url\": \"git+https://github.com/howlil/ez-agents.git\"\n  },\n  \"homepage\": \"https://github.com/howlil/ez-agents\",\n  \"bugs\": {\n    \"url\": \"https://github.com/howlil/ez-agents/issues\"\n  },\n  \"engines\": {\n    \"node\": \">=16.7.0\"\n  },\n  \"devDependencies\": {\n    \"@types/micromatch\": \"4.0.0\",\n    \"@types/node\": \"20.0.0\",\n    \"@typescript-eslint/eslint-plugin\": \"8.57.2\",\n    \"@typescript-eslint/parser\": \"8.57.2\",\n    \"c8\": \"11.0.0\",\n    \"complexity-report\": \"2.0.0-alpha\",\n    \"dependency-cruiser\": \"17.3.9\",\n    \"eslint\": \"8.57.0\",\n    \"eslint-plugin-complexity\": \"1.0.2\",\n    \"eslint-plugin-sonarjs\": \"4.0.2\",\n    \"eslint-plugin-tsdoc\": \"0.5.2\",\n    \"expect-type\": \"1.3.0\",\n    \"fast-check\": \"3.0.0\",\n    \"glob\": \"13.0.6\",\n    \"husky\": \"9.1.7\",\n    \"js-yaml\": \"4.1.1\",\n    \"jscpd\": \"4.0.8\",\n    \"jsdoc\": \"4.0.2\",\n    \"k6\": \"0.0.0\",\n    \"lint-staged\": \"15.5.2\",\n    \"madge\": \"8.0.0\",\n    \"markdownlint\": \"0.40.0\",\n    \"ts-morph\": \"27.0.2\",\n    \"ts-node\": \"10.9.0\",\n    \"ts-prune\": \"0.10.3\",\n    \"tsup\": \"8.0.0\",\n    \"typedoc\": \"0.28.18\",\n    \"typedoc-plugin-missing-exports\": \"4.1.2\",\n    \"typescript\": \"5.4.5\",\n    \"vitest\": \"4.1.0\"\n  },\n  \"scripts\": {\n    \"build\": \"tsup\",\n    \"build:watch\": \"tsup --watch\",\n    \"typecheck\": \"tsc --noEmit\",\n    \"typecheck:strict\": \"tsc --noEmit --strict\",\n    \"dev\": \"tsx bin/install.ts\",\n    \"lint\": \"eslint . --ext .ts --max-warnings=0\",\n    \"lint:fix\": \"eslint . --ext .ts --fix\",\n    \"build:hooks\": \"node dist/scripts/build-hooks.js\",\n    \"prepublishOnly\": \"npm run build && npm run build:hooks\",\n    \"test\": \"tsx scripts/run-tests.ts\",\n    \"test:coverage\": \"c8 --check-coverage --lines 70 --reporter text --include 'bin/lib/**/*.ts' --exclude 'tests/**' --all tsx scripts/run-tests.ts\",\n    \"test:a11y\": \"node tests/accessibility/run-a11y-tests.cjs\",\n    \"test:qwen:code\": \"sh tests/staging/qwen-code-battle-full.sh\",\n    \"test:qwen:code:simple\": \"sh tests/staging/qwen-code-simple-test.sh\",\n    \"test:qwen:code:edge\": \"sh tests/staging/qwen-code-edge-cases.sh\",\n    \"test:docker:qwen:code\": \"docker-compose -f docker-compose.staging.yml up --build ez-agents-qwen-code-battle\",\n    \"test:docker:qwen:edge\": \"docker-compose -f docker-compose.staging.yml up --build ez-agents-qwen-code-edge-cases\",\n    \"test:docker:all\": \"docker-compose -f docker-compose.staging.yml up --build --abort-on-container-exit\",\n    \"install:production\": \"node --loader ts-node/esm scripts/install-production.ts\",\n    \"install:production:latest\": \"node --loader ts-node/esm scripts/install-production.ts latest\",\n    \"install:production:v4\": \"node --loader ts-node/esm scripts/install-production.ts 4.0.0\",\n    \"validate:token\": \"node --loader ts-node/esm scripts/validate-token.ts\",\n    \"security:scan\": \"npm audit --audit-level=moderate\",\n    \"docs:generate\": \"jsdoc -c jsdoc.json -r bin/ ez-agents/bin/lib/\",\n    \"docs:api\": \"typedoc\",\n    \"analyze:duplicates\": \"jscpd --config .jscpd.json\",\n    \"check:duplicates\": \"jscpd --config .jscpd.json\",\n    \"check:complexity\": \"node --loader ts-node/esm scripts/check-complexity.ts\",\n    \"check:tsdoc\": \"node --loader ts-node/esm scripts/check-tsdoc-coverage.ts\",\n    \"test:package\": \"node --loader ts-node/esm scripts/test-package-docker.ts\",\n    \"check:coupling\": \"madge --circular bin/lib/\",\n    \"check:abstractions\": \"ts-prune\",\n    \"check:all-metrics\": \"npm run lint && npm run check:duplicates && npm run check:coupling\",\n    \"prepare\": \"husky\",\n    \"gates:local\": \"node ez-agents/bin/lib/gate-executor.cjs --ci --cache\",\n    \"gates:status\": \"node ez-agents/bin/lib/gate-executor.cjs --status\",\n    \"gates:stats\": \"node ez-agents/bin/lib/gate-executor.cjs --stats\",\n    \"deps:audit\": \"tsx bin/dependency-audit.ts\",\n    \"deps:audit:json\": \"tsx bin/dependency-audit.ts --json\",\n    \"deps:check\": \"tsx bin/dependency-audit.ts --quiet\"\n  },\n  \"lint-staged\": {\n    \"*.ts\": [\n      \"node --test\"\n    ],\n    \"*.md\": [\n      \"tsx scripts/run-tests.ts\"\n    ]\n  },\n  \"dependencies\": {\n    \"micromatch\": \"4.0.5\",\n    \"tiktoken\": \"1.0.22\",\n    \"zod\": \"4.3.6\"\n  }\n}\n"],
  "mappings": "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",
  "names": ["runtimeLabel", "scanForLeakedPaths"]
}

2706
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../bin/install.ts", "../../package.json"],
  "sourcesContent": ["#!/usr/bin/env node\n\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as os from 'os';\nimport * as readline from 'readline';\nimport * as crypto from 'crypto';\nimport { fileURLToPath } from 'url';\n\nconst __filename = fileURLToPath(import.meta.url);\nconst __dirname = path.dirname(__filename);\n\n// Colors\nconst cyan = '\\x1b[36m';\nconst green = '\\x1b[32m';\nconst yellow = '\\x1b[33m';\nconst dim = '\\x1b[2m';\nconst reset = '\\x1b[0m';\n\n// Codex config.toml constants\nconst EZ_CODEX_MARKER = '# EZ_Agents Agent Configuration \\u2014 managed by ez-agents installer';\n\n// Copilot instructions marker constants\nconst EZ_COPILOT_INSTRUCTIONS_MARKER = '<!-- EZ_AGENTS Configuration \\u2014 managed by ez-agents installer -->';\nconst EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER = '<!-- /EZ_AGENTS Configuration -->';\n// Legacy markers kept for backward compatibility with older installs.\nconst LEGACY_EZ_COPILOT_INSTRUCTIONS_MARKER = '<!-- EZ_Agents Configuration \\u2014 managed by ez-agents installer -->';\nconst LEGACY_EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER = '<!-- /EZ_Agents Configuration -->';\n\nconst CODEX_AGENT_SANDBOX = {\n  'ez-executor': 'workspace-write',\n  'ez-planner': 'workspace-write',\n  'ez-phase-researcher': 'workspace-write',\n  'ez-project-researcher': 'workspace-write',\n  'ez-research-synthesizer': 'workspace-write',\n  'ez-verifier': 'workspace-write',\n  'ez-codebase-mapper': 'workspace-write',\n  'ez-roadmapper': 'workspace-write',\n  'ez-debugger': 'workspace-write',\n  'ez-plan-checker': 'read-only',\n  'ez-integration-checker': 'read-only',\n};\n\n// Copilot tool name mapping \u2014 Claude Code tools to GitHub Copilot tools\n// Tool mapping applies ONLY to agents, NOT to skills (per CONTEXT.md decision)\nconst claudeToCopilotTools = {\n  Read: 'read',\n  Write: 'edit',\n  Edit: 'edit',\n  Bash: 'execute',\n  Grep: 'search',\n  Glob: 'search',\n  Task: 'agent',\n  WebSearch: 'web',\n  WebFetch: 'web',\n  TodoWrite: 'todo',\n  AskUserQuestion: 'ask_user',\n  SlashCommand: 'skill',\n};\n\n// Get version from package.json\nimport pkg from '../package.json' with { type: 'json' };\n\n// Parse args\nconst args = process.argv.slice(2);\nconst hasGlobal = args.includes('--global') || args.includes('-g');\nconst hasLocal = args.includes('--local') || args.includes('-l');\nconst hasOpencode = args.includes('--opencode');\nconst hasClaude = args.includes('--claude');\nconst hasGemini = args.includes('--gemini');\nconst hasCodex = args.includes('--codex');\nconst hasCopilot = args.includes('--copilot');\nconst hasQwen = args.includes('--qwen');\nconst hasKimi = args.includes('--kimi');\nconst hasBoth = args.includes('--both'); // Legacy flag, keeps working\nconst hasAll = args.includes('--all');\nconst hasUninstall = args.includes('--uninstall') || args.includes('-u');\n\n// Runtime selection - can be set by flags or interactive prompt\nlet selectedRuntimes: string[] = [];\nif (hasAll) {\n  selectedRuntimes = ['claude', 'opencode', 'gemini', 'codex', 'copilot', 'qwen', 'kimi'];\n} else if (hasBoth) {\n  selectedRuntimes = ['claude', 'opencode'];\n} else {\n  if (hasOpencode) selectedRuntimes.push('opencode');\n  if (hasClaude) selectedRuntimes.push('claude');\n  if (hasGemini) selectedRuntimes.push('gemini');\n  if (hasCodex) selectedRuntimes.push('codex');\n  if (hasCopilot) selectedRuntimes.push('copilot');\n  if (hasQwen) selectedRuntimes.push('qwen');\n  if (hasKimi) selectedRuntimes.push('kimi');\n}\n\n// WSL + Windows Node.js detection\n// When Windows-native Node runs on WSL, os.homedir() and path.join() produce\n// backslash paths that don't resolve correctly on the Linux filesystem.\nif (process.platform === 'win32') {\n  let isWSL = false;\n  try {\n    if (process.env.WSL_DISTRO_NAME) {\n      isWSL = true;\n    } else if (fs.existsSync('/proc/version')) {\n      const procVersion = fs.readFileSync('/proc/version', 'utf8').toLowerCase();\n      if (procVersion.includes('microsoft') || procVersion.includes('wsl')) {\n        isWSL = true;\n      }\n    }\n  } catch {\n    // Ignore read errors \u2014 not WSL\n  }\n\n  if (isWSL) {\n    console.error(`\n${yellow}\u26A0 Detected WSL with Windows-native Node.js.${reset}\n\nThis causes path resolution issues that prevent correct installation.\nPlease install a Linux-native Node.js inside WSL:\n\n  curl -fsSL https://fnm.vercel.app/install | bash\n  fnm install --lts\n\nThen re-run: npx ez-agents\n`);\n    process.exit(1);\n  }\n}\n\n/**\n * Convert a pathPrefix (which uses absolute paths for global installs) to a\n * $HOME-relative form for replacing $HOME/.claude/ references in bash code blocks.\n * Preserves $HOME as a shell variable so paths remain portable across machines.\n */\nfunction toHomePrefix(pathPrefix: string): string {\n  const home = os.homedir().replace(/\\\\/g, '/');\n  const normalized = pathPrefix.replace(/\\\\/g, '/');\n  if (normalized.startsWith(home)) {\n    return '$HOME' + normalized.slice(home.length);\n  }\n  // Convert tilde-based paths to $HOME-based paths for bash code blocks\n  if (normalized.startsWith('~/')) {\n    return '$HOME' + normalized.slice(1);\n  }\n  // For relative paths or paths not under $HOME, return as-is\n  return normalized;\n}\n\n// Helper to get directory name for a runtime (used for local/project installs)\nfunction getDirName(runtime: string): string {\n  if (runtime === 'copilot') return '.github';\n  if (runtime === 'opencode') return '.opencode';\n  if (runtime === 'gemini') return '.gemini';\n  if (runtime === 'codex') return '.codex';\n  if (runtime === 'qwen') return '.qwen';\n  if (runtime === 'kimi') return '.kimi';\n  return '.claude';\n}\n\n/**\n * Get the config directory path relative to home directory for a runtime\n * Used for templating hooks that use path.join(homeDir, '<configDir>', ...)\n * @param {string} runtime - 'claude', 'opencode', 'gemini', 'codex', 'copilot', or 'qwen'\n * @param {boolean} isGlobal - Whether this is a global install\n */\nfunction getConfigDirFromHome(runtime: string, isGlobal: boolean): string {\n  if (!isGlobal) {\n    // Local installs use the same dir name pattern\n    return `'${getDirName(runtime)}'`;\n  }\n  // Global installs - OpenCode uses XDG path structure\n  if (runtime === 'copilot') return \"'.copilot'\";\n  if (runtime === 'opencode') {\n    // OpenCode: ~/.config/opencode -> '.config', 'opencode'\n    // Return as comma-separated for path.join() replacement\n    return \"'.config', 'opencode'\";\n  }\n  if (runtime === 'gemini') return \"'.gemini'\";\n  if (runtime === 'codex') return \"'.codex'\";\n  if (runtime === 'qwen') return \"'.qwen'\";\n  if (runtime === 'kimi') return \"'.kimi'\";\n  return \"'.claude'\";\n}\n\n/**\n * Get the global config directory for OpenCode\n * OpenCode follows XDG Base Directory spec and uses ~/.config/opencode/\n * Priority: OPENCODE_CONFIG_DIR > dirname(OPENCODE_CONFIG) > XDG_CONFIG_HOME/opencode > ~/.config/opencode\n */\nfunction getOpencodeGlobalDir() {\n  // 1. Explicit OPENCODE_CONFIG_DIR env var\n  if (process.env.OPENCODE_CONFIG_DIR) {\n    return expandTilde(process.env.OPENCODE_CONFIG_DIR);\n  }\n  \n  // 2. OPENCODE_CONFIG env var (use its directory)\n  if (process.env.OPENCODE_CONFIG) {\n    return path.dirname(expandTilde(process.env.OPENCODE_CONFIG));\n  }\n  \n  // 3. XDG_CONFIG_HOME/opencode\n  if (process.env.XDG_CONFIG_HOME) {\n    return path.join(expandTilde(process.env.XDG_CONFIG_HOME), 'opencode');\n  }\n  \n  // 4. Default: ~/.config/opencode (XDG default)\n  return path.join(os.homedir(), '.config', 'opencode');\n}\n\n/**\n * Get the global config directory for a runtime\n * @param {string} runtime - 'claude', 'opencode', 'gemini', 'codex', or 'copilot'\n * @param {string|null} explicitDir - Explicit directory from --config-dir flag\n */\nfunction getGlobalDir(runtime: string, explicitDir: string | null = null): string {\n  if (runtime === 'opencode') {\n    // For OpenCode, --config-dir overrides env vars\n    if (explicitDir) {\n      return expandTilde(explicitDir);\n    }\n    return getOpencodeGlobalDir();\n  }\n  \n  if (runtime === 'gemini') {\n    // Gemini: --config-dir > GEMINI_CONFIG_DIR > ~/.gemini\n    if (explicitDir) {\n      return expandTilde(explicitDir);\n    }\n    if (process.env.GEMINI_CONFIG_DIR) {\n      return expandTilde(process.env.GEMINI_CONFIG_DIR);\n    }\n    return path.join(os.homedir(), '.gemini');\n  }\n\n  if (runtime === 'codex') {\n    // Codex: --config-dir > CODEX_HOME > ~/.codex\n    if (explicitDir) {\n      return expandTilde(explicitDir);\n    }\n    if (process.env.CODEX_HOME) {\n      return expandTilde(process.env.CODEX_HOME);\n    }\n    return path.join(os.homedir(), '.codex');\n  }\n\n  if (runtime === 'copilot') {\n    // Copilot: --config-dir > COPILOT_CONFIG_DIR > ~/.copilot\n    if (explicitDir) {\n      return expandTilde(explicitDir);\n    }\n    if (process.env.COPILOT_CONFIG_DIR) {\n      return expandTilde(process.env.COPILOT_CONFIG_DIR);\n    }\n    return path.join(os.homedir(), '.copilot');\n  }\n\n  if (runtime === 'qwen') {\n    // Qwen Code: --config-dir > QWEN_CONFIG_DIR > ~/.qwen\n    if (explicitDir) {\n      return expandTilde(explicitDir);\n    }\n    if (process.env.QWEN_CONFIG_DIR) {\n      return expandTilde(process.env.QWEN_CONFIG_DIR);\n    }\n    return path.join(os.homedir(), '.qwen');\n  }\n\n  if (runtime === 'kimi') {\n    // Kimi Code: --config-dir > KIMI_CONFIG_DIR > ~/.kimi\n    if (explicitDir) {\n      return expandTilde(explicitDir);\n    }\n    if (process.env.KIMI_CONFIG_DIR) {\n      return expandTilde(process.env.KIMI_CONFIG_DIR);\n    }\n    return path.join(os.homedir(), '.kimi');\n  }\n\n  // Claude Code: --config-dir > CLAUDE_CONFIG_DIR > ~/.claude\n  if (explicitDir) {\n    return expandTilde(explicitDir);\n  }\n  if (process.env.CLAUDE_CONFIG_DIR) {\n    return expandTilde(process.env.CLAUDE_CONFIG_DIR);\n  }\n  return path.join(os.homedir(), '.claude');\n}\n\nconst banner = '\\n' +\n  cyan + '  \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557\u2588\u2588\u2557  \u2588\u2588\u2557\\n' +\n  '  \u2588\u2588\u2554\u2550\u2550\u2550\u2550\u255D\u255A\u2588\u2588\u2557\u2588\u2588\u2554\u255D\\n' +\n  '  \u2588\u2588\u2588\u2588\u2588\u2557   \u255A\u2588\u2588\u2588\u2554\u255D \\n' +\n  '  \u2588\u2588\u2554\u2550\u2550\u255D   \u2588\u2588\u2554\u2588\u2588\u2557 \\n' +\n  '  \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557\u2588\u2588\u2554\u255D\u255A\u2588\u2588\u2557\\n' +\n  '  \u255A\u2550\u2550\u2550\u2550\u2550\u2550\u255D\u255A\u2550\u255D \u255A\u2550\u255D' + reset + '\\n' +\n  '\\n' +\n  '  EZ_Agents ' + dim + 'v' + pkg.version + reset + '\\n' +\n  '  Multi-Model Edition' + reset + '\\n' +\n  '  A meta-prompting, context engineering and spec-driven\\n' +\n  '  development system for Claude Code, OpenCode, Gemini, Codex, Copilot,\\n' +\n  '  Qwen Code, Kimi Code, with multi-model support.\\n' +\n  '  Original by T\u00C2CHES | Multi-Model Fork by @howlil.\\n';\n\n\n// Parse --config-dir argument\nfunction parseConfigDirArg() {\n  const configDirIndex = args.findIndex(arg => arg === '--config-dir' || arg === '-c');\n  if (configDirIndex !== -1) {\n    const nextArg = args[configDirIndex + 1];\n    // Error if --config-dir is provided without a value or next arg is another flag\n    if (!nextArg || nextArg.startsWith('-')) {\n      console.error(`  ${yellow}--config-dir requires a path argument${reset}`);\n      process.exit(1);\n    }\n    return nextArg;\n  }\n  // Also handle --config-dir=value format\n  const configDirArg = args.find(arg => arg.startsWith('--config-dir=') || arg.startsWith('-c='));\n  if (configDirArg) {\n    const value = configDirArg.split('=')[1];\n    if (!value) {\n      console.error(`  ${yellow}--config-dir requires a non-empty path${reset}`);\n      process.exit(1);\n    }\n    return value;\n  }\n  return null;\n}\nconst explicitConfigDir = parseConfigDirArg();\nconst hasHelp = args.includes('--help') || args.includes('-h');\nconst hasVersion = args.includes('--version') || args.includes('-v');\nconst forceStatusline = args.includes('--force-statusline');\n\n// Show version if requested (skip banner, just output version)\nif (hasVersion) {\n  console.log(pkg.version);\n  process.exit(0);\n}\n\nconsole.log(banner);\n\nif (hasUninstall) {\n  console.log('  Mode: Uninstall\\n');\n}\n\n// Show help if requested\nif (hasHelp) {\n  console.log(`  ${yellow}Usage:${reset} npx ez-agents [options]\\n\\n  ${yellow}Options:${reset}\\n    ${cyan}-g, --global${reset}              Install globally (to config directory)\\n    ${cyan}-l, --local${reset}               Install locally (to current directory)\\n    ${cyan}--claude${reset}                  Install for Claude Code only\\n    ${cyan}--opencode${reset}                Install for OpenCode only\\n    ${cyan}--gemini${reset}                  Install for Gemini only\\n    ${cyan}--codex${reset}                   Install for Codex only\\n    ${cyan}--copilot${reset}                 Install for Copilot only\\n    ${cyan}--qwen${reset}                    Install for Qwen Code only\\n    ${cyan}--kimi${reset}                    Install for Kimi Code only\\n    ${cyan}--all${reset}                     Install for all runtimes\\n    ${cyan}-u, --uninstall${reset}           Uninstall EZ_Agents (remove all EZ_Agents files)\\n    ${cyan}-c, --config-dir <path>${reset}   Specify custom config directory\\n    ${cyan}-v, --version${reset}             Show version number\\n    ${cyan}-h, --help${reset}                Show this help message\\n    ${cyan}--force-statusline${reset}        Replace existing statusline config\\n\\n  ${yellow}Examples:${reset}\\n    ${dim}# Interactive install (prompts for runtime and location)${reset}\\n    npx ez-agents\\n\\n    ${dim}# Install for Claude Code globally${reset}\\n    npx ez-agents --claude --global\\n\\n    ${dim}# Install for Qwen Code globally${reset}\\n    npx ez-agents --qwen --global\\n\\n    ${dim}# Install for Kimi Code globally${reset}\\n    npx ez-agents --kimi --global\\n\\n    ${dim}# Install for all runtimes globally${reset}\\n    npx ez-agents --all --global\\n\\n    ${dim}# Uninstall EZ_Agents globally${reset}\\n    npx ez-agents --all --global --uninstall\\n\\n    ${dim}# Show version${reset}\\n    npx ez-agents --version\\n\\n  ${yellow}Notes:${reset}\\n    The --config-dir option is useful when you have multiple configurations.\\n    It takes priority over CLAUDE_CONFIG_DIR / GEMINI_CONFIG_DIR / CODEX_HOME / COPILOT_CONFIG_DIR / QWEN_CONFIG_DIR / KIMI_CONFIG_DIR environment variables.\\n\\n  ${yellow}Model Providers:${reset}\\n    OpenAI and Anthropic are model providers configured in settings.json,\\n    not separate CLI runtimes. See README.md for model configuration.\\n`);\n  process.exit(0);\n}\n\n/**\n * Expand ~ to home directory (shell doesn't expand in env vars passed to node)\n */\nfunction expandTilde(filePath) {\n  if (filePath && filePath.startsWith('~/')) {\n    return path.join(os.homedir(), filePath.slice(2));\n  }\n  return filePath;\n}\n\n/**\n * Build a hook command path using forward slashes for cross-platform compatibility.\n * On Windows, $HOME is not expanded by cmd.exe/PowerShell, so we use the actual path.\n */\nfunction buildHookCommand(configDir, hookName) {\n  // Use forward slashes for Node.js compatibility on all platforms\n  const hooksPath = configDir.replace(/\\\\/g, '/') + '/hooks/' + hookName;\n  return `node \"${hooksPath}\"`;\n}\n\n/**\n * Read and parse settings.json, returning empty object if it doesn't exist\n */\nfunction readSettings(settingsPath) {\n  if (fs.existsSync(settingsPath)) {\n    try {\n      return JSON.parse(fs.readFileSync(settingsPath, 'utf8'));\n    } catch (e) {\n      return {};\n    }\n  }\n  return {};\n}\n\n/**\n * Write settings.json with proper formatting\n */\nfunction writeSettings(settingsPath, settings) {\n  fs.writeFileSync(settingsPath, JSON.stringify(settings, null, 2) + '\\n');\n}\n\n// Cache for attribution settings (populated once per runtime during install)\nconst attributionCache = new Map<string, null | undefined | string>();\n\n/**\n * Get commit attribution setting for a runtime\n * @param {string} runtime - 'claude', 'opencode', 'gemini', 'codex', or 'copilot'\n * @returns {null|undefined|string} null = remove, undefined = keep default, string = custom\n */\nfunction getCommitAttribution(runtime: string): null | undefined | string {\n  // Return cached value if available\n  if (attributionCache.has(runtime)) {\n    return attributionCache.get(runtime);\n  }\n\n  let result: null | undefined | string;\n\n  if (runtime === 'opencode') {\n    const config = readSettings(path.join(getGlobalDir('opencode', null), 'opencode.json'));\n    result = config.disable_ai_attribution === true ? null : undefined;\n  } else if (runtime === 'gemini') {\n    // Gemini: check gemini settings.json for attribution config\n    const settings = readSettings(path.join(getGlobalDir('gemini', explicitConfigDir), 'settings.json'));\n    if (!settings.attribution || settings.attribution.commit === undefined) {\n      result = undefined;\n    } else if (settings.attribution.commit === '') {\n      result = null;\n    } else {\n      result = settings.attribution.commit;\n    }\n  } else if (runtime === 'claude') {\n    // Claude Code\n    const settings = readSettings(path.join(getGlobalDir('claude', explicitConfigDir), 'settings.json'));\n    if (!settings.attribution || settings.attribution.commit === undefined) {\n      result = undefined;\n    } else if (settings.attribution.commit === '') {\n      result = null;\n    } else {\n      result = settings.attribution.commit;\n    }\n  } else {\n    // Codex and Copilot currently have no attribution setting equivalent\n    result = undefined;\n  }\n\n  // Cache and return\n  attributionCache.set(runtime, result);\n  return result;\n}\n\n/**\n * Process Co-Authored-By lines based on attribution setting\n * @param {string} content - File content to process\n * @param {null|undefined|string} attribution - null=remove, undefined=keep, string=replace\n * @returns {string} Processed content\n */\nfunction processAttribution(content: string, attribution: null | undefined | string): string {\n  if (attribution === null) {\n    // Remove Co-Authored-By lines and the preceding blank line\n    return content.replace(/(\\r?\\n){2}Co-Authored-By:.*$/gim, '');\n  }\n  if (attribution === undefined) {\n    return content;\n  }\n  // Replace with custom attribution (escape $ to prevent backreference injection)\n  const safeAttribution = attribution.replace(/\\$/g, '$$$$');\n  return content.replace(/Co-Authored-By:.*$/gim, `Co-Authored-By: ${safeAttribution}`);\n}\n\n/**\n * Convert Claude Code frontmatter to opencode format\n * - Converts 'allowed-tools:' array to 'permission:' object\n * @param {string} content - Markdown file content with YAML frontmatter\n * @returns {string} - Content with converted frontmatter\n */\n// Color name to hex mapping for opencode compatibility\nconst colorNameToHex = {\n  cyan: '#00FFFF',\n  red: '#FF0000',\n  green: '#00FF00',\n  blue: '#0000FF',\n  yellow: '#FFFF00',\n  magenta: '#FF00FF',\n  orange: '#FFA500',\n  purple: '#800080',\n  pink: '#FFC0CB',\n  white: '#FFFFFF',\n  black: '#000000',\n  gray: '#808080',\n  grey: '#808080',\n};\n\n// Tool name mapping from Claude Code to OpenCode\n// OpenCode uses lowercase tool names; special mappings for renamed tools\nconst claudeToOpencodeTools = {\n  AskUserQuestion: 'question',\n  SlashCommand: 'skill',\n  TodoWrite: 'todowrite',\n  WebFetch: 'webfetch',\n  WebSearch: 'websearch',  // Plugin/MCP - keep for compatibility\n};\n\n// Tool name mapping from Claude Code to Gemini CLI\n// Gemini CLI uses snake_case built-in tool names\nconst claudeToGeminiTools = {\n  Read: 'read_file',\n  Write: 'write_file',\n  Edit: 'replace',\n  Bash: 'run_shell_command',\n  Glob: 'glob',\n  Grep: 'search_file_content',\n  WebSearch: 'google_web_search',\n  WebFetch: 'web_fetch',\n  TodoWrite: 'write_todos',\n  AskUserQuestion: 'ask_user',\n};\n\n/**\n * Convert a Claude Code tool name to OpenCode format\n * - Applies special mappings (AskUserQuestion -> question, etc.)\n * - Converts to lowercase (except MCP tools which keep their format)\n */\nfunction convertToolName(claudeTool) {\n  // Check for special mapping first\n  if (claudeToOpencodeTools[claudeTool]) {\n    return claudeToOpencodeTools[claudeTool];\n  }\n  // MCP tools (mcp__*) keep their format\n  if (claudeTool.startsWith('mcp__')) {\n    return claudeTool;\n  }\n  // Default: convert to lowercase\n  return claudeTool.toLowerCase();\n}\n\n/**\n * Convert a Claude Code tool name to Gemini CLI format\n * - Applies Claude\u2192Gemini mapping (Read\u2192read_file, Bash\u2192run_shell_command, etc.)\n * - Filters out MCP tools (mcp__*) \u2014 they are auto-discovered at runtime in Gemini\n * - Filters out Task \u2014 agents are auto-registered as tools in Gemini\n * @returns {string|null} Gemini tool name, or null if tool should be excluded\n */\nfunction convertGeminiToolName(claudeTool) {\n  // MCP tools: exclude \u2014 auto-discovered from mcpServers config at runtime\n  if (claudeTool.startsWith('mcp__')) {\n    return null;\n  }\n  // Task: exclude \u2014 agents are auto-registered as callable tools\n  if (claudeTool === 'Task') {\n    return null;\n  }\n  // Check for explicit mapping\n  if (claudeToGeminiTools[claudeTool]) {\n    return claudeToGeminiTools[claudeTool];\n  }\n  // Default: lowercase\n  return claudeTool.toLowerCase();\n}\n\n/**\n * Convert a Claude Code tool name to GitHub Copilot format.\n * - Applies explicit mapping from claudeToCopilotTools\n * - Handles mcp__context7__* prefix \u2192 io.github.upstash/context7/*\n * - Falls back to lowercase for unknown tools\n */\nfunction convertCopilotToolName(claudeTool) {\n  // mcp__context7__* wildcard \u2192 io.github.upstash/context7/*\n  if (claudeTool.startsWith('mcp__context7__')) {\n    return 'io.github.upstash/context7/' + claudeTool.slice('mcp__context7__'.length);\n  }\n  // Check explicit mapping\n  if (claudeToCopilotTools[claudeTool]) {\n    return claudeToCopilotTools[claudeTool];\n  }\n  // Default: lowercase\n  return claudeTool.toLowerCase();\n}\n\n/**\n * Apply Copilot-specific content conversion \u2014 CONV-06 (paths) + CONV-07 (command names).\n * Path mappings depend on install mode:\n *   Global: ~/.claude/ \u2192 ~/.copilot/, ./.claude/ \u2192 ./.github/\n *   Local:  ~/.claude/ \u2192 ./.github/, ./.claude/ \u2192 ./.github/\n * Applied to ALL Copilot content (skills, agents, engine files).\n * @param {string} content - Source content to convert\n * @param {boolean} [isGlobal=false] - Whether this is a global install\n */\nfunction convertClaudeToCopilotContent(content, isGlobal = false) {\n  let c = content;\n  // CONV-06: Path replacement \u2014 most specific first to avoid substring matches\n  if (isGlobal) {\n    c = c.replace(/\\$HOME\\/\\.claude\\//g, '$HOME/.copilot/');\n    c = c.replace(/~\\/\\.claude\\//g, '~/.copilot/');\n  } else {\n    c = c.replace(/\\$HOME\\/\\.claude\\//g, '.github/');\n    c = c.replace(/~\\/\\.claude\\//g, '.github/');\n  }\n  c = c.replace(/\\.\\/\\.claude\\//g, './.github/');\n  c = c.replace(/\\.claude\\//g, '.github/');\n  // CONV-07: Command name conversion (all ez: references \u2192 ez-)\n  c = c.replace(/ez:/g, 'ez-');\n  return c;\n}\n\n/**\n * Convert a Claude command (.md) to a Copilot skill (SKILL.md).\n * Transforms frontmatter only \u2014 body passes through with CONV-06/07 applied.\n * Skills keep original tool names (no mapping) per CONTEXT.md decision.\n */\nfunction convertClaudeCommandToCopilotSkill(content, skillName, isGlobal = false) {\n  const converted = convertClaudeToCopilotContent(content, isGlobal);\n  const { frontmatter, body } = extractFrontmatterAndBody(converted);\n  if (!frontmatter) return converted;\n\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\n  const argumentHint = extractFrontmatterField(frontmatter, 'argument-hint');\n  const agent = extractFrontmatterField(frontmatter, 'agent');\n\n  // CONV-02: Extract allowed-tools YAML multiline list \u2192 comma-separated string\n  const toolsMatch = frontmatter.match(/^allowed-tools:\\s*\\n((?:\\s+-\\s+.+\\n?)*)/m);\n  let toolsLine = '';\n  if (toolsMatch) {\n    const tools = toolsMatch[1].match(/^\\s+-\\s+(.+)/gm);\n    if (tools) {\n      toolsLine = tools.map(t => t.replace(/^\\s+-\\s+/, '').trim()).join(', ');\n    }\n  }\n\n  // Reconstruct frontmatter in Copilot format\n  let fm = `---\\nname: ${skillName}\\ndescription: ${description}\\n`;\n  if (argumentHint) fm += `argument-hint: ${yamlQuote(argumentHint)}\\n`;\n  if (agent) fm += `agent: ${agent}\\n`;\n  if (toolsLine) fm += `allowed-tools: ${toolsLine}\\n`;\n  fm += '---';\n\n  return `${fm}\\n${body}`;\n}\n\n/**\n * Convert a Claude agent (.md) to a Copilot agent (.agent.md).\n * Applies tool mapping + deduplication, formats tools as JSON array.\n * CONV-04: JSON array format. CONV-05: Tool name mapping.\n */\nfunction convertClaudeAgentToCopilotAgent(content, isGlobal = false) {\n  const converted = convertClaudeToCopilotContent(content, isGlobal);\n  const { frontmatter, body } = extractFrontmatterAndBody(converted);\n  if (!frontmatter) return converted;\n\n  const name = extractFrontmatterField(frontmatter, 'name') || 'unknown';\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\n  const color = extractFrontmatterField(frontmatter, 'color');\n  const toolsRaw = extractFrontmatterField(frontmatter, 'tools') || '';\n\n  // CONV-04 + CONV-05: Map tools, deduplicate, format as JSON array\n  const claudeTools = toolsRaw.split(',').map(t => t.trim()).filter(Boolean);\n  const mappedTools = claudeTools.map(t => convertCopilotToolName(t));\n  const uniqueTools = [...new Set(mappedTools)];\n  const toolsArray = uniqueTools.length > 0\n    ? \"['\" + uniqueTools.join(\"', '\") + \"']\"\n    : '[]';\n\n  // Reconstruct frontmatter in Copilot format\n  let fm = `---\\nname: ${name}\\ndescription: ${description}\\ntools: ${toolsArray}\\n`;\n  if (color) fm += `color: ${color}\\n`;\n  fm += '---';\n\n  return `${fm}\\n${body}`;\n}\n\nfunction toSingleLine(value) {\n  return value.replace(/\\s+/g, ' ').trim();\n}\n\nfunction yamlQuote(value) {\n  return JSON.stringify(value);\n}\n\nfunction extractFrontmatterAndBody(content) {\n  if (!content.startsWith('---')) {\n    return { frontmatter: null, body: content };\n  }\n\n  const endIndex = content.indexOf('---', 3);\n  if (endIndex === -1) {\n    return { frontmatter: null, body: content };\n  }\n\n  return {\n    frontmatter: content.substring(3, endIndex).trim(),\n    body: content.substring(endIndex + 3),\n  };\n}\n\nfunction extractFrontmatterField(frontmatter, fieldName) {\n  const regex = new RegExp(`^${fieldName}:\\\\s*(.+)$`, 'm');\n  const match = frontmatter.match(regex);\n  if (!match) return null;\n  return match[1].trim().replace(/^['\"]|['\"]$/g, '');\n}\n\nfunction convertSlashCommandsToCodexSkillMentions(content) {\n  let converted = content.replace(/\\/ez:([a-z0-9-]+)/gi, (_, commandName) => {\n    return `$ez-${String(commandName).toLowerCase()}`;\n  });\n  converted = converted.replace(/\\/ez-help\\b/g, '$ez-help');\n  return converted;\n}\n\nfunction convertClaudeToCodexMarkdown(content) {\n  let converted = convertSlashCommandsToCodexSkillMentions(content);\n  converted = converted.replace(/\\$ARGUMENTS\\b/g, '{{EZ_ARGS}}');\n  return converted;\n}\n\nfunction getCodexSkillAdapterHeader(skillName) {\n  const invocation = `$${skillName}`;\n  return `<codex_skill_adapter>\n## A. Skill Invocation\n- This skill is invoked by mentioning \\`${invocation}\\`.\n- Treat all user text after \\`${invocation}\\` as \\`{{EZ_ARGS}}\\`.\n- If no arguments are present, treat \\`{{EZ_ARGS}}\\` as empty.\n\n## B. AskUserQuestion \u2192 request_user_input Mapping\nEZ_Agents workflows use \\`AskUserQuestion\\` (Claude Code syntax). Translate to Codex \\`request_user_input\\`:\n\nParameter mapping:\n- \\`header\\` \u2192 \\`header\\`\n- \\`question\\` \u2192 \\`question\\`\n- Options formatted as \\`\"Label\" \u2014 description\\` \u2192 \\`{label: \"Label\", description: \"description\"}\\`\n- Generate \\`id\\` from header: lowercase, replace spaces with underscores\n\nBatched calls:\n- \\`AskUserQuestion([q1, q2])\\` \u2192 single \\`request_user_input\\` with multiple entries in \\`questions[]\\`\n\nMulti-select workaround:\n- Codex has no \\`multiSelect\\`. Use sequential single-selects, or present a numbered freeform list asking the user to enter comma-separated numbers.\n\nExecute mode fallback:\n- When \\`request_user_input\\` is rejected (Execute mode), present a plain-text numbered list and pick a reasonable default.\n\n## C. Task() \u2192 spawn_agent Mapping\nEZ_Agents workflows use \\`Task(...)\\` (Claude Code syntax). Translate to Codex collaboration tools:\n\nDirect mapping:\n- \\`Task(subagent_type=\"X\", prompt=\"Y\")\\` \u2192 \\`spawn_agent(agent_type=\"X\", message=\"Y\")\\`\n- \\`Task(model=\"...\")\\` \u2192 omit (Codex uses per-role config, not inline model selection)\n- \\`fork_context: false\\` by default \u2014 EZ_Agents agents load their own context via \\`<files_to_read>\\` blocks\n\nParallel fan-out:\n- Spawn multiple agents \u2192 collect agent IDs \u2192 \\`wait(ids)\\` for all to complete\n\nResult parsing:\n- Look for structured markers in agent output: \\`CHECKPOINT\\`, \\`PLAN COMPLETE\\`, \\`SUMMARY\\`, etc.\n- \\`close_agent(id)\\` after collecting results from each agent\n</codex_skill_adapter>`;\n}\n\nfunction convertClaudeCommandToCodexSkill(content, skillName) {\n  const converted = convertClaudeToCodexMarkdown(content);\n  const { frontmatter, body } = extractFrontmatterAndBody(converted);\n  let description = `Run EZ_Agents workflow ${skillName}.`;\n  if (frontmatter) {\n    const maybeDescription = extractFrontmatterField(frontmatter, 'description');\n    if (maybeDescription) {\n      description = maybeDescription;\n    }\n  }\n  description = toSingleLine(description);\n  const shortDescription = description.length > 180 ? `${description.slice(0, 177)}...` : description;\n  const adapter = getCodexSkillAdapterHeader(skillName);\n\n  return `---\\nname: ${yamlQuote(skillName)}\\ndescription: ${yamlQuote(description)}\\nmetadata:\\n  short-description: ${yamlQuote(shortDescription)}\\n---\\n\\n${adapter}\\n\\n${body.trimStart()}`;\n}\n\nfunction convertClaudeToQwenAgent(content) {\n  const { frontmatter, body } = extractFrontmatterAndBody(content);\n  if (!frontmatter) return content;\n\n  const name = extractFrontmatterField(frontmatter, 'name') || 'unknown';\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\n\n  // Qwen Code uses simple markdown. We'll put name and description in the body.\n  return `# Agent: ${name}\\n\\n${description}\\n\\n${body.trimStart()}`;\n}\n\nfunction convertClaudeToKimiSkill(content, skillName) {\n  const { frontmatter, body } = extractFrontmatterAndBody(content);\n  if (!frontmatter) return content;\n\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\n\n  // Reconstruct as simple Markdown\n  return `# Skill: ${skillName}\\n\\n${description}\\n\\n${body.trimStart()}`;\n}\n\nfunction convertClaudeToKimiAgent(content) {\n  const { frontmatter, body } = extractFrontmatterAndBody(content);\n  if (!frontmatter) return content;\n\n  const name = extractFrontmatterField(frontmatter, 'name') || 'unknown';\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\n\n  // Kimi Code uses simple markdown\n  return `# Agent: ${name}\\n\\n${description}\\n\\n${body.trimStart()}`;\n}\n\n/**\n * Convert a Claude command (.md) to a Qwen skill (SKILL.md).\n * Strips frontmatter and formats as simple Markdown.\n */\nfunction convertClaudeToQwenSkill(content, skillName) {\n  const { frontmatter, body } = extractFrontmatterAndBody(content);\n  if (!frontmatter) return content;\n\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\n\n  // Reconstruct as simple Markdown\n  return `# Skill: ${skillName}\\n\\n${description}\\n\\n${body.trimStart()}`;\n}\n\n/**\n * Convert Claude Code agent markdown to Codex agent format.\n * Applies base markdown conversions, then adds a <codex_agent_role> header\n * and cleans up frontmatter (removes tools/color fields).\n */\nfunction convertClaudeAgentToCodexAgent(content) {\n  let converted = convertClaudeToCodexMarkdown(content);\n\n  const { frontmatter, body } = extractFrontmatterAndBody(converted);\n  if (!frontmatter) return converted;\n\n  const name = extractFrontmatterField(frontmatter, 'name') || 'unknown';\n  const description = extractFrontmatterField(frontmatter, 'description') || '';\n  const tools = extractFrontmatterField(frontmatter, 'tools') || '';\n\n  const roleHeader = `<codex_agent_role>\nrole: ${name}\ntools: ${tools}\npurpose: ${toSingleLine(description)}\n</codex_agent_role>`;\n\n  const cleanFrontmatter = `---\\nname: ${yamlQuote(name)}\\ndescription: ${yamlQuote(toSingleLine(description))}\\n---`;\n\n  return `${cleanFrontmatter}\\n\\n${roleHeader}\\n${body}`;\n}\n\n/**\n * Generate a per-agent .toml config file for Codex.\n * Sets sandbox_mode and developer_instructions from the agent markdown body.\n */\nfunction generateCodexAgentToml(agentName, agentContent) {\n  const sandboxMode = CODEX_AGENT_SANDBOX[agentName] || 'read-only';\n  const { body } = extractFrontmatterAndBody(agentContent);\n  const instructions = body.trim();\n\n  const lines = [\n    `sandbox_mode = \"${sandboxMode}\"`,\n    // Agent prompts contain raw backslashes in regexes and shell snippets.\n    // TOML literal multiline strings preserve them without escape parsing.\n    `developer_instructions = '''`,\n    instructions,\n    `'''`,\n  ];\n  return lines.join('\\n') + '\\n';\n}\n\n/**\n * Generate the EZ_Agents config block for Codex config.toml.\n * @param {Array<{name: string, description: string}>} agents\n */\nfunction generateCodexConfigBlock(agents) {\n  const lines = [\n    EZ_CODEX_MARKER,\n    '',\n  ];\n\n  for (const { name, description } of agents) {\n    lines.push(`[agents.${name}]`);\n    lines.push(`description = ${JSON.stringify(description)}`);\n    lines.push(`config_file = \"agents/${name}.toml\"`);\n    lines.push('');\n  }\n\n  return lines.join('\\n');\n}\n\n/**\n * Strip EZ_Agents sections from Codex config.toml content.\n * Returns cleaned content, or null if file would be empty.\n */\nfunction stripEzAgentsFromCodexConfig(content) {\n  const markerIndex = content.indexOf(EZ_CODEX_MARKER);\n\n  if (markerIndex !== -1) {\n    // Has EZ_Agents marker \u2014 remove everything from marker to EOF\n    let before = content.substring(0, markerIndex).trimEnd();\n    // Also strip EZ-injected feature keys above the marker (Case 3 inject)\n    before = before.replace(/^multi_agent\\s*=\\s*true\\s*\\n?/m, '');\n    before = before.replace(/^default_mode_request_user_input\\s*=\\s*true\\s*\\n?/m, '');\n    before = before.replace(/^\\[features\\]\\s*\\n(?=\\[|$)/m, '');\n    before = before.replace(/\\n{3,}/g, '\\n\\n').trim();\n    if (!before) return null;\n    return before + '\\n';\n  }\n\n  // No marker but may have EZ-injected feature keys\n  let cleaned = content;\n  cleaned = cleaned.replace(/^multi_agent\\s*=\\s*true\\s*\\n?/m, '');\n  cleaned = cleaned.replace(/^default_mode_request_user_input\\s*=\\s*true\\s*\\n?/m, '');\n\n  // Remove [agents.ez-*] sections (from header to next section or EOF)\n  cleaned = cleaned.replace(/^\\[agents\\.ez-[^\\]]+\\]\\n(?:(?!\\[)[^\\n]*\\n?)*/gm, '');\n\n  // Remove [features] section if now empty (only header, no keys before next section)\n  cleaned = cleaned.replace(/^\\[features\\]\\s*\\n(?=\\[|$)/m, '');\n\n  // Remove [agents] section if now empty\n  cleaned = cleaned.replace(/^\\[agents\\]\\s*\\n(?=\\[|$)/m, '');\n\n  // Clean up excessive blank lines\n  cleaned = cleaned.replace(/\\n{3,}/g, '\\n\\n').trim();\n\n  if (!cleaned) return null;\n  return cleaned + '\\n';\n}\n\n/**\n * Merge EZ_Agents config block into an existing or new config.toml.\n * Three cases: new file, existing with EZ_Agents marker, existing without marker.\n */\nfunction mergeCodexConfig(configPath, EZ_AgentsBlock) {\n  // Case 1: No config.toml \u2014 create fresh\n  if (!fs.existsSync(configPath)) {\n    fs.writeFileSync(configPath, EZ_AgentsBlock + '\\n');\n    return;\n  }\n\n  const existing = fs.readFileSync(configPath, 'utf8');\n  const markerIndex = existing.indexOf(EZ_CODEX_MARKER);\n\n  // Case 2: Has EZ_Agents marker \u2014 truncate and re-append\n  if (markerIndex !== -1) {\n    let before = existing.substring(0, markerIndex).trimEnd();\n    if (before) {\n      // Strip any EZ-managed sections that leaked above the marker from previous installs\n      before = before.replace(/^\\[agents\\.ez-[^\\]]+\\]\\n(?:(?!\\[)[^\\n]*\\n?)*/gm, '');\n      before = before.replace(/^\\[agents\\]\\n(?:(?!\\[)[^\\n]*\\n?)*/m, '');\n      before = before.replace(/\\n{3,}/g, '\\n\\n').trimEnd();\n\n      fs.writeFileSync(configPath, before + '\\n\\n' + EZ_AgentsBlock + '\\n');\n    } else {\n      fs.writeFileSync(configPath, EZ_AgentsBlock + '\\n');\n    }\n    return;\n  }\n\n  // Case 3: No marker \u2014 append EZ_Agents block\n  let content = existing;\n  content = content.trimEnd() + '\\n\\n' + EZ_AgentsBlock + '\\n';\n\n  fs.writeFileSync(configPath, content);\n}\n\nfunction findCopilotInstructionsMarkerPair(content: string): { openIndex: number; closeIndex: number; closeMarker: string } | null {\n  const markerPairs: [string, string][] = [\n    [EZ_COPILOT_INSTRUCTIONS_MARKER, EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER],\n    [LEGACY_EZ_COPILOT_INSTRUCTIONS_MARKER, LEGACY_EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER],\n  ];\n\n  for (const [openMarker, closeMarker] of markerPairs) {\n    const openIndex = content.indexOf(openMarker);\n    const closeIndex = content.indexOf(closeMarker);\n    if (openIndex !== -1 && closeIndex !== -1 && closeIndex >= openIndex) {\n      return { openIndex, closeIndex, closeMarker };\n    }\n  }\n\n  return null;\n}\n\n/**\n * Merge EZ_Agents instructions into copilot-instructions.md.\n * Three cases: new file, existing with markers, existing without markers.\n * @param {string} filePath - Full path to copilot-instructions.md\n * @param {string} EZ_AgentsContent - Template content (without markers)\n */\nfunction mergeCopilotInstructions(filePath, EZ_AgentsContent) {\n  const EZ_AgentsBlock = EZ_COPILOT_INSTRUCTIONS_MARKER + '\\n' +\n    EZ_AgentsContent.trim() + '\\n' +\n    EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER;\n\n  // Case 1: No file \u2014 create fresh\n  if (!fs.existsSync(filePath)) {\n    fs.writeFileSync(filePath, EZ_AgentsBlock + '\\n');\n    return;\n  }\n\n  const existing = fs.readFileSync(filePath, 'utf8');\n  const markerMatch = findCopilotInstructionsMarkerPair(existing);\n\n  // Case 2: Has EZ_Agents markers \u2014 replace between markers\n  if (markerMatch) {\n    const before = existing.substring(0, markerMatch.openIndex).trimEnd();\n    const after = existing.substring(markerMatch.closeIndex + markerMatch.closeMarker.length).trimStart();\n    let newContent = '';\n    if (before) newContent += before + '\\n\\n';\n    newContent += EZ_AgentsBlock;\n    if (after) newContent += '\\n\\n' + after;\n    newContent += '\\n';\n    fs.writeFileSync(filePath, newContent);\n    return;\n  }\n\n  // Case 3: No markers \u2014 append at end\n  const content = existing.trimEnd() + '\\n\\n' + EZ_AgentsBlock + '\\n';\n  fs.writeFileSync(filePath, content);\n}\n\n/**\n * Strip EZ_Agents section from copilot-instructions.md content.\n * Returns cleaned content, or null if file should be deleted (was EZ-only).\n * @param {string} content - File content\n * @returns {string|null} - Cleaned content or null if empty\n */\nfunction stripEzAgentsFromCopilotInstructions(content) {\n  const markerMatch = findCopilotInstructionsMarkerPair(content);\n\n  if (markerMatch) {\n    const before = content.substring(0, markerMatch.openIndex).trimEnd();\n    const after = content.substring(markerMatch.closeIndex + markerMatch.closeMarker.length).trimStart();\n    const cleaned = (before + (before && after ? '\\n\\n' : '') + after).trim();\n    if (!cleaned) return null;\n    return cleaned + '\\n';\n  }\n\n  // No markers found \u2014 nothing to strip\n  return content;\n}\n\n/**\n * Generate config.toml and per-agent .toml files for Codex.\n * Reads agent .md files from source, extracts metadata, writes .toml configs.\n */\nfunction installCodexConfig(targetDir: string, agentsSrc: string): number {\n  const configPath = path.join(targetDir, 'config.toml');\n  const agentsTomlDir = path.join(targetDir, 'agents');\n  fs.mkdirSync(agentsTomlDir, { recursive: true });\n\n  const agentEntries = fs.readdirSync(agentsSrc).filter(f => f.startsWith('ez-') && f.endsWith('.md'));\n  const agents: Array<{ name: string; description: string }> = [];\n\n  // Compute the Codex pathPrefix for replacing .claude paths\n  // Use tilde-based path to avoid baking absolute paths into templates\n  const codexPathPrefix = `${targetDir.replace(/\\\\/g, '/').replace(os.homedir().replace(/\\\\/g, '/'), '~')}/`;\n\n  for (const file of agentEntries) {\n    let content = fs.readFileSync(path.join(agentsSrc, file), 'utf8');\n    // Replace .claude paths before generating TOML (source files use ~/.claude and $HOME/.claude)\n    content = content.replace(/~\\/\\.claude\\//g, codexPathPrefix);\n    content = content.replace(/\\$HOME\\/\\.claude\\//g, toHomePrefix(codexPathPrefix));\n    const { frontmatter } = extractFrontmatterAndBody(content);\n    const name = extractFrontmatterField(frontmatter, 'name') || file.replace('.md', '');\n    const description = extractFrontmatterField(frontmatter, 'description') || '';\n\n    agents.push({ name, description: toSingleLine(description) });\n\n    const tomlContent = generateCodexAgentToml(name, content);\n    fs.writeFileSync(path.join(agentsTomlDir, `${name}.toml`), tomlContent);\n  }\n\n  const EZ_AgentsBlock = generateCodexConfigBlock(agents);\n  mergeCodexConfig(configPath, EZ_AgentsBlock);\n\n  return agents.length;\n}\n\n/**\n * Strip HTML <sub> tags for Gemini CLI output\n * Terminals don't support subscript \u2014 Gemini renders these as raw HTML.\n * Converts <sub>text</sub> to italic *(text)* for readable terminal output.\n */\nfunction stripSubTags(content) {\n  return content.replace(/<sub>(.*?)<\\/sub>/g, '*($1)*');\n}\n\n/**\n * Convert Claude Code agent frontmatter to Gemini CLI format\n * Gemini agents use .md files with YAML frontmatter, same as Claude,\n * but with different field names and formats:\n * - tools: must be a YAML array (not comma-separated string)\n * - tool names: must use Gemini built-in names (read_file, not Read)\n * - color: must be removed (causes validation error)\n * - skills: must be removed (causes validation error)\n * - mcp__* tools: must be excluded (auto-discovered at runtime)\n */\nfunction convertClaudeToGeminiAgent(content: string): string {\n  if (!content.startsWith('---')) return content;\n\n  const endIndex = content.indexOf('---', 3);\n  if (endIndex === -1) return content;\n\n  const frontmatter = content.substring(3, endIndex).trim();\n  const body = content.substring(endIndex + 3);\n\n  const lines = frontmatter.split('\\n');\n  const newLines: string[] = [];\n  let inAllowedTools = false;\n  let inSkippedArrayField = false;\n  const tools: string[] = [];\n\n  for (const line of lines) {\n    const trimmed = line.trim();\n\n    if (inSkippedArrayField) {\n      if (!trimmed || trimmed.startsWith('- ')) {\n        continue;\n      }\n      inSkippedArrayField = false;\n    }\n\n    // Convert allowed-tools YAML array to tools list\n    if (trimmed.startsWith('allowed-tools:')) {\n      inAllowedTools = true;\n      continue;\n    }\n\n    // Handle inline tools: field (comma-separated string)\n    if (trimmed.startsWith('tools:')) {\n      const toolsValue = trimmed.substring(6).trim();\n      if (toolsValue) {\n        const parsed = toolsValue.split(',').map(t => t.trim()).filter(t => t);\n        for (const t of parsed) {\n          const mapped = convertGeminiToolName(t);\n          if (mapped) tools.push(mapped);\n        }\n      } else {\n        // tools: with no value means YAML array follows\n        inAllowedTools = true;\n      }\n      continue;\n    }\n\n    // Strip color field (not supported by Gemini CLI, causes validation error)\n    if (trimmed.startsWith('color:')) continue;\n\n    // Strip skills field (not supported by Gemini CLI, causes validation error)\n    if (trimmed.startsWith('skills:')) {\n      inSkippedArrayField = true;\n      continue;\n    }\n\n    // Collect allowed-tools/tools array items\n    if (inAllowedTools) {\n      if (trimmed.startsWith('- ')) {\n        const mapped = convertGeminiToolName(trimmed.substring(2).trim());\n        if (mapped) tools.push(mapped);\n        continue;\n      } else if (trimmed && !trimmed.startsWith('-')) {\n        inAllowedTools = false;\n      }\n    }\n\n    if (!inAllowedTools) {\n      newLines.push(line);\n    }\n  }\n\n  // Add tools as YAML array (Gemini requires array format)\n  if (tools.length > 0) {\n    newLines.push('tools:');\n    for (const tool of tools) {\n      newLines.push(`  - ${tool}`);\n    }\n  }\n\n  const newFrontmatter = newLines.join('\\n').trim();\n\n  // Escape ${VAR} patterns in agent body for Gemini CLI compatibility.\n  // Gemini's templateString() treats all ${word} patterns as template variables\n  // and throws \"Template validation failed: Missing required input parameters\"\n  // when they can't be resolved. EZ_Agents agents use ${PHASE}, ${PLAN}, etc. as\n  // shell variables in bash code blocks \u2014 convert to $VAR (no braces) which\n  // is equivalent bash and invisible to Gemini's /\\$\\{(\\w+)\\}/g regex.\n  const escapedBody = body.replace(/\\$\\{(\\w+)\\}/g, '$$$1');\n\n  return `---\\n${newFrontmatter}\\n---${stripSubTags(escapedBody)}`;\n}\n\nfunction convertClaudeToOpencodeFrontmatter(content: string): string {\n  // Replace tool name references in content (applies to all files)\n  let convertedContent = content;\n  convertedContent = convertedContent.replace(/\\bAskUserQuestion\\b/g, 'question');\n  convertedContent = convertedContent.replace(/\\bSlashCommand\\b/g, 'skill');\n  convertedContent = convertedContent.replace(/\\bTodoWrite\\b/g, 'todowrite');\n  // Replace /ez:command with /ez-command for opencode (flat command structure)\n  convertedContent = convertedContent.replace(/\\/ez:/g, '/ez-');\n  // Replace ~/.claude and $HOME/.claude with OpenCode's config location\n  convertedContent = convertedContent.replace(/~\\/\\.claude\\b/g, '~/.config/opencode');\n  convertedContent = convertedContent.replace(/\\$HOME\\/\\.claude\\b/g, '$HOME/.config/opencode');\n  // Replace general-purpose subagent type with OpenCode's equivalent \"general\"\n  convertedContent = convertedContent.replace(/subagent_type=\"general-purpose\"/g, 'subagent_type=\"general\"');\n\n  // Check if content has frontmatter\n  if (!convertedContent.startsWith('---')) {\n    return convertedContent;\n  }\n\n  // Find the end of frontmatter\n  const endIndex = convertedContent.indexOf('---', 3);\n  if (endIndex === -1) {\n    return convertedContent;\n  }\n\n  const frontmatter = convertedContent.substring(3, endIndex).trim();\n  const body = convertedContent.substring(endIndex + 3);\n\n  // Parse frontmatter line by line (simple YAML parsing)\n  const lines = frontmatter.split('\\n');\n  const newLines: string[] = [];\n  let inAllowedTools = false;\n  const allowedTools: string[] = [];\n\n  for (const line of lines) {\n    const trimmed = line.trim();\n\n    // Detect start of allowed-tools array\n    if (trimmed.startsWith('allowed-tools:')) {\n      inAllowedTools = true;\n      continue;\n    }\n\n    // Detect inline tools: field (comma-separated string)\n    if (trimmed.startsWith('tools:')) {\n      const toolsValue = trimmed.substring(6).trim();\n      if (toolsValue) {\n        // Parse comma-separated tools\n        const tools = toolsValue.split(',').map(t => t.trim()).filter(t => t);\n        allowedTools.push(...tools);\n      }\n      continue;\n    }\n\n    // Remove name: field - opencode uses filename for command name\n    if (trimmed.startsWith('name:')) {\n      continue;\n    }\n\n    // Convert color names to hex for opencode\n    if (trimmed.startsWith('color:')) {\n      const colorValue = trimmed.substring(6).trim().toLowerCase();\n      const hexColor = colorNameToHex[colorValue];\n      if (hexColor) {\n        newLines.push(`color: \"${hexColor}\"`);\n      } else if (colorValue.startsWith('#')) {\n        // Validate hex color format (#RGB or #RRGGBB)\n        if (/^#[0-9a-f]{3}$|^#[0-9a-f]{6}$/i.test(colorValue)) {\n          // Already hex and valid, keep as is\n          newLines.push(line);\n        }\n        // Skip invalid hex colors\n      }\n      // Skip unknown color names\n      continue;\n    }\n\n    // Collect allowed-tools items\n    if (inAllowedTools) {\n      if (trimmed.startsWith('- ')) {\n        allowedTools.push(trimmed.substring(2).trim());\n        continue;\n      } else if (trimmed && !trimmed.startsWith('-')) {\n        // End of array, new field started\n        inAllowedTools = false;\n      }\n    }\n\n    // Keep other fields (including name: which opencode ignores)\n    if (!inAllowedTools) {\n      newLines.push(line);\n    }\n  }\n\n  // Add tools object if we had allowed-tools or tools\n  if (allowedTools.length > 0) {\n    newLines.push('tools:');\n    for (const tool of allowedTools) {\n      newLines.push(`  ${convertToolName(tool)}: true`);\n    }\n  }\n\n  // Rebuild frontmatter (body already has tool names converted)\n  const newFrontmatter = newLines.join('\\n').trim();\n  return `---\\n${newFrontmatter}\\n---${body}`;\n}\n\n/**\n * Convert Claude Code markdown command to Gemini TOML format\n * @param {string} content - Markdown file content with YAML frontmatter\n * @returns {string} - TOML content\n */\nfunction convertClaudeToGeminiToml(content) {\n  // Check if content has frontmatter\n  if (!content.startsWith('---')) {\n    return `prompt = ${JSON.stringify(content)}\\n`;\n  }\n\n  const endIndex = content.indexOf('---', 3);\n  if (endIndex === -1) {\n    return `prompt = ${JSON.stringify(content)}\\n`;\n  }\n\n  const frontmatter = content.substring(3, endIndex).trim();\n  const body = content.substring(endIndex + 3).trim();\n  \n  // Extract description from frontmatter\n  let description = '';\n  const lines = frontmatter.split('\\n');\n  for (const line of lines) {\n    const trimmed = line.trim();\n    if (trimmed.startsWith('description:')) {\n      description = trimmed.substring(12).trim();\n      break;\n    }\n  }\n\n  // Construct TOML\n  let toml = '';\n  if (description) {\n    toml += `description = ${JSON.stringify(description)}\\n`;\n  }\n  \n  toml += `prompt = ${JSON.stringify(body)}\\n`;\n  \n  return toml;\n}\n\n/**\n * Copy commands to a flat structure for OpenCode\n * OpenCode expects: command/ez-help.md (invoked as /ez-help)\n * Source structure: commands/ez/help.md\n *\n * @param {string} srcDir - Source directory (e.g., commands/ez/)\n * @param {string} destDir - Destination directory (e.g., command/)\n * @param {string} prefix - Prefix for filenames (e.g., 'ez')\n * @param {string} pathPrefix - Path prefix for file references\n * @param {string} runtime - Target runtime ('claude' or 'opencode')\n */\nfunction copyFlattenedCommands(srcDir, destDir, prefix, pathPrefix, runtime) {\n  if (!fs.existsSync(srcDir)) {\n    return;\n  }\n\n  // Remove old ez-*.md files before copying new ones\n  if (fs.existsSync(destDir)) {\n    for (const file of fs.readdirSync(destDir)) {\n      if (file.startsWith(`${prefix}-`) && file.endsWith('.md')) {\n        fs.unlinkSync(path.join(destDir, file));\n      }\n    }\n  } else {\n    fs.mkdirSync(destDir, { recursive: true });\n  }\n\n  const entries = fs.readdirSync(srcDir, { withFileTypes: true });\n\n  for (const entry of entries) {\n    const srcPath = path.join(srcDir, entry.name);\n\n    if (entry.isDirectory()) {\n      // Recurse into subdirectories, adding to prefix\n      // e.g., commands/ez/debug/start.md -> command/ez-debug-start.md\n      copyFlattenedCommands(srcPath, destDir, `${prefix}-${entry.name}`, pathPrefix, runtime);\n    } else if (entry.name.endsWith('.md')) {\n      // Flatten: help.md -> ez-help.md\n      const baseName = entry.name.replace('.md', '');\n      const destName = `${prefix}-${baseName}.md`;\n      const destPath = path.join(destDir, destName);\n\n      let content = fs.readFileSync(srcPath, 'utf8');\n      const globalClaudeRegex = /~\\/\\.claude\\//g;\n      const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\n      const localClaudeRegex = /\\.\\/\\.claude\\//g;\n      const opencodeDirRegex = /~\\/\\.opencode\\//g;\n      content = content.replace(globalClaudeRegex, pathPrefix);\n      content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\n      content = content.replace(localClaudeRegex, `./${getDirName(runtime)}/`);\n      content = content.replace(opencodeDirRegex, pathPrefix);\n      content = processAttribution(content, getCommitAttribution(runtime));\n      content = convertClaudeToOpencodeFrontmatter(content);\n\n      fs.writeFileSync(destPath, content);\n    }\n  }\n}\n\n/**\n * Copy EZ Agents skills to global skills directory\n * Skills are installed to $HOME/.skills/ez-agents/ for universal access\n * @param {string} srcDir - Source directory (ez-agents root)\n * @param {boolean} isGlobal - Whether this is a global install\n * @param {string} runtime - Runtime identifier\n */\nfunction copySkills(srcDir: string, isGlobal: boolean, runtime: string): void {\n  // Only install skills for global installs\n  if (!isGlobal) {\n    console.log(`  ${yellow}\u26A0${reset} Skipping skills (local install)`);\n    return;\n  }\n\n  const homeDir = os.homedir().replace(/\\\\/g, '/');\n  const skillsDestDir = path.posix.join(homeDir, '.skills', 'ez-agents');\n  const srcSkillsDir = path.join(srcDir, 'skills');\n\n  console.log(`  Installing skills to ${cyan}${skillsDestDir}${reset}...`);\n\n  // Create skills directory\n  fs.mkdirSync(skillsDestDir, { recursive: true });\n\n  // Copy all skills from source\n  if (fs.existsSync(srcSkillsDir)) {\n    fs.cpSync(srcSkillsDir, skillsDestDir, {\n      recursive: true,\n      force: true\n    });\n\n    // Count installed skills\n    let skillCount = 0;\n    const categories = ['stack', 'architecture', 'domain', 'operational', 'governance', 'testing', 'observability', 'devops', 'security', 'ai', 'data', 'product'];\n\n    for (const category of categories) {\n      const categoryPath = path.join(skillsDestDir, category);\n      if (fs.existsSync(categoryPath)) {\n        const skills = fs.readdirSync(categoryPath, { withFileTypes: true })\n          .filter(d => d.isDirectory()).length;\n        skillCount += skills;\n      }\n    }\n\n    console.log(`  ${green}\u2713${reset} Installed ${skillCount} skills to ${skillsDestDir}`);\n  } else {\n    console.log(`  ${yellow}\u26A0${reset} Skills directory not found: ${srcSkillsDir}`);\n  }\n}\n\nfunction listCodexSkillNames(skillsDir, prefix = 'ez-') {\n  if (!fs.existsSync(skillsDir)) return [];\n  const entries = fs.readdirSync(skillsDir, { withFileTypes: true });\n  return entries\n    .filter(entry => entry.isDirectory() && entry.name.startsWith(prefix))\n    .filter(entry => fs.existsSync(path.join(skillsDir, entry.name, 'SKILL.md')))\n    .map(entry => entry.name)\n    .sort();\n}\n\nfunction copyCommandsAsCodexSkills(srcDir, skillsDir, prefix, pathPrefix, runtime) {\n  if (!fs.existsSync(srcDir)) {\n    return;\n  }\n\n  fs.mkdirSync(skillsDir, { recursive: true });\n\n  // Remove previous EZ Codex skills to avoid stale command skills.\n  const existing = fs.readdirSync(skillsDir, { withFileTypes: true });\n  for (const entry of existing) {\n    if (entry.isDirectory() && entry.name.startsWith(`${prefix}-`)) {\n      fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\n    }\n  }\n\n  function recurse(currentSrcDir, currentPrefix) {\n    const entries = fs.readdirSync(currentSrcDir, { withFileTypes: true });\n\n    for (const entry of entries) {\n      const srcPath = path.join(currentSrcDir, entry.name);\n      if (entry.isDirectory()) {\n        recurse(srcPath, `${currentPrefix}-${entry.name}`);\n        continue;\n      }\n\n      if (!entry.name.endsWith('.md')) {\n        continue;\n      }\n\n      const baseName = entry.name.replace('.md', '');\n      const skillName = `${currentPrefix}-${baseName}`;\n      const skillDir = path.join(skillsDir, skillName);\n      fs.mkdirSync(skillDir, { recursive: true });\n\n      let content = fs.readFileSync(srcPath, 'utf8');\n      const globalClaudeRegex = /~\\/\\.claude\\//g;\n      const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\n      const localClaudeRegex = /\\.\\/\\.claude\\//g;\n      const codexDirRegex = /~\\/\\.codex\\//g;\n      content = content.replace(globalClaudeRegex, pathPrefix);\n      content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\n      content = content.replace(localClaudeRegex, `./${getDirName(runtime)}/`);\n      content = content.replace(codexDirRegex, pathPrefix);\n      content = processAttribution(content, getCommitAttribution(runtime));\n      content = convertClaudeCommandToCodexSkill(content, skillName);\n\n      fs.writeFileSync(path.join(skillDir, 'SKILL.md'), content);\n    }\n  }\n\n  recurse(srcDir, prefix);\n}\n\n/**\n * Copy Claude commands as Copilot skills \u2014 one folder per skill with SKILL.md.\n * Applies CONV-01 (structure), CONV-02 (allowed-tools), CONV-06 (paths), CONV-07 (command names).\n */\nfunction copyCommandsAsCopilotSkills(srcDir, skillsDir, prefix, isGlobal = false) {\n  if (!fs.existsSync(srcDir)) {\n    return;\n  }\n\n  fs.mkdirSync(skillsDir, { recursive: true });\n\n  // Remove previous EZ Copilot skills\n  const existing = fs.readdirSync(skillsDir, { withFileTypes: true });\n  for (const entry of existing) {\n    if (entry.isDirectory() && entry.name.startsWith(`${prefix}-`)) {\n      fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\n    }\n  }\n\n  function recurse(currentSrcDir, currentPrefix) {\n    const entries = fs.readdirSync(currentSrcDir, { withFileTypes: true });\n\n    for (const entry of entries) {\n      const srcPath = path.join(currentSrcDir, entry.name);\n      if (entry.isDirectory()) {\n        recurse(srcPath, `${currentPrefix}-${entry.name}`);\n        continue;\n      }\n\n      if (!entry.name.endsWith('.md')) {\n        continue;\n      }\n\n      const baseName = entry.name.replace('.md', '');\n      const skillName = `${currentPrefix}-${baseName}`;\n      const skillDir = path.join(skillsDir, skillName);\n      fs.mkdirSync(skillDir, { recursive: true });\n\n      let content = fs.readFileSync(srcPath, 'utf8');\n      content = convertClaudeCommandToCopilotSkill(content, skillName, isGlobal);\n      content = processAttribution(content, getCommitAttribution('copilot'));\n\n      fs.writeFileSync(path.join(skillDir, 'SKILL.md'), content);\n    }\n  }\n\n  recurse(srcDir, prefix);\n}\n\n/**\n * Copy Claude commands as Kimi Code skills \u2014 one folder per skill with SKILL.md.\n */\nfunction copyCommandsAsKimiSkills(srcDir, skillsDir, prefix, pathPrefix, runtime) {\n  if (!fs.existsSync(srcDir)) {\n    return;\n  }\n\n  fs.mkdirSync(skillsDir, { recursive: true });\n\n  // Remove previous EZ Kimi skills\n  const existing = fs.readdirSync(skillsDir, { withFileTypes: true });\n  for (const entry of existing) {\n    if (entry.isDirectory() && entry.name.startsWith(`${prefix}-`)) {\n      fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\n    }\n  }\n\n  function recurse(currentSrcDir, currentPrefix) {\n    const entries = fs.readdirSync(currentSrcDir, { withFileTypes: true });\n\n    for (const entry of entries) {\n      const srcPath = path.join(currentSrcDir, entry.name);\n      if (entry.isDirectory()) {\n        recurse(srcPath, `${currentPrefix}-${entry.name}`);\n        continue;\n      }\n\n      if (!entry.name.endsWith('.md')) {\n        continue;\n      }\n\n      const baseName = entry.name.replace('.md', '');\n      const skillName = `${currentPrefix}-${baseName}`;\n      const skillDir = path.join(skillsDir, skillName);\n      fs.mkdirSync(skillDir, { recursive: true });\n\n      let content = fs.readFileSync(srcPath, 'utf8');\n      const globalClaudeRegex = /~\\/\\.claude\\//g;\n      const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\n      const localClaudeRegex = /\\.\\/\\.claude\\//g;\n      const kimiDirRegex = /~\\/\\.kimi\\//g;\n      content = content.replace(globalClaudeRegex, pathPrefix);\n      content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\n      content = content.replace(localClaudeRegex, `./${getDirName(runtime)}/`);\n      content = content.replace(kimiDirRegex, pathPrefix);\n      content = processAttribution(content, getCommitAttribution(runtime));\n      content = convertClaudeToKimiSkill(content, skillName);\n      fs.writeFileSync(path.join(skillDir, 'SKILL.md'), content);\n    }\n  }\n\n  recurse(srcDir, prefix);\n}\n\n/**\n * Copy Claude commands as Qwen Code commands \u2014 flat structure in commands/ez/.\n * Qwen Code uses ~/.qwen/commands/ez/*.md (like Claude Code ~/.claude/commands/ez/)\n */\nfunction copyCommandsAsQwenCommands(srcDir, commandsDir, prefix, pathPrefix, runtime) {\n  if (!fs.existsSync(srcDir)) {\n    return;\n  }\n\n  fs.mkdirSync(commandsDir, { recursive: true });\n\n  // Remove previous EZ Qwen commands\n  const existing = fs.readdirSync(commandsDir, { withFileTypes: true });\n  for (const entry of existing) {\n    if (entry.isFile() && entry.name.startsWith(`${prefix}-`) && entry.name.endsWith('.md')) {\n      fs.rmSync(path.join(commandsDir, entry.name));\n    }\n  }\n\n  function recurse(currentSrcDir, currentPrefix) {\n    const entries = fs.readdirSync(currentSrcDir, { withFileTypes: true });\n\n    for (const entry of entries) {\n      const srcPath = path.join(currentSrcDir, entry.name);\n      if (entry.isDirectory()) {\n        recurse(srcPath, `${currentPrefix}-${entry.name}`);\n        continue;\n      }\n\n      if (!entry.name.endsWith('.md')) {\n        continue;\n      }\n\n      const baseName = entry.name.replace('.md', '');\n      const commandName = `${currentPrefix}-${baseName}`;\n      const commandFile = path.join(commandsDir, `${commandName}.md`);\n\n      let content = fs.readFileSync(srcPath, 'utf8');\n      const globalClaudeRegex = /~\\/\\.claude\\//g;\n      const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\n      const localClaudeRegex = /\\.\\/\\.claude\\//g;\n      const qwenDirRegex = /~\\/\\.qwen\\//g;\n      content = content.replace(globalClaudeRegex, pathPrefix);\n      content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\n      content = content.replace(localClaudeRegex, `./${getDirName(runtime)}/`);\n      content = content.replace(qwenDirRegex, pathPrefix);\n      content = processAttribution(content, getCommitAttribution(runtime));\n      \n      // Qwen Code uses simple markdown commands (no SKILL.md wrapper)\n      // Just copy the content as-is with path replacements\n      fs.writeFileSync(commandFile, content);\n    }\n  }\n\n  recurse(srcDir, prefix);\n}\n\n/**\n * Copy Claude commands as Qwen Code skills \u2014 one folder per skill with SKILL.md.\n * Qwen Code uses the same skills/ structure as Codex but with simpler format (no adapters needed).\n * @deprecated Use copyCommandsAsQwenCommands instead - Qwen Code uses commands/ez/ not skills/\n */\nfunction copyCommandsAsQwenSkills(srcDir, skillsDir, prefix, pathPrefix, runtime) {\n  if (!fs.existsSync(srcDir)) {\n    return;\n  }\n\n  fs.mkdirSync(skillsDir, { recursive: true });\n\n  // Remove previous EZ Qwen skills\n  const existing = fs.readdirSync(skillsDir, { withFileTypes: true });\n  for (const entry of existing) {\n    if (entry.isDirectory() && entry.name.startsWith(`${prefix}-`)) {\n      fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\n    }\n  }\n\n  function recurse(currentSrcDir, currentPrefix) {\n    const entries = fs.readdirSync(currentSrcDir, { withFileTypes: true });\n\n    for (const entry of entries) {\n      const srcPath = path.join(currentSrcDir, entry.name);\n      if (entry.isDirectory()) {\n        recurse(srcPath, `${currentPrefix}-${entry.name}`);\n        continue;\n      }\n\n      if (!entry.name.endsWith('.md')) {\n        continue;\n      }\n\n      const baseName = entry.name.replace('.md', '');\n      const skillName = `${currentPrefix}-${baseName}`;\n      const skillDir = path.join(skillsDir, skillName);\n      fs.mkdirSync(skillDir, { recursive: true });\n\n      let content = fs.readFileSync(srcPath, 'utf8');\n      const globalClaudeRegex = /~\\/\\.claude\\//g;\n      const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\n      const localClaudeRegex = /\\.\\/\\.claude\\//g;\n      const qwenDirRegex = /~\\/\\.qwen\\//g;\n      content = content.replace(globalClaudeRegex, pathPrefix);\n      content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\n      content = content.replace(localClaudeRegex, `./${getDirName(runtime)}/`);\n      content = content.replace(qwenDirRegex, pathPrefix);\n      content = processAttribution(content, getCommitAttribution(runtime));\n      // Qwen Code uses simple markdown skills without frontmatter adapters\n      content = convertClaudeToQwenSkill(content, skillName);\n      fs.writeFileSync(path.join(skillDir, 'SKILL.md'), content);\n    }\n  }\n\n  recurse(srcDir, prefix);\n}\n\n/**\n * Recursively copy directory, replacing paths in .md files\n * Deletes existing destDir first to remove orphaned files from previous versions\n * @param {string} srcDir - Source directory\n * @param {string} destDir - Destination directory\n * @param {string} pathPrefix - Path prefix for file references\n * @param {string} runtime - Target runtime ('claude', 'opencode', 'gemini', 'codex')\n */\nfunction copyWithPathReplacement(srcDir, destDir, pathPrefix, runtime, isCommand = false, isGlobal = false) {\n  const isOpencode = runtime === 'opencode';\n  const isCodex = runtime === 'codex';\n  const isCopilot = runtime === 'copilot';\n  const dirName = getDirName(runtime);\n\n  // Clean install: remove existing destination to prevent orphaned files\n  if (fs.existsSync(destDir)) {\n    fs.rmSync(destDir, { recursive: true });\n  }\n  fs.mkdirSync(destDir, { recursive: true });\n\n  const entries = fs.readdirSync(srcDir, { withFileTypes: true });\n\n  for (const entry of entries) {\n    const srcPath = path.join(srcDir, entry.name);\n    const destPath = path.join(destDir, entry.name);\n\n    if (entry.isDirectory()) {\n      copyWithPathReplacement(srcPath, destPath, pathPrefix, runtime, isCommand, isGlobal);\n    } else if (entry.name.endsWith('.md')) {\n      // Replace ~/.claude/ and $HOME/.claude/ and ./.claude/ with runtime-appropriate paths\n      // Skip generic replacement for Copilot \u2014 convertClaudeToCopilotContent handles all paths\n      let content = fs.readFileSync(srcPath, 'utf8');\n      if (!isCopilot) {\n        const globalClaudeRegex = /~\\/\\.claude\\//g;\n        const globalClaudeHomeRegex = /\\$HOME\\/\\.claude\\//g;\n        const localClaudeRegex = /\\.\\/\\.claude\\//g;\n        content = content.replace(globalClaudeRegex, pathPrefix);\n        content = content.replace(globalClaudeHomeRegex, toHomePrefix(pathPrefix));\n        content = content.replace(localClaudeRegex, `./${dirName}/`);\n      }\n      content = processAttribution(content, getCommitAttribution(runtime));\n\n      // Convert frontmatter for opencode compatibility\n      if (isOpencode) {\n        content = convertClaudeToOpencodeFrontmatter(content);\n        fs.writeFileSync(destPath, content);\n      } else if (runtime === 'gemini') {\n        if (isCommand) {\n          // Convert to TOML for Gemini (strip <sub> tags \u2014 terminals can't render subscript)\n          content = stripSubTags(content);\n          const tomlContent = convertClaudeToGeminiToml(content);\n          // Replace extension with .toml\n          const tomlPath = destPath.replace(/\\.md$/, '.toml');\n          fs.writeFileSync(tomlPath, tomlContent);\n        } else {\n          fs.writeFileSync(destPath, content);\n        }\n      } else if (isCodex) {\n        content = convertClaudeToCodexMarkdown(content);\n        fs.writeFileSync(destPath, content);\n      } else if (isCopilot) {\n        content = convertClaudeToCopilotContent(content, isGlobal);\n        content = processAttribution(content, getCommitAttribution(runtime));\n        fs.writeFileSync(destPath, content);\n      } else {\n        fs.writeFileSync(destPath, content);\n      }\n    } else if (isCopilot && (entry.name.endsWith('.cjs') || entry.name.endsWith('.js'))) {\n      // Copilot: also transform .cjs/.js files for CONV-06 and CONV-07\n      let content = fs.readFileSync(srcPath, 'utf8');\n      content = convertClaudeToCopilotContent(content, isGlobal);\n      fs.writeFileSync(destPath, content);\n    } else {\n      fs.copyFileSync(srcPath, destPath);\n    }\n  }\n}\n\n/**\n * Clean up orphaned files from previous versions\n */\nfunction cleanupOrphanedFiles(configDir) {\n  const orphanedFiles = [\n    'hooks/ez-notify.sh',  // Removed in v1.6.x\n    'hooks/statusline.js',  // Renamed to ez-statusline.js in v1.9.0\n  ];\n\n  for (const relPath of orphanedFiles) {\n    const fullPath = path.join(configDir, relPath);\n    if (fs.existsSync(fullPath)) {\n      fs.unlinkSync(fullPath);\n      console.log(`  ${green}\u2713${reset} Removed orphaned ${relPath}`);\n    }\n  }\n}\n\n/**\n * Clean up orphaned hook registrations from settings.json\n */\nfunction cleanupOrphanedHooks(settings) {\n  const orphanedHookPatterns = [\n    'ez-notify.sh',  // Removed in v1.6.x\n    'hooks/statusline.js',  // Renamed to ez-statusline.js in v1.9.0\n    'ez-intel-index.js',  // Removed in v1.9.2\n    'ez-intel-session.js',  // Removed in v1.9.2\n    'ez-intel-prune.js',  // Removed in v1.9.2\n  ];\n\n  let cleanedHooks = false;\n\n  // Check all hook event types (Stop, SessionStart, etc.)\n  if (settings.hooks) {\n    for (const eventType of Object.keys(settings.hooks)) {\n      const hookEntries = settings.hooks[eventType];\n      if (Array.isArray(hookEntries)) {\n        // Filter out entries that contain orphaned hooks\n        const filtered = hookEntries.filter(entry => {\n          if (entry.hooks && Array.isArray(entry.hooks)) {\n            // Check if any hook in this entry matches orphaned patterns\n            const hasOrphaned = entry.hooks.some(h =>\n              h.command && orphanedHookPatterns.some(pattern => h.command.includes(pattern))\n            );\n            if (hasOrphaned) {\n              cleanedHooks = true;\n              return false;  // Remove this entry\n            }\n          }\n          return true;  // Keep this entry\n        });\n        settings.hooks[eventType] = filtered;\n      }\n    }\n  }\n\n  if (cleanedHooks) {\n    console.log(`  ${green}\u2713${reset} Removed orphaned hook registrations`);\n  }\n\n  // Fix #330: Update statusLine if it points to old statusline.js path\n  // Only match the specific old path pattern (hooks/statusline.js),\n  // not third-party statusline scripts that happen to contain 'statusline.js'\n  if (settings.statusLine && settings.statusLine.command &&\n      /hooks[\\/\\\\]statusline\\.js/.test(settings.statusLine.command)) {\n    settings.statusLine.command = settings.statusLine.command.replace(\n      /hooks([\\/\\\\])statusline\\.js/,\n      'hooks$1ez-statusline.js'\n    );\n    console.log(`  ${green}\u2713${reset} Updated statusline path (hooks/statusline.js \u2192 hooks/ez-statusline.js)`);\n  }\n\n  return settings;\n}\n\n/**\n * Uninstall EZ_Agents from the specified directory for a specific runtime\n * Removes only EZ-specific files/directories, preserves user content\n * @param {boolean} isGlobal - Whether to uninstall from global or local\n * @param {string} runtime - Target runtime ('claude', 'opencode', 'gemini', 'codex', 'copilot')\n */\nfunction uninstall(isGlobal, runtime = 'claude') {\n  const isOpencode = runtime === 'opencode';\n  const isCodex = runtime === 'codex';\n  const isCopilot = runtime === 'copilot';\n  const isQwen = runtime === 'qwen';\n  const isKimi = runtime === 'kimi';\n  const dirName = getDirName(runtime);\n\n  // Get the target directory based on runtime and install type\n  const targetDir = isGlobal\n    ? getGlobalDir(runtime, explicitConfigDir)\n    : path.join(process.cwd(), dirName);\n\n  const locationLabel = isGlobal\n    ? targetDir.replace(os.homedir(), '~')\n    : targetDir.replace(process.cwd(), '.');\n\n  let runtimeLabel = 'Claude Code';\n  if (runtime === 'opencode') runtimeLabel = 'OpenCode';\n  if (runtime === 'gemini') runtimeLabel = 'Gemini';\n  if (runtime === 'codex') runtimeLabel = 'Codex';\n  if (runtime === 'copilot') runtimeLabel = 'Copilot';\n  if (runtime === 'qwen') runtimeLabel = 'Qwen Code';\n\n  console.log(`  Uninstalling EZ_Agents from ${cyan}${runtimeLabel}${reset} at ${cyan}${locationLabel}${reset}\\n`);\n\n  // Check if target directory exists\n  if (!fs.existsSync(targetDir)) {\n    console.log(`  ${yellow}\u26A0${reset} Directory does not exist: ${locationLabel}`);\n    console.log(`  Nothing to uninstall.\\n`);\n    return;\n  }\n\n  let removedCount = 0;\n\n  // 1. Remove EZ commands/skills\n  if (isOpencode) {\n    // OpenCode: remove command/ez-*.md files\n    const commandDir = path.join(targetDir, 'command');\n    if (fs.existsSync(commandDir)) {\n      const files = fs.readdirSync(commandDir);\n      for (const file of files) {\n        if (file.startsWith('ez-') && file.endsWith('.md')) {\n          fs.unlinkSync(path.join(commandDir, file));\n          removedCount++;\n        }\n      }\n      console.log(`  ${green}\u2713${reset} Removed EZ_Agents commands from command/`);\n    }\n  } else if (isCodex || isQwen || isKimi) {\n    // Codex/Qwen/Kimi: remove skills/ez-*/SKILL.md skill directories\n    const skillsDir = path.join(targetDir, 'skills');\n    if (fs.existsSync(skillsDir)) {\n      let skillCount = 0;\n      const entries = fs.readdirSync(skillsDir, { withFileTypes: true });\n      for (const entry of entries) {\n        if (entry.isDirectory() && entry.name.startsWith('ez-')) {\n          fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\n          skillCount++;\n        }\n      }\n      if (skillCount > 0) {\n        removedCount++;\n        let runtimeLabel = 'Codex';\n        if (isQwen) runtimeLabel = 'Qwen Code';\n        if (isKimi) runtimeLabel = 'Kimi Code';\n        console.log(`  ${green}\u2713${reset} Removed ${skillCount} ${runtimeLabel} skills`);\n      }\n    }\n\n    if (isCodex) {\n      // Codex: remove EZ agent .toml config files\n      const codexAgentsDir = path.join(targetDir, 'agents');\n      if (fs.existsSync(codexAgentsDir)) {\n        const tomlFiles = fs.readdirSync(codexAgentsDir);\n        let tomlCount = 0;\n        for (const file of tomlFiles) {\n          if (file.startsWith('ez-') && file.endsWith('.toml')) {\n            fs.unlinkSync(path.join(codexAgentsDir, file));\n            tomlCount++;\n          }\n        }\n        if (tomlCount > 0) {\n          removedCount++;\n          console.log(`  ${green}\u2713${reset} Removed ${tomlCount} agent .toml configs`);\n        }\n      }\n\n      // Codex: clean EZ sections from config.toml\n      const configPath = path.join(targetDir, 'config.toml');\n      if (fs.existsSync(configPath)) {\n        const content = fs.readFileSync(configPath, 'utf8');\n        const cleaned = stripEzAgentsFromCodexConfig(content);\n        if (cleaned === null) {\n          // File is empty after stripping \u2014 delete it\n          fs.unlinkSync(configPath);\n          removedCount++;\n          console.log(`  ${green}\u2713${reset} Removed config.toml (was EZ-only)`);\n        } else if (cleaned !== content) {\n          fs.writeFileSync(configPath, cleaned);\n          removedCount++;\n          console.log(`  ${green}\u2713${reset} Cleaned EZ_Agents sections from config.toml`);\n        }\n      }\n    }\n  } else if (isCopilot) {\n    // Copilot: remove skills/ez-*/ directories (same layout as Codex skills)\n    const skillsDir = path.join(targetDir, 'skills');\n    if (fs.existsSync(skillsDir)) {\n      let skillCount = 0;\n      const entries = fs.readdirSync(skillsDir, { withFileTypes: true });\n      for (const entry of entries) {\n        if (entry.isDirectory() && entry.name.startsWith('ez-')) {\n          fs.rmSync(path.join(skillsDir, entry.name), { recursive: true });\n          skillCount++;\n        }\n      }\n      if (skillCount > 0) {\n        removedCount++;\n        console.log(`  ${green}\u2713${reset} Removed ${skillCount} Copilot skills`);\n      }\n    }\n\n    // Copilot: clean EZ section from copilot-instructions.md\n    const instructionsPath = path.join(targetDir, 'copilot-instructions.md');\n    if (fs.existsSync(instructionsPath)) {\n      const content = fs.readFileSync(instructionsPath, 'utf8');\n      const cleaned = stripEzAgentsFromCopilotInstructions(content);\n      if (cleaned === null) {\n        fs.unlinkSync(instructionsPath);\n        removedCount++;\n        console.log(`  ${green}\u2713${reset} Removed copilot-instructions.md (was EZ-only)`);\n      } else if (cleaned !== content) {\n        fs.writeFileSync(instructionsPath, cleaned);\n        removedCount++;\n        console.log(`  ${green}\u2713${reset} Cleaned EZ_Agents section from copilot-instructions.md`);\n      }\n    }\n  } else {\n    const ezCommandsDir = path.join(targetDir, 'commands', 'ez');\n    if (fs.existsSync(ezCommandsDir)) {\n      fs.rmSync(ezCommandsDir, { recursive: true });\n      removedCount++;\n      console.log(`  ${green}\u2713${reset} Removed commands/ez/`);\n    }\n  }\n\n  // 2. Remove ez-agents directory\n  const ezDir = path.join(targetDir, 'ez-agents');\n  if (fs.existsSync(ezDir)) {\n    fs.rmSync(ezDir, { recursive: true });\n    removedCount++;\n    console.log(`  ${green}\u2713${reset} Removed ez-agents/`);\n  }\n\n  // 3. Remove EZ_Agents agents (ez-*.md files only)\n  const agentsDir = path.join(targetDir, 'agents');\n  if (fs.existsSync(agentsDir)) {\n    const files = fs.readdirSync(agentsDir);\n    let agentCount = 0;\n    for (const file of files) {\n      if (file.startsWith('ez-') && file.endsWith('.md')) {\n        fs.unlinkSync(path.join(agentsDir, file));\n        agentCount++;\n      }\n    }\n    if (agentCount > 0) {\n      removedCount++;\n      console.log(`  ${green}\u2713${reset} Removed ${agentCount} EZ_Agents agents`);\n    }\n  }\n\n  // 4. Remove EZ hooks\n  const hooksDir = path.join(targetDir, 'hooks');\n  if (fs.existsSync(hooksDir)) {\n    const ezHooks = ['ez-statusline.js', 'ez-check-update.js', 'ez-check-update.sh', 'ez-context-monitor.js'];\n    let hookCount = 0;\n    for (const hook of ezHooks) {\n      const hookPath = path.join(hooksDir, hook);\n      if (fs.existsSync(hookPath)) {\n        fs.unlinkSync(hookPath);\n        hookCount++;\n      }\n    }\n    if (hookCount > 0) {\n      removedCount++;\n      console.log(`  ${green}\u2713${reset} Removed ${hookCount} EZ hooks`);\n    }\n  }\n\n  // 5. Remove EZ package.json (CommonJS mode marker)\n  const pkgJsonPath = path.join(targetDir, 'package.json');\n  if (fs.existsSync(pkgJsonPath)) {\n    try {\n      const content = fs.readFileSync(pkgJsonPath, 'utf8').trim();\n      // Only remove if it's our minimal CommonJS marker\n      if (content === '{\"type\":\"commonjs\"}') {\n        fs.unlinkSync(pkgJsonPath);\n        removedCount++;\n        console.log(`  ${green}\u2713${reset} Removed EZ package.json`);\n      }\n    } catch (e) {\n      // Ignore read errors\n    }\n  }\n\n  // 6. Clean up settings.json (remove EZ hooks and statusline)\n  const settingsPath = path.join(targetDir, 'settings.json');\n  if (fs.existsSync(settingsPath)) {\n    let settings = readSettings(settingsPath);\n    let settingsModified = false;\n\n    // Remove EZ statusline if it references our hook\n    if (settings.statusLine && settings.statusLine.command &&\n        settings.statusLine.command.includes('ez-statusline')) {\n      delete settings.statusLine;\n      settingsModified = true;\n      console.log(`  ${green}\u2713${reset} Removed EZ statusline from settings`);\n    }\n\n    // Remove EZ hooks from SessionStart\n    if (settings.hooks && settings.hooks.SessionStart) {\n      const before = settings.hooks.SessionStart.length;\n      settings.hooks.SessionStart = settings.hooks.SessionStart.filter(entry => {\n        if (entry.hooks && Array.isArray(entry.hooks)) {\n          // Filter out EZ hooks\n          const hasEzHook = entry.hooks.some(h =>\n            h.command && (h.command.includes('ez-check-update') || h.command.includes('ez-statusline'))\n          );\n          return !hasEzHook;\n        }\n        return true;\n      });\n      if (settings.hooks.SessionStart.length < before) {\n        settingsModified = true;\n        console.log(`  ${green}\u2713${reset} Removed EZ hooks from settings`);\n      }\n      // Clean up empty array\n      if (settings.hooks.SessionStart.length === 0) {\n        delete settings.hooks.SessionStart;\n      }\n    }\n\n    // Remove EZ hooks from PostToolUse and AfterTool (Gemini uses AfterTool)\n    for (const eventName of ['PostToolUse', 'AfterTool']) {\n      if (settings.hooks && settings.hooks[eventName]) {\n        const before = settings.hooks[eventName].length;\n        settings.hooks[eventName] = settings.hooks[eventName].filter(entry => {\n          if (entry.hooks && Array.isArray(entry.hooks)) {\n            const hasEzHook = entry.hooks.some(h =>\n              h.command && h.command.includes('ez-context-monitor')\n            );\n            return !hasEzHook;\n          }\n          return true;\n        });\n        if (settings.hooks[eventName].length < before) {\n          settingsModified = true;\n          console.log(`  ${green}\u2713${reset} Removed context monitor hook from settings`);\n        }\n        if (settings.hooks[eventName].length === 0) {\n          delete settings.hooks[eventName];\n        }\n      }\n    }\n\n    // Clean up empty hooks object\n    if (settings.hooks && Object.keys(settings.hooks).length === 0) {\n      delete settings.hooks;\n    }\n\n    if (settingsModified) {\n      writeSettings(settingsPath, settings);\n      removedCount++;\n    }\n  }\n\n  // 6. For OpenCode, clean up permissions from opencode.json\n  if (isOpencode) {\n    // For local uninstalls, clean up ./.opencode/opencode.json\n    // For global uninstalls, clean up ~/.config/opencode/opencode.json\n    const opencodeConfigDir = isGlobal\n      ? getOpencodeGlobalDir()\n      : path.join(process.cwd(), '.opencode');\n    const configPath = path.join(opencodeConfigDir, 'opencode.json');\n    if (fs.existsSync(configPath)) {\n      try {\n        const config = JSON.parse(fs.readFileSync(configPath, 'utf8'));\n        let modified = false;\n\n        // Remove EZ permission entries\n        if (config.permission) {\n          for (const permType of ['read', 'external_directory']) {\n            if (config.permission[permType]) {\n              const keys = Object.keys(config.permission[permType]);\n              for (const key of keys) {\n                if (key.includes('ez-agents')) {\n                  delete config.permission[permType][key];\n                  modified = true;\n                }\n              }\n              // Clean up empty objects\n              if (Object.keys(config.permission[permType]).length === 0) {\n                delete config.permission[permType];\n              }\n            }\n          }\n          if (Object.keys(config.permission).length === 0) {\n            delete config.permission;\n          }\n        }\n\n        if (modified) {\n          fs.writeFileSync(configPath, JSON.stringify(config, null, 2) + '\\n');\n          removedCount++;\n          console.log(`  ${green}\u2713${reset} Removed EZ permissions from opencode.json`);\n        }\n      } catch (e) {\n        // Ignore JSON parse errors\n      }\n    }\n  }\n\n  if (removedCount === 0) {\n    console.log(`  ${yellow}\u26A0${reset} No EZ_Agents files found to remove.`);\n  }\n\n  console.log(`\n  ${green}Done!${reset} EZ_Agents has been uninstalled from ${runtimeLabel}.\n  Your other files and settings have been preserved.\n`);\n}\n\n/**\n * Parse JSONC (JSON with Comments) by stripping comments and trailing commas.\n * OpenCode supports JSONC format via jsonc-parser, so users may have comments.\n * This is a lightweight inline parser to avoid adding dependencies.\n */\nfunction parseJsonc(content) {\n  // Strip BOM if present\n  if (content.charCodeAt(0) === 0xFEFF) {\n    content = content.slice(1);\n  }\n\n  // Remove single-line and block comments while preserving strings\n  let result = '';\n  let inString = false;\n  let i = 0;\n  while (i < content.length) {\n    const char = content[i];\n    const next = content[i + 1];\n\n    if (inString) {\n      result += char;\n      // Handle escape sequences\n      if (char === '\\\\' && i + 1 < content.length) {\n        result += next;\n        i += 2;\n        continue;\n      }\n      if (char === '\"') {\n        inString = false;\n      }\n      i++;\n    } else {\n      if (char === '\"') {\n        inString = true;\n        result += char;\n        i++;\n      } else if (char === '/' && next === '/') {\n        // Skip single-line comment until end of line\n        while (i < content.length && content[i] !== '\\n') {\n          i++;\n        }\n      } else if (char === '/' && next === '*') {\n        // Skip block comment\n        i += 2;\n        while (i < content.length - 1 && !(content[i] === '*' && content[i + 1] === '/')) {\n          i++;\n        }\n        i += 2; // Skip closing */\n      } else {\n        result += char;\n        i++;\n      }\n    }\n  }\n\n  // Remove trailing commas before } or ]\n  result = result.replace(/,(\\s*[}\\]])/g, '$1');\n\n  return JSON.parse(result);\n}\n\n/**\n * Configure OpenCode permissions to allow reading EZ reference docs\n * This prevents permission prompts when EZ accesses the ez-agents directory\n * @param {boolean} isGlobal - Whether this is a global or local install\n */\nfunction configureOpencodePermissions(isGlobal = true) {\n  // For local installs, use ./.opencode/opencode.json\n  // For global installs, use ~/.config/opencode/opencode.json\n  const opencodeConfigDir = isGlobal\n    ? getOpencodeGlobalDir()\n    : path.join(process.cwd(), '.opencode');\n  const configPath = path.join(opencodeConfigDir, 'opencode.json');\n\n  // Ensure config directory exists\n  fs.mkdirSync(opencodeConfigDir, { recursive: true });\n\n  // Read existing config or create empty object\n  let config: Record<string, unknown> = {};\n  if (fs.existsSync(configPath)) {\n    try {\n      const content = fs.readFileSync(configPath, 'utf8');\n      config = parseJsonc(content);\n    } catch (e) {\n      // Cannot parse - DO NOT overwrite user's config\n      console.log(`  ${yellow}\u26A0${reset} Could not parse opencode.json - skipping permission config`);\n      console.log(`    ${dim}Reason: ${(e as Error).message}${reset}`);\n      console.log(`    ${dim}Your config was NOT modified. Fix the syntax manually if needed.${reset}`);\n      return;\n    }\n  }\n\n  // Ensure permission structure exists\n  if (!config.permission) {\n    config.permission = {};\n  }\n\n  // Build the EZ path using the actual config directory\n  // Use ~ shorthand if it's in the default location, otherwise use full path\n  const defaultConfigDir = path.join(os.homedir(), '.config', 'opencode');\n  const ezPath = opencodeConfigDir === defaultConfigDir\n    ? '~/.config/opencode/ez-agents/*'\n    : `${opencodeConfigDir.replace(/\\\\/g, '/')}/ez-agents/*`;\n\n  let modified = false;\n\n  // Configure read permission\n  const permission = config.permission as Record<string, unknown> || {};\n  config.permission = permission;\n  if (!permission.read || typeof permission.read !== 'object') {\n    permission.read = {};\n  }\n  const readPerm = permission.read as Record<string, unknown>;\n  if (readPerm[ezPath] !== 'allow') {\n    readPerm[ezPath] = 'allow';\n    modified = true;\n  }\n\n  // Configure external_directory permission (the safety guard for paths outside project)\n  if (!permission.external_directory || typeof permission.external_directory !== 'object') {\n    permission.external_directory = {};\n  }\n  const extDirPerm = permission.external_directory as Record<string, unknown>;\n  if (extDirPerm[ezPath] !== 'allow') {\n    extDirPerm[ezPath] = 'allow';\n    modified = true;\n  }\n\n  if (!modified) {\n    return; // Already configured\n  }\n\n  // Write config back\n  fs.writeFileSync(configPath, JSON.stringify(config, null, 2) + '\\n');\n  console.log(`  ${green}\u2713${reset} Configured read permission for EZ_Agents docs`);\n}\n\n/**\n * Verify a directory exists and contains files\n */\nfunction verifyInstalled(dirPath, description) {\n  if (!fs.existsSync(dirPath)) {\n    console.error(`  ${yellow}\u2717${reset} Failed to install ${description}: directory not created`);\n    return false;\n  }\n  try {\n    const entries = fs.readdirSync(dirPath);\n    if (entries.length === 0) {\n      console.error(`  ${yellow}\u2717${reset} Failed to install ${description}: directory is empty`);\n      return false;\n    }\n  } catch (e) {\n    console.error(`  ${yellow}\u2717${reset} Failed to install ${description}: ${(e as Error).message}`);\n    return false;\n  }\n  return true;\n}\n\n/**\n * Verify a file exists\n */\nfunction verifyFileInstalled(filePath, description) {\n  if (!fs.existsSync(filePath)) {\n    console.error(`  ${yellow}\u2717${reset} Failed to install ${description}: file not created`);\n    return false;\n  }\n  return true;\n}\n\n/**\n * Install to the specified directory for a specific runtime\n * @param {boolean} isGlobal - Whether to install globally or locally\n * @param {string} runtime - Target runtime ('claude', 'opencode', 'gemini', 'codex')\n */\n\n// \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n// Local Patch Persistence\n// \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst PATCHES_DIR_NAME = 'ez-local-patches';\nconst MANIFEST_NAME = 'ez-file-manifest.json';\n\n/**\n * Compute SHA256 hash of file contents\n */\nfunction fileHash(filePath: string): string {\n  const content = fs.readFileSync(filePath);\n  // Convert to Uint8Array to satisfy BinaryLike type requirement\n  return crypto.createHash('sha256').update(new Uint8Array(content)).digest('hex');\n}\n\n/**\n * Recursively collect all files in dir with their hashes\n */\nfunction generateManifest(dir: string, baseDir?: string): Record<string, string> {\n  if (!baseDir) baseDir = dir;\n  const manifest: Record<string, string> = {};\n  if (!fs.existsSync(dir)) return manifest;\n  const entries = fs.readdirSync(dir, { withFileTypes: true });\n  for (const entry of entries) {\n    const fullPath = path.join(dir, entry.name);\n    const relPath = path.relative(baseDir, fullPath).replace(/\\\\/g, '/');\n    if (entry.isDirectory()) {\n      Object.assign(manifest, generateManifest(fullPath, baseDir));\n    } else {\n      manifest[relPath] = fileHash(fullPath);\n    }\n  }\n  return manifest;\n}\n\n/**\n * Write file manifest after installation for future modification detection\n */\nfunction writeManifest(configDir: string, runtime = 'claude') {\n  const isOpencode = runtime === 'opencode';\n  const isCodex = runtime === 'codex';\n  const isCopilot = runtime === 'copilot';\n  const ezDir = path.join(configDir, 'ez-agents');\n  const commandsDir = path.join(configDir, 'commands', 'ez');\n  const opencodeCommandDir = path.join(configDir, 'command');\n  const codexSkillsDir = path.join(configDir, 'skills');\n  const agentsDir = path.join(configDir, 'agents');\n  const manifest = { version: pkg.version, timestamp: new Date().toISOString(), files: {} as Record<string, string> };\n\n  const ezHashes = generateManifest(ezDir);\n  for (const [rel, hash] of Object.entries(ezHashes)) {\n    manifest.files['ez-agents/' + rel] = hash;\n  }\n  if (!isOpencode && !isCodex && !isCopilot && fs.existsSync(commandsDir)) {\n    const cmdHashes = generateManifest(commandsDir);\n    for (const [rel, hash] of Object.entries(cmdHashes)) {\n      manifest.files['commands/ez/' + rel] = hash;\n    }\n  }\n  if (isOpencode && fs.existsSync(opencodeCommandDir)) {\n    for (const file of fs.readdirSync(opencodeCommandDir)) {\n      if (file.startsWith('ez-') && file.endsWith('.md')) {\n        manifest.files['command/' + file] = fileHash(path.join(opencodeCommandDir, file));\n      }\n    }\n  }\n  if ((isCodex || isCopilot) && fs.existsSync(codexSkillsDir)) {\n    for (const skillName of listCodexSkillNames(codexSkillsDir)) {\n      const skillRoot = path.join(codexSkillsDir, skillName);\n      const skillHashes = generateManifest(skillRoot);\n      for (const [rel, hash] of Object.entries(skillHashes)) {\n        manifest.files[`skills/${skillName}/${rel}`] = hash;\n      }\n    }\n  }\n  if (fs.existsSync(agentsDir)) {\n    for (const file of fs.readdirSync(agentsDir)) {\n      if (file.startsWith('ez-') && file.endsWith('.md')) {\n        manifest.files['agents/' + file] = fileHash(path.join(agentsDir, file));\n      }\n    }\n  }\n\n  fs.writeFileSync(path.join(configDir, MANIFEST_NAME), JSON.stringify(manifest, null, 2));\n  return manifest;\n}\n\n/**\n * Detect user-modified EZ_Agents files by comparing against install manifest.\n * Backs up modified files to ez-local-patches/ for reapply after update.\n */\nfunction saveLocalPatches(configDir) {\n  const manifestPath = path.join(configDir, MANIFEST_NAME);\n  if (!fs.existsSync(manifestPath)) return [];\n\n  let manifest;\n  try { manifest = JSON.parse(fs.readFileSync(manifestPath, 'utf8')); } catch { return []; }\n\n  const patchesDir = path.join(configDir, PATCHES_DIR_NAME);\n  const modified: string[] = [];\n\n  for (const [relPath, originalHash] of Object.entries(manifest.files || {})) {\n    const fullPath = path.join(configDir, relPath);\n    if (!fs.existsSync(fullPath)) continue;\n    const currentHash = fileHash(fullPath);\n    if (currentHash !== originalHash) {\n      const backupPath = path.join(patchesDir, relPath);\n      fs.mkdirSync(path.dirname(backupPath), { recursive: true });\n      fs.copyFileSync(fullPath, backupPath);\n      modified.push(relPath);\n    }\n  }\n\n  if (modified.length > 0) {\n    const meta = {\n      backed_up_at: new Date().toISOString(),\n      from_version: manifest.version,\n      files: modified\n    };\n    fs.writeFileSync(path.join(patchesDir, 'backup-meta.json'), JSON.stringify(meta, null, 2));\n    console.log('  ' + yellow + 'i' + reset + '  Found ' + modified.length + ' locally modified EZ file(s) \u2014 backed up to ' + PATCHES_DIR_NAME + '/');\n    for (const f of modified) {\n      console.log('     ' + dim + f + reset);\n    }\n  }\n  return modified;\n}\n\n/**\n * After install, report backed-up patches for user to reapply.\n */\nfunction reportLocalPatches(configDir, runtime = 'claude') {\n  const patchesDir = path.join(configDir, PATCHES_DIR_NAME);\n  const metaPath = path.join(patchesDir, 'backup-meta.json');\n  if (!fs.existsSync(metaPath)) return [];\n\n  let meta;\n  try { meta = JSON.parse(fs.readFileSync(metaPath, 'utf8')); } catch { return []; }\n\n  if (meta.files && meta.files.length > 0) {\n    const reapplyCommand = (runtime === 'opencode' || runtime === 'copilot')\n      ? '/ez-reapply-patches'\n      : runtime === 'codex'\n        ? '$ez-reapply-patches'\n        : '/ez:reapply-patches';\n    console.log('');\n    console.log('  ' + yellow + 'Local patches detected' + reset + ' (from v' + meta.from_version + '):');\n    for (const f of meta.files) {\n      console.log('     ' + cyan + f + reset);\n    }\n    console.log('');\n    console.log('  Your modifications are saved in ' + cyan + PATCHES_DIR_NAME + '/' + reset);\n    console.log('  Run ' + cyan + reapplyCommand + reset + ' to merge them into the new version.');\n    console.log('  Or manually compare and merge the files.');\n    console.log('');\n  }\n  return meta.files || [];\n}\n\nfunction install(isGlobal: boolean, runtime = 'claude'): { settingsPath: string | null; settings: Record<string, unknown> | null; statuslineCommand: string | null; runtime: string } {\n  const isOpencode = runtime === 'opencode';\n  const isGemini = runtime === 'gemini';\n  const isCodex = runtime === 'codex';\n  const isCopilot = runtime === 'copilot';\n  const isQwen = runtime === 'qwen';\n  const isKimi = runtime === 'kimi';\n  const dirName = getDirName(runtime);\n  \n  // Resolve package root by finding directory containing package.json\n  // This works whether running from dist/bin/ (installed) or bin/ (development)\n  let pkgRoot = path.join(__dirname, '..');\n  while (!fs.existsSync(path.join(pkgRoot, 'package.json'))) {\n    const parent = path.dirname(pkgRoot);\n    if (parent === pkgRoot) {\n      // Reached filesystem root, fallback to __dirname\n      pkgRoot = path.join(__dirname, '..');\n      break;\n    }\n    pkgRoot = parent;\n  }\n  const src = pkgRoot;\n\n  // Get the target directory based on runtime and install type\n  const targetDir = isGlobal\n    ? getGlobalDir(runtime, explicitConfigDir)\n    : path.join(process.cwd(), dirName);\n\n  const locationLabel = isGlobal\n    ? targetDir.replace(os.homedir(), '~')\n    : targetDir.replace(process.cwd(), '.');\n\n  // Path prefix for file references in markdown content\n  // For global installs: use tilde-based path (~/.claude/) to avoid baking\n  // absolute paths (containing OS username) into templates\n  // For local installs: use relative\n  const pathPrefix = isGlobal\n    ? `${targetDir.replace(/\\\\/g, '/').replace(os.homedir().replace(/\\\\/g, '/'), '~')}/`\n    : `./${dirName}/`;\n\n  let runtimeLabel = 'Claude Code';\n  if (isOpencode) runtimeLabel = 'OpenCode';\n  if (isGemini) runtimeLabel = 'Gemini';\n  if (isCodex) runtimeLabel = 'Codex';\n  if (isCopilot) runtimeLabel = 'Copilot';\n  if (isQwen) runtimeLabel = 'Qwen Code';\n  if (isKimi) runtimeLabel = 'Kimi Code';\n\n  console.log(`  Installing for ${cyan}${runtimeLabel}${reset} to ${cyan}${locationLabel}${reset}\\n`);\n\n  // Track installation failures\n  const failures: string[] = [];\n\n  // Save any locally modified EZ_Agents files before they get wiped\n  saveLocalPatches(targetDir);\n\n  // Clean up orphaned files from previous versions\n  cleanupOrphanedFiles(targetDir);\n\n  // OpenCode uses command/ (flat), Codex/Copilot/Qwen use skills/, Claude/Gemini use commands/ez/\n  if (isOpencode) {\n    // OpenCode: flat structure in command/ directory\n    const commandDir = path.join(targetDir, 'command');\n    fs.mkdirSync(commandDir, { recursive: true });\n\n    // Copy commands/ez/*.md as command/ez-*.md (flatten structure)\n    const ezSrc = path.join(src, 'commands', 'ez');\n    copyFlattenedCommands(ezSrc, commandDir, 'ez', pathPrefix, runtime);\n    if (verifyInstalled(commandDir, 'command/ez-*')) {\n      const count = fs.readdirSync(commandDir).filter(f => f.startsWith('ez-')).length;\n      console.log(`  ${green}\u2713${reset} Installed ${count} commands to command/`);\n    } else {\n      failures.push('command/ez-*');\n    }\n  } else if (isCodex || isKimi) {\n    // Codex, Kimi: skills structure in skills/ directory\n    const skillsDir = path.join(targetDir, 'skills');\n    const ezSrc = path.join(src, 'commands', 'ez');\n    if (isKimi) {\n      copyCommandsAsKimiSkills(ezSrc, skillsDir, 'ez', pathPrefix, runtime);\n    } else {\n      copyCommandsAsCodexSkills(ezSrc, skillsDir, 'ez', pathPrefix, runtime);\n    }\n    const installedSkillNames = listCodexSkillNames(skillsDir);\n    if (installedSkillNames.length > 0) {\n      console.log(`  ${green}\u2713${reset} Installed ${installedSkillNames.length} skills to skills/`);\n    } else {\n      failures.push('skills/ez-*');\n    }\n  } else if (isQwen) {\n    // Qwen Code: commands structure in commands/ez/ directory (like Claude Code/Gemini)\n    // Qwen Code uses ~/.qwen/commands/ez/*.md, NOT ~/.qwen/skills/\n    const commandsDir = path.join(targetDir, 'commands');\n    const ezCommandsDir = path.join(commandsDir, 'ez');\n    fs.mkdirSync(ezCommandsDir, { recursive: true });\n    \n    const ezSrc = path.join(src, 'commands', 'ez');\n    copyCommandsAsQwenCommands(ezSrc, ezCommandsDir, 'ez', pathPrefix, runtime);\n    \n    if (fs.existsSync(ezCommandsDir)) {\n      const count = fs.readdirSync(ezCommandsDir).filter(f => f.endsWith('.md')).length;\n      if (count > 0) {\n        console.log(`  ${green}\u2713${reset} Installed ${count} commands to commands/ez/`);\n      } else {\n        failures.push('commands/ez/*.md');\n      }\n    } else {\n      failures.push('commands/ez/*.md');\n    }\n  } else if (isCopilot) {\n    const skillsDir = path.join(targetDir, 'skills');\n    const ezSrc = path.join(src, 'commands', 'ez');\n    copyCommandsAsCopilotSkills(ezSrc, skillsDir, 'ez', isGlobal);\n    if (fs.existsSync(skillsDir)) {\n      const count = fs.readdirSync(skillsDir, { withFileTypes: true })\n        .filter(e => e.isDirectory() && e.name.startsWith('ez-')).length;\n      if (count > 0) {\n        console.log(`  ${green}\u2713${reset} Installed ${count} skills to skills/`);\n      } else {\n        failures.push('skills/ez-*');\n      }\n    } else {\n      failures.push('skills/ez-*');\n    }\n  } else {\n    // Claude Code & Gemini: nested structure in commands/ directory\n    const commandsDir = path.join(targetDir, 'commands');\n    fs.mkdirSync(commandsDir, { recursive: true });\n\n    const ezSrc = path.join(src, 'commands', 'ez');\n    const ezDest = path.join(commandsDir, 'ez');\n    copyWithPathReplacement(ezSrc, ezDest, pathPrefix, runtime, true, isGlobal);\n    if (verifyInstalled(ezDest, 'commands/ez')) {\n      console.log(`  ${green}\u2713${reset} Installed commands/ez`);\n    } else {\n      failures.push('commands/ez');\n    }\n  }\n\n  // Copy ez-agents skill with path replacement\n  const skillSrc = path.join(src, 'ez-agents');\n  const skillDest = path.join(targetDir, 'ez-agents');\n  copyWithPathReplacement(skillSrc, skillDest, pathPrefix, runtime, false, isGlobal);\n  if (verifyInstalled(skillDest, 'ez-agents')) {\n    console.log(`  ${green}\u2713${reset} Installed ez-agents`);\n  } else {\n    failures.push('ez-agents');\n  }\n\n  // Copy EZ Agents skills to global directory ($HOME/.skills/ez-agents/)\n  copySkills(src, isGlobal, runtime);\n\n  // Copy agents to agents directory\n  const agentsSrc = path.join(src, 'agents');\n  if (fs.existsSync(agentsSrc)) {\n    const agentsDest = path.join(targetDir, 'agents');\n    fs.mkdirSync(agentsDest, { recursive: true });\n\n    // Remove old EZ_Agents agents (ez-*.md) before copying new ones\n    if (fs.existsSync(agentsDest)) {\n      for (const file of fs.readdirSync(agentsDest)) {\n        if (file.startsWith('ez-') && file.endsWith('.md')) {\n          fs.unlinkSync(path.join(agentsDest, file));\n        }\n      }\n    }\n\n    // Copy new agents\n    const agentEntries = fs.readdirSync(agentsSrc, { withFileTypes: true });\n    for (const entry of agentEntries) {\n      if (entry.isFile() && entry.name.endsWith('.md')) {\n        let content = fs.readFileSync(path.join(agentsSrc, entry.name), 'utf8');\n        // Replace ~/.claude/ and $HOME/.claude/ as they are the source of truth in the repo\n        const dirRegex = /~\\/\\.claude\\//g;\n        const homeDirRegex = /\\$HOME\\/\\.claude\\//g;\n        if (!isCopilot) {\n          content = content.replace(dirRegex, pathPrefix);\n          content = content.replace(homeDirRegex, toHomePrefix(pathPrefix));\n        }\n        content = processAttribution(content, getCommitAttribution(runtime));\n        // Convert frontmatter for runtime compatibility\n        if (isOpencode) {\n          content = convertClaudeToOpencodeFrontmatter(content);\n        } else if (isGemini) {\n          content = convertClaudeToGeminiAgent(content);\n        } else if (isCodex) {\n          content = convertClaudeAgentToCodexAgent(content);\n        } else if (isCopilot) {\n          content = convertClaudeAgentToCopilotAgent(content, isGlobal);\n        } else if (isQwen) {\n          content = convertClaudeToQwenAgent(content);\n        } else if (isKimi) {\n          content = convertClaudeToKimiAgent(content);\n        }\n        const destName = isCopilot ? entry.name.replace('.md', '.agent.md') : entry.name;\n        fs.writeFileSync(path.join(agentsDest, destName), content);\n      }\n    }\n    if (verifyInstalled(agentsDest, 'agents')) {\n      console.log(`  ${green}\u2713${reset} Installed agents`);\n    } else {\n      failures.push('agents');\n    }\n  }\n\n  // Copy CHANGELOG.md\n  const changelogSrc = path.join(src, 'CHANGELOG.md');\n  const changelogDest = path.join(targetDir, 'ez-agents', 'CHANGELOG.md');\n  if (fs.existsSync(changelogSrc)) {\n    fs.copyFileSync(changelogSrc, changelogDest);\n    if (verifyFileInstalled(changelogDest, 'CHANGELOG.md')) {\n      console.log(`  ${green}\u2713${reset} Installed CHANGELOG.md`);\n    } else {\n      failures.push('CHANGELOG.md');\n    }\n  }\n\n  // Write VERSION file\n  const versionDest = path.join(targetDir, 'ez-agents', 'VERSION');\n  fs.writeFileSync(versionDest, pkg.version);\n  if (verifyFileInstalled(versionDest, 'VERSION')) {\n    console.log(`  ${green}\u2713${reset} Wrote VERSION (${pkg.version})`);\n  } else {\n    failures.push('VERSION');\n  }\n\n  if (!isCodex && !isCopilot && !isQwen && !isKimi) {\n    // Write package.json to force CommonJS mode for EZ scripts\n    // Prevents \"require is not defined\" errors when project has \"type\": \"module\"\n    // Node.js walks up looking for package.json - this stops inheritance from project\n    const pkgJsonDest = path.join(targetDir, 'package.json');\n    fs.writeFileSync(pkgJsonDest, '{\"type\":\"commonjs\"}\\n');\n    console.log(`  ${green}\u2713${reset} Wrote package.json (CommonJS mode)`);\n\n    // Copy hooks from dist/ (bundled with dependencies)\n    // Template paths for the target runtime (replaces '.claude' with correct config dir)\n    const hooksSrc = path.join(src, 'hooks', 'dist');\n    if (fs.existsSync(hooksSrc)) {\n      const hooksDest = path.join(targetDir, 'hooks');\n      fs.mkdirSync(hooksDest, { recursive: true });\n      const hookEntries = fs.readdirSync(hooksSrc);\n      const configDirReplacement = getConfigDirFromHome(runtime, isGlobal);\n      for (const entry of hookEntries) {\n        const srcFile = path.join(hooksSrc, entry);\n        if (fs.statSync(srcFile).isFile()) {\n          const destFile = path.join(hooksDest, entry);\n          // Template .js files to replace '.claude' with runtime-specific config dir\n          if (entry.endsWith('.js')) {\n            let content = fs.readFileSync(srcFile, 'utf8');\n            content = content.replace(/'\\.claude'/g, configDirReplacement);\n            fs.writeFileSync(destFile, content);\n          } else {\n            fs.copyFileSync(srcFile, destFile);\n          }\n        }\n      }\n      if (verifyInstalled(hooksDest, 'hooks')) {\n        console.log(`  ${green}\u2713${reset} Installed hooks (bundled)`);\n      } else {\n        failures.push('hooks');\n      }\n    }\n  }\n\n  if (failures.length > 0) {\n    console.error(`\\n  ${yellow}Installation incomplete!${reset} Failed: ${failures.join(', ')}`);\n    process.exit(1);\n  }\n\n  // Write file manifest for future modification detection\n  writeManifest(targetDir, runtime);\n  console.log(`  ${green}\u2713${reset} Wrote file manifest (${MANIFEST_NAME})`);\n\n  // Report any backed-up local patches\n  reportLocalPatches(targetDir, runtime);\n\n  // Verify no leaked .claude paths in non-Claude runtimes\n  if (runtime !== 'claude') {\n    const leakedPaths: Array<{ file: string; count: number }> = [];\n    function scanForLeakedPaths(dir: string) {\n      if (!fs.existsSync(dir)) return;\n      const entries = fs.readdirSync(dir, { withFileTypes: true });\n      for (const entry of entries) {\n        const fullPath = path.join(dir, entry.name);\n        if (entry.isDirectory()) {\n          scanForLeakedPaths(fullPath);\n        } else if ((entry.name.endsWith('.md') || entry.name.endsWith('.toml')) && entry.name !== 'CHANGELOG.md') {\n          const content = fs.readFileSync(fullPath, 'utf8');\n          const matches = content.match(/(?:~|\\$HOME)\\/\\.claude\\b/g);\n          if (matches) {\n            leakedPaths.push({ file: fullPath.replace(targetDir + '/', ''), count: matches.length });\n          }\n        }\n      }\n    }\n    scanForLeakedPaths(targetDir);\n    if (leakedPaths.length > 0) {\n      const totalLeaks = leakedPaths.reduce((sum, l) => sum + l.count, 0);\n      console.warn(`\\n  ${yellow}\u26A0${reset}  Found ${totalLeaks} unreplaced .claude path reference(s) in ${leakedPaths.length} file(s):`);\n      for (const leak of leakedPaths.slice(0, 5)) {\n        console.warn(`     ${dim}${leak.file}${reset} (${leak.count})`);\n      }\n      if (leakedPaths.length > 5) {\n        console.warn(`     ${dim}... and ${leakedPaths.length - 5} more file(s)${reset}`);\n      }\n      console.warn(`  ${dim}These paths may not resolve correctly for ${runtimeLabel}.${reset}`);\n    }\n  }\n\n  if (isCodex) {\n    // Generate Codex config.toml and per-agent .toml files\n    const agentCount = installCodexConfig(targetDir, agentsSrc);\n    console.log(`  ${green}\u2713${reset} Generated config.toml with ${agentCount} agent roles`);\n    console.log(`  ${green}\u2713${reset} Generated ${agentCount} agent .toml config files`);\n    return { settingsPath: null, settings: null, statuslineCommand: null, runtime };\n  }\n\n  if (isCopilot) {\n    // Generate copilot-instructions.md\n    const templatePath = path.join(targetDir, 'ez-agents', 'templates', 'copilot-instructions.md');\n    const instructionsPath = path.join(targetDir, 'copilot-instructions.md');\n    if (fs.existsSync(templatePath)) {\n      const template = fs.readFileSync(templatePath, 'utf8');\n      mergeCopilotInstructions(instructionsPath, template);\n      console.log(`  ${green}\u2713${reset} Generated copilot-instructions.md`);\n    }\n    // Copilot: no settings.json, no hooks, no statusline (like Codex)\n    return { settingsPath: null, settings: null, statuslineCommand: null, runtime };\n  }\n\n  if (isQwen || isKimi) {\n    // Qwen/Kimi: no settings.json hooks, no statusline (like Codex/Copilot)\n    // Skills are installed directly in skills/ez-*/SKILL.md\n    return { settingsPath: null, settings: null, statuslineCommand: null, runtime };\n  }\n\n  // Configure statusline and hooks in settings.json\n  // Gemini uses AfterTool instead of PostToolUse for post-tool hooks\n  const postToolEvent = runtime === 'gemini' ? 'AfterTool' : 'PostToolUse';\n  const settingsPath = path.join(targetDir, 'settings.json');\n  const settings = cleanupOrphanedHooks(readSettings(settingsPath));\n  const statuslineCommand = isGlobal\n    ? buildHookCommand(targetDir, 'ez-statusline.js')\n    : 'node ' + dirName + '/hooks/ez-statusline.js';\n  const updateCheckCommand = isGlobal\n    ? buildHookCommand(targetDir, 'ez-check-update.js')\n    : 'node ' + dirName + '/hooks/ez-check-update.js';\n  const contextMonitorCommand = isGlobal\n    ? buildHookCommand(targetDir, 'ez-context-monitor.js')\n    : 'node ' + dirName + '/hooks/ez-context-monitor.js';\n\n  // Enable experimental agents for Gemini CLI (required for custom sub-agents)\n  if (isGemini) {\n    if (!settings.experimental) {\n      settings.experimental = {};\n    }\n    if (!settings.experimental.enableAgents) {\n      settings.experimental.enableAgents = true;\n      console.log(`  ${green}\u2713${reset} Enabled experimental agents`);\n    }\n  }\n\n  // Configure SessionStart hook for update checking (skip for opencode)\n  if (!isOpencode) {\n    if (!settings.hooks) {\n      settings.hooks = {};\n    }\n    if (!settings.hooks.SessionStart) {\n      settings.hooks.SessionStart = [];\n    }\n\n    const hasEzUpdateHook = settings.hooks.SessionStart.some(entry =>\n      entry.hooks && entry.hooks.some(h => h.command && h.command.includes('ez-check-update'))\n    );\n\n    if (!hasEzUpdateHook) {\n      settings.hooks.SessionStart.push({\n        hooks: [\n          {\n            type: 'command',\n            command: updateCheckCommand\n          }\n        ]\n      });\n      console.log(`  ${green}\u2713${reset} Configured update check hook`);\n    }\n\n    // Configure post-tool hook for context window monitoring\n    if (!settings.hooks[postToolEvent]) {\n      settings.hooks[postToolEvent] = [];\n    }\n\n    const hasContextMonitorHook = settings.hooks[postToolEvent].some(entry =>\n      entry.hooks && entry.hooks.some(h => h.command && h.command.includes('ez-context-monitor'))\n    );\n\n    if (!hasContextMonitorHook) {\n      settings.hooks[postToolEvent].push({\n        hooks: [\n          {\n            type: 'command',\n            command: contextMonitorCommand\n          }\n        ]\n      });\n      console.log(`  ${green}\u2713${reset} Configured context window monitor hook`);\n    }\n  }\n\n  return { settingsPath, settings, statuslineCommand, runtime };\n}\n\n/**\n * Apply statusline config, then print completion message\n */\nfunction finishInstall(settingsPath, settings, statuslineCommand, shouldInstallStatusline, runtime = 'claude', isGlobal = true) {\n  const isOpencode = runtime === 'opencode';\n  const isCodex = runtime === 'codex';\n  const isCopilot = runtime === 'copilot';\n  const isQwen = runtime === 'qwen';\n  const isKimi = runtime === 'kimi';\n\n  if (shouldInstallStatusline && !isOpencode && !isCodex && !isCopilot && !isQwen && !isKimi) {\n    settings.statusLine = {\n      type: 'command',\n      command: statuslineCommand\n    };\n    console.log(`  ${green}\u2713${reset} Configured statusline`);\n  }\n\n  // Write settings when runtime supports settings.json\n  if (!isCodex && !isCopilot && !isQwen && !isKimi) {\n    writeSettings(settingsPath, settings);\n  }\n\n  // Configure OpenCode permissions\n  if (isOpencode) {\n    configureOpencodePermissions(isGlobal);\n  }\n\n  let program = 'Claude Code';\n  if (runtime === 'opencode') program = 'OpenCode';\n  if (runtime === 'gemini') program = 'Gemini';\n  if (runtime === 'codex') program = 'Codex';\n  if (runtime === 'copilot') program = 'Copilot';\n  if (runtime === 'qwen') program = 'Qwen Code';\n  if (runtime === 'kimi') program = 'Kimi Code';\n\n  let command = '/ez:new-project';\n  if (runtime === 'opencode' || runtime === 'gemini' || runtime === 'copilot' || runtime === 'qwen' || runtime === 'kimi') command = '/ez-new-project';\n  if (runtime === 'codex') command = '$ez-new-project';\n  console.log(`\n  ${green}Done!${reset} Open a blank directory in ${program} and run ${cyan}${command}${reset}.\n\n  ${cyan}Join the community:${reset} https://discord.gg/ez-agents\n`);\n}\n\n/**\n * Handle statusline configuration with optional prompt\n */\nfunction handleStatusline(settings, isInteractive, callback) {\n  const hasExisting = settings.statusLine != null;\n\n  if (!hasExisting) {\n    callback(true);\n    return;\n  }\n\n  if (forceStatusline) {\n    callback(true);\n    return;\n  }\n\n  if (!isInteractive) {\n    console.log(`  ${yellow}\u26A0${reset} Skipping statusline (already configured)`);\n    console.log(`    Use ${cyan}--force-statusline${reset} to replace\\n`);\n    callback(false);\n    return;\n  }\n\n  const existingCmd = settings.statusLine.command || settings.statusLine.url || '(custom)';\n\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout\n  });\n\n  console.log(`\n  ${yellow}\u26A0${reset} Existing statusline detected\\n\n  Your current statusline:\n    ${dim}command: ${existingCmd}${reset}\n\n  EZ_Agents includes a statusline showing:\n    \u2022 Model name\n    \u2022 Current task (from todo list)\n    \u2022 Context window usage (color-coded)\n\n  ${cyan}1${reset}) Keep existing\n  ${cyan}2${reset}) Replace with EZ_Agents statusline\n`);\n\n  rl.question(`  Choice ${dim}[1]${reset}: `, (answer) => {\n    rl.close();\n    const choice = answer.trim() || '1';\n    callback(choice === '2');\n  });\n}\n\n/**\n * Prompt for runtime selection\n */\nfunction promptRuntime(callback) {\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout\n  });\n\n  let answered = false;\n\n  rl.on('close', () => {\n    if (!answered) {\n      answered = true;\n      console.log(`\\n  ${yellow}Installation cancelled${reset}\\n`);\n      process.exit(0);\n    }\n  });\n\n  console.log(`  ${yellow}Which runtime(s) would you like to install for?${reset}\\n\\n  ${cyan}1${reset}) Claude Code ${dim}(~/.claude)${reset}\n  ${cyan}2${reset}) OpenCode    ${dim}(~/.config/opencode)${reset} - open source, free models\n  ${cyan}3${reset}) Gemini      ${dim}(~/.gemini)${reset}\n  ${cyan}4${reset}) Codex       ${dim}(~/.codex)${reset}\n  ${cyan}5${reset}) Copilot     ${dim}(~/.copilot)${reset}\n  ${cyan}6${reset}) Qwen Code   ${dim}(~/.qwen)${reset} - Alibaba Qwen official CLI\n  ${cyan}7${reset}) Kimi Code   ${dim}(~/.kimi)${reset} - Moonshot Kimi official CLI\n  ${cyan}8${reset}) All ${dim}(all 7 runtimes above)${reset}\n\n  ${dim}Note: OpenAI and Anthropic are model providers configured within each runtime's settings,\n  not separate CLI runtimes. See README.md for model configuration.${reset}\n`);\n\n  rl.question(`  Choice ${dim}[1]${reset}: `, (answer) => {\n    answered = true;\n    rl.close();\n    const choice = answer.trim() || '1';\n    if (choice === '8') {\n      callback(['claude', 'opencode', 'gemini', 'codex', 'copilot', 'qwen', 'kimi']);\n    } else if (choice === '7') {\n      callback(['kimi']);\n    } else if (choice === '6') {\n      callback(['qwen']);\n    } else if (choice === '5') {\n      callback(['copilot']);\n    } else if (choice === '4') {\n      callback(['codex']);\n    } else if (choice === '3') {\n      callback(['gemini']);\n    } else if (choice === '2') {\n      callback(['opencode']);\n    } else {\n      callback(['claude']);\n    }\n  });\n}\n\n/**\n * Prompt for install location\n */\nfunction promptLocation(runtimes) {\n  if (!process.stdin.isTTY) {\n    console.log(`  ${yellow}Non-interactive terminal detected, defaulting to global install${reset}\\n`);\n    installAllRuntimes(runtimes, true, false);\n    return;\n  }\n\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout\n  });\n\n  let answered = false;\n\n  rl.on('close', () => {\n    if (!answered) {\n      answered = true;\n      console.log(`\\n  ${yellow}Installation cancelled${reset}\\n`);\n      process.exit(0);\n    }\n  });\n\n  const pathExamples = runtimes.map(r => {\n    const globalPath = getGlobalDir(r, explicitConfigDir);\n    return globalPath.replace(os.homedir(), '~');\n  }).join(', ');\n\n  const localExamples = runtimes.map(r => `./${getDirName(r)}`).join(', ');\n\n  console.log(`  ${yellow}Where would you like to install?${reset}\\n\\n  ${cyan}1${reset}) Global ${dim}(${pathExamples})${reset} - available in all projects\n  ${cyan}2${reset}) Local  ${dim}(${localExamples})${reset} - this project only\n`);\n\n  rl.question(`  Choice ${dim}[1]${reset}: `, (answer) => {\n    answered = true;\n    rl.close();\n    const choice = answer.trim() || '1';\n    const isGlobal = choice !== '2';\n    installAllRuntimes(runtimes, isGlobal, true);\n  });\n}\n\n/**\n * Install EZ_Agents for all selected runtimes\n */\nfunction installAllRuntimes(runtimes: string[], isGlobal: boolean, isInteractive: boolean) {\n  const results: Array<{ settingsPath: string | null; settings: Record<string, unknown> | null; statuslineCommand: string | null; runtime: string }> = [];\n\n  for (const runtime of runtimes) {\n    const result = install(isGlobal, runtime);\n    results.push(result);\n  }\n\n  const statuslineRuntimes = ['claude', 'gemini'];\n  const statuslineResults = results.filter(r => statuslineRuntimes.includes(r.runtime));\n\n  // Handle statusline per-runtime to avoid conflicts\n  // Each runtime with settings.json gets its own statusline prompt\n  const finalizePerRuntime = (runtimeResult: { settingsPath: string | null; settings: Record<string, unknown> | null; statuslineCommand: string | null; runtime: string }, shouldInstallStatusline: boolean) => {\n    const useStatusline = statuslineRuntimes.includes(runtimeResult.runtime) && shouldInstallStatusline;\n    finishInstall(\n      runtimeResult.settingsPath,\n      runtimeResult.settings,\n      runtimeResult.statuslineCommand,\n      useStatusline,\n      runtimeResult.runtime,\n      isGlobal\n    );\n  };\n\n  if (statuslineResults.length === 0) {\n    // No runtimes support statusline - finalize all without statusline\n    for (const result of results) {\n      finalizePerRuntime(result, false);\n    }\n  } else if (statuslineResults.length === 1) {\n    // Single runtime with statusline support - prompt once\n    const singleResult = statuslineResults[0];\n    if (singleResult) {\n      handleStatusline(singleResult.settings, isInteractive, (shouldInstall) => {\n        finalizePerRuntime(singleResult, shouldInstall);\n        // Finalize other runtimes without statusline\n        for (const result of results) {\n          if (!statuslineRuntimes.includes(result.runtime)) {\n            finalizePerRuntime(result, false);\n          }\n        }\n      });\n    }\n  } else {\n    // Multiple runtimes with statusline support - prompt for each\n    let currentIndex = 0;\n    const statuslineChoices = new Array(statuslineResults.length).fill(false);\n\n    const promptNextStatusline = () => {\n      if (currentIndex >= statuslineResults.length) {\n        // All statusline prompts done - finalize all runtimes\n        for (let i = 0; i < results.length; i++) {\n          const result = results[i];\n          if (!result) continue;\n          const statuslineIndex = statuslineResults.findIndex(r => r.runtime === result.runtime);\n          const shouldInstall = statuslineIndex !== -1 ? statuslineChoices[statuslineIndex] : false;\n          finalizePerRuntime(result, shouldInstall);\n        }\n        return;\n      }\n\n      const currentResult = statuslineResults[currentIndex];\n      if (currentResult) {\n        const currentRuntimeIndex = currentIndex;\n        handleStatusline(currentResult.settings, isInteractive, (shouldInstall) => {\n          statuslineChoices[currentRuntimeIndex] = shouldInstall;\n          currentIndex++;\n          promptNextStatusline();\n        });\n      }\n    };\n\n    promptNextStatusline();\n  }\n}\n\n// Test-only exports \u2014 skip main logic when loaded as a module for testing\nconst testExports = {\n    getCodexSkillAdapterHeader,\n    convertClaudeToGeminiAgent,\n    convertClaudeAgentToCodexAgent,\n    generateCodexAgentToml,\n    generateCodexConfigBlock,\n    stripEzAgentsFromCodexConfig,\n    mergeCodexConfig,\n    installCodexConfig,\n    convertClaudeCommandToCodexSkill,\n    EZ_CODEX_MARKER,\n    CODEX_AGENT_SANDBOX,\n    getDirName,\n    getGlobalDir,\n    getConfigDirFromHome,\n    claudeToCopilotTools,\n    convertCopilotToolName,\n    convertClaudeToCopilotContent,\n    convertClaudeCommandToCopilotSkill,\n    convertClaudeAgentToCopilotAgent,\n    copyCommandsAsCopilotSkills,\n    EZ_COPILOT_INSTRUCTIONS_MARKER,\n    EZ_COPILOT_INSTRUCTIONS_CLOSE_MARKER,\n    mergeCopilotInstructions,\n    stripEzAgentsFromCopilotInstructions,\n    writeManifest,\n    reportLocalPatches,\n    convertClaudeToQwenSkill,\n    convertClaudeToQwenAgent,\n    convertClaudeToKimiSkill,\n    convertClaudeToKimiAgent,\n  };\n\n// Main logic\nif (hasGlobal && hasLocal) {\n  console.error(`  ${yellow}Cannot specify both --global and --local${reset}`);\n  process.exit(1);\n} else if (explicitConfigDir && hasLocal) {\n  console.error(`  ${yellow}Cannot use --config-dir with --local${reset}`);\n  process.exit(1);\n} else if (hasUninstall) {\n  if (!hasGlobal && !hasLocal) {\n    console.error(`  ${yellow}--uninstall requires --global or --local${reset}`);\n    process.exit(1);\n  }\n  const runtimes = selectedRuntimes.length > 0 ? selectedRuntimes : ['claude'];\n  for (const runtime of runtimes) {\n    uninstall(hasGlobal, runtime);\n  }\n} else if (selectedRuntimes.length > 0) {\n  if (!hasGlobal && !hasLocal) {\n    promptLocation(selectedRuntimes);\n  } else {\n    installAllRuntimes(selectedRuntimes, hasGlobal, false);\n  }\n} else if (hasGlobal || hasLocal) {\n  // Default to Claude if no runtime specified but location is\n  installAllRuntimes(['claude'], hasGlobal, false);\n} else {\n  // Interactive\n  if (!process.stdin.isTTY) {\n    console.log(`  ${yellow}Non-interactive terminal detected, defaulting to Claude Code global install${reset}\\n`);\n    installAllRuntimes(['claude'], true, false);\n  } else {\n    promptRuntime((runtimes) => {\n      promptLocation(runtimes);\n    });\n  }\n}\n\n// Export for testing\nexport { testExports };\n", "{\n  \"name\": \"@howlil/ez-agents\",\n  \"version\": \"5.0.2\",\n  \"description\": \"EZ Agents \u2014 Meta-prompting with multi-model support (Qwen, Kimi, OpenAI)\",\n  \"type\": \"module\",\n  \"sideEffects\": false,\n  \"bin\": {\n    \"ez-agents\": \"dist/bin/install.js\",\n    \"ez-agents-update\": \"dist/bin/update.js\"\n  },\n  \"exports\": {\n    \".\": {\n      \"types\": \"./dist/index.d.ts\",\n      \"import\": \"./dist/index.js\"\n    },\n    \"./factories\": {\n      \"types\": \"./dist/lib/factories/index.d.ts\",\n      \"import\": \"./dist/lib/factories/index.js\"\n    },\n    \"./strategies\": {\n      \"types\": \"./dist/lib/strategies/index.d.ts\",\n      \"import\": \"./dist/lib/strategies/index.js\"\n    },\n    \"./adapters\": {\n      \"types\": \"./dist/lib/adapters/index.d.ts\",\n      \"import\": \"./dist/lib/adapters/index.js\"\n    },\n    \"./observer\": {\n      \"types\": \"./dist/lib/observer/index.d.ts\",\n      \"import\": \"./dist/lib/observer/index.js\"\n    },\n    \"./decorators\": {\n      \"types\": \"./dist/lib/decorators/index.d.ts\",\n      \"import\": \"./dist/lib/decorators/index.js\"\n    },\n    \"./facades\": {\n      \"types\": \"./dist/lib/facades/index.d.ts\",\n      \"import\": \"./dist/lib/facades/index.js\"\n    },\n    \"./commands\": {\n      \"types\": \"./dist/lib/commands/index.d.ts\",\n      \"import\": \"./dist/lib/commands/index.js\"\n    },\n    \"./services\": {\n      \"types\": \"./dist/lib/services/index.d.ts\",\n      \"import\": \"./dist/lib/services/index.js\"\n    },\n    \"./repositories\": {\n      \"types\": \"./dist/lib/repositories/index.d.ts\",\n      \"import\": \"./dist/lib/repositories/index.js\"\n    },\n    \"./fp\": {\n      \"types\": \"./dist/lib/fp/index.d.ts\",\n      \"import\": \"./dist/lib/fp/index.js\"\n    }\n  },\n  \"main\": \"./dist/index.js\",\n  \"types\": \"./dist/index.d.ts\",\n  \"files\": [\n    \"dist\",\n    \"commands\",\n    \"ez-agents\",\n    \"agents\",\n    \"hooks/dist\",\n    \"scripts\"\n  ],\n  \"keywords\": [\n    \"claude\",\n    \"claude-code\",\n    \"ai\",\n    \"meta-prompting\",\n    \"context-engineering\",\n    \"spec-driven-development\",\n    \"gemini\",\n    \"gemini-cli\",\n    \"codex\",\n    \"codex-cli\",\n    \"ez-agents\",\n    \"agent-orchestration\"\n  ],\n  \"author\": \"T\u00C2CHES\",\n  \"license\": \"MIT\",\n  \"repository\": {\n    \"type\": \"git\",\n    \"url\": \"git+https://github.com/howlil/ez-agents.git\"\n  },\n  \"homepage\": \"https://github.com/howlil/ez-agents\",\n  \"bugs\": {\n    \"url\": \"https://github.com/howlil/ez-agents/issues\"\n  },\n  \"engines\": {\n    \"node\": \">=16.7.0\"\n  },\n  \"devDependencies\": {\n    \"@types/micromatch\": \"4.0.0\",\n    \"@types/node\": \"20.0.0\",\n    \"@typescript-eslint/eslint-plugin\": \"8.57.2\",\n    \"@typescript-eslint/parser\": \"8.57.2\",\n    \"c8\": \"11.0.0\",\n    \"complexity-report\": \"2.0.0-alpha\",\n    \"dependency-cruiser\": \"17.3.9\",\n    \"eslint\": \"8.57.0\",\n    \"eslint-plugin-complexity\": \"1.0.2\",\n    \"eslint-plugin-sonarjs\": \"4.0.2\",\n    \"eslint-plugin-tsdoc\": \"0.5.2\",\n    \"expect-type\": \"1.3.0\",\n    \"fast-check\": \"3.0.0\",\n    \"glob\": \"13.0.6\",\n    \"husky\": \"9.1.7\",\n    \"js-yaml\": \"4.1.1\",\n    \"jscpd\": \"4.0.8\",\n    \"jsdoc\": \"4.0.2\",\n    \"k6\": \"0.0.0\",\n    \"lint-staged\": \"15.5.2\",\n    \"madge\": \"8.0.0\",\n    \"markdownlint\": \"0.40.0\",\n    \"ts-morph\": \"27.0.2\",\n    \"ts-node\": \"10.9.0\",\n    \"ts-prune\": \"0.10.3\",\n    \"tsup\": \"8.0.0\",\n    \"typedoc\": \"0.28.18\",\n    \"typedoc-plugin-missing-exports\": \"4.1.2\",\n    \"typescript\": \"5.4.5\",\n    \"vitest\": \"4.1.0\"\n  },\n  \"scripts\": {\n    \"build\": \"tsup\",\n    \"build:watch\": \"tsup --watch\",\n    \"typecheck\": \"tsc --noEmit\",\n    \"typecheck:strict\": \"tsc --noEmit --strict\",\n    \"dev\": \"tsx bin/install.ts\",\n    \"lint\": \"eslint . --ext .ts --max-warnings=0\",\n    \"lint:fix\": \"eslint . --ext .ts --fix\",\n    \"build:hooks\": \"node dist/scripts/build-hooks.js\",\n    \"prepublishOnly\": \"npm run build && npm run build:hooks\",\n    \"test\": \"tsx scripts/run-tests.ts\",\n    \"test:coverage\": \"c8 --check-coverage --lines 70 --reporter text --include 'bin/lib/**/*.ts' --exclude 'tests/**' --all tsx scripts/run-tests.ts\",\n    \"test:a11y\": \"node tests/accessibility/run-a11y-tests.cjs\",\n    \"test:qwen:code\": \"sh tests/staging/qwen-code-battle-full.sh\",\n    \"test:qwen:code:simple\": \"sh tests/staging/qwen-code-simple-test.sh\",\n    \"test:qwen:code:edge\": \"sh tests/staging/qwen-code-edge-cases.sh\",\n    \"test:docker:qwen:code\": \"docker-compose -f docker-compose.staging.yml up --build ez-agents-qwen-code-battle\",\n    \"test:docker:qwen:edge\": \"docker-compose -f docker-compose.staging.yml up --build ez-agents-qwen-code-edge-cases\",\n    \"test:docker:all\": \"docker-compose -f docker-compose.staging.yml up --build --abort-on-container-exit\",\n    \"install:production\": \"node --loader ts-node/esm scripts/install-production.ts\",\n    \"install:production:latest\": \"node --loader ts-node/esm scripts/install-production.ts latest\",\n    \"install:production:v4\": \"node --loader ts-node/esm scripts/install-production.ts 4.0.0\",\n    \"validate:token\": \"node --loader ts-node/esm scripts/validate-token.ts\",\n    \"security:scan\": \"npm audit --audit-level=moderate\",\n    \"docs:generate\": \"jsdoc -c jsdoc.json -r bin/ ez-agents/bin/lib/\",\n    \"docs:api\": \"typedoc\",\n    \"analyze:duplicates\": \"jscpd --config .jscpd.json\",\n    \"check:duplicates\": \"jscpd --config .jscpd.json\",\n    \"check:complexity\": \"node --loader ts-node/esm scripts/check-complexity.ts\",\n    \"check:tsdoc\": \"node --loader ts-node/esm scripts/check-tsdoc-coverage.ts\",\n    \"test:package\": \"node --loader ts-node/esm scripts/test-package-docker.ts\",\n    \"check:coupling\": \"madge --circular bin/lib/\",\n    \"check:abstractions\": \"ts-prune\",\n    \"check:all-metrics\": \"npm run lint && npm run check:duplicates && npm run check:coupling\",\n    \"prepare\": \"husky\",\n    \"gates:local\": \"node ez-agents/bin/lib/gate-executor.cjs --ci --cache\",\n    \"gates:status\": \"node ez-agents/bin/lib/gate-executor.cjs --status\",\n    \"gates:stats\": \"node ez-agents/bin/lib/gate-executor.cjs --stats\",\n    \"deps:audit\": \"tsx bin/dependency-audit.ts\",\n    \"deps:audit:json\": \"tsx bin/dependency-audit.ts --json\",\n    \"deps:check\": \"tsx bin/dependency-audit.ts --quiet\"\n  },\n  \"lint-staged\": {\n    \"*.ts\": [\n      \"node --test\"\n    ],\n    \"*.md\": [\n      \"tsx scripts/run-tests.ts\"\n    ]\n  },\n  \"dependencies\": {\n    \"micromatch\": \"4.0.5\",\n    \"tiktoken\": \"1.0.22\",\n    \"zod\": \"4.3.6\"\n  }\n}\n"],
  "mappings": "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",
  "names": ["runtimeLabel", "scanForLeakedPaths"]
}
