poe-code 3.0.154 → 3.0.156
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.
- package/dist/cli/commands/auth.js +7 -2
- package/dist/cli/commands/auth.js.map +1 -1
- package/dist/cli/commands/configure.js +3 -2
- package/dist/cli/commands/configure.js.map +1 -1
- package/dist/cli/commands/experiment.js +2 -1
- package/dist/cli/commands/experiment.js.map +1 -1
- package/dist/cli/commands/generate.js +3 -3
- package/dist/cli/commands/generate.js.map +1 -1
- package/dist/cli/commands/install.js +3 -4
- package/dist/cli/commands/install.js.map +1 -1
- package/dist/cli/commands/launch.js +15 -7
- package/dist/cli/commands/launch.js.map +1 -1
- package/dist/cli/commands/logout.js +1 -1
- package/dist/cli/commands/logout.js.map +1 -1
- package/dist/cli/commands/mcp.js +11 -11
- package/dist/cli/commands/mcp.js.map +1 -1
- package/dist/cli/commands/pipeline.js +3 -2
- package/dist/cli/commands/pipeline.js.map +1 -1
- package/dist/cli/commands/plan.d.ts +3 -0
- package/dist/cli/commands/plan.js +285 -0
- package/dist/cli/commands/plan.js.map +1 -0
- package/dist/cli/commands/ralph.js +2 -1
- package/dist/cli/commands/ralph.js.map +1 -1
- package/dist/cli/commands/shared.d.ts +4 -0
- package/dist/cli/commands/shared.js +19 -3
- package/dist/cli/commands/shared.js.map +1 -1
- package/dist/cli/commands/skill.js +10 -18
- package/dist/cli/commands/skill.js.map +1 -1
- package/dist/cli/commands/spawn.js +24 -7
- package/dist/cli/commands/spawn.js.map +1 -1
- package/dist/cli/commands/test.js +3 -4
- package/dist/cli/commands/test.js.map +1 -1
- package/dist/cli/commands/unconfigure.js +2 -2
- package/dist/cli/commands/unconfigure.js.map +1 -1
- package/dist/cli/commands/usage.js +1 -1
- package/dist/cli/commands/usage.js.map +1 -1
- package/dist/cli/commands/utils.js +2 -1
- package/dist/cli/commands/utils.js.map +1 -1
- package/dist/cli/program.js +139 -188
- package/dist/cli/program.js.map +1 -1
- package/dist/index.js +1324 -443
- package/dist/index.js.map +4 -4
- package/dist/providers/claude-code.js +1 -1
- package/dist/providers/claude-code.js.map +2 -2
- package/dist/providers/codex.js +1 -1
- package/dist/providers/codex.js.map +2 -2
- package/dist/providers/kimi.js +1 -1
- package/dist/providers/kimi.js.map +2 -2
- package/dist/providers/opencode.js +1 -1
- package/dist/providers/opencode.js.map +2 -2
- package/dist/providers/poe-agent.js +18 -9
- package/dist/providers/poe-agent.js.map +2 -2
- package/package.json +3 -2
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../node_modules/sisteransi/src/index.js", "../../src/templates/py-poe-spawn/env.hbs", "../../src/templates/py-poe-spawn/main.py.hbs", "../../src/templates/py-poe-spawn/requirements.txt.hbs", "../../src/templates/codex/config.toml.hbs", "../../packages/agent-spawn/src/run-command.ts", "../../packages/agent-defs/src/agents/claude-code.ts", "../../packages/agent-defs/src/agents/claude-desktop.ts", "../../packages/agent-defs/src/agents/codex.ts", "../../packages/agent-defs/src/agents/opencode.ts", "../../packages/agent-defs/src/agents/kimi.ts", "../../packages/agent-defs/src/registry.ts", "../../packages/agent-spawn/src/configs/mcp.ts", "../../packages/agent-spawn/src/configs/claude-code.ts", "../../packages/agent-spawn/src/configs/codex.ts", "../../packages/agent-spawn/src/configs/opencode.ts", "../../packages/agent-spawn/src/configs/kimi.ts", "../../packages/agent-spawn/src/configs/index.ts", "../../packages/agent-spawn/src/spawn.ts", "../../packages/agent-spawn/src/configs/resolve-config.ts", "../../packages/agent-spawn/src/mcp-args.ts", "../../packages/agent-spawn/src/model-utils.ts", "../../packages/agent-spawn/src/spawn-interactive.ts", "../../packages/design-system/src/tokens/colors.ts", "../../packages/design-system/src/tokens/typography.ts", "../../packages/design-system/src/components/text.ts", "../../packages/design-system/src/internal/output-format.ts", "../../packages/design-system/src/internal/theme-detect.ts", "../../packages/design-system/src/components/symbols.ts", "../../packages/design-system/src/components/logger.ts", "../../packages/design-system/src/prompts/primitives/log.ts", "../../packages/design-system/src/internal/strip-ansi.ts", "../../packages/design-system/src/components/table.ts", "../../packages/design-system/src/acp/components.ts", "../../packages/design-system/src/prompts/index.ts", "../../packages/design-system/src/prompts/primitives/cancel.ts", "../../node_modules/node_modules/.pnpm/fast-string-truncated-width@1.2.1/node_modules/fast-string-truncated-width/dist/utils.js", "../../node_modules/node_modules/.pnpm/fast-string-truncated-width@1.2.1/node_modules/fast-string-truncated-width/dist/index.js", "../../node_modules/node_modules/.pnpm/fast-string-width@1.1.0/node_modules/fast-string-width/dist/index.js", "../../node_modules/node_modules/.pnpm/fast-wrap-ansi@0.1.3/node_modules/fast-wrap-ansi/lib/main.js", "../../node_modules/@clack/core/src/utils/settings.ts", "../../node_modules/@clack/core/src/utils/string.ts", "../../node_modules/@clack/core/src/utils/index.ts", "../../node_modules/@clack/core/src/prompts/prompt.ts", "../../node_modules/@clack/core/src/prompts/autocomplete.ts", "../../node_modules/@clack/core/src/prompts/confirm.ts", "../../node_modules/@clack/core/src/prompts/group-multiselect.ts", "../../node_modules/@clack/core/src/utils/cursor.ts", "../../node_modules/@clack/core/src/prompts/multi-select.ts", "../../node_modules/@clack/core/src/prompts/password.ts", "../../node_modules/@clack/core/src/prompts/select.ts", "../../node_modules/@clack/core/src/prompts/select-key.ts", "../../node_modules/@clack/core/src/prompts/text.ts", "../../packages/design-system/src/prompts/primitives/intro.ts", "../../packages/design-system/src/prompts/primitives/note.ts", "../../packages/design-system/src/prompts/primitives/outro.ts", "../../packages/design-system/src/prompts/primitives/spinner.ts", "../../packages/design-system/src/static/spinner.ts", "../../packages/design-system/src/static/menu.ts", "../../packages/agent-spawn/src/acp/spawn.ts", "../../packages/poe-acp-client/src/acp-client.ts", "../../packages/poe-acp-client/src/acp-transport.ts", "../../packages/poe-acp-client/src/run-report.ts", "../../packages/agent-spawn/src/acp/middlewares/spawn-log.ts", "../../src/utils/command-checks.ts", "../../packages/config-mutations/src/mutations/config-mutation.ts", "../../packages/config-mutations/src/mutations/file-mutation.ts", "../../packages/config-mutations/src/execution/apply-mutation.ts", "../../packages/config-mutations/src/formats/json.ts", "../../packages/config-mutations/src/formats/toml.ts", "../../packages/config-mutations/src/formats/index.ts", "../../packages/config-mutations/src/execution/path-utils.ts", "../../packages/config-mutations/src/fs-utils.ts", "../../packages/config-mutations/src/execution/run-mutations.ts", "../../packages/config-mutations/src/template/render.ts", "../../src/cli/constants.ts", "../../src/services/service-install.ts", "../../src/providers/create-provider.ts", "../../src/providers/kimi.ts"],
|
|
4
|
-
"sourcesContent": ["'use strict';\n\nconst ESC = '\\x1B';\nconst CSI = `${ESC}[`;\nconst beep = '\\u0007';\n\nconst cursor = {\n to(x, y) {\n if (!y) return `${CSI}${x + 1}G`;\n return `${CSI}${y + 1};${x + 1}H`;\n },\n move(x, y) {\n let ret = '';\n\n if (x < 0) ret += `${CSI}${-x}D`;\n else if (x > 0) ret += `${CSI}${x}C`;\n\n if (y < 0) ret += `${CSI}${-y}A`;\n else if (y > 0) ret += `${CSI}${y}B`;\n\n return ret;\n },\n up: (count = 1) => `${CSI}${count}A`,\n down: (count = 1) => `${CSI}${count}B`,\n forward: (count = 1) => `${CSI}${count}C`,\n backward: (count = 1) => `${CSI}${count}D`,\n nextLine: (count = 1) => `${CSI}E`.repeat(count),\n prevLine: (count = 1) => `${CSI}F`.repeat(count),\n left: `${CSI}G`,\n hide: `${CSI}?25l`,\n show: `${CSI}?25h`,\n save: `${ESC}7`,\n restore: `${ESC}8`\n}\n\nconst scroll = {\n up: (count = 1) => `${CSI}S`.repeat(count),\n down: (count = 1) => `${CSI}T`.repeat(count)\n}\n\nconst erase = {\n screen: `${CSI}2J`,\n up: (count = 1) => `${CSI}1J`.repeat(count),\n down: (count = 1) => `${CSI}J`.repeat(count),\n line: `${CSI}2K`,\n lineEnd: `${CSI}K`,\n lineStart: `${CSI}1K`,\n lines(count) {\n let clear = '';\n for (let i = 0; i < count; i++)\n clear += this.line + (i < count - 1 ? cursor.up() : '');\n if (count)\n clear += cursor.left;\n return clear;\n }\n}\n\nmodule.exports = { cursor, scroll, erase, beep };\n", "POE_API_KEY={{apiKey}}\nPOE_BASE_URL=https://api.poe.com/v1\nMODEL={{model}}\n", "import os\nfrom openai import OpenAI\nfrom dotenv import load_dotenv\n\nload_dotenv()\n\nclient = OpenAI(\n api_key=os.getenv(\"POE_API_KEY\"),\n base_url=os.getenv(\"POE_BASE_URL\")\n)\n\nresponse = client.chat.completions.create(\n model=os.getenv(\"MODEL\", \"{{model}}\"),\n messages=[{\"role\": \"user\", \"content\": \"Tell me a joke\"}]\n)\n\nprint(response.choices[0].message.content)\n", "openai>=1.0.0\npython-dotenv>=1.0.0\n", "model_provider = \"poe\"\n\n[profiles.\"{{{profileName}}}\"]\nmodel = \"{{{model}}}\"\nmodel_provider = \"poe\"\nmodel_reasoning_effort = \"{{reasoningEffort}}\"\nmodel_verbosity = \"medium\"\n\n[model_providers.poe]\nname = \"poe\"\nbase_url = \"{{{baseUrl}}}\"\nwire_api = \"responses\"\nexperimental_bearer_token = \"{{apiKey}}\"\nrequires_openai_auth = false\nsupports_websockets = false\n", "import { spawn } from \"node:child_process\";\n\nexport interface CommandRunnerResult {\n stdout: string;\n stderr: string;\n exitCode: number;\n}\n\nexport interface CommandRunnerOptions {\n cwd?: string;\n env?: Record<string, string | undefined>;\n stdin?: string | Buffer;\n}\n\nexport type CommandRunner = (\n command: string,\n args: string[],\n options?: CommandRunnerOptions\n) => Promise<CommandRunnerResult>;\n\nexport function runCommand(\n command: string,\n args: string[],\n options?: CommandRunnerOptions\n): Promise<CommandRunnerResult> {\n return new Promise((resolve) => {\n const hasStdin = options?.stdin != null;\n const child = spawn(command, args, {\n stdio: [hasStdin ? \"pipe\" : \"ignore\", \"pipe\", \"pipe\"],\n cwd: options?.cwd,\n env: options?.env\n ? {\n ...(process.env as Record<string, string | undefined>),\n ...options.env\n }\n : undefined\n });\n let stdout = \"\";\n let stderr = \"\";\n\n if (hasStdin && child.stdin) {\n child.stdin.on(\"error\", () => {});\n child.stdin.end(options!.stdin);\n }\n\n child.stdout?.setEncoding(\"utf8\");\n child.stdout?.on(\"data\", (chunk: string | Buffer) => {\n stdout += chunk.toString();\n });\n\n child.stderr?.setEncoding(\"utf8\");\n child.stderr?.on(\"data\", (chunk: string | Buffer) => {\n stderr += chunk.toString();\n });\n\n child.on(\"error\", (error: NodeJS.ErrnoException) => {\n const exitCode =\n typeof error.code === \"number\"\n ? error.code\n : typeof error.errno === \"number\"\n ? error.errno\n : 127;\n const message =\n error instanceof Error ? error.message : String(error ?? \"error\");\n resolve({\n stdout,\n stderr: stderr ? `${stderr}${message}` : message,\n exitCode\n });\n });\n\n child.on(\"close\", (code) => {\n resolve({\n stdout,\n stderr,\n exitCode: code ?? 0\n });\n });\n });\n}\n", "import type { AgentDefinition } from \"../types.js\";\n\nexport const claudeCodeAgent: AgentDefinition = {\n id: \"claude-code\",\n name: \"claude-code\",\n label: \"Claude Code\",\n summary: \"Configure Claude Code to route through Poe.\",\n aliases: [\"claude\"],\n binaryName: \"claude\",\n configPath: \"~/.claude/settings.json\",\n branding: {\n colors: {\n dark: \"#C15F3C\",\n light: \"#C15F3C\"\n }\n }\n};\n", "import type { AgentDefinition } from \"../types.js\";\n\nexport const claudeDesktopAgent: AgentDefinition = {\n id: \"claude-desktop\",\n name: \"claude-desktop\",\n label: \"Claude Desktop\",\n summary: \"Anthropic's official desktop application for Claude\",\n configPath: \"~/.claude/settings.json\",\n branding: {\n colors: {\n dark: \"#D97757\",\n light: \"#D97757\"\n }\n }\n};\n", "import type { AgentDefinition } from \"../types.js\";\n\nexport const codexAgent: AgentDefinition = {\n id: \"codex\",\n name: \"codex\",\n label: \"Codex\",\n summary: \"Configure Codex to use Poe as the model provider.\",\n binaryName: \"codex\",\n configPath: \"~/.codex/config.toml\",\n branding: {\n colors: {\n dark: \"#D5D9DF\",\n light: \"#7A7F86\"\n }\n }\n};\n", "import type { AgentDefinition } from \"../types.js\";\n\nexport const openCodeAgent: AgentDefinition = {\n id: \"opencode\",\n name: \"opencode\",\n label: \"OpenCode CLI\",\n summary: \"Configure OpenCode CLI to use the Poe API.\",\n binaryName: \"opencode\",\n configPath: \"~/.config/opencode/config.json\",\n branding: {\n colors: {\n dark: \"#4A4F55\",\n light: \"#2F3338\"\n }\n }\n};\n", "import type { AgentDefinition } from \"../types.js\";\n\nexport const kimiAgent: AgentDefinition = {\n id: \"kimi\",\n name: \"kimi\",\n label: \"Kimi\",\n summary: \"Configure Kimi CLI to use Poe API\",\n aliases: [\"kimi-cli\"],\n binaryName: \"kimi\",\n configPath: \"~/.kimi/config.toml\",\n branding: {\n colors: {\n dark: \"#7B68EE\",\n light: \"#6A5ACD\"\n }\n }\n};\n", "import type { AgentDefinition } from \"./types.js\";\nimport {\n claudeCodeAgent,\n claudeDesktopAgent,\n codexAgent,\n openCodeAgent,\n kimiAgent\n} from \"./agents/index.js\";\n\nexport const allAgents: AgentDefinition[] = [\n claudeCodeAgent,\n claudeDesktopAgent,\n codexAgent,\n openCodeAgent,\n kimiAgent\n];\n\nconst lookup = new Map<string, string>();\n\nfor (const agent of allAgents) {\n const values = [agent.id, agent.name, ...(agent.aliases ?? [])];\n for (const value of values) {\n const normalized = value.toLowerCase();\n if (!lookup.has(normalized)) {\n lookup.set(normalized, agent.id);\n }\n }\n}\n\nexport function resolveAgentId(input: string): string | undefined {\n if (!input) {\n return undefined;\n }\n return lookup.get(input.toLowerCase());\n}\n", "import type { McpSpawnConfig } from \"../types.js\";\n\ninterface JsonMcpServer {\n command: string;\n args?: string[];\n env?: Record<string, string>;\n}\n\nfunction toJsonMcpServers(servers: McpSpawnConfig): Record<string, JsonMcpServer> {\n const out: Record<string, JsonMcpServer> = {};\n\n for (const [name, server] of Object.entries(servers)) {\n const mapped: JsonMcpServer = { command: server.command };\n if (server.args && server.args.length > 0) {\n mapped.args = server.args;\n }\n if (server.env && Object.keys(server.env).length > 0) {\n mapped.env = server.env;\n }\n out[name] = mapped;\n }\n\n return out;\n}\n\nfunction toTomlString(value: string): string {\n return JSON.stringify(value);\n}\n\nfunction toTomlArray(values: string[]): string {\n const serialized = values.map((value) => toTomlString(value));\n return `[${serialized.join(\", \")}]`;\n}\n\nfunction toTomlInlineTable(values: Record<string, string>): string {\n const parts: string[] = [];\n for (const [key, value] of Object.entries(values)) {\n parts.push(`${JSON.stringify(key)}=${toTomlString(value)}`);\n }\n return `{${parts.join(\", \")}}`;\n}\n\nexport function serializeJsonMcpArgs(servers: McpSpawnConfig): string[] {\n return [\"--mcp-servers\", JSON.stringify({ mcpServers: toJsonMcpServers(servers) })];\n}\n\nexport function serializeOpenCodeMcpEnv(servers: McpSpawnConfig): Record<string, string> {\n const mcp: Record<\n string,\n { type: \"local\"; command: string[]; environment?: Record<string, string> }\n > = {};\n for (const [name, server] of Object.entries(servers)) {\n const entry: {\n type: \"local\";\n command: string[];\n environment?: Record<string, string>;\n } = { type: \"local\", command: [server.command, ...(server.args ?? [])] };\n if (server.env && Object.keys(server.env).length > 0) {\n entry.environment = server.env;\n }\n mcp[name] = entry;\n }\n return { OPENCODE_CONFIG_CONTENT: JSON.stringify({ mcp }) };\n}\n\nexport function serializeCodexMcpArgs(servers: McpSpawnConfig): string[] {\n const args: string[] = [];\n\n for (const [name, server] of Object.entries(servers)) {\n const prefix = `mcp_servers.${name}`;\n args.push(\"-c\", `${prefix}.command=${toTomlString(server.command)}`);\n\n if (server.args && server.args.length > 0) {\n args.push(\"-c\", `${prefix}.args=${toTomlArray(server.args)}`);\n }\n\n if (server.env && Object.keys(server.env).length > 0) {\n args.push(\"-c\", `${prefix}.env=${toTomlInlineTable(server.env)}`);\n }\n }\n\n return args;\n}\n", "import type { CliSpawnConfig } from \"../types.js\";\nimport { serializeJsonMcpArgs } from \"./mcp.js\";\n\nexport const claudeCodeSpawnConfig: CliSpawnConfig = {\n kind: \"cli\",\n agentId: \"claude-code\",\n // ACP adapter support: yes (adapter: \"claude\")\n adapter: \"claude\",\n promptFlag: \"-p\",\n modelFlag: \"--model\",\n modelStripProviderPrefix: true,\n modelTransform: (model) => model.replaceAll(\".\", \"-\"),\n defaultArgs: [\n \"--output-format\",\n \"stream-json\",\n \"--verbose\"\n ],\n mcpArgs: serializeJsonMcpArgs,\n modes: {\n yolo: [\"--dangerously-skip-permissions\"],\n edit: [\"--permission-mode\", \"acceptEdits\", \"--allowedTools\", \"Bash,Read,Write,Edit,Glob,Grep,NotebookEdit\"],\n read: [\"--permission-mode\", \"plan\"]\n },\n stdinMode: {\n omitPrompt: true,\n extraArgs: [\"--input-format\", \"text\"]\n },\n interactive: {\n defaultArgs: []\n },\n resumeCommand: (threadId) => [\"--resume\", threadId]\n};\n", "import type { CliSpawnConfig } from \"../types.js\";\nimport { serializeCodexMcpArgs } from \"./mcp.js\";\n\nexport const codexSpawnConfig: CliSpawnConfig = {\n kind: \"cli\",\n agentId: \"codex\",\n // ACP adapter support: yes (adapter: \"codex\")\n adapter: \"codex\",\n promptFlag: \"exec\",\n modelFlag: \"--model\",\n modelStripProviderPrefix: true,\n defaultArgs: [\"--skip-git-repo-check\", \"--json\"],\n mcpArgs: serializeCodexMcpArgs,\n mcpArgsBeforeCommand: true,\n modes: {\n yolo: [\"-s\", \"danger-full-access\"],\n edit: [\"-s\", \"workspace-write\"],\n read: [\"-s\", \"read-only\"]\n },\n stdinMode: {\n omitPrompt: true,\n extraArgs: [\"-\"]\n },\n interactive: {\n defaultArgs: [\"-a\", \"never\"]\n },\n resumeCommand: (threadId, cwd) => [\"resume\", \"-C\", cwd, threadId]\n};\n", "import type { AcpSpawnConfig, CliSpawnConfig } from \"../types.js\";\nimport { serializeOpenCodeMcpEnv } from \"./mcp.js\";\n\n/**\n * OpenCode JSON output format (empirically observed)\n *\n * OpenCode can emit \"raw JSON events\" when running a prompt via:\n * - `opencode run \"<prompt>\" --format json ...`\n *\n * Key observations (OpenCode CLI v1.1.47):\n * - Output is **NDJSON / line-delimited JSON**: one JSON object per stdout line.\n * - Each line is an event object with a top-level `type` string (NOT ACP's `{ event: ... }`).\n * - Common top-level fields:\n * - `type`: `\"step_start\" | \"text\" | \"tool_use\" | \"step_finish\" | ...`\n * - `timestamp`: number (ms since epoch)\n * - `sessionID`: string (e.g. `\"ses_...\"`)\n * - `part`: object with event-specific payload\n *\n * `text` events:\n * ```ts\n * {\n * type: \"text\",\n * sessionID: \"ses_...\",\n * part: {\n * type: \"text\",\n * messageID: \"msg_...\",\n * text: \"Hello ...\",\n * time: { start: 1770000000000, end: 1770000000000 }\n * }\n * }\n * ```\n *\n * Tool calls (`tool_use`):\n * - Represented as a single event with `part.type: \"tool\"` and `state.status`.\n * - `state.input` includes tool arguments; `state.output` is the tool result string.\n * ```ts\n * {\n * type: \"tool_use\",\n * sessionID: \"ses_...\",\n * part: {\n * type: \"tool\",\n * callID: \"call_...\",\n * tool: \"bash\",\n * state: {\n * status: \"completed\",\n * input: { command: \"echo hello\", description: \"...\" },\n * output: \"hello\\n\"\n * }\n * }\n * }\n * ```\n *\n * Step boundaries:\n * - `step_start` and `step_finish` wrap a single model/tool turn.\n * - `step_finish.part.tokens` contains token accounting:\n * `{ input, output, reasoning, cache: { read, write } }`\n *\n * Negative cases / gotchas:\n * - Some failures (e.g. invalid `--model` / unknown provider) can print a non-JSON stack trace\n * before any JSON events are emitted, even with `--format json`.\n * - If `--format json` is ever removed upstream, OpenCode will need a text-mode fallback\n * (no streaming event adapter).\n */\nexport const openCodeSpawnConfig: CliSpawnConfig = {\n kind: \"cli\",\n agentId: \"opencode\",\n // ACP adapter support: yes (adapter: \"opencode\").\n // OpenCode's `--format json` emits NDJSON events with `{ type, sessionID, part }`\n // (no `{ event, ... }` field), so it needs the OpenCode adapter (not \"native\").\n adapter: \"opencode\",\n promptFlag: \"run\",\n modelFlag: \"--model\",\n modelStripProviderPrefix: false,\n modelTransform: (model) => {\n return model.startsWith(\"poe/\") ? model : `poe/${model}`;\n },\n defaultArgs: [\"--format\", \"json\"],\n modes: {\n yolo: [],\n edit: [],\n read: [\"--agent\", \"plan\"]\n },\n interactive: {\n defaultArgs: [],\n promptFlag: \"--prompt\"\n },\n resumeCommand: (threadId, cwd) => [cwd, \"--session\", threadId],\n mcpEnv: serializeOpenCodeMcpEnv\n};\n\nexport const openCodeAcpSpawnConfig: AcpSpawnConfig = {\n kind: \"acp\",\n agentId: \"opencode\",\n acpArgs: [\"acp\"],\n skipAuth: true,\n mcpEnv: serializeOpenCodeMcpEnv,\n};\n", "import type { AcpSpawnConfig, CliSpawnConfig } from \"../types.js\";\nimport { serializeJsonMcpArgs } from \"./mcp.js\";\n\nexport const kimiSpawnConfig: CliSpawnConfig = {\n kind: \"cli\",\n agentId: \"kimi\",\n // ACP adapter support: yes (adapter: \"kimi\").\n // Kimi's `--output-format stream-json` emits OpenAI-style `{ role, content }` JSON\n // (no `{ event, ... }` field), so it needs the Kimi adapter (not \"native\").\n adapter: \"kimi\",\n promptFlag: \"-p\",\n modelStripProviderPrefix: true,\n defaultArgs: [\"--print\", \"--output-format\", \"stream-json\"],\n mcpArgs: serializeJsonMcpArgs,\n modes: {\n yolo: [\"--yolo\"],\n edit: [],\n read: []\n },\n stdinMode: {\n omitPrompt: true,\n extraArgs: [\"--input-format\", \"stream-json\"]\n },\n interactive: {\n defaultArgs: [],\n promptFlag: \"-p\"\n },\n resumeCommand: (threadId, cwd) => [\"--session\", threadId, \"--work-dir\", cwd]\n};\n\nexport const kimiAcpSpawnConfig: AcpSpawnConfig = {\n kind: \"acp\",\n agentId: \"kimi\",\n acpArgs: [\"acp\"],\n};\n", "import { resolveAgentId } from \"@poe-code/agent-defs\";\nimport type { AcpSpawnConfig, SpawnConfig } from \"../types.js\";\nimport { claudeCodeSpawnConfig } from \"./claude-code.js\";\nimport { codexSpawnConfig } from \"./codex.js\";\nimport { openCodeSpawnConfig, openCodeAcpSpawnConfig } from \"./opencode.js\";\nimport { kimiSpawnConfig, kimiAcpSpawnConfig } from \"./kimi.js\";\n\n// ACP adapter support (spawn streaming):\n// - Supported (has `adapter`): claude-code, codex, opencode, kimi\nexport const allSpawnConfigs: readonly SpawnConfig[] = [\n claudeCodeSpawnConfig,\n codexSpawnConfig,\n openCodeSpawnConfig,\n kimiSpawnConfig\n];\n\nconst lookup = new Map<string, SpawnConfig>();\n\nfor (const config of allSpawnConfigs) {\n lookup.set(config.agentId, config);\n}\n\nconst acpLookup = new Map<string, AcpSpawnConfig>();\nacpLookup.set(openCodeAcpSpawnConfig.agentId, openCodeAcpSpawnConfig);\nacpLookup.set(kimiAcpSpawnConfig.agentId, kimiAcpSpawnConfig);\n\nexport function getSpawnConfig(input: string): SpawnConfig | undefined {\n const resolvedId = resolveAgentId(input);\n if (!resolvedId) {\n return undefined;\n }\n return lookup.get(resolvedId);\n}\n\nexport function getAcpSpawnConfig(input: string): AcpSpawnConfig | undefined {\n const resolvedId = resolveAgentId(input);\n if (!resolvedId) {\n return undefined;\n }\n return acpLookup.get(resolvedId);\n}\n\nexport function supportsMcpAtSpawn(input: string): boolean {\n const config = getSpawnConfig(input);\n return (\n !!config &&\n config.kind === \"cli\" &&\n (typeof config.mcpArgs === \"function\" || typeof config.mcpEnv === \"function\")\n );\n}\n\nexport function listMcpSupportedAgents(): string[] {\n const supported: string[] = [];\n\n for (const config of allSpawnConfigs) {\n if (\n config.kind !== \"cli\" ||\n (typeof config.mcpArgs !== \"function\" && typeof config.mcpEnv !== \"function\")\n ) {\n continue;\n }\n supported.push(config.agentId);\n }\n\n return supported;\n}\n", "import { spawn as spawnChildProcess } from \"node:child_process\";\nimport { resolveConfig } from \"./configs/resolve-config.js\";\nimport { getMcpArgs } from \"./mcp-args.js\";\nimport { stripModelNamespace } from \"./model-utils.js\";\nimport type {\n CliSpawnConfig,\n McpSpawnConfig,\n SpawnContext,\n SpawnMode,\n SpawnOptions,\n SpawnResult,\n StdinMode\n} from \"./types.js\";\n\nfunction createAbortError(): Error {\n const error = new Error(\"Agent spawn aborted\");\n error.name = \"AbortError\";\n return error;\n}\n\nfunction createActivityTimeoutError(timeoutMs: number): Error {\n const error = new Error(\n `Agent spawn timed out after ${timeoutMs / 1000}s of inactivity`\n );\n error.name = \"ActivityTimeoutError\";\n return error;\n}\n\nexport function isActivityTimeoutError(error: unknown): boolean {\n return error instanceof Error && error.name === \"ActivityTimeoutError\";\n}\n\nexport interface BuildSpawnArgsOptions {\n prompt: string;\n model?: string;\n mode?: SpawnMode;\n args?: string[];\n mcpServers?: McpSpawnConfig;\n useStdin?: boolean;\n}\n\nexport interface BuildSpawnArgsResult {\n binaryName: string;\n args: string[];\n}\n\nfunction resolveCliConfig(agentId: string) {\n const resolved = resolveConfig(agentId);\n\n if (!resolved.spawnConfig) {\n throw new Error(`Agent \"${resolved.agentId}\" has no spawn config.`);\n }\n\n if (resolved.spawnConfig.kind !== \"cli\") {\n throw new Error(`Agent \"${resolved.agentId}\" does not support CLI spawn.`);\n }\n\n if (!resolved.binaryName) {\n throw new Error(`Agent \"${resolved.agentId}\" has no binaryName.`);\n }\n\n return {\n agentId: resolved.agentId,\n binaryName: resolved.binaryName,\n spawnConfig: resolved.spawnConfig\n };\n}\n\nfunction buildCliArgs(\n config: CliSpawnConfig,\n options: BuildSpawnArgsOptions,\n stdinMode?: StdinMode\n): string[] {\n const mcpArgs = getMcpArgs(config, options.mcpServers);\n\n const args: string[] = [];\n\n if (config.mcpArgsBeforeCommand) {\n args.push(...mcpArgs);\n }\n\n if (stdinMode) {\n args.push(\n config.promptFlag,\n ...(stdinMode.omitPrompt ? [] : [options.prompt]),\n ...stdinMode.extraArgs\n );\n } else {\n args.push(config.promptFlag, options.prompt);\n }\n\n if (options.model && config.modelFlag) {\n let model = config.modelStripProviderPrefix\n ? stripModelNamespace(options.model)\n : options.model;\n if (config.modelTransform) model = config.modelTransform(model);\n args.push(config.modelFlag, model);\n }\n\n args.push(...config.defaultArgs);\n\n if (!config.mcpArgsBeforeCommand) {\n args.push(...mcpArgs);\n }\n\n args.push(...config.modes[options.mode ?? \"yolo\"]);\n\n if (options.args && options.args.length > 0) {\n args.push(...options.args);\n }\n\n return args;\n}\n\nexport function buildSpawnArgs(\n agentId: string,\n options: BuildSpawnArgsOptions\n): BuildSpawnArgsResult {\n const { binaryName, spawnConfig } = resolveCliConfig(agentId);\n const stdinMode =\n options.useStdin && spawnConfig.stdinMode ? spawnConfig.stdinMode : undefined;\n return { binaryName, args: buildCliArgs(spawnConfig, options, stdinMode) };\n}\n\nexport async function spawn(\n agentId: string,\n options: SpawnOptions,\n context?: SpawnContext\n): Promise<SpawnResult> {\n if (options.signal?.aborted) {\n throw createAbortError();\n }\n\n const { agentId: resolvedId, binaryName, spawnConfig } = resolveCliConfig(agentId);\n\n const stdinMode =\n options.useStdin && spawnConfig.stdinMode ? spawnConfig.stdinMode : undefined;\n\n const spawnArgs = buildCliArgs(spawnConfig, options, stdinMode);\n\n if (context?.dryRun) {\n const rendered = [binaryName, ...spawnArgs].join(\" \");\n context.logger?.dryRun(rendered);\n return { stdout: \"\", stderr: \"\", exitCode: 0 };\n }\n\n const child = spawnChildProcess(binaryName, spawnArgs, {\n cwd: options.cwd,\n stdio: [stdinMode ? \"pipe\" : \"inherit\", \"pipe\", \"pipe\"]\n });\n\n if (!child.stdout || !child.stderr) {\n throw new Error(`Failed to spawn \"${resolvedId}\": missing stdio pipes.`);\n }\n\n const stdoutStream = child.stdout;\n const stderrStream = child.stderr;\n\n if (stdinMode) {\n if (!child.stdin) {\n throw new Error(`Failed to spawn \"${resolvedId}\": missing stdin pipe.`);\n }\n child.stdin.setDefaultEncoding(\"utf8\");\n child.stdin.write(options.prompt);\n child.stdin.end();\n }\n\n return new Promise<SpawnResult>((resolve, reject) => {\n let stdout = \"\";\n let stderr = \"\";\n let aborted = false;\n let timedOut = false;\n\n const onAbort = () => {\n aborted = true;\n child.kill(\"SIGTERM\");\n };\n\n options.signal?.addEventListener(\"abort\", onAbort, { once: true });\n\n let activityTimer: ReturnType<typeof setTimeout> | undefined;\n const resetActivityTimer = options.activityTimeoutMs\n ? () => {\n if (activityTimer) clearTimeout(activityTimer);\n activityTimer = setTimeout(() => {\n timedOut = true;\n child.kill(\"SIGTERM\");\n }, options.activityTimeoutMs);\n }\n : undefined;\n\n resetActivityTimer?.();\n\n const cleanup = () => {\n options.signal?.removeEventListener(\"abort\", onAbort);\n if (activityTimer) clearTimeout(activityTimer);\n };\n\n stdoutStream.setEncoding(\"utf8\");\n stdoutStream.on(\"data\", (chunk) => {\n stdout += chunk;\n resetActivityTimer?.();\n if (options.tee?.stdout) options.tee.stdout.write(chunk);\n });\n\n stderrStream.setEncoding(\"utf8\");\n stderrStream.on(\"data\", (chunk) => {\n stderr += chunk;\n resetActivityTimer?.();\n if (options.tee?.stderr) options.tee.stderr.write(chunk);\n });\n\n child.on(\"error\", (error) => {\n cleanup();\n if (aborted) {\n reject(createAbortError());\n return;\n }\n reject(error);\n });\n\n child.on(\"close\", (code) => {\n cleanup();\n if (aborted) {\n reject(createAbortError());\n return;\n }\n if (timedOut) {\n reject(createActivityTimeoutError(options.activityTimeoutMs!));\n return;\n }\n resolve({\n stdout,\n stderr,\n exitCode: code ?? 1\n });\n });\n });\n}\n", "import { allAgents, resolveAgentId } from \"@poe-code/agent-defs\";\nimport type { SpawnConfig } from \"../types.js\";\nimport { getSpawnConfig } from \"./index.js\";\n\nexport interface ResolvedSpawnConfig {\n agentId: string;\n binaryName?: string;\n spawnConfig?: SpawnConfig;\n}\n\nexport function resolveConfig(agentId: string): ResolvedSpawnConfig {\n const resolvedAgentId = resolveAgentId(agentId);\n if (!resolvedAgentId) {\n throw new Error(`Unknown agent \"${agentId}\".`);\n }\n\n const agentDefinition = allAgents.find((agent) => agent.id === resolvedAgentId);\n if (!agentDefinition) {\n throw new Error(`Unknown agent \"${agentId}\".`);\n }\n\n const spawnConfig = getSpawnConfig(resolvedAgentId);\n const binaryName = agentDefinition.binaryName;\n\n return { agentId: resolvedAgentId, binaryName, spawnConfig };\n}\n", "import { listMcpSupportedAgents } from \"./configs/index.js\";\nimport type { CliSpawnConfig, McpSpawnConfig } from \"./types.js\";\n\nexport function hasMcpServers(servers?: McpSpawnConfig): servers is McpSpawnConfig {\n if (!servers) {\n return false;\n }\n return Object.keys(servers).length > 0;\n}\n\nexport function getMcpArgs(\n config: CliSpawnConfig,\n servers?: McpSpawnConfig\n): string[] {\n if (!hasMcpServers(servers)) {\n return [];\n }\n if (!config.mcpArgs && !config.mcpEnv) {\n throw new Error(formatUnsupportedMcpSpawnMessage(config.agentId));\n }\n if (!config.mcpArgs) {\n return [];\n }\n return config.mcpArgs(servers);\n}\n\nexport function getMcpEnv(\n config: CliSpawnConfig,\n servers?: McpSpawnConfig\n): Record<string, string> {\n if (!hasMcpServers(servers) || !config.mcpEnv) {\n return {};\n }\n return config.mcpEnv(servers);\n}\n\nexport function formatUnsupportedMcpSpawnMessage(agentId: string): string {\n const supported = listMcpSupportedAgents();\n const supportedText = supported.length > 0 ? supported.join(\", \") : \"(none)\";\n return (\n `Agent \"${agentId}\" does not support MCP servers at spawn time.\\n` +\n `Agents with spawn-time MCP support: ${supportedText}`\n );\n}\n", "/**\n * Strips the provider namespace prefix from a model identifier.\n * e.g., \"anthropic/claude-opus-4.6\" \u2192 \"claude-opus-4.6\"\n *\n * CLI binaries (claude, codex, opencode, kimi) only accept bare model IDs.\n * Passing a namespaced model like \"anthropic/claude-opus-4.6\" causes the\n * binary to fail with \"model not found\". This function MUST be called\n * before passing any model to a CLI binary via spawn args.\n */\nexport function stripModelNamespace(model: string): string {\n const slashIndex = model.indexOf(\"/\");\n return slashIndex === -1 ? model : model.slice(slashIndex + 1);\n}\n", "import { spawn as spawnChildProcess } from \"node:child_process\";\nimport { resolveConfig } from \"./configs/resolve-config.js\";\nimport { getMcpArgs } from \"./mcp-args.js\";\nimport { stripModelNamespace } from \"./model-utils.js\";\nimport type { SpawnOptions, SpawnResult } from \"./types.js\";\n\nexport async function spawnInteractive(\n agentId: string,\n options: SpawnOptions\n): Promise<SpawnResult> {\n const resolved = resolveConfig(agentId);\n const spawnConfig = resolved.spawnConfig;\n\n if (!spawnConfig) {\n throw new Error(`Agent \"${resolved.agentId}\" has no spawn config.`);\n }\n\n if (spawnConfig.kind !== \"cli\") {\n throw new Error(`Agent \"${resolved.agentId}\" does not support CLI spawn.`);\n }\n\n if (!resolved.binaryName) {\n throw new Error(`Agent \"${resolved.agentId}\" has no binaryName.`);\n }\n\n if (!spawnConfig.interactive) {\n throw new Error(`Agent \"${resolved.agentId}\" does not support interactive mode.`);\n }\n\n const { interactive } = spawnConfig;\n\n const args: string[] = [];\n\n if (options.prompt) {\n if (interactive.promptFlag) {\n args.push(interactive.promptFlag, options.prompt);\n } else {\n args.push(options.prompt);\n }\n }\n\n if (options.model && spawnConfig.modelFlag) {\n let model = spawnConfig.modelStripProviderPrefix\n ? stripModelNamespace(options.model)\n : options.model;\n if (spawnConfig.modelTransform) model = spawnConfig.modelTransform(model);\n args.push(spawnConfig.modelFlag, model);\n }\n\n args.push(...interactive.defaultArgs);\n args.push(...getMcpArgs(spawnConfig, options.mcpServers));\n\n const mode = options.mode ?? \"yolo\";\n args.push(...spawnConfig.modes[mode]);\n\n if (options.args && options.args.length > 0) {\n args.push(...options.args);\n }\n\n const child = spawnChildProcess(resolved.binaryName, args, {\n cwd: options.cwd,\n stdio: \"inherit\"\n });\n\n return new Promise<SpawnResult>((resolve, reject) => {\n child.on(\"error\", (error) => {\n reject(error);\n });\n\n child.on(\"close\", (code) => {\n resolve({\n stdout: \"\",\n stderr: \"\",\n exitCode: code ?? 1\n });\n });\n });\n}\n", "import chalk from \"chalk\";\n\nexport const brand = \"#a200ff\";\n\nexport const dark = {\n header: (text: string) => chalk.magentaBright.bold(text),\n divider: (text: string) => chalk.dim(text),\n prompt: (text: string) => chalk.cyan(text),\n number: (text: string) => chalk.cyanBright(text),\n intro: (text: string) => chalk.bgMagenta.white(` Poe - ${text} `),\n resolvedSymbol: chalk.magenta(\"\u25C7\"),\n errorSymbol: chalk.red(\"\u25A0\"),\n accent: (text: string) => chalk.cyan(text),\n muted: (text: string) => chalk.dim(text),\n success: (text: string) => chalk.green(text),\n warning: (text: string) => chalk.yellow(text),\n error: (text: string) => chalk.red(text),\n info: (text: string) => chalk.magenta(text),\n badge: (text: string) => chalk.bgYellow.black(` ${text} `)\n};\n\nexport const light = {\n header: (text: string) => chalk.hex(\"#a200ff\").bold(text),\n divider: (text: string) => chalk.hex(\"#666666\")(text),\n prompt: (text: string) => chalk.hex(\"#006699\").bold(text),\n number: (text: string) => chalk.hex(\"#0077cc\").bold(text),\n intro: (text: string) => chalk.bgHex(\"#a200ff\").white(` Poe - ${text} `),\n resolvedSymbol: chalk.hex(\"#a200ff\")(\"\u25C7\"),\n errorSymbol: chalk.hex(\"#cc0000\")(\"\u25A0\"),\n accent: (text: string) => chalk.hex(\"#006699\").bold(text),\n muted: (text: string) => chalk.hex(\"#666666\")(text),\n success: (text: string) => chalk.hex(\"#008800\")(text),\n warning: (text: string) => chalk.hex(\"#cc6600\")(text),\n error: (text: string) => chalk.hex(\"#cc0000\")(text),\n info: (text: string) => chalk.hex(\"#a200ff\")(text),\n badge: (text: string) => chalk.bgHex(\"#cc6600\").white(` ${text} `)\n};\n\nexport type ThemeName = \"dark\" | \"light\";\nexport type ThemePalette = typeof dark;\n", "import chalk from \"chalk\";\n\nexport const typography = {\n bold: (text: string) => chalk.bold(text),\n dim: (text: string) => chalk.dim(text),\n italic: (text: string) => chalk.italic(text),\n underline: (text: string) => chalk.underline(text),\n strikethrough: (text: string) => chalk.strikethrough(text)\n} as const;\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { getTheme } from \"../internal/theme-detect.js\";\nimport { typography } from \"../tokens/typography.js\";\n\nexport const text = {\n intro(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `**${content}**`;\n return getTheme().intro(content);\n },\n heading(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `## ${content}`;\n return getTheme().header(content);\n },\n section(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `**${content}**`;\n return typography.bold(content);\n },\n command(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `\\`${content}\\``;\n return getTheme().accent(content);\n },\n argument(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `<${content}>`;\n return getTheme().muted(content);\n },\n option(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `\\`${content}\\``;\n return chalk.yellow(content);\n },\n example(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `\\`${content}\\``;\n return getTheme().muted(content);\n },\n usageCommand(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `\\`${content}\\``;\n return chalk.green(content);\n },\n link(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `[${content}](${content})`;\n return getTheme().accent(content);\n },\n muted(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `*${content}*`;\n return getTheme().muted(content);\n },\n badge(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `[${content}]`;\n return getTheme().badge(content);\n },\n selectLabel(label: string, detail?: string): string {\n if (!detail) {\n return label;\n }\n const format = resolveOutputFormat();\n if (format !== \"terminal\") {\n return `${label} \u2014 ${detail}`;\n }\n return `${label} ${typography.dim(\"\u2014\")} ${typography.dim(detail)}`;\n }\n} as const;\n", "import { AsyncLocalStorage } from \"node:async_hooks\";\n\nexport type OutputFormat = \"terminal\" | \"markdown\" | \"json\";\n\nconst VALID_FORMATS = new Set<OutputFormat>([\"terminal\", \"markdown\", \"json\"]);\nconst formatStorage = new AsyncLocalStorage<OutputFormat>();\n\nlet cached: OutputFormat | undefined;\n\nexport function resolveOutputFormat(\n env: { OUTPUT_FORMAT?: string } = process.env as { OUTPUT_FORMAT?: string }\n): OutputFormat {\n const scoped = formatStorage.getStore();\n if (scoped) {\n return scoped;\n }\n if (cached) {\n return cached;\n }\n const raw = env.OUTPUT_FORMAT?.toLowerCase();\n cached = VALID_FORMATS.has(raw as OutputFormat) ? (raw as OutputFormat) : \"terminal\";\n return cached;\n}\n\nexport function withOutputFormat<T>(format: OutputFormat, fn: () => T): T {\n return formatStorage.run(format, fn);\n}\n\nexport function resetOutputFormatCache(): void {\n cached = undefined;\n}\n", "import { dark, light, type ThemeName, type ThemePalette } from \"../tokens/colors.js\";\n\nexport interface ThemeEnv {\n POE_CODE_THEME?: string;\n POE_THEME?: string;\n APPLE_INTERFACE_STYLE?: string;\n VSCODE_COLOR_THEME_KIND?: string;\n COLORFGBG?: string;\n}\n\nfunction detectThemeFromEnv(env: ThemeEnv): ThemeName | undefined {\n const apple = env.APPLE_INTERFACE_STYLE;\n if (typeof apple === \"string\") {\n return apple.toLowerCase() === \"dark\" ? \"dark\" : \"light\";\n }\n\n const vscodeKind = env.VSCODE_COLOR_THEME_KIND;\n if (typeof vscodeKind === \"string\") {\n const normalized = vscodeKind.toLowerCase();\n if (normalized.includes(\"light\")) {\n return \"light\";\n }\n if (normalized.includes(\"dark\")) {\n return \"dark\";\n }\n }\n\n const colorFGBG = env.COLORFGBG;\n if (typeof colorFGBG === \"string\") {\n const parts = colorFGBG.split(\";\").map((part) => Number.parseInt(part, 10));\n const background = parts.at(-1);\n if (Number.isFinite(background)) {\n return background! >= 8 ? \"light\" : \"dark\";\n }\n }\n\n return undefined;\n}\n\nexport function resolveThemeName(env: ThemeEnv = process.env as ThemeEnv): ThemeName {\n const raw = (env.POE_CODE_THEME ?? env.POE_THEME)?.toLowerCase();\n if (raw === \"light\" || raw === \"dark\") {\n return raw;\n }\n const detected = detectThemeFromEnv(env);\n if (detected) {\n return detected;\n }\n return \"dark\";\n}\n\nlet cachedTheme: ThemePalette | undefined;\n\nexport function getTheme(env?: ThemeEnv): ThemePalette {\n if (cachedTheme) {\n return cachedTheme;\n }\n const themeName = resolveThemeName(env);\n cachedTheme = themeName === \"light\" ? light : dark;\n return cachedTheme;\n}\n\nexport function resetThemeCache(): void {\n cachedTheme = undefined;\n}\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { getTheme } from \"../internal/theme-detect.js\";\n\nexport const symbols = {\n get info(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"info\";\n if (format === \"markdown\") return \"(i)\";\n return chalk.magenta(\"\u25CF\");\n },\n get success(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"success\";\n if (format === \"markdown\") return \"[ok]\";\n return chalk.magenta(\"\u25C6\");\n },\n get resolved(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"resolved\";\n if (format === \"markdown\") return \">\";\n return getTheme().resolvedSymbol;\n },\n get errorResolved(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"error\";\n if (format === \"markdown\") return \"[!]\";\n return getTheme().errorSymbol;\n },\n get bar(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"\";\n if (format === \"markdown\") return \"|\";\n return \"\u2502\";\n },\n cornerTopRight: \"\u256E\",\n cornerBottomRight: \"\u256F\",\n get warning(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"warning\";\n if (format === \"markdown\") return \"[!]\";\n return \"\u25B2\";\n },\n get active(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"active\";\n if (format === \"markdown\") return \"[x]\";\n return \"\u25C6\";\n },\n get inactive(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"inactive\";\n if (format === \"markdown\") return \"[ ]\";\n return \"\u25CB\";\n }\n} as const;\n", "import chalk from \"chalk\";\nimport { log } from \"../prompts/primitives/log.js\";\nimport { symbols } from \"./symbols.js\";\n\nexport interface LoggerOutput {\n info(message: string): void;\n success(message: string): void;\n warn(message: string): void;\n error(message: string): void;\n resolved(label: string, value: string): void;\n errorResolved(label: string, value: string): void;\n message(message: string, symbol?: string): void;\n}\n\nexport function createLogger(emitter?: (message: string) => void): LoggerOutput {\n const emit = (\n level: \"info\" | \"success\" | \"warn\" | \"error\",\n message: string\n ): void => {\n if (emitter) {\n emitter(message);\n return;\n }\n if (level === \"success\") {\n log.success(message);\n return;\n }\n if (level === \"warn\") {\n log.warn(message);\n return;\n }\n if (level === \"error\") {\n log.error(message);\n return;\n }\n log.info(message);\n };\n\n return {\n info(message: string): void {\n emit(\"info\", message);\n },\n success(message: string): void {\n emit(\"success\", message);\n },\n warn(message: string): void {\n emit(\"warn\", message);\n },\n error(message: string): void {\n emit(\"error\", message);\n },\n resolved(label: string, value: string): void {\n if (emitter) {\n emitter(`${label}: ${value}`);\n return;\n }\n log.message(`${label}\\n ${value}`, { symbol: symbols.resolved });\n },\n errorResolved(label: string, value: string): void {\n if (emitter) {\n emitter(`${label}: ${value}`);\n return;\n }\n log.message(`${label}\\n ${value}`, { symbol: symbols.errorResolved });\n },\n message(message: string, symbol?: string): void {\n if (emitter) {\n emitter(message);\n return;\n }\n log.message(message, { symbol: symbol ?? chalk.gray(\"\u2502\") });\n }\n };\n}\n\nexport const logger = createLogger();\n", "import chalk from \"chalk\";\nimport { symbols } from \"../../components/symbols.js\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\nimport { stripAnsi } from \"../../internal/strip-ansi.js\";\n\nexport interface LogMessageOptions {\n symbol?: string;\n secondarySymbol?: string;\n spacing?: number;\n withGuide?: boolean;\n}\n\nfunction writeTerminalMessage(\n msg: string,\n {\n symbol = chalk.gray(\"\u2502\"),\n secondarySymbol = chalk.gray(\"\u2502\"),\n spacing = 1,\n withGuide = true\n }: LogMessageOptions = {}\n): void {\n const lines: string[] = [];\n const showGuide = withGuide !== false;\n const contentLines = msg.split(\"\\n\");\n const prefix = showGuide ? `${symbol} ` : \"\";\n const continuationPrefix = showGuide ? `${secondarySymbol} ` : \"\";\n const emptyGuide = showGuide ? secondarySymbol : \"\";\n\n for (let index = 0; index < spacing; index += 1) {\n lines.push(emptyGuide);\n }\n\n if (contentLines.length === 0) {\n process.stdout.write(\"\\n\");\n return;\n }\n\n const [firstLine = \"\", ...continuationLines] = contentLines;\n if (firstLine.length > 0) {\n lines.push(`${prefix}${firstLine}`);\n } else {\n lines.push(showGuide ? symbol : \"\");\n }\n\n for (const line of continuationLines) {\n if (line.length > 0) {\n lines.push(`${continuationPrefix}${line}`);\n continue;\n }\n lines.push(emptyGuide);\n }\n\n process.stdout.write(`${lines.join(\"\\n\")}\\n`);\n}\n\nexport function message(msg: string, options?: LogMessageOptions): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`- ${stripAnsi(msg)}\\n`);\n return;\n }\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ level: \"message\", message: stripAnsi(msg) })}\\n`\n );\n return;\n }\n\n writeTerminalMessage(msg, options);\n}\n\nexport function info(msg: string): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`- **info:** ${stripAnsi(msg)}\\n`);\n return;\n }\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ level: \"info\", message: stripAnsi(msg) })}\\n`\n );\n return;\n }\n\n message(msg, { symbol: symbols.info });\n}\n\nexport function success(msg: string): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`- **success:** ${stripAnsi(msg)}\\n`);\n return;\n }\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ level: \"success\", message: stripAnsi(msg) })}\\n`\n );\n return;\n }\n\n message(msg, { symbol: symbols.success });\n}\n\nexport function warn(msg: string): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`- **warning:** ${stripAnsi(msg)}\\n`);\n return;\n }\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ level: \"warn\", message: stripAnsi(msg) })}\\n`\n );\n return;\n }\n\n message(msg, { symbol: chalk.yellow(\"\u25B2\") });\n}\n\nexport function error(msg: string): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`- **error:** ${stripAnsi(msg)}\\n`);\n return;\n }\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ level: \"error\", message: stripAnsi(msg) })}\\n`\n );\n return;\n }\n\n message(msg, { symbol: chalk.red(\"\u25A0\") });\n}\n\nexport const log = {\n info,\n success,\n message,\n warn,\n error\n};\n", "export function stripAnsi(value: string): string {\n return value.replace(/\\u001b\\[[0-9;]*m/g, \"\");\n}\n", "import { Table } from \"console-table-printer\";\nimport type { ThemePalette } from \"../tokens/colors.js\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { stripAnsi } from \"../internal/strip-ansi.js\";\n\nexport interface TableColumn {\n name: string;\n title: string;\n alignment: \"left\" | \"right\";\n maxLen: number;\n}\n\nexport interface RenderTableOptions {\n theme: ThemePalette;\n columns: TableColumn[];\n rows: Record<string, string>[];\n}\n\nfunction renderTableTerminal(options: RenderTableOptions): string {\n const { theme, columns, rows } = options;\n\n const table = new Table({\n style: {\n headerTop: {\n left: theme.muted(\"\u250C\"),\n mid: theme.muted(\"\u252C\"),\n right: theme.muted(\"\u2510\"),\n other: theme.muted(\"\u2500\")\n },\n headerBottom: {\n left: theme.muted(\"\u251C\"),\n mid: theme.muted(\"\u253C\"),\n right: theme.muted(\"\u2524\"),\n other: theme.muted(\"\u2500\")\n },\n tableBottom: {\n left: theme.muted(\"\u2514\"),\n mid: theme.muted(\"\u2534\"),\n right: theme.muted(\"\u2518\"),\n other: theme.muted(\"\u2500\")\n },\n vertical: theme.muted(\"\u2502\"),\n rowSeparator: {\n left: theme.muted(\"\u251C\"),\n mid: theme.muted(\"\u253C\"),\n right: theme.muted(\"\u2524\"),\n other: theme.muted(\"\u2500\")\n }\n },\n columns: columns.map((col) => ({\n name: col.name,\n title: theme.header(col.title),\n alignment: col.alignment,\n maxLen: col.maxLen\n }))\n });\n\n for (const row of rows) {\n table.addRow(row);\n }\n\n return table.render();\n}\n\nfunction renderTableMarkdown(options: RenderTableOptions): string {\n const { columns, rows } = options;\n\n const header = `| ${columns.map((c) => c.title).join(\" | \")} |`;\n const separator = `| ${columns.map((c) => (c.alignment === \"right\" ? \"---:\" : \":---\")).join(\" | \")} |`;\n\n const dataRows = rows.map(\n (row) =>\n `| ${columns.map((c) => stripAnsi(row[c.name] ?? \"\").replace(/\\|/g, \"\\\\|\")).join(\" | \")} |`\n );\n\n return [header, separator, ...dataRows].join(\"\\n\");\n}\n\nfunction renderTableJson(options: RenderTableOptions): string {\n const { columns, rows } = options;\n\n const cleaned = rows.map((row) => {\n const obj: Record<string, string> = {};\n for (const col of columns) {\n obj[col.name] = stripAnsi(row[col.name] ?? \"\");\n }\n return obj;\n });\n\n return JSON.stringify(cleaned, null, 2);\n}\n\nexport function renderTable(options: RenderTableOptions): string {\n const format = resolveOutputFormat();\n switch (format) {\n case \"markdown\":\n return renderTableMarkdown(options);\n case \"json\":\n return renderTableJson(options);\n default:\n return renderTableTerminal(options);\n }\n}\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { renderMarkdown } from \"../terminal-markdown/index.js\";\n\nfunction truncate(text: string, maxLength: number): string {\n if (text.length <= maxLength) return text;\n if (maxLength <= 3) return text.slice(0, maxLength);\n return `${text.slice(0, maxLength - 3)}...`;\n}\n\nconst KIND_COLORS: Record<string, (text: string) => string> = {\n exec: (text) => chalk.yellow(text),\n edit: (text) => chalk.magenta(text),\n read: (text) => chalk.cyan(text),\n search: (text) => chalk.blue(text),\n think: (text) => chalk.dim(text),\n other: (text) => chalk.dim(text)\n};\n\nfunction colorForKind(kind: string): (text: string) => string {\n return KIND_COLORS[kind] ?? ((text) => chalk.dim(text));\n}\n\nfunction writeLine(line: string): void {\n process.stdout.write(`${line}\\n`);\n}\n\nconst AGENT_PREFIX = `${chalk.green.bold(\"\u2713\")} agent: `;\n\nfunction formatCost(costUsd: number): string {\n return new Intl.NumberFormat(\"en-US\", {\n style: \"currency\",\n currency: \"USD\",\n minimumFractionDigits: 2,\n maximumFractionDigits: 6\n }).format(costUsd);\n}\n\nexport function renderAgentMessage(text: string): void {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n writeLine(`- **agent:** ${text}`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(JSON.stringify({ event: \"agent_message\", text }));\n return;\n }\n\n const rendered = renderMarkdown(text).trimEnd();\n writeLine(`${AGENT_PREFIX}${rendered}`);\n}\n\nexport function renderToolStart(kind: string, title: string): void {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n writeLine(`- *\u2192 ${kind}: ${title}*`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(JSON.stringify({ event: \"tool_start\", kind, title }));\n return;\n }\n\n const color = colorForKind(kind);\n writeLine(color(` \u2192 ${kind}: ${title}`));\n}\n\nexport function renderToolComplete(kind: string): void {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n writeLine(`- *\u2713 ${kind}*`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(JSON.stringify({ event: \"tool_complete\", kind }));\n return;\n }\n\n const color = colorForKind(kind);\n writeLine(color(` \u2713 ${kind}`));\n}\n\nexport function renderReasoning(text: string): void {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n writeLine(`- *thinking:* ${truncate(text, 80)}`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(JSON.stringify({ event: \"reasoning\", text }));\n return;\n }\n\n writeLine(chalk.dim(` \u2713 ${truncate(text, 80)}`));\n}\n\nexport function renderUsage(tokens: {\n input: number;\n output: number;\n cached?: number;\n costUsd?: number;\n}): void {\n const format = resolveOutputFormat();\n const cached =\n typeof tokens.cached === \"number\" && tokens.cached > 0 ? ` (${tokens.cached} cached)` : \"\";\n\n let cost = \"\";\n if (typeof tokens.costUsd === \"number\") {\n cost = ` (${formatCost(tokens.costUsd)})`;\n }\n\n if (format === \"markdown\") {\n writeLine(`- **tokens:** ${tokens.input} in \u2192 ${tokens.output} out${cost}`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(\n JSON.stringify({\n event: \"usage\",\n inputTokens: tokens.input,\n outputTokens: tokens.output,\n cachedTokens: tokens.cached ?? 0,\n costUsd: tokens.costUsd ?? 0\n })\n );\n return;\n }\n\n writeLine(chalk.green(`\u2713 tokens: ${tokens.input} in${cached} \u2192 ${tokens.output} out${cost}`));\n}\n\nexport function renderError(message: string): void {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n writeLine(`- **error:** ${message}`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(JSON.stringify({ event: \"error\", message }));\n return;\n }\n\n writeLine(chalk.red(`\u2717 ${message}`));\n}\n", "import chalk from \"chalk\";\nimport * as clack from \"@clack/prompts\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { stripAnsi } from \"../internal/strip-ansi.js\";\nimport { cancel, isCancel } from \"./primitives/cancel.js\";\nimport { intro } from \"./primitives/intro.js\";\nimport { log } from \"./primitives/log.js\";\nimport { note } from \"./primitives/note.js\";\nimport { outro } from \"./primitives/outro.js\";\nimport { spinner } from \"./primitives/spinner.js\";\n\nexport { isCancel, cancel, log };\nexport { intro, outro, note, spinner };\n\nexport function introPlain(title: string): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`# ${stripAnsi(title)}\\n\\n`);\n return;\n }\n if (format === \"json\") {\n return;\n }\n process.stdout.write(`${chalk.gray(\"\u250C\")} ${title}\\n`);\n}\n\nexport interface SelectOptions<Value> {\n message: string;\n options: Array<{ value: Value; label: string; hint?: string }>;\n initialValue?: Value;\n}\n\nexport async function select<Value>(\n opts: SelectOptions<Value>\n): Promise<Value | symbol> {\n return clack.select(opts as Parameters<typeof clack.select<Value>>[0]);\n}\n\nexport type MultiselectOptions<Value> = Parameters<typeof clack.multiselect<Value>>[0];\n\n/**\n * Prompts the user to select one or more values from a list.\n *\n * Returns the selected values as an array, or a cancellation symbol if the\n * user cancels. Use `isCancel` to check for cancellation.\n *\n * @example\n * const result = await multiselect({\n * message: \"Pick workflows to run\",\n * options: [{ label: \"Fix Vulnerabilities\", value: \"fix-vulnerabilities\" }],\n * required: true\n * });\n * if (!isCancel(result)) {\n * // result is Value[]\n * }\n */\nexport async function multiselect<Value>(\n opts: MultiselectOptions<Value>\n): Promise<Value[] | symbol> {\n return clack.multiselect(opts);\n}\n\nexport type TextOptions = Parameters<typeof clack.text>[0];\n\nexport async function text(opts: TextOptions): Promise<string | symbol> {\n return clack.text(opts as Parameters<typeof clack.text>[0]);\n}\n\nexport interface ConfirmOptions {\n message: string;\n initialValue?: boolean;\n}\n\nexport async function confirm(opts: ConfirmOptions): Promise<boolean | symbol> {\n return clack.confirm(opts as Parameters<typeof clack.confirm>[0]);\n}\n\nexport class PromptCancelledError extends Error {\n constructor(message = \"Operation cancelled.\") {\n super(message);\n this.name = \"PromptCancelledError\";\n if (Error.captureStackTrace) {\n Error.captureStackTrace(this, this.constructor);\n }\n }\n}\n\nexport async function confirmOrCancel(opts: ConfirmOptions): Promise<boolean> {\n const result = await confirm(opts);\n if (isCancel(result)) {\n cancel(\"Operation cancelled.\");\n throw new PromptCancelledError();\n }\n return result === true;\n}\n\nexport interface PasswordOptions {\n message: string;\n validate?: (value: string) => string | undefined;\n}\n\nexport async function password(opts: PasswordOptions): Promise<string | symbol> {\n return clack.password(opts as Parameters<typeof clack.password>[0]);\n}\n\nexport type SpinnerOptions = {\n start: (message?: string) => void;\n stop: (message?: string, code?: number) => void;\n message: (message?: string) => void;\n};\n\nexport interface WithSpinnerOptions<T> {\n message: string;\n fn: () => Promise<T>;\n stopMessage?: (result: T) => string;\n subtext?: (result: T) => string | undefined;\n}\n\nfunction formatElapsed(ms: number): string {\n const totalSeconds = Math.floor(ms / 1000);\n if (totalSeconds < 60) {\n return `${totalSeconds}s`;\n }\n const minutes = Math.floor(totalSeconds / 60);\n const seconds = totalSeconds % 60;\n return `${minutes}m ${seconds}s`;\n}\n\nexport async function withSpinner<T>(options: WithSpinnerOptions<T>): Promise<T> {\n const { message, fn, stopMessage, subtext } = options;\n\n if (resolveOutputFormat() === \"json\") {\n const result = await fn();\n const sub = subtext ? subtext(result) : undefined;\n if (sub) {\n process.stdout.write(sub + \"\\n\");\n }\n return result;\n }\n\n const noSpinner = process.env.POE_NO_SPINNER === \"1\";\n const isTTY = process.stdout.isTTY;\n\n if (noSpinner || !isTTY) {\n const result = await fn();\n const msg = stopMessage ? stopMessage(result) : undefined;\n if (msg) {\n process.stdout.write(`\\x1b[32m\u25C6\\x1b[0m ${msg}\\n`);\n }\n const sub = subtext ? subtext(result) : undefined;\n if (sub) {\n for (const line of sub.split(\"\\n\")) {\n process.stdout.write(`\\x1b[90m\u2502\\x1b[0m ${line}\\n`);\n }\n }\n return result;\n }\n\n const s = spinner();\n const start = Date.now();\n s.start(message);\n\n const timer = setInterval(() => {\n s.message(`${message} [${formatElapsed(Date.now() - start)}]`);\n }, 1000);\n\n try {\n const result = await fn();\n clearInterval(timer);\n const elapsed = formatElapsed(Date.now() - start);\n const msg = stopMessage ? stopMessage(result) : undefined;\n s.stop(msg ? `${msg} [${elapsed}]` : `Done [${elapsed}]`);\n\n const sub = subtext ? subtext(result) : undefined;\n if (sub) {\n for (const line of sub.split(\"\\n\")) {\n process.stdout.write(`\\x1b[90m\u2502\\x1b[0m ${line}\\n`);\n }\n }\n\n return result;\n } catch (error) {\n clearInterval(timer);\n s.stop(\"\", 1);\n throw error;\n }\n}\n", "import chalk from \"chalk\";\nexport { isCancel } from \"@clack/core\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\n\nexport function cancel(msg = \"\"): void {\n if (resolveOutputFormat() !== \"terminal\") {\n return;\n }\n\n process.stdout.write(`${chalk.gray(\"\u2514\")} ${chalk.red(msg)}\\n\\n`);\n}\n", "/* MAIN */\n//URL: https://github.com/sindresorhus/get-east-asian-width/blob/main/lookup.js\n//LICENSE: https://github.com/sindresorhus/get-east-asian-width/blob/main/license\n//TODO: Replace these with some unicode property classes, if the ones we need exist\nconst isAmbiguous = (x) => {\n return x === 0xA1 || x === 0xA4 || x === 0xA7 || x === 0xA8 || x === 0xAA || x === 0xAD || x === 0xAE || x >= 0xB0 && x <= 0xB4 || x >= 0xB6 && x <= 0xBA || x >= 0xBC && x <= 0xBF || x === 0xC6 || x === 0xD0 || x === 0xD7 || x === 0xD8 || x >= 0xDE && x <= 0xE1 || x === 0xE6 || x >= 0xE8 && x <= 0xEA || x === 0xEC || x === 0xED || x === 0xF0 || x === 0xF2 || x === 0xF3 || x >= 0xF7 && x <= 0xFA || x === 0xFC || x === 0xFE || x === 0x101 || x === 0x111 || x === 0x113 || x === 0x11B || x === 0x126 || x === 0x127 || x === 0x12B || x >= 0x131 && x <= 0x133 || x === 0x138 || x >= 0x13F && x <= 0x142 || x === 0x144 || x >= 0x148 && x <= 0x14B || x === 0x14D || x === 0x152 || x === 0x153 || x === 0x166 || x === 0x167 || x === 0x16B || x === 0x1CE || x === 0x1D0 || x === 0x1D2 || x === 0x1D4 || x === 0x1D6 || x === 0x1D8 || x === 0x1DA || x === 0x1DC || x === 0x251 || x === 0x261 || x === 0x2C4 || x === 0x2C7 || x >= 0x2C9 && x <= 0x2CB || x === 0x2CD || x === 0x2D0 || x >= 0x2D8 && x <= 0x2DB || x === 0x2DD || x === 0x2DF || x >= 0x300 && x <= 0x36F || x >= 0x391 && x <= 0x3A1 || x >= 0x3A3 && x <= 0x3A9 || x >= 0x3B1 && x <= 0x3C1 || x >= 0x3C3 && x <= 0x3C9 || x === 0x401 || x >= 0x410 && x <= 0x44F || x === 0x451 || x === 0x2010 || x >= 0x2013 && x <= 0x2016 || x === 0x2018 || x === 0x2019 || x === 0x201C || x === 0x201D || x >= 0x2020 && x <= 0x2022 || x >= 0x2024 && x <= 0x2027 || x === 0x2030 || x === 0x2032 || x === 0x2033 || x === 0x2035 || x === 0x203B || x === 0x203E || x === 0x2074 || x === 0x207F || x >= 0x2081 && x <= 0x2084 || x === 0x20AC || x === 0x2103 || x === 0x2105 || x === 0x2109 || x === 0x2113 || x === 0x2116 || x === 0x2121 || x === 0x2122 || x === 0x2126 || x === 0x212B || x === 0x2153 || x === 0x2154 || x >= 0x215B && x <= 0x215E || x >= 0x2160 && x <= 0x216B || x >= 0x2170 && x <= 0x2179 || x === 0x2189 || x >= 0x2190 && x <= 0x2199 || x === 0x21B8 || x === 0x21B9 || x === 0x21D2 || x === 0x21D4 || x === 0x21E7 || x === 0x2200 || x === 0x2202 || x === 0x2203 || x === 0x2207 || x === 0x2208 || x === 0x220B || x === 0x220F || x === 0x2211 || x === 0x2215 || x === 0x221A || x >= 0x221D && x <= 0x2220 || x === 0x2223 || x === 0x2225 || x >= 0x2227 && x <= 0x222C || x === 0x222E || x >= 0x2234 && x <= 0x2237 || x === 0x223C || x === 0x223D || x === 0x2248 || x === 0x224C || x === 0x2252 || x === 0x2260 || x === 0x2261 || x >= 0x2264 && x <= 0x2267 || x === 0x226A || x === 0x226B || x === 0x226E || x === 0x226F || x === 0x2282 || x === 0x2283 || x === 0x2286 || x === 0x2287 || x === 0x2295 || x === 0x2299 || x === 0x22A5 || x === 0x22BF || x === 0x2312 || x >= 0x2460 && x <= 0x24E9 || x >= 0x24EB && x <= 0x254B || x >= 0x2550 && x <= 0x2573 || x >= 0x2580 && x <= 0x258F || x >= 0x2592 && x <= 0x2595 || x === 0x25A0 || x === 0x25A1 || x >= 0x25A3 && x <= 0x25A9 || x === 0x25B2 || x === 0x25B3 || x === 0x25B6 || x === 0x25B7 || x === 0x25BC || x === 0x25BD || x === 0x25C0 || x === 0x25C1 || x >= 0x25C6 && x <= 0x25C8 || x === 0x25CB || x >= 0x25CE && x <= 0x25D1 || x >= 0x25E2 && x <= 0x25E5 || x === 0x25EF || x === 0x2605 || x === 0x2606 || x === 0x2609 || x === 0x260E || x === 0x260F || x === 0x261C || x === 0x261E || x === 0x2640 || x === 0x2642 || x === 0x2660 || x === 0x2661 || x >= 0x2663 && x <= 0x2665 || x >= 0x2667 && x <= 0x266A || x === 0x266C || x === 0x266D || x === 0x266F || x === 0x269E || x === 0x269F || x === 0x26BF || x >= 0x26C6 && x <= 0x26CD || x >= 0x26CF && x <= 0x26D3 || x >= 0x26D5 && x <= 0x26E1 || x === 0x26E3 || x === 0x26E8 || x === 0x26E9 || x >= 0x26EB && x <= 0x26F1 || x === 0x26F4 || x >= 0x26F6 && x <= 0x26F9 || x === 0x26FB || x === 0x26FC || x === 0x26FE || x === 0x26FF || x === 0x273D || x >= 0x2776 && x <= 0x277F || x >= 0x2B56 && x <= 0x2B59 || x >= 0x3248 && x <= 0x324F || x >= 0xE000 && x <= 0xF8FF || x >= 0xFE00 && x <= 0xFE0F || x === 0xFFFD || x >= 0x1F100 && x <= 0x1F10A || x >= 0x1F110 && x <= 0x1F12D || x >= 0x1F130 && x <= 0x1F169 || x >= 0x1F170 && x <= 0x1F18D || x === 0x1F18F || x === 0x1F190 || x >= 0x1F19B && x <= 0x1F1AC || x >= 0xE0100 && x <= 0xE01EF || x >= 0xF0000 && x <= 0xFFFFD || x >= 0x100000 && x <= 0x10FFFD;\n};\nconst isFullWidth = (x) => {\n return x === 0x3000 || x >= 0xFF01 && x <= 0xFF60 || x >= 0xFFE0 && x <= 0xFFE6;\n};\nconst isWide = (x) => {\n return x >= 0x1100 && x <= 0x115F || x === 0x231A || x === 0x231B || x === 0x2329 || x === 0x232A || x >= 0x23E9 && x <= 0x23EC || x === 0x23F0 || x === 0x23F3 || x === 0x25FD || x === 0x25FE || x === 0x2614 || x === 0x2615 || x >= 0x2648 && x <= 0x2653 || x === 0x267F || x === 0x2693 || x === 0x26A1 || x === 0x26AA || x === 0x26AB || x === 0x26BD || x === 0x26BE || x === 0x26C4 || x === 0x26C5 || x === 0x26CE || x === 0x26D4 || x === 0x26EA || x === 0x26F2 || x === 0x26F3 || x === 0x26F5 || x === 0x26FA || x === 0x26FD || x === 0x2705 || x === 0x270A || x === 0x270B || x === 0x2728 || x === 0x274C || x === 0x274E || x >= 0x2753 && x <= 0x2755 || x === 0x2757 || x >= 0x2795 && x <= 0x2797 || x === 0x27B0 || x === 0x27BF || x === 0x2B1B || x === 0x2B1C || x === 0x2B50 || x === 0x2B55 || x >= 0x2E80 && x <= 0x2E99 || x >= 0x2E9B && x <= 0x2EF3 || x >= 0x2F00 && x <= 0x2FD5 || x >= 0x2FF0 && x <= 0x2FFF || x >= 0x3001 && x <= 0x303E || x >= 0x3041 && x <= 0x3096 || x >= 0x3099 && x <= 0x30FF || x >= 0x3105 && x <= 0x312F || x >= 0x3131 && x <= 0x318E || x >= 0x3190 && x <= 0x31E3 || x >= 0x31EF && x <= 0x321E || x >= 0x3220 && x <= 0x3247 || x >= 0x3250 && x <= 0x4DBF || x >= 0x4E00 && x <= 0xA48C || x >= 0xA490 && x <= 0xA4C6 || x >= 0xA960 && x <= 0xA97C || x >= 0xAC00 && x <= 0xD7A3 || x >= 0xF900 && x <= 0xFAFF || x >= 0xFE10 && x <= 0xFE19 || x >= 0xFE30 && x <= 0xFE52 || x >= 0xFE54 && x <= 0xFE66 || x >= 0xFE68 && x <= 0xFE6B || x >= 0x16FE0 && x <= 0x16FE4 || x === 0x16FF0 || x === 0x16FF1 || x >= 0x17000 && x <= 0x187F7 || x >= 0x18800 && x <= 0x18CD5 || x >= 0x18D00 && x <= 0x18D08 || x >= 0x1AFF0 && x <= 0x1AFF3 || x >= 0x1AFF5 && x <= 0x1AFFB || x === 0x1AFFD || x === 0x1AFFE || x >= 0x1B000 && x <= 0x1B122 || x === 0x1B132 || x >= 0x1B150 && x <= 0x1B152 || x === 0x1B155 || x >= 0x1B164 && x <= 0x1B167 || x >= 0x1B170 && x <= 0x1B2FB || x === 0x1F004 || x === 0x1F0CF || x === 0x1F18E || x >= 0x1F191 && x <= 0x1F19A || x >= 0x1F200 && x <= 0x1F202 || x >= 0x1F210 && x <= 0x1F23B || x >= 0x1F240 && x <= 0x1F248 || x === 0x1F250 || x === 0x1F251 || x >= 0x1F260 && x <= 0x1F265 || x >= 0x1F300 && x <= 0x1F320 || x >= 0x1F32D && x <= 0x1F335 || x >= 0x1F337 && x <= 0x1F37C || x >= 0x1F37E && x <= 0x1F393 || x >= 0x1F3A0 && x <= 0x1F3CA || x >= 0x1F3CF && x <= 0x1F3D3 || x >= 0x1F3E0 && x <= 0x1F3F0 || x === 0x1F3F4 || x >= 0x1F3F8 && x <= 0x1F43E || x === 0x1F440 || x >= 0x1F442 && x <= 0x1F4FC || x >= 0x1F4FF && x <= 0x1F53D || x >= 0x1F54B && x <= 0x1F54E || x >= 0x1F550 && x <= 0x1F567 || x === 0x1F57A || x === 0x1F595 || x === 0x1F596 || x === 0x1F5A4 || x >= 0x1F5FB && x <= 0x1F64F || x >= 0x1F680 && x <= 0x1F6C5 || x === 0x1F6CC || x >= 0x1F6D0 && x <= 0x1F6D2 || x >= 0x1F6D5 && x <= 0x1F6D7 || x >= 0x1F6DC && x <= 0x1F6DF || x === 0x1F6EB || x === 0x1F6EC || x >= 0x1F6F4 && x <= 0x1F6FC || x >= 0x1F7E0 && x <= 0x1F7EB || x === 0x1F7F0 || x >= 0x1F90C && x <= 0x1F93A || x >= 0x1F93C && x <= 0x1F945 || x >= 0x1F947 && x <= 0x1F9FF || x >= 0x1FA70 && x <= 0x1FA7C || x >= 0x1FA80 && x <= 0x1FA88 || x >= 0x1FA90 && x <= 0x1FABD || x >= 0x1FABF && x <= 0x1FAC5 || x >= 0x1FACE && x <= 0x1FADB || x >= 0x1FAE0 && x <= 0x1FAE8 || x >= 0x1FAF0 && x <= 0x1FAF8 || x >= 0x20000 && x <= 0x2FFFD || x >= 0x30000 && x <= 0x3FFFD;\n};\n/* EXPORT */\nexport { isAmbiguous, isFullWidth, isWide };\n", "/* IMPORT */\nimport { isAmbiguous, isFullWidth, isWide } from './utils.js';\n/* HELPERS */\nconst ANSI_RE = /[\\u001b\\u009b][[()#;?]*(?:[0-9]{1,4}(?:;[0-9]{0,4})*)?[0-9A-ORZcf-nqry=><]/y;\nconst CONTROL_RE = /[\\x00-\\x08\\x0A-\\x1F\\x7F-\\x9F]{1,1000}/y;\nconst TAB_RE = /\\t{1,1000}/y;\nconst EMOJI_RE = /[\\u{1F1E6}-\\u{1F1FF}]{2}|\\u{1F3F4}[\\u{E0061}-\\u{E007A}]{2}[\\u{E0030}-\\u{E0039}\\u{E0061}-\\u{E007A}]{1,3}\\u{E007F}|(?:\\p{Emoji}\\uFE0F\\u20E3?|\\p{Emoji_Modifier_Base}\\p{Emoji_Modifier}?|\\p{Emoji_Presentation})(?:\\u200D(?:\\p{Emoji_Modifier_Base}\\p{Emoji_Modifier}?|\\p{Emoji_Presentation}|\\p{Emoji}\\uFE0F\\u20E3?))*/yu;\nconst LATIN_RE = /(?:[\\x20-\\x7E\\xA0-\\xFF](?!\\uFE0F)){1,1000}/y;\nconst MODIFIER_RE = /\\p{M}+/gu;\nconst NO_TRUNCATION = { limit: Infinity, ellipsis: '' };\n/* MAIN */\n//TODO: Optimize matching non-latin letters\nconst getStringTruncatedWidth = (input, truncationOptions = {}, widthOptions = {}) => {\n /* CONSTANTS */\n const LIMIT = truncationOptions.limit ?? Infinity;\n const ELLIPSIS = truncationOptions.ellipsis ?? '';\n const ELLIPSIS_WIDTH = truncationOptions?.ellipsisWidth ?? (ELLIPSIS ? getStringTruncatedWidth(ELLIPSIS, NO_TRUNCATION, widthOptions).width : 0);\n const ANSI_WIDTH = widthOptions.ansiWidth ?? 0;\n const CONTROL_WIDTH = widthOptions.controlWidth ?? 0;\n const TAB_WIDTH = widthOptions.tabWidth ?? 8;\n const AMBIGUOUS_WIDTH = widthOptions.ambiguousWidth ?? 1;\n const EMOJI_WIDTH = widthOptions.emojiWidth ?? 2;\n const FULL_WIDTH_WIDTH = widthOptions.fullWidthWidth ?? 2;\n const REGULAR_WIDTH = widthOptions.regularWidth ?? 1;\n const WIDE_WIDTH = widthOptions.wideWidth ?? 2;\n /* STATE */\n let indexPrev = 0;\n let index = 0;\n let length = input.length;\n let lengthExtra = 0;\n let truncationEnabled = false;\n let truncationIndex = length;\n let truncationLimit = Math.max(0, LIMIT - ELLIPSIS_WIDTH);\n let unmatchedStart = 0;\n let unmatchedEnd = 0;\n let width = 0;\n let widthExtra = 0;\n /* PARSE LOOP */\n outer: while (true) {\n /* UNMATCHED */\n if ((unmatchedEnd > unmatchedStart) || (index >= length && index > indexPrev)) {\n const unmatched = input.slice(unmatchedStart, unmatchedEnd) || input.slice(indexPrev, index);\n lengthExtra = 0;\n for (const char of unmatched.replaceAll(MODIFIER_RE, '')) {\n const codePoint = char.codePointAt(0) || 0;\n if (isFullWidth(codePoint)) {\n widthExtra = FULL_WIDTH_WIDTH;\n }\n else if (isWide(codePoint)) {\n widthExtra = WIDE_WIDTH;\n }\n else if (AMBIGUOUS_WIDTH !== REGULAR_WIDTH && isAmbiguous(codePoint)) {\n widthExtra = AMBIGUOUS_WIDTH;\n }\n else {\n widthExtra = REGULAR_WIDTH;\n }\n if ((width + widthExtra) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, Math.max(unmatchedStart, indexPrev) + lengthExtra);\n }\n if ((width + widthExtra) > LIMIT) {\n truncationEnabled = true;\n break outer;\n }\n lengthExtra += char.length;\n width += widthExtra;\n }\n unmatchedStart = unmatchedEnd = 0;\n }\n /* EXITING */\n if (index >= length)\n break;\n /* LATIN */\n LATIN_RE.lastIndex = index;\n if (LATIN_RE.test(input)) {\n lengthExtra = LATIN_RE.lastIndex - index;\n widthExtra = lengthExtra * REGULAR_WIDTH;\n if ((width + widthExtra) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, index + Math.floor((truncationLimit - width) / REGULAR_WIDTH));\n }\n if ((width + widthExtra) > LIMIT) {\n truncationEnabled = true;\n break;\n }\n width += widthExtra;\n unmatchedStart = indexPrev;\n unmatchedEnd = index;\n index = indexPrev = LATIN_RE.lastIndex;\n continue;\n }\n /* ANSI */\n ANSI_RE.lastIndex = index;\n if (ANSI_RE.test(input)) {\n if ((width + ANSI_WIDTH) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, index);\n }\n if ((width + ANSI_WIDTH) > LIMIT) {\n truncationEnabled = true;\n break;\n }\n width += ANSI_WIDTH;\n unmatchedStart = indexPrev;\n unmatchedEnd = index;\n index = indexPrev = ANSI_RE.lastIndex;\n continue;\n }\n /* CONTROL */\n CONTROL_RE.lastIndex = index;\n if (CONTROL_RE.test(input)) {\n lengthExtra = CONTROL_RE.lastIndex - index;\n widthExtra = lengthExtra * CONTROL_WIDTH;\n if ((width + widthExtra) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, index + Math.floor((truncationLimit - width) / CONTROL_WIDTH));\n }\n if ((width + widthExtra) > LIMIT) {\n truncationEnabled = true;\n break;\n }\n width += widthExtra;\n unmatchedStart = indexPrev;\n unmatchedEnd = index;\n index = indexPrev = CONTROL_RE.lastIndex;\n continue;\n }\n /* TAB */\n TAB_RE.lastIndex = index;\n if (TAB_RE.test(input)) {\n lengthExtra = TAB_RE.lastIndex - index;\n widthExtra = lengthExtra * TAB_WIDTH;\n if ((width + widthExtra) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, index + Math.floor((truncationLimit - width) / TAB_WIDTH));\n }\n if ((width + widthExtra) > LIMIT) {\n truncationEnabled = true;\n break;\n }\n width += widthExtra;\n unmatchedStart = indexPrev;\n unmatchedEnd = index;\n index = indexPrev = TAB_RE.lastIndex;\n continue;\n }\n /* EMOJI */\n EMOJI_RE.lastIndex = index;\n if (EMOJI_RE.test(input)) {\n if ((width + EMOJI_WIDTH) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, index);\n }\n if ((width + EMOJI_WIDTH) > LIMIT) {\n truncationEnabled = true;\n break;\n }\n width += EMOJI_WIDTH;\n unmatchedStart = indexPrev;\n unmatchedEnd = index;\n index = indexPrev = EMOJI_RE.lastIndex;\n continue;\n }\n /* UNMATCHED INDEX */\n index += 1;\n }\n /* RETURN */\n return {\n width: truncationEnabled ? truncationLimit : width,\n index: truncationEnabled ? truncationIndex : length,\n truncated: truncationEnabled,\n ellipsed: truncationEnabled && LIMIT >= ELLIPSIS_WIDTH\n };\n};\n/* EXPORT */\nexport default getStringTruncatedWidth;\n", "/* IMPORT */\nimport fastStringTruncatedWidth from 'fast-string-truncated-width';\n/* HELPERS */\nconst NO_TRUNCATION = {\n limit: Infinity,\n ellipsis: '',\n ellipsisWidth: 0,\n};\n/* MAIN */\nconst fastStringWidth = (input, options = {}) => {\n return fastStringTruncatedWidth(input, NO_TRUNCATION, options).width;\n};\n/* EXPORT */\nexport default fastStringWidth;\n", "import stringWidth from 'fast-string-width';\nconst ESC = '\\x1B';\nconst CSI = '\\x9B';\nconst END_CODE = 39;\nconst ANSI_ESCAPE_BELL = '\\u0007';\nconst ANSI_CSI = '[';\nconst ANSI_OSC = ']';\nconst ANSI_SGR_TERMINATOR = 'm';\nconst ANSI_ESCAPE_LINK = `${ANSI_OSC}8;;`;\nconst GROUP_REGEX = new RegExp(`(?:\\\\${ANSI_CSI}(?<code>\\\\d+)m|\\\\${ANSI_ESCAPE_LINK}(?<uri>.*)${ANSI_ESCAPE_BELL})`, 'y');\nconst getClosingCode = (openingCode) => {\n if (openingCode >= 30 && openingCode <= 37)\n return 39;\n if (openingCode >= 90 && openingCode <= 97)\n return 39;\n if (openingCode >= 40 && openingCode <= 47)\n return 49;\n if (openingCode >= 100 && openingCode <= 107)\n return 49;\n if (openingCode === 1 || openingCode === 2)\n return 22;\n if (openingCode === 3)\n return 23;\n if (openingCode === 4)\n return 24;\n if (openingCode === 7)\n return 27;\n if (openingCode === 8)\n return 28;\n if (openingCode === 9)\n return 29;\n if (openingCode === 0)\n return 0;\n return undefined;\n};\nconst wrapAnsiCode = (code) => `${ESC}${ANSI_CSI}${code}${ANSI_SGR_TERMINATOR}`;\nconst wrapAnsiHyperlink = (url) => `${ESC}${ANSI_ESCAPE_LINK}${url}${ANSI_ESCAPE_BELL}`;\nconst wordLengths = (words) => words.map((character) => stringWidth(character));\nconst wrapWord = (rows, word, columns) => {\n const characters = word[Symbol.iterator]();\n let isInsideEscape = false;\n let isInsideLinkEscape = false;\n let lastRow = rows.at(-1);\n let visible = lastRow === undefined ? 0 : stringWidth(lastRow);\n let currentCharacter = characters.next();\n let nextCharacter = characters.next();\n let rawCharacterIndex = 0;\n while (!currentCharacter.done) {\n const character = currentCharacter.value;\n const characterLength = stringWidth(character);\n if (visible + characterLength <= columns) {\n rows[rows.length - 1] += character;\n }\n else {\n rows.push(character);\n visible = 0;\n }\n if (character === ESC || character === CSI) {\n isInsideEscape = true;\n isInsideLinkEscape = word.startsWith(ANSI_ESCAPE_LINK, rawCharacterIndex + 1);\n }\n if (isInsideEscape) {\n if (isInsideLinkEscape) {\n if (character === ANSI_ESCAPE_BELL) {\n isInsideEscape = false;\n isInsideLinkEscape = false;\n }\n }\n else if (character === ANSI_SGR_TERMINATOR) {\n isInsideEscape = false;\n }\n }\n else {\n visible += characterLength;\n if (visible === columns && !nextCharacter.done) {\n rows.push('');\n visible = 0;\n }\n }\n currentCharacter = nextCharacter;\n nextCharacter = characters.next();\n rawCharacterIndex += character.length;\n }\n lastRow = rows.at(-1);\n if (!visible &&\n lastRow !== undefined &&\n lastRow.length > 0 &&\n rows.length > 1) {\n rows[rows.length - 2] += rows.pop();\n }\n};\nconst stringVisibleTrimSpacesRight = (string) => {\n const words = string.split(' ');\n let last = words.length;\n while (last > 0) {\n if (stringWidth(words[last - 1]) > 0) {\n break;\n }\n last--;\n }\n if (last === words.length) {\n return string;\n }\n return words.slice(0, last).join(' ') + words.slice(last).join('');\n};\nconst exec = (string, columns, options = {}) => {\n if (options.trim !== false && string.trim() === '') {\n return '';\n }\n let returnValue = '';\n let escapeCode;\n let escapeUrl;\n const words = string.split(' ');\n const lengths = wordLengths(words);\n let rows = [''];\n for (const [index, word] of words.entries()) {\n if (options.trim !== false) {\n rows[rows.length - 1] = (rows.at(-1) ?? '').trimStart();\n }\n let rowLength = stringWidth(rows.at(-1) ?? '');\n if (index !== 0) {\n if (rowLength >= columns &&\n (options.wordWrap === false || options.trim === false)) {\n rows.push('');\n rowLength = 0;\n }\n if (rowLength > 0 || options.trim === false) {\n rows[rows.length - 1] += ' ';\n rowLength++;\n }\n }\n if (options.hard && lengths[index] > columns) {\n const remainingColumns = columns - rowLength;\n const breaksStartingThisLine = 1 + Math.floor((lengths[index] - remainingColumns - 1) / columns);\n const breaksStartingNextLine = Math.floor((lengths[index] - 1) / columns);\n if (breaksStartingNextLine < breaksStartingThisLine) {\n rows.push('');\n }\n wrapWord(rows, word, columns);\n continue;\n }\n if (rowLength + lengths[index] > columns &&\n rowLength > 0 &&\n lengths[index] > 0) {\n if (options.wordWrap === false && rowLength < columns) {\n wrapWord(rows, word, columns);\n continue;\n }\n rows.push('');\n }\n if (rowLength + lengths[index] > columns && options.wordWrap === false) {\n wrapWord(rows, word, columns);\n continue;\n }\n rows[rows.length - 1] += word;\n }\n if (options.trim !== false) {\n rows = rows.map((row) => stringVisibleTrimSpacesRight(row));\n }\n const preString = rows.join('\\n');\n const pre = preString[Symbol.iterator]();\n let currentPre = pre.next();\n let nextPre = pre.next();\n // We need to keep a separate index as `String#slice()` works on Unicode code units, while `pre` is an array of codepoints.\n let preStringIndex = 0;\n while (!currentPre.done) {\n const character = currentPre.value;\n const nextCharacter = nextPre.value;\n returnValue += character;\n if (character === ESC || character === CSI) {\n GROUP_REGEX.lastIndex = preStringIndex + 1;\n const groupsResult = GROUP_REGEX.exec(preString);\n const groups = groupsResult?.groups;\n if (groups?.code !== undefined) {\n const code = Number.parseFloat(groups.code);\n escapeCode = code === END_CODE ? undefined : code;\n }\n else if (groups?.uri !== undefined) {\n escapeUrl = groups.uri.length === 0 ? undefined : groups.uri;\n }\n }\n const closingCode = escapeCode ? getClosingCode(escapeCode) : undefined;\n if (nextCharacter === '\\n') {\n if (escapeUrl) {\n returnValue += wrapAnsiHyperlink('');\n }\n if (escapeCode && closingCode) {\n returnValue += wrapAnsiCode(closingCode);\n }\n }\n else if (character === '\\n') {\n if (escapeCode && closingCode) {\n returnValue += wrapAnsiCode(escapeCode);\n }\n if (escapeUrl) {\n returnValue += wrapAnsiHyperlink(escapeUrl);\n }\n }\n preStringIndex += character.length;\n currentPre = nextPre;\n nextPre = pre.next();\n }\n return returnValue;\n};\nexport function wrapAnsi(string, columns, options) {\n return String(string)\n .normalize()\n .replaceAll('\\r\\n', '\\n')\n .split('\\n')\n .map((line) => exec(line, columns, options))\n .join('\\n');\n}\n//# sourceMappingURL=main.js.map", "const actions = ['up', 'down', 'left', 'right', 'space', 'enter', 'cancel'] as const;\nexport type Action = (typeof actions)[number];\n\n/** Global settings for Clack programs, stored in memory */\ninterface InternalClackSettings {\n\tactions: Set<Action>;\n\taliases: Map<string, Action>;\n\tmessages: {\n\t\tcancel: string;\n\t\terror: string;\n\t};\n\twithGuide: boolean;\n}\n\nexport const settings: InternalClackSettings = {\n\tactions: new Set(actions),\n\taliases: new Map<string, Action>([\n\t\t// vim support\n\t\t['k', 'up'],\n\t\t['j', 'down'],\n\t\t['h', 'left'],\n\t\t['l', 'right'],\n\t\t['\\x03', 'cancel'],\n\t\t// opinionated defaults!\n\t\t['escape', 'cancel'],\n\t]),\n\tmessages: {\n\t\tcancel: 'Canceled',\n\t\terror: 'Something went wrong',\n\t},\n\twithGuide: true,\n};\n\nexport interface ClackSettings {\n\t/**\n\t * Set custom global aliases for the default actions.\n\t * This will not overwrite existing aliases, it will only add new ones!\n\t *\n\t * @param aliases - An object that maps aliases to actions\n\t * @default { k: 'up', j: 'down', h: 'left', l: 'right', '\\x03': 'cancel', 'escape': 'cancel' }\n\t */\n\taliases?: Record<string, Action>;\n\n\t/**\n\t * Custom messages for prompts\n\t */\n\tmessages?: {\n\t\t/**\n\t\t * Custom message to display when a spinner is cancelled\n\t\t * @default \"Canceled\"\n\t\t */\n\t\tcancel?: string;\n\t\t/**\n\t\t * Custom message to display when a spinner encounters an error\n\t\t * @default \"Something went wrong\"\n\t\t */\n\t\terror?: string;\n\t};\n\n\twithGuide?: boolean;\n}\n\nexport function updateSettings(updates: ClackSettings) {\n\t// Handle each property in the updates\n\tif (updates.aliases !== undefined) {\n\t\tconst aliases = updates.aliases;\n\t\tfor (const alias in aliases) {\n\t\t\tif (!Object.hasOwn(aliases, alias)) continue;\n\n\t\t\tconst action = aliases[alias];\n\t\t\tif (!settings.actions.has(action)) continue;\n\n\t\t\tif (!settings.aliases.has(alias)) {\n\t\t\t\tsettings.aliases.set(alias, action);\n\t\t\t}\n\t\t}\n\t}\n\n\tif (updates.messages !== undefined) {\n\t\tconst messages = updates.messages;\n\t\tif (messages.cancel !== undefined) {\n\t\t\tsettings.messages.cancel = messages.cancel;\n\t\t}\n\t\tif (messages.error !== undefined) {\n\t\t\tsettings.messages.error = messages.error;\n\t\t}\n\t}\n\n\tif (updates.withGuide !== undefined) {\n\t\tsettings.withGuide = updates.withGuide !== false;\n\t}\n}\n\n/**\n * Check if a key is an alias for a default action\n * @param key - The raw key which might match to an action\n * @param action - The action to match\n * @returns boolean\n */\nexport function isActionKey(key: string | Array<string | undefined>, action: Action) {\n\tif (typeof key === 'string') {\n\t\treturn settings.aliases.get(key) === action;\n\t}\n\n\tfor (const value of key) {\n\t\tif (value === undefined) continue;\n\t\tif (isActionKey(value, action)) {\n\t\t\treturn true;\n\t\t}\n\t}\n\treturn false;\n}\n", "export function diffLines(a: string, b: string) {\n\tif (a === b) return;\n\n\tconst aLines = a.split('\\n');\n\tconst bLines = b.split('\\n');\n\tconst numLines = Math.max(aLines.length, bLines.length);\n\tconst diff: number[] = [];\n\n\tfor (let i = 0; i < numLines; i++) {\n\t\tif (aLines[i] !== bLines[i]) diff.push(i);\n\t}\n\n\treturn {\n\t\tlines: diff,\n\t\tnumLinesBefore: aLines.length,\n\t\tnumLinesAfter: bLines.length,\n\t\tnumLines,\n\t};\n}\n", "import { stdin, stdout } from 'node:process';\nimport type { Key } from 'node:readline';\nimport * as readline from 'node:readline';\nimport type { Readable, Writable } from 'node:stream';\nimport { ReadStream } from 'node:tty';\nimport { wrapAnsi } from 'fast-wrap-ansi';\nimport { cursor } from 'sisteransi';\nimport { isActionKey } from './settings.js';\n\nexport * from './settings.js';\nexport * from './string.js';\n\nconst isWindows = globalThis.process.platform.startsWith('win');\n\nexport const CANCEL_SYMBOL = Symbol('clack:cancel');\n\nexport function isCancel(value: unknown): value is symbol {\n\treturn value === CANCEL_SYMBOL;\n}\n\nexport function setRawMode(input: Readable, value: boolean) {\n\tconst i = input as typeof stdin;\n\n\tif (i.isTTY) i.setRawMode(value);\n}\n\ninterface BlockOptions {\n\tinput?: Readable;\n\toutput?: Writable;\n\toverwrite?: boolean;\n\thideCursor?: boolean;\n}\n\nexport function block({\n\tinput = stdin,\n\toutput = stdout,\n\toverwrite = true,\n\thideCursor = true,\n}: BlockOptions = {}) {\n\tconst rl = readline.createInterface({\n\t\tinput,\n\t\toutput,\n\t\tprompt: '',\n\t\ttabSize: 1,\n\t});\n\treadline.emitKeypressEvents(input, rl);\n\n\tif (input instanceof ReadStream && input.isTTY) {\n\t\tinput.setRawMode(true);\n\t}\n\n\tconst clear = (data: Buffer, { name, sequence }: Key) => {\n\t\tconst str = String(data);\n\t\tif (isActionKey([str, name, sequence], 'cancel')) {\n\t\t\tif (hideCursor) output.write(cursor.show);\n\t\t\tprocess.exit(0);\n\t\t\treturn;\n\t\t}\n\t\tif (!overwrite) return;\n\t\tconst dx = name === 'return' ? 0 : -1;\n\t\tconst dy = name === 'return' ? -1 : 0;\n\n\t\treadline.moveCursor(output, dx, dy, () => {\n\t\t\treadline.clearLine(output, 1, () => {\n\t\t\t\tinput.once('keypress', clear);\n\t\t\t});\n\t\t});\n\t};\n\tif (hideCursor) output.write(cursor.hide);\n\tinput.once('keypress', clear);\n\n\treturn () => {\n\t\tinput.off('keypress', clear);\n\t\tif (hideCursor) output.write(cursor.show);\n\n\t\t// Prevent Windows specific issues: https://github.com/bombshell-dev/clack/issues/176\n\t\tif (input instanceof ReadStream && input.isTTY && !isWindows) {\n\t\t\tinput.setRawMode(false);\n\t\t}\n\n\t\t// @ts-expect-error fix for https://github.com/nodejs/node/issues/31762#issuecomment-1441223907\n\t\trl.terminal = false;\n\t\trl.close();\n\t};\n}\n\nexport const getColumns = (output: Writable): number => {\n\tif ('columns' in output && typeof output.columns === 'number') {\n\t\treturn output.columns;\n\t}\n\treturn 80;\n};\n\nexport const getRows = (output: Writable): number => {\n\tif ('rows' in output && typeof output.rows === 'number') {\n\t\treturn output.rows;\n\t}\n\treturn 20;\n};\n\nexport function wrapTextWithPrefix(\n\toutput: Writable | undefined,\n\ttext: string,\n\tprefix: string,\n\tstartPrefix: string = prefix\n): string {\n\tconst columns = getColumns(output ?? stdout);\n\tconst wrapped = wrapAnsi(text, columns - prefix.length, {\n\t\thard: true,\n\t\ttrim: false,\n\t});\n\tconst lines = wrapped\n\t\t.split('\\n')\n\t\t.map((line, index) => {\n\t\t\treturn `${index === 0 ? startPrefix : prefix}${line}`;\n\t\t})\n\t\t.join('\\n');\n\treturn lines;\n}\n", "import { stdin, stdout } from 'node:process';\nimport readline, { type Key, type ReadLine } from 'node:readline';\nimport type { Readable, Writable } from 'node:stream';\nimport { wrapAnsi } from 'fast-wrap-ansi';\nimport { cursor, erase } from 'sisteransi';\nimport type { ClackEvents, ClackState } from '../types.js';\nimport type { Action } from '../utils/index.js';\nimport {\n\tCANCEL_SYMBOL,\n\tdiffLines,\n\tgetRows,\n\tisActionKey,\n\tsetRawMode,\n\tsettings,\n} from '../utils/index.js';\n\nexport interface PromptOptions<TValue, Self extends Prompt<TValue>> {\n\trender(this: Omit<Self, 'prompt'>): string | undefined;\n\tinitialValue?: any;\n\tinitialUserInput?: string;\n\tvalidate?: ((value: TValue | undefined) => string | Error | undefined) | undefined;\n\tinput?: Readable;\n\toutput?: Writable;\n\tdebug?: boolean;\n\tsignal?: AbortSignal;\n}\n\nexport default class Prompt<TValue> {\n\tprotected input: Readable;\n\tprotected output: Writable;\n\tprivate _abortSignal?: AbortSignal;\n\n\tprivate rl: ReadLine | undefined;\n\tprivate opts: Omit<PromptOptions<TValue, Prompt<TValue>>, 'render' | 'input' | 'output'>;\n\tprivate _render: (context: Omit<Prompt<TValue>, 'prompt'>) => string | undefined;\n\tprivate _track = false;\n\tprivate _prevFrame = '';\n\tprivate _subscribers = new Map<string, { cb: (...args: any) => any; once?: boolean }[]>();\n\tprotected _cursor = 0;\n\n\tpublic state: ClackState = 'initial';\n\tpublic error = '';\n\tpublic value: TValue | undefined;\n\tpublic userInput = '';\n\n\tconstructor(options: PromptOptions<TValue, Prompt<TValue>>, trackValue = true) {\n\t\tconst { input = stdin, output = stdout, render, signal, ...opts } = options;\n\n\t\tthis.opts = opts;\n\t\tthis.onKeypress = this.onKeypress.bind(this);\n\t\tthis.close = this.close.bind(this);\n\t\tthis.render = this.render.bind(this);\n\t\tthis._render = render.bind(this);\n\t\tthis._track = trackValue;\n\t\tthis._abortSignal = signal;\n\n\t\tthis.input = input;\n\t\tthis.output = output;\n\t}\n\n\t/**\n\t * Unsubscribe all listeners\n\t */\n\tprotected unsubscribe() {\n\t\tthis._subscribers.clear();\n\t}\n\n\t/**\n\t * Set a subscriber with opts\n\t * @param event - The event name\n\t */\n\tprivate setSubscriber<T extends keyof ClackEvents<TValue>>(\n\t\tevent: T,\n\t\topts: { cb: ClackEvents<TValue>[T]; once?: boolean }\n\t) {\n\t\tconst params = this._subscribers.get(event) ?? [];\n\t\tparams.push(opts);\n\t\tthis._subscribers.set(event, params);\n\t}\n\n\t/**\n\t * Subscribe to an event\n\t * @param event - The event name\n\t * @param cb - The callback\n\t */\n\tpublic on<T extends keyof ClackEvents<TValue>>(event: T, cb: ClackEvents<TValue>[T]) {\n\t\tthis.setSubscriber(event, { cb });\n\t}\n\n\t/**\n\t * Subscribe to an event once\n\t * @param event - The event name\n\t * @param cb - The callback\n\t */\n\tpublic once<T extends keyof ClackEvents<TValue>>(event: T, cb: ClackEvents<TValue>[T]) {\n\t\tthis.setSubscriber(event, { cb, once: true });\n\t}\n\n\t/**\n\t * Emit an event with data\n\t * @param event - The event name\n\t * @param data - The data to pass to the callback\n\t */\n\tpublic emit<T extends keyof ClackEvents<TValue>>(\n\t\tevent: T,\n\t\t...data: Parameters<ClackEvents<TValue>[T]>\n\t) {\n\t\tconst cbs = this._subscribers.get(event) ?? [];\n\t\tconst cleanup: (() => void)[] = [];\n\n\t\tfor (const subscriber of cbs) {\n\t\t\tsubscriber.cb(...data);\n\n\t\t\tif (subscriber.once) {\n\t\t\t\tcleanup.push(() => cbs.splice(cbs.indexOf(subscriber), 1));\n\t\t\t}\n\t\t}\n\n\t\tfor (const cb of cleanup) {\n\t\t\tcb();\n\t\t}\n\t}\n\n\tpublic prompt() {\n\t\treturn new Promise<TValue | symbol | undefined>((resolve) => {\n\t\t\tif (this._abortSignal) {\n\t\t\t\tif (this._abortSignal.aborted) {\n\t\t\t\t\tthis.state = 'cancel';\n\n\t\t\t\t\tthis.close();\n\t\t\t\t\treturn resolve(CANCEL_SYMBOL);\n\t\t\t\t}\n\n\t\t\t\tthis._abortSignal.addEventListener(\n\t\t\t\t\t'abort',\n\t\t\t\t\t() => {\n\t\t\t\t\t\tthis.state = 'cancel';\n\t\t\t\t\t\tthis.close();\n\t\t\t\t\t},\n\t\t\t\t\t{ once: true }\n\t\t\t\t);\n\t\t\t}\n\n\t\t\tthis.rl = readline.createInterface({\n\t\t\t\tinput: this.input,\n\t\t\t\ttabSize: 2,\n\t\t\t\tprompt: '',\n\t\t\t\tescapeCodeTimeout: 50,\n\t\t\t\tterminal: true,\n\t\t\t});\n\t\t\tthis.rl.prompt();\n\n\t\t\tif (this.opts.initialUserInput !== undefined) {\n\t\t\t\tthis._setUserInput(this.opts.initialUserInput, true);\n\t\t\t}\n\n\t\t\tthis.input.on('keypress', this.onKeypress);\n\t\t\tsetRawMode(this.input, true);\n\t\t\tthis.output.on('resize', this.render);\n\n\t\t\tthis.render();\n\n\t\t\tthis.once('submit', () => {\n\t\t\t\tthis.output.write(cursor.show);\n\t\t\t\tthis.output.off('resize', this.render);\n\t\t\t\tsetRawMode(this.input, false);\n\t\t\t\tresolve(this.value);\n\t\t\t});\n\t\t\tthis.once('cancel', () => {\n\t\t\t\tthis.output.write(cursor.show);\n\t\t\t\tthis.output.off('resize', this.render);\n\t\t\t\tsetRawMode(this.input, false);\n\t\t\t\tresolve(CANCEL_SYMBOL);\n\t\t\t});\n\t\t});\n\t}\n\n\tprotected _isActionKey(char: string | undefined, _key: Key): boolean {\n\t\treturn char === '\\t';\n\t}\n\n\tprotected _setValue(value: TValue | undefined): void {\n\t\tthis.value = value;\n\t\tthis.emit('value', this.value);\n\t}\n\n\tprotected _setUserInput(value: string | undefined, write?: boolean): void {\n\t\tthis.userInput = value ?? '';\n\t\tthis.emit('userInput', this.userInput);\n\t\tif (write && this._track && this.rl) {\n\t\t\tthis.rl.write(this.userInput);\n\t\t\tthis._cursor = this.rl.cursor;\n\t\t}\n\t}\n\n\tprotected _clearUserInput(): void {\n\t\tthis.rl?.write(null, { ctrl: true, name: 'u' });\n\t\tthis._setUserInput('');\n\t}\n\n\tprivate onKeypress(char: string | undefined, key: Key) {\n\t\tif (this._track && key.name !== 'return') {\n\t\t\tif (key.name && this._isActionKey(char, key)) {\n\t\t\t\tthis.rl?.write(null, { ctrl: true, name: 'h' });\n\t\t\t}\n\t\t\tthis._cursor = this.rl?.cursor ?? 0;\n\t\t\tthis._setUserInput(this.rl?.line);\n\t\t}\n\n\t\tif (this.state === 'error') {\n\t\t\tthis.state = 'active';\n\t\t}\n\t\tif (key?.name) {\n\t\t\tif (!this._track && settings.aliases.has(key.name)) {\n\t\t\t\tthis.emit('cursor', settings.aliases.get(key.name));\n\t\t\t}\n\t\t\tif (settings.actions.has(key.name as Action)) {\n\t\t\t\tthis.emit('cursor', key.name as Action);\n\t\t\t}\n\t\t}\n\t\tif (char && (char.toLowerCase() === 'y' || char.toLowerCase() === 'n')) {\n\t\t\tthis.emit('confirm', char.toLowerCase() === 'y');\n\t\t}\n\n\t\t// Call the key event handler and emit the key event\n\t\tthis.emit('key', char?.toLowerCase(), key);\n\n\t\tif (key?.name === 'return') {\n\t\t\tif (this.opts.validate) {\n\t\t\t\tconst problem = this.opts.validate(this.value);\n\t\t\t\tif (problem) {\n\t\t\t\t\tthis.error = problem instanceof Error ? problem.message : problem;\n\t\t\t\t\tthis.state = 'error';\n\t\t\t\t\tthis.rl?.write(this.userInput);\n\t\t\t\t}\n\t\t\t}\n\t\t\tif (this.state !== 'error') {\n\t\t\t\tthis.state = 'submit';\n\t\t\t}\n\t\t}\n\n\t\tif (isActionKey([char, key?.name, key?.sequence], 'cancel')) {\n\t\t\tthis.state = 'cancel';\n\t\t}\n\n\t\tif (this.state === 'submit' || this.state === 'cancel') {\n\t\t\tthis.emit('finalize');\n\t\t}\n\t\tthis.render();\n\t\tif (this.state === 'submit' || this.state === 'cancel') {\n\t\t\tthis.close();\n\t\t}\n\t}\n\n\tprotected close() {\n\t\tthis.input.unpipe();\n\t\tthis.input.removeListener('keypress', this.onKeypress);\n\t\tthis.output.write('\\n');\n\t\tsetRawMode(this.input, false);\n\t\tthis.rl?.close();\n\t\tthis.rl = undefined;\n\t\tthis.emit(`${this.state}`, this.value);\n\t\tthis.unsubscribe();\n\t}\n\n\tprivate restoreCursor() {\n\t\tconst lines =\n\t\t\twrapAnsi(this._prevFrame, process.stdout.columns, { hard: true, trim: false }).split('\\n')\n\t\t\t\t.length - 1;\n\t\tthis.output.write(cursor.move(-999, lines * -1));\n\t}\n\n\tprivate render() {\n\t\tconst frame = wrapAnsi(this._render(this) ?? '', process.stdout.columns, {\n\t\t\thard: true,\n\t\t\ttrim: false,\n\t\t});\n\t\tif (frame === this._prevFrame) return;\n\n\t\tif (this.state === 'initial') {\n\t\t\tthis.output.write(cursor.hide);\n\t\t} else {\n\t\t\tconst diff = diffLines(this._prevFrame, frame);\n\t\t\tconst rows = getRows(this.output);\n\t\t\tthis.restoreCursor();\n\t\t\tif (diff) {\n\t\t\t\tconst diffOffsetAfter = Math.max(0, diff.numLinesAfter - rows);\n\t\t\t\tconst diffOffsetBefore = Math.max(0, diff.numLinesBefore - rows);\n\t\t\t\tlet diffLine = diff.lines.find((line) => line >= diffOffsetAfter);\n\n\t\t\t\tif (diffLine === undefined) {\n\t\t\t\t\tthis._prevFrame = frame;\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\t// If a single line has changed, only update that line\n\t\t\t\tif (diff.lines.length === 1) {\n\t\t\t\t\tthis.output.write(cursor.move(0, diffLine - diffOffsetBefore));\n\t\t\t\t\tthis.output.write(erase.lines(1));\n\t\t\t\t\tconst lines = frame.split('\\n');\n\t\t\t\t\tthis.output.write(lines[diffLine]);\n\t\t\t\t\tthis._prevFrame = frame;\n\t\t\t\t\tthis.output.write(cursor.move(0, lines.length - diffLine - 1));\n\t\t\t\t\treturn;\n\t\t\t\t\t// If many lines have changed, rerender everything past the first line\n\t\t\t\t} else if (diff.lines.length > 1) {\n\t\t\t\t\tif (diffOffsetAfter < diffOffsetBefore) {\n\t\t\t\t\t\tdiffLine = diffOffsetAfter;\n\t\t\t\t\t} else {\n\t\t\t\t\t\tconst adjustedDiffLine = diffLine - diffOffsetBefore;\n\t\t\t\t\t\tif (adjustedDiffLine > 0) {\n\t\t\t\t\t\t\tthis.output.write(cursor.move(0, adjustedDiffLine));\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\tthis.output.write(erase.down());\n\t\t\t\t\tconst lines = frame.split('\\n');\n\t\t\t\t\tconst newLines = lines.slice(diffLine);\n\t\t\t\t\tthis.output.write(newLines.join('\\n'));\n\t\t\t\t\tthis._prevFrame = frame;\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tthis.output.write(erase.down());\n\t\t}\n\n\t\tthis.output.write(frame);\n\t\tif (this.state === 'initial') {\n\t\t\tthis.state = 'active';\n\t\t}\n\t\tthis._prevFrame = frame;\n\t}\n}\n", "import type { Key } from 'node:readline';\nimport color from 'picocolors';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\ninterface OptionLike {\n\tvalue: unknown;\n\tlabel?: string;\n}\n\ntype FilterFunction<T extends OptionLike> = (search: string, opt: T) => boolean;\n\nfunction getCursorForValue<T extends OptionLike>(\n\tselected: T['value'] | undefined,\n\titems: T[]\n): number {\n\tif (selected === undefined) {\n\t\treturn 0;\n\t}\n\n\tconst currLength = items.length;\n\n\t// If filtering changed the available options, update cursor\n\tif (currLength === 0) {\n\t\treturn 0;\n\t}\n\n\t// Try to maintain the same selected item\n\tconst index = items.findIndex((item) => item.value === selected);\n\treturn index !== -1 ? index : 0;\n}\n\nfunction defaultFilter<T extends OptionLike>(input: string, option: T): boolean {\n\tconst label = option.label ?? String(option.value);\n\treturn label.toLowerCase().includes(input.toLowerCase());\n}\n\nfunction normalisedValue<T>(multiple: boolean, values: T[] | undefined): T | T[] | undefined {\n\tif (!values) {\n\t\treturn undefined;\n\t}\n\tif (multiple) {\n\t\treturn values;\n\t}\n\treturn values[0];\n}\n\nexport interface AutocompleteOptions<T extends OptionLike>\n\textends PromptOptions<T['value'] | T['value'][], AutocompletePrompt<T>> {\n\toptions: T[] | ((this: AutocompletePrompt<T>) => T[]);\n\tfilter?: FilterFunction<T>;\n\tmultiple?: boolean;\n}\n\nexport default class AutocompletePrompt<T extends OptionLike> extends Prompt<\n\tT['value'] | T['value'][]\n> {\n\tfilteredOptions: T[];\n\tmultiple: boolean;\n\tisNavigating = false;\n\tselectedValues: Array<T['value']> = [];\n\n\tfocusedValue: T['value'] | undefined;\n\t#cursor = 0;\n\t#lastUserInput = '';\n\t#filterFn: FilterFunction<T>;\n\t#options: T[] | (() => T[]);\n\n\tget cursor(): number {\n\t\treturn this.#cursor;\n\t}\n\n\tget userInputWithCursor() {\n\t\tif (!this.userInput) {\n\t\t\treturn color.inverse(color.hidden('_'));\n\t\t}\n\t\tif (this._cursor >= this.userInput.length) {\n\t\t\treturn `${this.userInput}█`;\n\t\t}\n\t\tconst s1 = this.userInput.slice(0, this._cursor);\n\t\tconst [s2, ...s3] = this.userInput.slice(this._cursor);\n\t\treturn `${s1}${color.inverse(s2)}${s3.join('')}`;\n\t}\n\n\tget options(): T[] {\n\t\tif (typeof this.#options === 'function') {\n\t\t\treturn this.#options();\n\t\t}\n\t\treturn this.#options;\n\t}\n\n\tconstructor(opts: AutocompleteOptions<T>) {\n\t\tsuper(opts);\n\n\t\tthis.#options = opts.options;\n\t\tconst options = this.options;\n\t\tthis.filteredOptions = [...options];\n\t\tthis.multiple = opts.multiple === true;\n\t\tthis.#filterFn = opts.filter ?? defaultFilter;\n\t\tlet initialValues: unknown[] | undefined;\n\t\tif (opts.initialValue && Array.isArray(opts.initialValue)) {\n\t\t\tif (this.multiple) {\n\t\t\t\tinitialValues = opts.initialValue;\n\t\t\t} else {\n\t\t\t\tinitialValues = opts.initialValue.slice(0, 1);\n\t\t\t}\n\t\t} else {\n\t\t\tif (!this.multiple && this.options.length > 0) {\n\t\t\t\tinitialValues = [this.options[0].value];\n\t\t\t}\n\t\t}\n\n\t\tif (initialValues) {\n\t\t\tfor (const selectedValue of initialValues) {\n\t\t\t\tconst selectedIndex = options.findIndex((opt) => opt.value === selectedValue);\n\t\t\t\tif (selectedIndex !== -1) {\n\t\t\t\t\tthis.toggleSelected(selectedValue);\n\t\t\t\t\tthis.#cursor = selectedIndex;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tthis.focusedValue = this.options[this.#cursor]?.value;\n\n\t\tthis.on('key', (char, key) => this.#onKey(char, key));\n\t\tthis.on('userInput', (value) => this.#onUserInputChanged(value));\n\t}\n\n\tprotected override _isActionKey(char: string | undefined, key: Key): boolean {\n\t\treturn (\n\t\t\tchar === '\\t' ||\n\t\t\t(this.multiple &&\n\t\t\t\tthis.isNavigating &&\n\t\t\t\tkey.name === 'space' &&\n\t\t\t\tchar !== undefined &&\n\t\t\t\tchar !== '')\n\t\t);\n\t}\n\n\t#onKey(_char: string | undefined, key: Key): void {\n\t\tconst isUpKey = key.name === 'up';\n\t\tconst isDownKey = key.name === 'down';\n\t\tconst isReturnKey = key.name === 'return';\n\n\t\t// Start navigation mode with up/down arrows\n\t\tif (isUpKey || isDownKey) {\n\t\t\tthis.#cursor = Math.max(\n\t\t\t\t0,\n\t\t\t\tMath.min(this.#cursor + (isUpKey ? -1 : 1), this.filteredOptions.length - 1)\n\t\t\t);\n\t\t\tthis.focusedValue = this.filteredOptions[this.#cursor]?.value;\n\t\t\tif (!this.multiple) {\n\t\t\t\tthis.selectedValues = [this.focusedValue];\n\t\t\t}\n\t\t\tthis.isNavigating = true;\n\t\t} else if (isReturnKey) {\n\t\t\tthis.value = normalisedValue(this.multiple, this.selectedValues);\n\t\t} else {\n\t\t\tif (this.multiple) {\n\t\t\t\tif (\n\t\t\t\t\tthis.focusedValue !== undefined &&\n\t\t\t\t\t(key.name === 'tab' || (this.isNavigating && key.name === 'space'))\n\t\t\t\t) {\n\t\t\t\t\tthis.toggleSelected(this.focusedValue);\n\t\t\t\t} else {\n\t\t\t\t\tthis.isNavigating = false;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tif (this.focusedValue) {\n\t\t\t\t\tthis.selectedValues = [this.focusedValue];\n\t\t\t\t}\n\t\t\t\tthis.isNavigating = false;\n\t\t\t}\n\t\t}\n\t}\n\n\tdeselectAll() {\n\t\tthis.selectedValues = [];\n\t}\n\n\ttoggleSelected(value: T['value']) {\n\t\tif (this.filteredOptions.length === 0) {\n\t\t\treturn;\n\t\t}\n\n\t\tif (this.multiple) {\n\t\t\tif (this.selectedValues.includes(value)) {\n\t\t\t\tthis.selectedValues = this.selectedValues.filter((v) => v !== value);\n\t\t\t} else {\n\t\t\t\tthis.selectedValues = [...this.selectedValues, value];\n\t\t\t}\n\t\t} else {\n\t\t\tthis.selectedValues = [value];\n\t\t}\n\t}\n\n\t#onUserInputChanged(value: string): void {\n\t\tif (value !== this.#lastUserInput) {\n\t\t\tthis.#lastUserInput = value;\n\n\t\t\tconst options = this.options;\n\n\t\t\tif (value) {\n\t\t\t\tthis.filteredOptions = options.filter((opt) => this.#filterFn(value, opt));\n\t\t\t} else {\n\t\t\t\tthis.filteredOptions = [...options];\n\t\t\t}\n\t\t\tthis.#cursor = getCursorForValue(this.focusedValue, this.filteredOptions);\n\t\t\tthis.focusedValue = this.filteredOptions[this.#cursor]?.value;\n\t\t\tif (!this.multiple) {\n\t\t\t\tif (this.focusedValue !== undefined) {\n\t\t\t\t\tthis.toggleSelected(this.focusedValue);\n\t\t\t\t} else {\n\t\t\t\t\tthis.deselectAll();\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n}\n", "import { cursor } from 'sisteransi';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface ConfirmOptions extends PromptOptions<boolean, ConfirmPrompt> {\n\tactive: string;\n\tinactive: string;\n\tinitialValue?: boolean;\n}\n\nexport default class ConfirmPrompt extends Prompt<boolean> {\n\tget cursor() {\n\t\treturn this.value ? 0 : 1;\n\t}\n\n\tprivate get _value() {\n\t\treturn this.cursor === 0;\n\t}\n\n\tconstructor(opts: ConfirmOptions) {\n\t\tsuper(opts, false);\n\t\tthis.value = !!opts.initialValue;\n\n\t\tthis.on('userInput', () => {\n\t\t\tthis.value = this._value;\n\t\t});\n\n\t\tthis.on('confirm', (confirm) => {\n\t\t\tthis.output.write(cursor.move(0, -1));\n\t\t\tthis.value = confirm;\n\t\t\tthis.state = 'submit';\n\t\t\tthis.close();\n\t\t});\n\n\t\tthis.on('cursor', () => {\n\t\t\tthis.value = !this.value;\n\t\t});\n\t}\n}\n", "import Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface GroupMultiSelectOptions<T extends { value: any }>\n\textends PromptOptions<T['value'][], GroupMultiSelectPrompt<T>> {\n\toptions: Record<string, T[]>;\n\tinitialValues?: T['value'][];\n\trequired?: boolean;\n\tcursorAt?: T['value'];\n\tselectableGroups?: boolean;\n}\nexport default class GroupMultiSelectPrompt<T extends { value: any }> extends Prompt<T['value'][]> {\n\toptions: (T & { group: string | boolean })[];\n\tcursor = 0;\n\t#selectableGroups: boolean;\n\n\tgetGroupItems(group: string): T[] {\n\t\treturn this.options.filter((o) => o.group === group);\n\t}\n\n\tisGroupSelected(group: string) {\n\t\tconst items = this.getGroupItems(group);\n\t\tconst value = this.value;\n\t\tif (value === undefined) {\n\t\t\treturn false;\n\t\t}\n\t\treturn items.every((i) => value.includes(i.value));\n\t}\n\n\tprivate toggleValue() {\n\t\tconst item = this.options[this.cursor];\n\t\tif (this.value === undefined) {\n\t\t\tthis.value = [];\n\t\t}\n\t\tif (item.group === true) {\n\t\t\tconst group = item.value;\n\t\t\tconst groupedItems = this.getGroupItems(group);\n\t\t\tif (this.isGroupSelected(group)) {\n\t\t\t\tthis.value = this.value.filter(\n\t\t\t\t\t(v: string) => groupedItems.findIndex((i) => i.value === v) === -1\n\t\t\t\t);\n\t\t\t} else {\n\t\t\t\tthis.value = [...this.value, ...groupedItems.map((i) => i.value)];\n\t\t\t}\n\t\t\tthis.value = Array.from(new Set(this.value));\n\t\t} else {\n\t\t\tconst selected = this.value.includes(item.value);\n\t\t\tthis.value = selected\n\t\t\t\t? this.value.filter((v: T['value']) => v !== item.value)\n\t\t\t\t: [...this.value, item.value];\n\t\t}\n\t}\n\n\tconstructor(opts: GroupMultiSelectOptions<T>) {\n\t\tsuper(opts, false);\n\t\tconst { options } = opts;\n\t\tthis.#selectableGroups = opts.selectableGroups !== false;\n\t\tthis.options = Object.entries(options).flatMap(([key, option]) => [\n\t\t\t{ value: key, group: true, label: key },\n\t\t\t...option.map((opt) => ({ ...opt, group: key })),\n\t\t]) as any;\n\t\tthis.value = [...(opts.initialValues ?? [])];\n\t\tthis.cursor = Math.max(\n\t\t\tthis.options.findIndex(({ value }) => value === opts.cursorAt),\n\t\t\tthis.#selectableGroups ? 0 : 1\n\t\t);\n\n\t\tthis.on('cursor', (key) => {\n\t\t\tswitch (key) {\n\t\t\t\tcase 'left':\n\t\t\t\tcase 'up': {\n\t\t\t\t\tthis.cursor = this.cursor === 0 ? this.options.length - 1 : this.cursor - 1;\n\t\t\t\t\tconst currentIsGroup = this.options[this.cursor]?.group === true;\n\t\t\t\t\tif (!this.#selectableGroups && currentIsGroup) {\n\t\t\t\t\t\tthis.cursor = this.cursor === 0 ? this.options.length - 1 : this.cursor - 1;\n\t\t\t\t\t}\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t\tcase 'down':\n\t\t\t\tcase 'right': {\n\t\t\t\t\tthis.cursor = this.cursor === this.options.length - 1 ? 0 : this.cursor + 1;\n\t\t\t\t\tconst currentIsGroup = this.options[this.cursor]?.group === true;\n\t\t\t\t\tif (!this.#selectableGroups && currentIsGroup) {\n\t\t\t\t\t\tthis.cursor = this.cursor === this.options.length - 1 ? 0 : this.cursor + 1;\n\t\t\t\t\t}\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t\tcase 'space':\n\t\t\t\t\tthis.toggleValue();\n\t\t\t\t\tbreak;\n\t\t\t}\n\t\t});\n\t}\n}\n", "export function findCursor<T extends { disabled?: boolean }>(\n\tcursor: number,\n\tdelta: number,\n\toptions: T[]\n) {\n\tconst newCursor = cursor + delta;\n\tconst maxCursor = Math.max(options.length - 1, 0);\n\tconst clampedCursor = newCursor < 0 ? maxCursor : newCursor > maxCursor ? 0 : newCursor;\n\tconst newOption = options[clampedCursor];\n\tif (newOption.disabled) {\n\t\treturn findCursor(clampedCursor, delta < 0 ? -1 : 1, options);\n\t}\n\treturn clampedCursor;\n}\n", "import { findCursor } from '../utils/cursor.js';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\ninterface OptionLike {\n\tvalue: any;\n\tdisabled?: boolean;\n}\n\nexport interface MultiSelectOptions<T extends OptionLike>\n\textends PromptOptions<T['value'][], MultiSelectPrompt<T>> {\n\toptions: T[];\n\tinitialValues?: T['value'][];\n\trequired?: boolean;\n\tcursorAt?: T['value'];\n}\nexport default class MultiSelectPrompt<T extends OptionLike> extends Prompt<T['value'][]> {\n\toptions: T[];\n\tcursor = 0;\n\n\tprivate get _value(): T['value'] {\n\t\treturn this.options[this.cursor].value;\n\t}\n\n\tprivate get _enabledOptions(): T[] {\n\t\treturn this.options.filter((option) => option.disabled !== true);\n\t}\n\n\tprivate toggleAll() {\n\t\tconst enabledOptions = this._enabledOptions;\n\t\tconst allSelected = this.value !== undefined && this.value.length === enabledOptions.length;\n\t\tthis.value = allSelected ? [] : enabledOptions.map((v) => v.value);\n\t}\n\n\tprivate toggleInvert() {\n\t\tconst value = this.value;\n\t\tif (!value) {\n\t\t\treturn;\n\t\t}\n\t\tconst notSelected = this._enabledOptions.filter((v) => !value.includes(v.value));\n\t\tthis.value = notSelected.map((v) => v.value);\n\t}\n\n\tprivate toggleValue() {\n\t\tif (this.value === undefined) {\n\t\t\tthis.value = [];\n\t\t}\n\t\tconst selected = this.value.includes(this._value);\n\t\tthis.value = selected\n\t\t\t? this.value.filter((value) => value !== this._value)\n\t\t\t: [...this.value, this._value];\n\t}\n\n\tconstructor(opts: MultiSelectOptions<T>) {\n\t\tsuper(opts, false);\n\n\t\tthis.options = opts.options;\n\t\tthis.value = [...(opts.initialValues ?? [])];\n\t\tconst cursor = Math.max(\n\t\t\tthis.options.findIndex(({ value }) => value === opts.cursorAt),\n\t\t\t0\n\t\t);\n\t\tthis.cursor = this.options[cursor].disabled ? findCursor<T>(cursor, 1, this.options) : cursor;\n\t\tthis.on('key', (char) => {\n\t\t\tif (char === 'a') {\n\t\t\t\tthis.toggleAll();\n\t\t\t}\n\t\t\tif (char === 'i') {\n\t\t\t\tthis.toggleInvert();\n\t\t\t}\n\t\t});\n\n\t\tthis.on('cursor', (key) => {\n\t\t\tswitch (key) {\n\t\t\t\tcase 'left':\n\t\t\t\tcase 'up':\n\t\t\t\t\tthis.cursor = findCursor<T>(this.cursor, -1, this.options);\n\t\t\t\t\tbreak;\n\t\t\t\tcase 'down':\n\t\t\t\tcase 'right':\n\t\t\t\t\tthis.cursor = findCursor<T>(this.cursor, 1, this.options);\n\t\t\t\t\tbreak;\n\t\t\t\tcase 'space':\n\t\t\t\t\tthis.toggleValue();\n\t\t\t\t\tbreak;\n\t\t\t}\n\t\t});\n\t}\n}\n", "import color from 'picocolors';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface PasswordOptions extends PromptOptions<string, PasswordPrompt> {\n\tmask?: string;\n}\nexport default class PasswordPrompt extends Prompt<string> {\n\tprivate _mask = '•';\n\tget cursor() {\n\t\treturn this._cursor;\n\t}\n\tget masked() {\n\t\treturn this.userInput.replaceAll(/./g, this._mask);\n\t}\n\tget userInputWithCursor() {\n\t\tif (this.state === 'submit' || this.state === 'cancel') {\n\t\t\treturn this.masked;\n\t\t}\n\t\tconst userInput = this.userInput;\n\t\tif (this.cursor >= userInput.length) {\n\t\t\treturn `${this.masked}${color.inverse(color.hidden('_'))}`;\n\t\t}\n\t\tconst masked = this.masked;\n\t\tconst s1 = masked.slice(0, this.cursor);\n\t\tconst s2 = masked.slice(this.cursor);\n\t\treturn `${s1}${color.inverse(s2[0])}${s2.slice(1)}`;\n\t}\n\tclear() {\n\t\tthis._clearUserInput();\n\t}\n\tconstructor({ mask, ...opts }: PasswordOptions) {\n\t\tsuper(opts);\n\t\tthis._mask = mask ?? '•';\n\t\tthis.on('userInput', (input) => {\n\t\t\tthis._setValue(input);\n\t\t});\n\t}\n}\n", "import { findCursor } from '../utils/cursor.js';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface SelectOptions<T extends { value: any; disabled?: boolean }>\n\textends PromptOptions<T['value'], SelectPrompt<T>> {\n\toptions: T[];\n\tinitialValue?: T['value'];\n}\nexport default class SelectPrompt<T extends { value: any; disabled?: boolean }> extends Prompt<\n\tT['value']\n> {\n\toptions: T[];\n\tcursor = 0;\n\n\tprivate get _selectedValue() {\n\t\treturn this.options[this.cursor];\n\t}\n\n\tprivate changeValue() {\n\t\tthis.value = this._selectedValue.value;\n\t}\n\n\tconstructor(opts: SelectOptions<T>) {\n\t\tsuper(opts, false);\n\n\t\tthis.options = opts.options;\n\n\t\tconst initialCursor = this.options.findIndex(({ value }) => value === opts.initialValue);\n\t\tconst cursor = initialCursor === -1 ? 0 : initialCursor;\n\t\tthis.cursor = this.options[cursor].disabled ? findCursor<T>(cursor, 1, this.options) : cursor;\n\t\tthis.changeValue();\n\n\t\tthis.on('cursor', (key) => {\n\t\t\tswitch (key) {\n\t\t\t\tcase 'left':\n\t\t\t\tcase 'up':\n\t\t\t\t\tthis.cursor = findCursor<T>(this.cursor, -1, this.options);\n\t\t\t\t\tbreak;\n\t\t\t\tcase 'down':\n\t\t\t\tcase 'right':\n\t\t\t\t\tthis.cursor = findCursor<T>(this.cursor, 1, this.options);\n\t\t\t\t\tbreak;\n\t\t\t}\n\t\t\tthis.changeValue();\n\t\t});\n\t}\n}\n", "import Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface SelectKeyOptions<T extends { value: string }>\n\textends PromptOptions<T['value'], SelectKeyPrompt<T>> {\n\toptions: T[];\n\tcaseSensitive?: boolean;\n}\nexport default class SelectKeyPrompt<T extends { value: string }> extends Prompt<T['value']> {\n\toptions: T[];\n\tcursor = 0;\n\n\tconstructor(opts: SelectKeyOptions<T>) {\n\t\tsuper(opts, false);\n\n\t\tthis.options = opts.options;\n\t\tconst caseSensitive = opts.caseSensitive === true;\n\t\tconst keys = this.options.map(({ value: [initial] }) => {\n\t\t\treturn caseSensitive ? initial : initial?.toLowerCase();\n\t\t});\n\t\tthis.cursor = Math.max(keys.indexOf(opts.initialValue), 0);\n\n\t\tthis.on('key', (key, keyInfo) => {\n\t\t\tif (!key) {\n\t\t\t\treturn;\n\t\t\t}\n\t\t\tconst casedKey = caseSensitive && keyInfo.shift ? key.toUpperCase() : key;\n\t\t\tif (!keys.includes(casedKey)) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst value = this.options.find(({ value: [initial] }) => {\n\t\t\t\treturn caseSensitive ? initial === casedKey : initial?.toLowerCase() === key;\n\t\t\t});\n\t\t\tif (value) {\n\t\t\t\tthis.value = value.value;\n\t\t\t\tthis.state = 'submit';\n\t\t\t\tthis.emit('submit');\n\t\t\t}\n\t\t});\n\t}\n}\n", "import color from 'picocolors';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface TextOptions extends PromptOptions<string, TextPrompt> {\n\tplaceholder?: string;\n\tdefaultValue?: string;\n}\n\nexport default class TextPrompt extends Prompt<string> {\n\tget userInputWithCursor() {\n\t\tif (this.state === 'submit') {\n\t\t\treturn this.userInput;\n\t\t}\n\t\tconst userInput = this.userInput;\n\t\tif (this.cursor >= userInput.length) {\n\t\t\treturn `${this.userInput}█`;\n\t\t}\n\t\tconst s1 = userInput.slice(0, this.cursor);\n\t\tconst [s2, ...s3] = userInput.slice(this.cursor);\n\t\treturn `${s1}${color.inverse(s2)}${s3.join('')}`;\n\t}\n\tget cursor() {\n\t\treturn this._cursor;\n\t}\n\tconstructor(opts: TextOptions) {\n\t\tsuper({\n\t\t\t...opts,\n\t\t\tinitialUserInput: opts.initialUserInput ?? opts.initialValue,\n\t\t});\n\n\t\tthis.on('userInput', (input) => {\n\t\t\tthis._setValue(input);\n\t\t});\n\t\tthis.on('finalize', () => {\n\t\t\tif (!this.value) {\n\t\t\t\tthis.value = opts.defaultValue;\n\t\t\t}\n\t\t\tif (this.value === undefined) {\n\t\t\t\tthis.value = '';\n\t\t\t}\n\t\t});\n\t}\n}\n", "import chalk from \"chalk\";\nimport { text } from \"../../components/text.js\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\nimport { stripAnsi } from \"../../internal/strip-ansi.js\";\n\nexport function intro(title: string): void {\n const format = resolveOutputFormat();\n if (format === \"json\") {\n return;\n }\n\n if (format === \"markdown\") {\n process.stdout.write(`# ${stripAnsi(title)}\\n\\n`);\n return;\n }\n\n process.stdout.write(`${chalk.gray(\"\u250C\")} ${text.intro(title)}\\n`);\n}\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\nimport { stripAnsi } from \"../../internal/strip-ansi.js\";\n\nfunction getVisibleWidth(value: string): number {\n return stripAnsi(value).length;\n}\n\nfunction renderTerminalNote(message: string, title?: string): string {\n const contentLines = [\"\", ...message.split(\"\\n\"), \"\"];\n const visibleTitle = stripAnsi(title ?? \"\");\n const contentWidth = Math.max(\n visibleTitle.length,\n ...contentLines.map((line) => getVisibleWidth(line))\n ) + 2;\n const titleLine = `${chalk.green(\"\u25C7\")} ${chalk.reset(title ?? \"\")} ${chalk.gray(\n `${\"\u2500\".repeat(Math.max(contentWidth - visibleTitle.length - 1, 1))}\u256E`\n )}`;\n const content = contentLines.map((line) => {\n const padding = \" \".repeat(contentWidth - getVisibleWidth(line));\n return `${chalk.gray(\"\u2502\")} ${line}${padding}${chalk.gray(\"\u2502\")}`;\n });\n const bottom = chalk.gray(`\u251C${\"\u2500\".repeat(contentWidth + 2)}\u256F`);\n\n return [chalk.gray(\"\u2502\"), titleLine, ...content, bottom].join(\"\\n\");\n}\n\nexport function note(message: string, title?: string): void {\n const format = resolveOutputFormat();\n const strippedMessage = stripAnsi(message);\n const strippedTitle = stripAnsi(title ?? \"\");\n\n if (format === \"markdown\") {\n const lines = strippedMessage.split(\"\\n\");\n const heading = strippedTitle ? `> **${strippedTitle}**\\n` : \"\";\n const body = lines.map((line) => `> ${line}`).join(\"\\n\");\n process.stdout.write(`${heading}${body}\\n`);\n return;\n }\n\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({\n type: \"note\",\n title: strippedTitle,\n message: strippedMessage\n })}\\n`\n );\n return;\n }\n\n process.stdout.write(`${renderTerminalNote(message, title)}\\n`);\n}\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\nimport { stripAnsi } from \"../../internal/strip-ansi.js\";\n\nexport function outro(message: string): void {\n const format = resolveOutputFormat();\n const stripped = stripAnsi(message);\n\n if (format === \"markdown\") {\n process.stdout.write(`---\\n${stripped}\\n`);\n return;\n }\n\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ type: \"outro\", message: stripped })}\\n`\n );\n return;\n }\n\n process.stdout.write(`${chalk.gray(\"\u2502\")}\\n${chalk.gray(\"\u2514\")} ${message}\\n\\n`);\n}\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\nimport { stripAnsi } from \"../../internal/strip-ansi.js\";\nimport { SPINNER_FRAMES } from \"../../static/spinner.js\";\n\nexport interface SpinnerOptions {\n start: (message?: string) => void;\n stop: (message?: string, code?: number) => void;\n message: (message?: string) => void;\n}\n\nfunction writeTerminalFrame(frame: string, message: string): void {\n process.stdout.write(`\\r\\x1b[K${frame} ${message}`);\n}\n\nexport function spinner(): SpinnerOptions {\n let currentMessage = \"\";\n let frameIndex = 0;\n let timer: ReturnType<typeof setInterval> | undefined;\n let fallback = false;\n\n const format = resolveOutputFormat();\n\n const renderFrame = (): void => {\n writeTerminalFrame(SPINNER_FRAMES[frameIndex % SPINNER_FRAMES.length], currentMessage);\n };\n\n const clearTimer = (): void => {\n if (!timer) {\n return;\n }\n clearInterval(timer);\n timer = undefined;\n };\n\n return {\n start(message = \"\"): void {\n currentMessage = stripAnsi(message);\n\n if (format === \"json\") {\n return;\n }\n\n if (format === \"markdown\") {\n process.stdout.write(`- ${currentMessage}...\\n`);\n return;\n }\n\n fallback = process.env.POE_NO_SPINNER === \"1\" || !process.stdout.isTTY;\n if (fallback) {\n process.stdout.write(`${currentMessage}\\n`);\n return;\n }\n\n frameIndex = 0;\n renderFrame();\n timer = setInterval(() => {\n frameIndex += 1;\n renderFrame();\n }, 16);\n },\n\n message(message = \"\"): void {\n currentMessage = stripAnsi(message);\n\n if (format !== \"terminal\" || fallback || !timer) {\n return;\n }\n\n renderFrame();\n },\n\n stop(message = currentMessage, code?: number): void {\n currentMessage = stripAnsi(message);\n\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ type: \"spinner\", state: \"stopped\", message: currentMessage })}\\n`\n );\n return;\n }\n\n if (format === \"markdown\") {\n process.stdout.write(`- ${currentMessage}\\n`);\n return;\n }\n\n clearTimer();\n\n const symbol = code === undefined || code === 0\n ? chalk.green(\"\u25C6\")\n : chalk.red(\"\u25A0\");\n\n if (fallback) {\n process.stdout.write(`${symbol} ${currentMessage}\\n`);\n return;\n }\n\n process.stdout.write(`\\r\\x1b[K${symbol} ${currentMessage}\\n`);\n }\n };\n}\n", "import chalk from \"chalk\";\nimport { symbols } from \"../components/symbols.js\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\n\nexport const SPINNER_FRAMES = [\"\u25D2\", \"\u25D0\", \"\u25D3\", \"\u25D1\"] as const;\n\nexport interface SpinnerFrameOptions {\n frame?: number;\n message: string;\n timer?: string;\n}\n\nexport function renderSpinnerFrame(options: SpinnerFrameOptions): string {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n return `- ${options.message}${options.timer ? ` [${options.timer}]` : \"\"}...\\n`;\n }\n\n if (format === \"json\") {\n return `${JSON.stringify({\n type: \"spinner\",\n state: \"running\",\n message: options.message,\n ...(options.timer ? { timer: options.timer } : {})\n })}\\n`;\n }\n\n const frame = options.frame ?? 0;\n const spinnerChar = chalk.magenta(SPINNER_FRAMES[frame % SPINNER_FRAMES.length]);\n const timerSuffix = options.timer ? chalk.dim(` [${options.timer}]`) : \"\";\n const bar = chalk.gray(symbols.bar);\n\n return `${spinnerChar} ${options.message}${timerSuffix}\\n${bar}`;\n}\n\nexport interface SpinnerStoppedOptions {\n message: string;\n code?: number;\n timer?: string;\n subtext?: string;\n}\n\nexport function renderSpinnerStopped(options: SpinnerStoppedOptions): string {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n return `- ${options.message}${options.timer ? ` [${options.timer}]` : \"\"}\\n`;\n }\n\n if (format === \"json\") {\n return `${JSON.stringify({\n type: \"spinner\",\n state: \"stopped\",\n message: options.message,\n ...(options.timer ? { timer: options.timer } : {})\n })}\\n`;\n }\n\n const code = options.code ?? 0;\n const symbol = code === 0 ? chalk.green(\"\u25C6\") : chalk.red(\"\u25A0\");\n const timerSuffix = options.timer ? chalk.dim(` [${options.timer}]`) : \"\";\n const bar = chalk.gray(symbols.bar);\n\n let output = `${symbol} ${options.message}${timerSuffix}`;\n if (options.subtext) {\n output += `\\n${bar} ${chalk.dim(options.subtext)}`;\n }\n return output;\n}\n", "import chalk from \"chalk\";\nimport { symbols } from \"../components/symbols.js\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { getTheme } from \"../internal/theme-detect.js\";\n\nexport interface MenuOption {\n label: string;\n value: string;\n hint?: string;\n}\n\nexport interface RenderMenuOptions {\n message: string;\n options: MenuOption[];\n selectedIndex?: number;\n}\n\nexport function renderMenu(opts: RenderMenuOptions): string {\n const format = resolveOutputFormat();\n const selectedIndex = opts.selectedIndex ?? 0;\n\n if (format === \"markdown\") {\n return [\n `**${opts.message}**`,\n ...opts.options.map(\n (option, index) => `- [${index === selectedIndex ? \"x\" : \" \"}] ${option.label}`\n )\n ].join(\"\\n\");\n }\n\n if (format === \"json\") {\n return JSON.stringify({\n type: \"menu\",\n message: opts.message,\n options: opts.options,\n selected: selectedIndex\n });\n }\n\n const theme = getTheme();\n const bar = chalk.gray(symbols.bar);\n const lines: string[] = [];\n\n lines.push(`${chalk.cyan(symbols.active)} ${opts.message}`);\n lines.push(bar);\n\n opts.options.forEach((option, index) => {\n const isSelected = index === selectedIndex;\n const prefix = isSelected ? chalk.cyan(symbols.active) : chalk.gray(symbols.inactive);\n const label = isSelected ? theme.accent(option.label) : option.label;\n const hint = option.hint ? chalk.dim(` (${option.hint})`) : \"\";\n lines.push(`${bar} ${prefix} ${label}${hint}`);\n });\n\n lines.push(`${bar}`);\n return lines.join(\"\\n\");\n}\n", "import { spawn as spawnChildProcess } from \"node:child_process\";\nimport { getAdapter } from \"../adapters/index.js\";\nimport type { AcpEvent } from \"./types.js\";\nimport { readLines } from \"./line-reader.js\";\nimport { resolveConfig } from \"../configs/resolve-config.js\";\nimport { getMcpArgs, getMcpEnv } from \"../mcp-args.js\";\nimport { stripModelNamespace } from \"../model-utils.js\";\nimport type { SpawnOptions, SpawnResult } from \"../types.js\";\n\nfunction createAbortError(): Error {\n const error = new Error(\"Agent spawn aborted\");\n error.name = \"AbortError\";\n return error;\n}\n\nfunction createActivityTimeoutError(timeoutMs: number): Error {\n const error = new Error(\n `Agent spawn timed out after ${timeoutMs / 1000}s of inactivity`\n );\n error.name = \"ActivityTimeoutError\";\n return error;\n}\n\nexport interface SpawnStreamingOptions extends SpawnOptions {\n agentId: string;\n}\n\nexport interface SpawnStreamingResult {\n events: AsyncIterable<AcpEvent>;\n done: Promise<SpawnResult>;\n}\n\nfunction isAcpEvent(value: unknown): value is AcpEvent {\n return !!value && typeof value === \"object\" && \"event\" in value;\n}\n\nexport function spawnStreaming(options: SpawnStreamingOptions): SpawnStreamingResult {\n if (options.signal?.aborted) {\n throw createAbortError();\n }\n\n const { agentId, binaryName, spawnConfig } = resolveConfig(options.agentId);\n\n if (spawnConfig === undefined) {\n throw new Error(`Agent \"${agentId}\" has no spawn config.`);\n }\n\n if (spawnConfig.kind !== \"cli\") {\n throw new Error(`Agent \"${agentId}\" does not support CLI spawn.`);\n }\n\n if (!binaryName) {\n throw new Error(`Agent \"${agentId}\" has no binaryName.`);\n }\n\n const mcpArgs = getMcpArgs(spawnConfig, options.mcpServers);\n const mcpEnvVars = getMcpEnv(spawnConfig, options.mcpServers);\n const args: string[] = [];\n\n if (spawnConfig.mcpArgsBeforeCommand) {\n args.push(...mcpArgs);\n }\n\n args.push(spawnConfig.promptFlag);\n\n const useStdin = !!options.useStdin && !!spawnConfig.stdinMode;\n if (!useStdin || !spawnConfig.stdinMode?.omitPrompt) {\n args.push(options.prompt);\n }\n\n if (options.model && spawnConfig.modelFlag) {\n let model = spawnConfig.modelStripProviderPrefix\n ? stripModelNamespace(options.model)\n : options.model;\n if (spawnConfig.modelTransform) model = spawnConfig.modelTransform(model);\n args.push(spawnConfig.modelFlag, model);\n }\n\n args.push(...spawnConfig.defaultArgs);\n\n if (!spawnConfig.mcpArgsBeforeCommand) {\n args.push(...mcpArgs);\n }\n\n const mode = options.mode ?? \"yolo\";\n args.push(...spawnConfig.modes[mode]);\n\n if (useStdin) {\n args.push(...spawnConfig.stdinMode!.extraArgs);\n }\n\n if (options.args && options.args.length > 0) {\n args.push(...options.args);\n }\n\n const child = spawnChildProcess(binaryName, args, {\n cwd: options.cwd,\n stdio: [\"pipe\", \"pipe\", \"pipe\"],\n env: Object.keys(mcpEnvVars).length > 0\n ? { ...process.env, ...mcpEnvVars }\n : undefined\n });\n let aborted = false;\n let timedOut = false;\n const onAbort = () => {\n aborted = true;\n child.kill(\"SIGTERM\");\n };\n options.signal?.addEventListener(\"abort\", onAbort, { once: true });\n\n let activityTimer: ReturnType<typeof setTimeout> | undefined;\n const resetActivityTimer = options.activityTimeoutMs\n ? () => {\n if (activityTimer) clearTimeout(activityTimer);\n activityTimer = setTimeout(() => {\n timedOut = true;\n child.kill(\"SIGTERM\");\n }, options.activityTimeoutMs);\n }\n : undefined;\n\n resetActivityTimer?.();\n\n const result: SpawnResult = { stdout: \"\", stderr: \"\", exitCode: 1 };\n child.stderr.setEncoding(\"utf8\");\n child.stderr.on(\"data\", (chunk) => {\n result.stderr += chunk;\n resetActivityTimer?.();\n });\n\n if (useStdin) {\n child.stdin.write(options.prompt);\n }\n child.stdin.end();\n\n const adapter = getAdapter(spawnConfig.adapter);\n\n const events: AsyncIterable<AcpEvent> = (async function* () {\n for await (const output of adapter(readLines(child.stdout))) {\n if (!isAcpEvent(output)) continue;\n resetActivityTimer?.();\n yield output;\n }\n })();\n\n const done = new Promise<SpawnResult>((resolve, reject) => {\n child.on(\"error\", (error) => {\n options.signal?.removeEventListener(\"abort\", onAbort);\n if (activityTimer) clearTimeout(activityTimer);\n if (aborted) {\n reject(createAbortError());\n return;\n }\n reject(error);\n });\n\n child.on(\"close\", (code) => {\n options.signal?.removeEventListener(\"abort\", onAbort);\n if (activityTimer) clearTimeout(activityTimer);\n if (aborted) {\n reject(createAbortError());\n return;\n }\n if (timedOut) {\n reject(createActivityTimeoutError(options.activityTimeoutMs!));\n return;\n }\n result.exitCode = code ?? 1;\n resolve(result);\n });\n });\n\n return { events, done };\n}\n", "import { isAbsolute } from \"node:path\";\nimport {\n AcpTransport,\n type AcpTransportClosedEvent,\n type AcpTransportOptions,\n} from \"./acp-transport.js\";\nimport type { JsonRpcRequestOptions } from \"./jsonrpc-message-layer.js\";\nimport {\n ACP_ERROR_CODE_INVALID_PARAMS,\n ACP_ERROR_CODE_RESOURCE_NOT_FOUND,\n AcpError,\n type AgentCapabilities,\n type AuthenticateResponse,\n type AuthMethod,\n type CancelNotification,\n type ClientCapabilities,\n type ContentBlock,\n type CreateTerminalRequest,\n type CreateTerminalResponse,\n type EnvVariable,\n type Implementation,\n type InitializeResponse,\n type KillTerminalCommandRequest,\n type KillTerminalCommandResponse,\n type LoadSessionResponse,\n type McpServer,\n type NewSessionResponse,\n type PermissionOption,\n type PromptResponse,\n type RequestId,\n type SessionConfigId,\n type SessionConfigOption,\n type SessionConfigValueId,\n type RequestPermissionOutcome,\n type RequestPermissionRequest,\n type RequestPermissionResponse,\n type ReadTextFileRequest,\n type ReadTextFileResponse,\n type ProtocolVersion,\n type SessionModeId,\n type SessionNotification,\n type SessionId,\n type SetSessionModeResponse,\n type SessionUpdateNotification,\n type ToolCallUpdate,\n type TerminalOutputRequest,\n type TerminalOutputResponse,\n type WaitForTerminalExitRequest,\n type WaitForTerminalExitResponse,\n type WriteTextFileRequest,\n type WriteTextFileResponse,\n type ReleaseTerminalRequest,\n type ReleaseTerminalResponse,\n} from \"./types.js\";\n\nexport type AcpClientState = \"uninitialized\" | \"initialized\" | \"ready\";\ntype ExtensionMethod = `_${string}`;\n\nexport interface PromptTurn extends AsyncIterable<SessionUpdateNotification> {\n response: Promise<PromptResponse>;\n}\n\ninterface AsyncQueue<T> extends AsyncIterable<T>, AsyncIterator<T> {\n push(value: T): void;\n complete(): void;\n fail(error: Error): void;\n}\n\nexport interface AcpClientFsHandler {\n readTextFile?: (args: {\n sessionId: SessionId;\n path: string;\n line?: number | null;\n limit?: number | null;\n }) => string | Promise<string>;\n writeTextFile?: (args: {\n sessionId: SessionId;\n path: string;\n content: string;\n }) => void | Promise<void>;\n}\n\nexport interface AcpClientTerminalHandler {\n create: (args: {\n sessionId: SessionId;\n command: string;\n args?: string[];\n cwd?: string | null;\n env?: EnvVariable[];\n outputByteLimit?: number | null;\n }) => string | Promise<string>;\n output: (args: {\n sessionId: SessionId;\n terminalId: string;\n }) => TerminalOutputResponse | Promise<TerminalOutputResponse>;\n waitForExit: (args: {\n sessionId: SessionId;\n terminalId: string;\n }) => WaitForTerminalExitResponse | Promise<WaitForTerminalExitResponse>;\n kill: (args: { sessionId: SessionId; terminalId: string }) => void | Promise<void>;\n release: (args: {\n sessionId: SessionId;\n terminalId: string;\n }) => void | Promise<void>;\n}\n\ntype AcpClientPermissionHandler = (args: {\n toolCall: ToolCallUpdate;\n options: PermissionOption[];\n}) => RequestPermissionOutcome | Promise<RequestPermissionOutcome>;\n\nexport interface AcpClientHandlers {\n permission?: AcpClientPermissionHandler;\n fs?: AcpClientFsHandler;\n terminal?: AcpClientTerminalHandler;\n}\n\ntype AcpClientTransport = Pick<\n AcpTransport,\n \"sendRequest\" | \"sendNotification\" | \"onRequest\" | \"onNotification\"\n> &\n Partial<Pick<AcpTransport, \"dispose\" | \"closed\">>;\n\ninterface AcpClientSharedOptions {\n protocolVersion?: ProtocolVersion;\n clientCapabilities?: ClientCapabilities;\n clientInfo?: Implementation | null;\n handlers?: AcpClientHandlers;\n permissionHandler?: AcpClientPermissionHandler;\n fsHandler?: AcpClientFsHandler;\n terminalHandler?: AcpClientTerminalHandler;\n skipAuth?: boolean;\n}\n\nexport interface AcpClientProcessOptions extends AcpClientSharedOptions {\n command: string;\n args?: readonly string[];\n cwd?: string;\n env?: NodeJS.ProcessEnv;\n requestTimeoutMs?: number;\n firstRequestId?: number;\n spawn?: AcpTransportOptions[\"spawn\"];\n}\n\nexport interface AcpClientInjectedTransportOptions extends AcpClientSharedOptions {\n transport: AcpClientTransport;\n}\n\nexport type AcpClientOptions = AcpClientProcessOptions | AcpClientInjectedTransportOptions;\n\nfunction toError(reason: unknown): Error {\n return reason instanceof Error ? reason : new Error(String(reason));\n}\n\nfunction invalidParams(message: string): AcpError {\n return new AcpError(\n ACP_ERROR_CODE_INVALID_PARAMS,\n `Invalid params: ${message}`\n );\n}\n\nfunction resourceNotFound(resource: string): AcpError {\n return new AcpError(\n ACP_ERROR_CODE_RESOURCE_NOT_FOUND,\n `Resource not found: ${resource}`\n );\n}\n\nfunction assertAbsolutePath(path: string): void {\n if (!isAbsolute(path)) {\n throw invalidParams('\"path\" must be an absolute path');\n }\n}\n\nfunction assertOneBasedLineNumber(line: number | null | undefined): void {\n if (line === null || line === undefined) {\n return;\n }\n\n if (!Number.isInteger(line) || line < 1) {\n throw invalidParams('\"line\" must be a 1-based integer');\n }\n}\n\nfunction assertExtensionMethod(method: string): asserts method is ExtensionMethod {\n if (!method.startsWith(\"_\")) {\n throw new Error('Extension method must start with \"_\"');\n }\n}\n\nfunction isInjectedTransportOptions(\n options: AcpClientOptions\n): options is AcpClientInjectedTransportOptions {\n return \"transport\" in options;\n}\n\nfunction createAsyncQueue<T>(): AsyncQueue<T> {\n const values: T[] = [];\n const waiters: Array<{\n resolve: (result: IteratorResult<T>) => void;\n reject: (error: Error) => void;\n }> = [];\n let closed = false;\n let failure: Error | null = null;\n\n const resolveOne = (value: T): boolean => {\n const waiter = waiters.shift();\n if (!waiter) {\n return false;\n }\n\n waiter.resolve({ done: false, value });\n return true;\n };\n\n const iterator: AsyncQueue<T> = {\n push(value: T): void {\n if (closed || failure) {\n return;\n }\n\n if (!resolveOne(value)) {\n values.push(value);\n }\n },\n complete(): void {\n if (closed || failure) {\n return;\n }\n\n closed = true;\n while (waiters.length > 0) {\n waiters.shift()?.resolve({ done: true, value: undefined });\n }\n },\n fail(error: Error): void {\n if (closed || failure) {\n return;\n }\n\n failure = error;\n while (waiters.length > 0) {\n waiters.shift()?.reject(error);\n }\n },\n async next(): Promise<IteratorResult<T>> {\n if (values.length > 0) {\n const value = values.shift() as T;\n return { done: false, value };\n }\n\n if (failure) {\n throw failure;\n }\n\n if (closed) {\n return { done: true, value: undefined };\n }\n\n return new Promise<IteratorResult<T>>((resolve, reject) => {\n waiters.push({ resolve, reject });\n });\n },\n async return(): Promise<IteratorResult<T>> {\n iterator.complete();\n return { done: true, value: undefined };\n },\n async throw(error: unknown): Promise<IteratorResult<T>> {\n const normalizedError = toError(error);\n iterator.fail(normalizedError);\n throw normalizedError;\n },\n [Symbol.asyncIterator](): AsyncIterator<T> {\n return iterator;\n },\n };\n\n return iterator;\n}\n\nexport class AcpClient {\n private readonly transport: AcpClientTransport;\n private readonly clientProtocolVersion: ProtocolVersion;\n private clientCapabilities?: ClientCapabilities;\n private readonly clientInfo?: Implementation | null;\n private readonly skipAuth: boolean;\n private readonly permissionHandler?: AcpClientPermissionHandler;\n private readonly fsHandler?: AcpClientFsHandler;\n private readonly terminalHandler?: AcpClientTerminalHandler;\n private readonly activePromptUpdates = new Map<\n SessionId,\n AsyncQueue<SessionUpdateNotification>\n >();\n private readonly trackedTerminalIds = new Map<SessionId, Set<string>>();\n private hasRegisteredFsReadHandler = false;\n private hasRegisteredFsWriteHandler = false;\n private hasRegisteredTerminalHandlers = false;\n private disposed = false;\n\n private lifecycleState: AcpClientState = \"uninitialized\";\n private negotiatedVersion: ProtocolVersion | null = null;\n private availableAuthMethods: AuthMethod[] = [];\n private negotiatedAgentCapabilities: AgentCapabilities | undefined;\n private negotiatedAgentInfo: Implementation | null | undefined;\n\n constructor(options: AcpClientOptions) {\n this.transport = isInjectedTransportOptions(options)\n ? options.transport\n : new AcpTransport({\n command: options.command,\n args: options.args,\n cwd: options.cwd,\n env: options.env,\n requestTimeoutMs: options.requestTimeoutMs,\n firstRequestId: options.firstRequestId,\n spawn: options.spawn,\n });\n this.clientProtocolVersion = options.protocolVersion ?? 1;\n this.clientCapabilities = options.clientCapabilities;\n this.clientInfo = options.clientInfo;\n this.skipAuth = options.skipAuth ?? false;\n this.permissionHandler = options.handlers?.permission ?? options.permissionHandler;\n this.fsHandler = options.handlers?.fs ?? options.fsHandler;\n this.terminalHandler = options.handlers?.terminal ?? options.terminalHandler;\n\n this.transport.onRequest(\n \"session/request_permission\",\n async (params: RequestPermissionRequest): Promise<RequestPermissionResponse> => {\n if (!this.permissionHandler) {\n return {\n outcome: { outcome: \"cancelled\" },\n };\n }\n\n const outcome = await this.permissionHandler({\n toolCall: params.toolCall,\n options: params.options,\n });\n return { outcome };\n }\n );\n\n this.registerCapabilityHandlers(this.clientCapabilities);\n\n this.transport.onNotification(\"session/update\", (params: SessionNotification) => {\n this.handleSessionUpdateNotification(params);\n });\n }\n\n get state(): AcpClientState {\n return this.lifecycleState;\n }\n\n get negotiatedProtocolVersion(): ProtocolVersion | null {\n return this.negotiatedVersion;\n }\n\n get authMethods(): AuthMethod[] {\n return [...this.availableAuthMethods];\n }\n\n get agentCapabilities(): AgentCapabilities | undefined {\n return this.negotiatedAgentCapabilities;\n }\n\n get agentInfo(): Implementation | null | undefined {\n return this.negotiatedAgentInfo;\n }\n\n get closed(): Promise<AcpTransportClosedEvent> | undefined {\n return this.transport.closed;\n }\n\n async initialize(clientCapabilities?: ClientCapabilities): Promise<InitializeResponse> {\n if (this.lifecycleState !== \"uninitialized\") {\n throw new Error(\"initialize() can only be called once.\");\n }\n\n if (clientCapabilities !== undefined) {\n this.clientCapabilities = clientCapabilities;\n this.registerCapabilityHandlers(clientCapabilities);\n }\n\n const response = await this.transport.sendRequest(\"initialize\", {\n protocolVersion: this.clientProtocolVersion,\n clientInfo: this.clientInfo,\n clientCapabilities: this.clientCapabilities,\n });\n\n const negotiatedProtocolVersion = Math.min(\n this.clientProtocolVersion,\n response.protocolVersion\n );\n\n this.negotiatedVersion = negotiatedProtocolVersion;\n this.negotiatedAgentCapabilities = response.agentCapabilities;\n this.negotiatedAgentInfo = response.agentInfo;\n this.availableAuthMethods = response.authMethods ? [...response.authMethods] : [];\n\n const requiresAuth = this.availableAuthMethods.length > 0 && !this.skipAuth;\n this.lifecycleState = requiresAuth ? \"initialized\" : \"ready\";\n\n return {\n protocolVersion: negotiatedProtocolVersion,\n ...(this.negotiatedAgentCapabilities !== undefined\n ? { agentCapabilities: this.negotiatedAgentCapabilities }\n : {}),\n ...(this.negotiatedAgentInfo !== undefined ? { agentInfo: this.negotiatedAgentInfo } : {}),\n ...(this.availableAuthMethods.length > 0 ? { authMethods: this.authMethods } : {}),\n };\n }\n\n async authenticate(methodId: string): Promise<AuthenticateResponse> {\n if (this.lifecycleState === \"uninitialized\") {\n throw new Error(\"Cannot authenticate before initialize().\");\n }\n\n if (this.lifecycleState === \"ready\") {\n throw new Error(\"Authentication is not required for this agent.\");\n }\n\n if (!this.availableAuthMethods.some((authMethod) => authMethod.id === methodId)) {\n throw new Error(`Unknown auth method \"${methodId}\".`);\n }\n\n const response = await this.transport.sendRequest(\"authenticate\", {\n methodId,\n });\n\n this.lifecycleState = \"ready\";\n return response;\n }\n\n async newSession(cwd: string, mcpServers: McpServer[]): Promise<NewSessionResponse> {\n this.assertReady(\"session/new\");\n this.assertMcpServerCapabilitySupport(mcpServers);\n\n return this.transport.sendRequest(\"session/new\", {\n cwd,\n mcpServers,\n });\n }\n\n async loadSession(\n sessionId: SessionId,\n cwd: string,\n mcpServers: McpServer[]\n ): Promise<LoadSessionResponse> {\n this.assertReady(\"session/load\");\n if (this.negotiatedAgentCapabilities?.loadSession !== true) {\n throw new Error(\n 'Cannot call \"session/load\" because the agent does not support session loading.'\n );\n }\n this.assertMcpServerCapabilitySupport(mcpServers);\n\n return this.transport.sendRequest(\"session/load\", {\n sessionId,\n cwd,\n mcpServers,\n });\n }\n\n async cancelSession(sessionId: SessionId): Promise<void> {\n this.assertReady(\"session/cancel\");\n const payload: CancelNotification = { sessionId };\n this.transport.sendNotification(\"session/cancel\", payload);\n }\n\n async setMode(\n sessionId: SessionId,\n modeId: SessionModeId\n ): Promise<SetSessionModeResponse> {\n this.assertReady(\"session/set_mode\");\n return this.transport.sendRequest(\"session/set_mode\", {\n sessionId,\n modeId,\n });\n }\n\n async setConfigOption(\n sessionId: SessionId,\n configId: SessionConfigId,\n value: SessionConfigValueId\n ): Promise<SessionConfigOption[]> {\n this.assertReady(\"session/set_config_option\");\n const response = await this.transport.sendRequest(\"session/set_config_option\", {\n sessionId,\n configId,\n value,\n });\n\n return response.configOptions;\n }\n\n prompt(sessionId: SessionId, content: ContentBlock[]): PromptTurn {\n this.assertReady(\"session/prompt\");\n this.assertPromptContentCapabilitySupport(content);\n\n if (this.activePromptUpdates.has(sessionId)) {\n throw new Error(\n `Cannot call \"session/prompt\" while another prompt is in progress for session \"${sessionId}\".`\n );\n }\n\n const updates = createAsyncQueue<SessionUpdateNotification>();\n this.activePromptUpdates.set(sessionId, updates);\n\n let requestPromise: Promise<PromptResponse>;\n try {\n requestPromise = this.transport.sendRequest(\"session/prompt\", {\n sessionId,\n prompt: content,\n });\n } catch (error) {\n const normalizedError = toError(error);\n this.activePromptUpdates.delete(sessionId);\n updates.fail(normalizedError);\n throw normalizedError;\n }\n\n const response = requestPromise\n .then((promptResponse) => {\n this.activePromptUpdates.delete(sessionId);\n updates.complete();\n return promptResponse;\n })\n .catch((error) => {\n const normalizedError = toError(error);\n this.activePromptUpdates.delete(sessionId);\n updates.fail(normalizedError);\n throw normalizedError;\n });\n\n return {\n response,\n [Symbol.asyncIterator](): AsyncIterator<SessionUpdateNotification> {\n return updates;\n },\n };\n }\n\n async sendExtRequest<TResult = unknown>(\n method: ExtensionMethod,\n params?: unknown,\n options?: JsonRpcRequestOptions\n ): Promise<TResult>;\n async sendExtRequest<TResult = unknown>(\n method: string,\n params?: unknown,\n options: JsonRpcRequestOptions = {}\n ): Promise<TResult> {\n assertExtensionMethod(method);\n return this.transport.sendRequest(method, params, options) as Promise<TResult>;\n }\n\n async sendExtNotification(method: ExtensionMethod, params?: unknown): Promise<void>;\n async sendExtNotification(method: string, params?: unknown): Promise<void> {\n assertExtensionMethod(method);\n this.transport.sendNotification(method, params);\n }\n\n onExtRequest<TMethod extends ExtensionMethod>(\n method: TMethod,\n handler: (\n params: unknown,\n context: { id: RequestId; method: TMethod }\n ) => unknown | Promise<unknown>\n ): void;\n onExtRequest(\n method: string,\n handler: (params: unknown, context: { id: RequestId; method: string }) => unknown\n ): void {\n assertExtensionMethod(method);\n this.transport.onRequest(method, handler);\n }\n\n onExtNotification<TMethod extends ExtensionMethod>(\n method: TMethod,\n handler: (params: unknown, context: { method: TMethod }) => void | Promise<void>\n ): void;\n onExtNotification(\n method: string,\n handler: (params: unknown, context: { method: string }) => void | Promise<void>\n ): void {\n assertExtensionMethod(method);\n this.transport.onNotification(method, handler);\n }\n\n async dispose(): Promise<void> {\n if (this.disposed) {\n if (this.transport.closed) {\n await this.transport.closed;\n }\n return;\n }\n\n this.disposed = true;\n const disposeReason = new Error(\"ACP client disposed\");\n for (const updates of this.activePromptUpdates.values()) {\n updates.fail(disposeReason);\n }\n this.activePromptUpdates.clear();\n\n if (typeof this.transport.dispose === \"function\") {\n this.transport.dispose(disposeReason);\n }\n\n if (this.transport.closed) {\n await this.transport.closed;\n }\n }\n\n assertReady(operation: string): void {\n if (this.lifecycleState === \"ready\") {\n return;\n }\n\n if (this.lifecycleState === \"uninitialized\") {\n throw new Error(`Cannot call \"${operation}\" before initialize().`);\n }\n\n throw new Error(`Cannot call \"${operation}\" before authentication completes.`);\n }\n\n private registerCapabilityHandlers(capabilities: ClientCapabilities | undefined): void {\n if (\n !this.hasRegisteredFsReadHandler &&\n capabilities?.fs?.readTextFile === true &&\n this.fsHandler?.readTextFile\n ) {\n const readTextFile = this.fsHandler.readTextFile;\n this.transport.onRequest(\n \"fs/read_text_file\",\n async (params: ReadTextFileRequest): Promise<ReadTextFileResponse> => {\n assertAbsolutePath(params.path);\n assertOneBasedLineNumber(params.line);\n\n const content = await readTextFile({\n sessionId: params.sessionId,\n path: params.path,\n line: params.line,\n limit: params.limit,\n });\n return { content };\n }\n );\n this.hasRegisteredFsReadHandler = true;\n }\n\n if (\n !this.hasRegisteredFsWriteHandler &&\n capabilities?.fs?.writeTextFile === true &&\n this.fsHandler?.writeTextFile\n ) {\n const writeTextFile = this.fsHandler.writeTextFile;\n this.transport.onRequest(\n \"fs/write_text_file\",\n async (params: WriteTextFileRequest): Promise<WriteTextFileResponse> => {\n assertAbsolutePath(params.path);\n\n await writeTextFile({\n sessionId: params.sessionId,\n path: params.path,\n content: params.content,\n });\n\n return {};\n }\n );\n this.hasRegisteredFsWriteHandler = true;\n }\n\n if (\n !this.hasRegisteredTerminalHandlers &&\n capabilities?.terminal === true &&\n this.terminalHandler\n ) {\n const terminalHandler = this.terminalHandler;\n this.transport.onRequest(\n \"terminal/create\",\n async (params: CreateTerminalRequest): Promise<CreateTerminalResponse> => {\n const terminalId = await terminalHandler.create({\n sessionId: params.sessionId,\n command: params.command,\n args: params.args,\n cwd: params.cwd,\n env: params.env,\n outputByteLimit: params.outputByteLimit,\n });\n this.trackTerminal(params.sessionId, terminalId);\n\n return { terminalId };\n }\n );\n\n this.transport.onRequest(\n \"terminal/output\",\n async (params: TerminalOutputRequest): Promise<TerminalOutputResponse> => {\n this.assertKnownTerminal(params.sessionId, params.terminalId);\n\n return terminalHandler.output({\n sessionId: params.sessionId,\n terminalId: params.terminalId,\n });\n }\n );\n\n this.transport.onRequest(\n \"terminal/wait_for_exit\",\n async (params: WaitForTerminalExitRequest): Promise<WaitForTerminalExitResponse> => {\n this.assertKnownTerminal(params.sessionId, params.terminalId);\n\n return terminalHandler.waitForExit({\n sessionId: params.sessionId,\n terminalId: params.terminalId,\n });\n }\n );\n\n this.transport.onRequest(\n \"terminal/kill\",\n async (params: KillTerminalCommandRequest): Promise<KillTerminalCommandResponse> => {\n this.assertKnownTerminal(params.sessionId, params.terminalId);\n\n await terminalHandler.kill({\n sessionId: params.sessionId,\n terminalId: params.terminalId,\n });\n\n return {};\n }\n );\n\n this.transport.onRequest(\n \"terminal/release\",\n async (params: ReleaseTerminalRequest): Promise<ReleaseTerminalResponse> => {\n this.assertKnownTerminal(params.sessionId, params.terminalId);\n\n await terminalHandler.release({\n sessionId: params.sessionId,\n terminalId: params.terminalId,\n });\n this.untrackTerminal(params.sessionId, params.terminalId);\n\n return {};\n }\n );\n this.hasRegisteredTerminalHandlers = true;\n }\n }\n\n private assertMcpServerCapabilitySupport(mcpServers: McpServer[]): void {\n const mcpCapabilities = this.negotiatedAgentCapabilities?.mcpCapabilities;\n\n for (const mcpServer of mcpServers) {\n if (!(\"type\" in mcpServer)) {\n continue;\n }\n\n if (mcpServer.type === \"http\" && mcpCapabilities?.http !== true) {\n throw new Error('Agent does not support MCP server type \"http\".');\n }\n\n if (mcpServer.type === \"sse\" && mcpCapabilities?.sse !== true) {\n throw new Error('Agent does not support MCP server type \"sse\".');\n }\n }\n }\n\n private handleSessionUpdateNotification(notification: SessionNotification): void {\n const activePrompt = this.activePromptUpdates.get(notification.sessionId);\n if (!activePrompt) {\n return;\n }\n\n activePrompt.push({\n jsonrpc: \"2.0\",\n method: \"session/update\",\n params: notification,\n });\n }\n\n private trackTerminal(sessionId: SessionId, terminalId: string): void {\n const sessionTerminals = this.trackedTerminalIds.get(sessionId);\n if (sessionTerminals) {\n sessionTerminals.add(terminalId);\n return;\n }\n\n this.trackedTerminalIds.set(sessionId, new Set([terminalId]));\n }\n\n private assertKnownTerminal(sessionId: SessionId, terminalId: string): void {\n const sessionTerminals = this.trackedTerminalIds.get(sessionId);\n if (sessionTerminals?.has(terminalId) === true) {\n return;\n }\n\n throw resourceNotFound(`terminal \"${terminalId}\"`);\n }\n\n private untrackTerminal(sessionId: SessionId, terminalId: string): void {\n const sessionTerminals = this.trackedTerminalIds.get(sessionId);\n if (!sessionTerminals) {\n return;\n }\n\n sessionTerminals.delete(terminalId);\n if (sessionTerminals.size === 0) {\n this.trackedTerminalIds.delete(sessionId);\n }\n }\n\n private assertPromptContentCapabilitySupport(content: ContentBlock[]): void {\n const promptCapabilities = this.negotiatedAgentCapabilities?.promptCapabilities;\n\n for (const block of content) {\n if (block.type === \"image\" && promptCapabilities?.image !== true) {\n throw new Error('Agent does not support prompt content type \"image\".');\n }\n\n if (block.type === \"audio\" && promptCapabilities?.audio !== true) {\n throw new Error('Agent does not support prompt content type \"audio\".');\n }\n\n if (block.type === \"resource\" && promptCapabilities?.embeddedContext !== true) {\n throw new Error('Agent does not support prompt content type \"resource\".');\n }\n }\n }\n}\n", "import {\n spawn as spawnChildProcess,\n type ChildProcessWithoutNullStreams,\n type SpawnOptionsWithoutStdio,\n} from \"node:child_process\";\nimport {\n JsonRpcMessageLayer,\n type JsonRpcNotificationHandler,\n type JsonRpcRequestHandler,\n type JsonRpcRequestOptions,\n} from \"./jsonrpc-message-layer.js\";\nimport type {\n AuthenticateRequest,\n AuthenticateResponse,\n CancelNotification,\n CreateTerminalRequest,\n CreateTerminalResponse,\n InitializeRequest,\n InitializeResponse,\n KillTerminalCommandRequest,\n KillTerminalCommandResponse,\n LoadSessionRequest,\n LoadSessionResponse,\n NewSessionRequest,\n NewSessionResponse,\n PromptRequest,\n PromptResponse,\n ReadTextFileRequest,\n ReadTextFileResponse,\n ReleaseTerminalRequest,\n ReleaseTerminalResponse,\n RequestId,\n RequestPermissionRequest,\n RequestPermissionResponse,\n SessionNotification,\n SetSessionConfigOptionRequest,\n SetSessionConfigOptionResponse,\n SetSessionModeRequest,\n SetSessionModeResponse,\n TerminalOutputRequest,\n TerminalOutputResponse,\n WaitForTerminalExitRequest,\n WaitForTerminalExitResponse,\n WriteTextFileRequest,\n WriteTextFileResponse,\n} from \"./types.js\";\n\ninterface AcpRequestShape<TParams, TResult> {\n params: TParams;\n result: TResult;\n}\n\nexport interface AcpAgentRequestMap {\n initialize: AcpRequestShape<InitializeRequest, InitializeResponse>;\n authenticate: AcpRequestShape<AuthenticateRequest, AuthenticateResponse>;\n \"session/new\": AcpRequestShape<NewSessionRequest, NewSessionResponse>;\n \"session/load\": AcpRequestShape<LoadSessionRequest, LoadSessionResponse>;\n \"session/prompt\": AcpRequestShape<PromptRequest, PromptResponse>;\n \"session/set_mode\": AcpRequestShape<SetSessionModeRequest, SetSessionModeResponse>;\n \"session/set_config_option\": AcpRequestShape<\n SetSessionConfigOptionRequest,\n SetSessionConfigOptionResponse\n >;\n}\n\nexport interface AcpAgentNotificationMap {\n \"session/cancel\": CancelNotification;\n}\n\nexport interface AcpClientRequestMap {\n \"session/request_permission\": AcpRequestShape<\n RequestPermissionRequest,\n RequestPermissionResponse\n >;\n \"fs/read_text_file\": AcpRequestShape<ReadTextFileRequest, ReadTextFileResponse>;\n \"fs/write_text_file\": AcpRequestShape<WriteTextFileRequest, WriteTextFileResponse>;\n \"terminal/create\": AcpRequestShape<CreateTerminalRequest, CreateTerminalResponse>;\n \"terminal/output\": AcpRequestShape<TerminalOutputRequest, TerminalOutputResponse>;\n \"terminal/wait_for_exit\": AcpRequestShape<\n WaitForTerminalExitRequest,\n WaitForTerminalExitResponse\n >;\n \"terminal/kill\": AcpRequestShape<\n KillTerminalCommandRequest,\n KillTerminalCommandResponse\n >;\n \"terminal/release\": AcpRequestShape<ReleaseTerminalRequest, ReleaseTerminalResponse>;\n}\n\nexport interface AcpClientNotificationMap {\n \"session/update\": SessionNotification;\n}\n\ntype ExtensionMethod = `_${string}`;\n\nfunction assertExtensionMethod(method: string): asserts method is ExtensionMethod {\n if (!method.startsWith(\"_\")) {\n throw new Error('Extension method must start with \"_\"');\n }\n}\n\ntype SpawnFunction = (\n command: string,\n args?: ReadonlyArray<string>,\n options?: SpawnOptionsWithoutStdio\n) => ChildProcessWithoutNullStreams;\n\nexport interface AcpTransportOptions {\n command: string;\n args?: readonly string[];\n cwd?: string;\n env?: NodeJS.ProcessEnv;\n requestTimeoutMs?: number;\n firstRequestId?: number;\n spawn?: SpawnFunction;\n}\n\nexport interface AcpTransportClosedEvent {\n code: number | null;\n signal: NodeJS.Signals | null;\n reason: Error;\n stderr: string;\n}\n\nexport class AcpTransport {\n readonly closed: Promise<AcpTransportClosedEvent>;\n\n private readonly command: string;\n private readonly child: ChildProcessWithoutNullStreams;\n private readonly layer: JsonRpcMessageLayer;\n private readonly stderrChunks: string[] = [];\n private resolveClosed: ((value: AcpTransportClosedEvent) => void) | null = null;\n private closeEvent: AcpTransportClosedEvent | null = null;\n private closeReason: Error | null = null;\n\n constructor(options: AcpTransportOptions) {\n const {\n command,\n args = [],\n cwd,\n env,\n requestTimeoutMs,\n firstRequestId,\n spawn = spawnChildProcess,\n } = options;\n\n this.command = command;\n this.closed = new Promise<AcpTransportClosedEvent>((resolve) => {\n this.resolveClosed = resolve;\n });\n\n this.child = spawn(command, [...args], {\n cwd,\n env,\n stdio: [\"pipe\", \"pipe\", \"pipe\"],\n });\n\n this.child.stderr.setEncoding(\"utf8\");\n this.child.stderr.on(\"data\", (chunk) => {\n this.stderrChunks.push(String(chunk));\n });\n\n this.layer = new JsonRpcMessageLayer({\n input: this.child.stdout,\n output: this.child.stdin,\n requestTimeoutMs,\n firstRequestId,\n });\n\n this.child.once(\"error\", (error) => {\n const reason = error instanceof Error ? error : new Error(String(error));\n this.close(reason, this.child.exitCode ?? null, this.child.signalCode ?? null);\n });\n\n this.child.once(\"close\", (code, signal) => {\n const reason =\n this.closeReason ??\n new Error(\n `ACP transport closed (command \"${this.command}\", code: ${code ?? \"null\"}${\n signal ? `, signal: ${signal}` : \"\"\n })`\n );\n this.close(reason, code ?? null, signal ?? null);\n });\n }\n\n sendRequest<TMethod extends keyof AcpAgentRequestMap>(\n method: TMethod,\n params: AcpAgentRequestMap[TMethod][\"params\"],\n options?: JsonRpcRequestOptions\n ): Promise<AcpAgentRequestMap[TMethod][\"result\"]>;\n sendRequest<TResult = unknown>(\n method: string,\n params?: unknown,\n options?: JsonRpcRequestOptions\n ): Promise<TResult>;\n sendRequest(\n method: string,\n params?: unknown,\n options: JsonRpcRequestOptions = {}\n ): Promise<unknown> {\n return this.layer.sendRequest(method, params, options);\n }\n\n sendExtRequest<TResult = unknown>(\n method: ExtensionMethod,\n params?: unknown,\n options?: JsonRpcRequestOptions\n ): Promise<TResult>;\n sendExtRequest<TResult = unknown>(\n method: string,\n params?: unknown,\n options: JsonRpcRequestOptions = {}\n ): Promise<TResult> {\n assertExtensionMethod(method);\n return this.layer.sendRequest(method, params, options) as Promise<TResult>;\n }\n\n sendNotification<TMethod extends keyof AcpAgentNotificationMap>(\n method: TMethod,\n params: AcpAgentNotificationMap[TMethod]\n ): void;\n sendNotification(method: string, params?: unknown): void;\n sendNotification(method: string, params?: unknown): void {\n this.layer.sendNotification(method, params);\n }\n\n sendExtNotification(method: ExtensionMethod, params?: unknown): void;\n sendExtNotification(method: string, params?: unknown): void;\n sendExtNotification(method: string, params?: unknown): void {\n assertExtensionMethod(method);\n this.layer.sendNotification(method, params);\n }\n\n onRequest<TMethod extends keyof AcpClientRequestMap>(\n method: TMethod,\n handler: (\n params: AcpClientRequestMap[TMethod][\"params\"],\n context: { id: RequestId; method: TMethod }\n ) =>\n | AcpClientRequestMap[TMethod][\"result\"]\n | Promise<AcpClientRequestMap[TMethod][\"result\"]>\n ): void;\n onRequest(method: string, handler: JsonRpcRequestHandler): void;\n onRequest(method: string, handler: JsonRpcRequestHandler): void {\n this.layer.onRequest(method, handler);\n }\n\n onExtRequest<TMethod extends ExtensionMethod>(\n method: TMethod,\n handler: (\n params: unknown,\n context: { id: RequestId; method: TMethod }\n ) => unknown | Promise<unknown>\n ): void;\n onExtRequest(method: string, handler: JsonRpcRequestHandler): void;\n onExtRequest(method: string, handler: JsonRpcRequestHandler): void {\n assertExtensionMethod(method);\n this.layer.onRequest(method, handler);\n }\n\n onNotification<TMethod extends keyof AcpClientNotificationMap>(\n method: TMethod,\n handler: (\n params: AcpClientNotificationMap[TMethod],\n context: { method: TMethod }\n ) => void | Promise<void>\n ): void;\n onNotification(method: string, handler: JsonRpcNotificationHandler): void;\n onNotification(method: string, handler: JsonRpcNotificationHandler): void {\n this.layer.onNotification(method, handler);\n }\n\n onExtNotification<TMethod extends ExtensionMethod>(\n method: TMethod,\n handler: (params: unknown, context: { method: TMethod }) => void | Promise<void>\n ): void;\n onExtNotification(method: string, handler: JsonRpcNotificationHandler): void;\n onExtNotification(method: string, handler: JsonRpcNotificationHandler): void {\n assertExtensionMethod(method);\n this.layer.onNotification(method, handler);\n }\n\n getStderrOutput(): string {\n return this.stderrChunks.join(\"\");\n }\n\n pendingRequestCount(): number {\n return this.layer.pendingRequestCount();\n }\n\n dispose(reason: Error = new Error(\"ACP transport disposed\")): void {\n if (this.closeEvent !== null) {\n return;\n }\n\n this.closeReason = reason;\n this.layer.dispose(reason);\n\n if (!this.child.stdin.destroyed && !this.child.stdin.writableEnded) {\n this.child.stdin.end();\n }\n\n if (this.child.exitCode !== null || this.child.signalCode !== null) {\n this.close(reason, this.child.exitCode, this.child.signalCode);\n return;\n }\n\n const killed = this.child.kill();\n if (!killed) {\n this.close(reason, this.child.exitCode, this.child.signalCode);\n }\n }\n\n private close(reason: Error, code: number | null, signal: NodeJS.Signals | null): void {\n if (this.closeEvent !== null) {\n return;\n }\n\n this.layer.dispose(reason);\n this.closeEvent = {\n code,\n signal,\n reason,\n stderr: this.getStderrOutput(),\n };\n this.resolveClosed?.(this.closeEvent);\n this.resolveClosed = null;\n }\n}\n", "import * as fsPromises from \"node:fs/promises\";\nimport { homedir } from \"node:os\";\nimport { join } from \"node:path\";\nimport {\n extractToolCallSummariesFromSessionUpdateStream,\n extractUsageFromSessionUpdateStream,\n type ToolCallSummary,\n} from \"./stream-helpers.js\";\nimport type { Cost, SessionUpdate, SessionUpdateNotification, UsageUpdate } from \"./types.js\";\n\ntype SessionUpdateStreamItem = SessionUpdateNotification | SessionUpdate;\n\nexport type RunExitStatus = \"success\" | \"failed\";\n\nexport interface RunReportUsage {\n used: number;\n size: number;\n updates: number;\n cost?: Cost | null;\n}\n\nexport interface RunReportError {\n message: string;\n toolCallId?: string;\n}\n\nexport interface RunReport {\n runId: string;\n startTime: string;\n endTime: string;\n exitStatus: RunExitStatus;\n toolCalls: ToolCallSummary[];\n usage: RunReportUsage;\n errors: RunReportError[];\n}\n\nexport interface GenerateRunReportOptions {\n runId?: string;\n startTime?: string | Date;\n endTime?: string | Date;\n exitStatus?: RunExitStatus;\n errors?: string[];\n now?: () => Date;\n}\n\nexport type RunReportFileSystem = {\n mkdir(path: string, options?: { recursive?: boolean }): Promise<void>;\n writeFile(\n path: string,\n data: string,\n options?: { encoding?: BufferEncoding },\n ): Promise<void>;\n};\n\nexport interface SaveRunReportOptions {\n fs?: RunReportFileSystem;\n homeDir?: string;\n now?: () => Date;\n}\n\nexport interface SavedRunReportPaths {\n reportsDir: string;\n jsonPath: string;\n summaryPath: string;\n}\n\nexport async function generateRunReportFromSessionUpdateStream(\n stream: AsyncIterable<SessionUpdateStreamItem> | Iterable<SessionUpdateStreamItem>,\n options: GenerateRunReportOptions = {},\n): Promise<RunReport> {\n const now = options.now ?? (() => new Date());\n\n const bufferedEntries: SessionUpdateStreamItem[] = [];\n let runIdFromStream: string | undefined;\n\n for await (const entry of stream) {\n bufferedEntries.push(entry);\n if (runIdFromStream) {\n continue;\n }\n\n if (isSessionUpdateNotification(entry)) {\n const sessionId = toNonEmptyString(entry.params.sessionId);\n if (sessionId) {\n runIdFromStream = sessionId;\n }\n }\n }\n\n const runId = toNonEmptyString(options.runId) ?? runIdFromStream;\n if (!runId) {\n throw new Error(\"Run id is required via options.runId or session/update stream items\");\n }\n\n const startTime = normalizeTime(options.startTime, now);\n const endTime = normalizeTime(options.endTime, now);\n\n const toolCalls = await extractToolCallSummariesFromSessionUpdateStream(bufferedEntries);\n const usageUpdates = await extractUsageFromSessionUpdateStream(bufferedEntries);\n\n const usage = summarizeUsage(usageUpdates);\n const errors = collectErrors(toolCalls, options.errors);\n const exitStatus = options.exitStatus ?? (errors.length > 0 ? \"failed\" : \"success\");\n\n return {\n runId,\n startTime,\n endTime,\n exitStatus,\n toolCalls,\n usage,\n errors,\n };\n}\n\nexport function formatRunReportSummary(report: RunReport): string {\n const lines = [\n `Run ID: ${report.runId}`,\n `Start time: ${report.startTime}`,\n `End time: ${report.endTime}`,\n `Duration: ${toDuration(report.startTime, report.endTime)}`,\n `Exit status: ${report.exitStatus}`,\n `Tool count: ${report.toolCalls.length}`,\n `Token usage: ${report.usage.used}/${report.usage.size}`,\n `Error count: ${report.errors.length}`,\n ];\n\n if (report.usage.cost) {\n lines.push(`Cost: ${report.usage.cost.amount} ${report.usage.cost.currency}`);\n }\n\n return lines.join(\"\\n\");\n}\n\nexport async function saveRunReport(\n report: RunReport,\n options: SaveRunReportOptions = {},\n): Promise<SavedRunReportPaths> {\n const fs = options.fs ?? fsPromises;\n const now = options.now ?? (() => new Date());\n\n const reportsDir = join(options.homeDir ?? homedir(), \".poe-code\", \"reports\");\n await fs.mkdir(reportsDir, { recursive: true });\n\n const timestamp = toTimestampForFileName(now());\n const safeRunId = toSafeFileSegment(report.runId);\n const baseFileName = `${timestamp}-${safeRunId}`;\n\n const jsonPath = join(reportsDir, `${baseFileName}.json`);\n const summaryPath = join(reportsDir, `${baseFileName}.txt`);\n\n await fs.writeFile(jsonPath, JSON.stringify(report, null, 2), { encoding: \"utf8\" });\n await fs.writeFile(summaryPath, formatRunReportSummary(report), { encoding: \"utf8\" });\n\n return {\n reportsDir,\n jsonPath,\n summaryPath,\n };\n}\n\nfunction isSessionUpdateNotification(entry: SessionUpdateStreamItem): entry is SessionUpdateNotification {\n return (\n typeof (entry as SessionUpdateNotification).jsonrpc === \"string\" &&\n (entry as SessionUpdateNotification).method === \"session/update\"\n );\n}\n\nfunction normalizeTime(value: string | Date | undefined, now: () => Date): string {\n if (value instanceof Date) {\n return value.toISOString();\n }\n\n if (typeof value === \"string\" && value.length > 0) {\n const parsed = new Date(value);\n if (!Number.isNaN(parsed.getTime())) {\n return parsed.toISOString();\n }\n }\n\n return now().toISOString();\n}\n\nfunction summarizeUsage(updates: readonly UsageUpdate[]): RunReportUsage {\n let used = 0;\n let size = 0;\n let cost: Cost | null | undefined;\n\n for (const update of updates) {\n used += update.used;\n size += update.size;\n\n if (update.cost !== undefined) {\n cost = update.cost;\n }\n }\n\n const usage: RunReportUsage = {\n used,\n size,\n updates: updates.length,\n };\n\n if (cost !== undefined) {\n usage.cost = cost;\n }\n\n return usage;\n}\n\nfunction collectErrors(\n toolCalls: readonly ToolCallSummary[],\n additionalErrors: readonly string[] | undefined,\n): RunReportError[] {\n const errors: RunReportError[] = [];\n\n for (const toolCall of toolCalls) {\n if (toolCall.status !== \"failed\") {\n continue;\n }\n\n errors.push({\n toolCallId: toolCall.toolCallId,\n message: toErrorMessage(toolCall),\n });\n }\n\n if (additionalErrors) {\n for (const message of additionalErrors) {\n const text = toNonEmptyString(message);\n if (text) {\n errors.push({ message: text });\n }\n }\n }\n\n return errors;\n}\n\nfunction toErrorMessage(toolCall: ToolCallSummary): string {\n if (typeof toolCall.rawOutput === \"string\" && toolCall.rawOutput.length > 0) {\n return toolCall.rawOutput;\n }\n\n if (toolCall.rawOutput instanceof Error && toolCall.rawOutput.message.length > 0) {\n return toolCall.rawOutput.message;\n }\n\n if (toolCall.rawOutput !== undefined && toolCall.rawOutput !== null) {\n const encoded = trySerialize(toolCall.rawOutput);\n if (encoded) {\n return encoded;\n }\n }\n\n return `${toolCall.title} failed`;\n}\n\nfunction trySerialize(value: unknown): string | undefined {\n try {\n const serialized = JSON.stringify(value);\n if (typeof serialized === \"string\" && serialized.length > 0) {\n return serialized;\n }\n } catch {\n return undefined;\n }\n\n return undefined;\n}\n\nfunction toDuration(startTime: string, endTime: string): string {\n const startMs = Date.parse(startTime);\n const endMs = Date.parse(endTime);\n\n if (Number.isNaN(startMs) || Number.isNaN(endMs) || endMs < startMs) {\n return \"unknown\";\n }\n\n const seconds = (endMs - startMs) / 1000;\n return `${Number(seconds.toFixed(3))}s`;\n}\n\nfunction toSafeFileSegment(value: string): string {\n let output = \"\";\n\n for (const char of value) {\n if (isAsciiLetterOrDigit(char) || char === \"-\" || char === \"_\") {\n output += char;\n continue;\n }\n\n output += \"-\";\n }\n\n return output.length > 0 ? output : \"run\";\n}\n\nfunction isAsciiLetterOrDigit(value: string): boolean {\n const code = value.charCodeAt(0);\n return (\n (code >= 48 && code <= 57) ||\n (code >= 65 && code <= 90) ||\n (code >= 97 && code <= 122)\n );\n}\n\nfunction toTimestampForFileName(value: Date): string {\n return [\n String(value.getUTCFullYear()),\n pad(value.getUTCMonth() + 1, 2),\n pad(value.getUTCDate(), 2),\n ].join(\"\")\n + \"-\"\n + [\n pad(value.getUTCHours(), 2),\n pad(value.getUTCMinutes(), 2),\n pad(value.getUTCSeconds(), 2),\n ].join(\"\")\n + \"-\"\n + pad(value.getUTCMilliseconds(), 3);\n}\n\nfunction pad(value: number, size: number): string {\n const text = String(value);\n if (text.length >= size) {\n return text;\n }\n\n return `${\"0\".repeat(size - text.length)}${text}`;\n}\n\nfunction toNonEmptyString(value: unknown): string | undefined {\n if (typeof value !== \"string\" || value.length === 0) {\n return undefined;\n }\n\n return value;\n}\n", "import path from \"node:path\";\nimport { homedir } from \"node:os\";\nimport { mkdir, open, type FileHandle } from \"node:fs/promises\";\nimport type { AcpEvent } from \"../types.js\";\nimport type { AcpMiddleware, SpawnContext } from \"../middleware.js\";\n\nfunction pad(value: number, width: number): string {\n return String(value).padStart(width, \"0\");\n}\n\nfunction formatTimestamp(date: Date): { day: string; time: string; milliseconds: string } {\n const day = `${date.getUTCFullYear()}${pad(date.getUTCMonth() + 1, 2)}${pad(date.getUTCDate(), 2)}`;\n const time = `${pad(date.getUTCHours(), 2)}${pad(date.getUTCMinutes(), 2)}${pad(date.getUTCSeconds(), 2)}`;\n const milliseconds = pad(date.getUTCMilliseconds(), 3);\n return { day, time, milliseconds };\n}\n\nfunction normalizeAgent(agent: string): string {\n let normalized = \"\";\n for (const char of agent) {\n const code = char.charCodeAt(0);\n const isLower = code >= 97 && code <= 122;\n const isUpper = code >= 65 && code <= 90;\n const isDigit = code >= 48 && code <= 57;\n\n if (isLower || isUpper || isDigit || char === \"-\" || char === \"_\") {\n normalized += char;\n } else {\n normalized += \"-\";\n }\n }\n\n return normalized.length > 0 ? normalized : \"agent\";\n}\n\nfunction resolveStartedAt(value: Date | undefined): Date {\n if (!(value instanceof Date) || Number.isNaN(value.getTime())) {\n return new Date();\n }\n\n return value;\n}\n\nfunction resolveLogFilePath(ctx: SpawnContext): string {\n const baseDir = ctx.logDir ?? path.join(homedir(), \".poe-code\", \"spawn-logs\");\n const startedAt = resolveStartedAt(ctx.startedAt);\n const { day, time, milliseconds } = formatTimestamp(startedAt);\n const fileName = `${day}-${time}-${milliseconds}-${normalizeAgent(ctx.agent)}.jsonl`;\n return path.join(baseDir, fileName);\n}\n\nclass SpawnLogWriter {\n private fileHandle: FileHandle | undefined;\n\n private isDisabled = false;\n\n private readonly logFilePath: string;\n\n private readonly logDirPath: string;\n\n constructor(ctx: SpawnContext) {\n this.logFilePath = resolveLogFilePath(ctx);\n this.logDirPath = path.dirname(this.logFilePath);\n }\n\n async writeEvent(event: AcpEvent): Promise<void> {\n if (this.isDisabled) {\n return;\n }\n\n try {\n await this.ensureOpen();\n if (!this.fileHandle) {\n return;\n }\n\n await this.fileHandle.appendFile(`${JSON.stringify(event)}\\n`, \"utf8\");\n } catch {\n this.isDisabled = true;\n await this.close();\n }\n }\n\n async close(): Promise<void> {\n if (!this.fileHandle) {\n return;\n }\n\n try {\n await this.fileHandle.close();\n } catch {\n // Ignore close errors to avoid disrupting event processing.\n } finally {\n this.fileHandle = undefined;\n }\n }\n\n private async ensureOpen(): Promise<void> {\n if (this.fileHandle || this.isDisabled) {\n return;\n }\n\n try {\n await mkdir(this.logDirPath, { recursive: true });\n this.fileHandle = await open(this.logFilePath, \"a\");\n } catch {\n this.isDisabled = true;\n }\n }\n}\n\nasync function writePreloadedEvents(writer: SpawnLogWriter, events: AcpEvent[]): Promise<void> {\n for (const event of events) {\n await writer.writeEvent(event);\n }\n}\n\nexport const spawnLog: AcpMiddleware = async (ctx, next) => {\n await next();\n\n const source = ctx.eventStream;\n const writer = new SpawnLogWriter(ctx);\n\n await writePreloadedEvents(writer, ctx.events);\n\n if (!source) {\n await writer.close();\n return;\n }\n\n ctx.eventStream = (async function* () {\n try {\n for await (const event of source) {\n await writer.writeEvent(event);\n yield event;\n }\n } finally {\n await writer.close();\n }\n })();\n};\n", "import type { CommandRunner, CommandRunnerResult } from \"@poe-code/agent-spawn\";\nimport { buildSpawnArgs } from \"@poe-code/agent-spawn\";\n\nexport type {\n CommandRunner,\n CommandRunnerOptions,\n CommandRunnerResult\n} from \"@poe-code/agent-spawn\";\n\nexport function formatCommandRunnerResult(\n result: CommandRunnerResult\n): string {\n const stdout =\n result.stdout.length > 0 ? result.stdout : \"<empty>\";\n const stderr =\n result.stderr.length > 0 ? result.stderr : \"<empty>\";\n return `stdout:\\n${stdout}\\nstderr:\\n${stderr}`;\n}\n\nexport interface RunAndMatchOutputOptions {\n command: string;\n args: string[];\n expectedOutput: string;\n skipOnDryRun?: boolean;\n}\n\nexport function describeCommandExpectation(\n command: string,\n args: string[],\n expectedOutput: string\n): string {\n return `${renderCommandLine(command, args)} (expecting \"${expectedOutput}\")`;\n}\n\nexport interface CommandExpectationCheckOptions\n extends RunAndMatchOutputOptions {\n id: string;\n}\n\nexport function createCommandExpectationCheck(\n options: CommandExpectationCheckOptions\n): CommandCheck {\n return {\n id: options.id,\n description: describeCommandExpectation(\n options.command,\n options.args,\n options.expectedOutput\n ),\n async run(context) {\n await runAndMatchOutput(context, options);\n }\n };\n}\n\nexport async function runAndMatchOutput(\n context: CommandCheckContext,\n options: RunAndMatchOutputOptions\n): Promise<void> {\n const rendered = renderCommandLine(options.command, options.args);\n if (options.skipOnDryRun !== false && context.isDryRun) {\n if (context.logDryRun) {\n context.logDryRun(\n `Dry run: ${rendered} (expecting \"${options.expectedOutput}\")`\n );\n }\n return;\n }\n\n const result = await context.runCommand(options.command, options.args);\n if (result.exitCode !== 0) {\n const detail = formatCommandRunnerResult(result);\n throw new Error(\n [`Command ${rendered} failed with exit code ${result.exitCode}.`, detail].join(\"\\n\")\n );\n }\n\n if (!stdoutMatchesExpected(result.stdout, options.expectedOutput)) {\n const detail = formatCommandRunnerResult(result);\n const received = result.stdout.trim();\n throw new Error(\n [\n `Command ${rendered} failed: expected \"${options.expectedOutput}\" but received \"${received}\".`,\n detail\n ].join(\"\\n\")\n );\n }\n}\n\nexport function stdoutMatchesExpected(stdout: string, expected: string): boolean {\n const trimmed = stdout.trim();\n if (trimmed === expected) {\n return true;\n }\n\n const lines = stdout\n .split(/\\r?\\n/)\n .map((line) => line.trim())\n .filter((line) => line.length > 0);\n\n if (lines.some((line) => line === expected)) {\n return true;\n }\n\n for (const line of lines) {\n if (line[0] !== \"{\") continue;\n try {\n const parsed = JSON.parse(line) as { type?: string; result?: string };\n if (parsed.type === \"result\" && parsed.result === expected) {\n return true;\n }\n } catch {\n continue;\n }\n }\n\n return false;\n}\n\nfunction renderCommandLine(command: string, args: string[]): string {\n return [command, ...args].map(quoteIfNeeded).join(\" \").trim();\n}\n\nfunction quoteIfNeeded(value: string): string {\n if (value.length === 0) {\n return '\"\"';\n }\n if (needsQuoting(value)) {\n return `\"${value.replaceAll('\"', '\\\\\"')}\"`;\n }\n return value;\n}\n\nfunction needsQuoting(value: string): boolean {\n return (\n value.includes(\" \") ||\n value.includes(\"\\t\") ||\n value.includes(\"\\n\")\n );\n}\n\nexport interface CommandCheckContext {\n isDryRun: boolean;\n runCommand: CommandRunner;\n logDryRun?: (message: string) => void;\n}\n\nexport interface CommandCheck {\n id: string;\n description?: string;\n run(context: CommandCheckContext): Promise<void>;\n}\n\nexport function createSpawnHealthCheck(\n agentId: string,\n options: { model?: string; expectedOutput: string }\n): CommandCheck {\n const prompt = `Output exactly: ${options.expectedOutput}`;\n const { binaryName, args } = buildSpawnArgs(agentId, {\n prompt,\n model: options.model,\n mode: \"yolo\"\n });\n return {\n id: `${agentId}-cli-health`,\n description: `spawn ${agentId} (expecting \"${options.expectedOutput}\")`,\n async run(context) {\n if (context.isDryRun) {\n context.logDryRun?.(\n `Dry run: ${[binaryName, ...args].join(\" \")} (expecting \"${options.expectedOutput}\")`\n );\n return;\n }\n\n const result = await context.runCommand(binaryName, args);\n\n if (result.exitCode !== 0) {\n throw new Error(\n `spawn ${agentId} failed with exit code ${result.exitCode}.\\n${formatCommandRunnerResult(result)}`\n );\n }\n\n if (!result.stdout.includes(options.expectedOutput)) {\n throw new Error(\n `spawn ${agentId}: expected \"${options.expectedOutput}\" in stdout.\\n${formatCommandRunnerResult(result)}`\n );\n }\n }\n };\n}\n\n/**\n * Creates a check that detects if a binary exists using multiple fallback methods.\n * This is useful in Docker/containerized environments where PATH may not be updated after npm install.\n *\n * @param binaryName - The name of the binary to check for (e.g., \"claude\", \"codex\")\n * @param id - Unique identifier for the check\n * @param description - Human-readable description of what's being checked\n * @returns A CommandCheck that verifies the binary using multiple detection methods\n */\nexport function createBinaryExistsCheck(\n binaryName: string,\n id: string,\n description: string\n): CommandCheck {\n return {\n id,\n description,\n async run({ runCommand }) {\n // Common installation paths for CLI tools\n const commonPaths = [\n `/usr/local/bin/${binaryName}`,\n `/usr/bin/${binaryName}`,\n `$HOME/.local/bin/${binaryName}`,\n `$HOME/.claude/local/bin/${binaryName}`\n ];\n\n const detectors: Array<{\n command: string;\n args: string[];\n validate: (result: CommandRunnerResult) => boolean;\n }> = [\n {\n command: \"which\",\n args: [binaryName],\n validate: (result) => result.exitCode === 0\n },\n {\n command: \"where\",\n args: [binaryName],\n validate: (result) =>\n result.exitCode === 0 && result.stdout.trim().length > 0\n },\n // Check common installation paths using shell expansion for $HOME\n {\n command: \"sh\",\n args: [\n \"-c\",\n commonPaths.map((p) => `test -f \"${p}\"`).join(\" || \")\n ],\n validate: (result) => result.exitCode === 0\n }\n ];\n\n for (const detector of detectors) {\n const result = await runCommand(detector.command, detector.args);\n if (detector.validate(result)) {\n return;\n }\n }\n\n throw new Error(`${binaryName} CLI binary not found on PATH.`);\n }\n };\n}\n", "import type {\n ConfigMergeMutation,\n ConfigPruneMutation,\n ConfigTransformMutation,\n ConfigObject,\n ValueResolver,\n MutationOptions\n} from \"../types.js\";\n\nexport interface MergeOptions {\n /** Target file path (must start with ~) */\n target: ValueResolver<string>;\n /** Value to merge into the config file */\n value: ValueResolver<ConfigObject>;\n /** Optional explicit format override */\n format?: \"json\" | \"toml\";\n /** Optional prune by prefix before merging (TOML) */\n pruneByPrefix?: Record<string, string>;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface PruneOptions {\n /** Target file path (must start with ~) */\n target: ValueResolver<string>;\n /** Shape to prune from the config file */\n shape: ValueResolver<ConfigObject>;\n /** Optional explicit format override */\n format?: \"json\" | \"toml\";\n /** Optional guard - only prune if predicate returns true */\n onlyIf?: (doc: ConfigObject, ctx: MutationOptions) => boolean;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface TransformOptions {\n /** Target file path (must start with ~) */\n target: ValueResolver<string>;\n /** Optional explicit format override */\n format?: \"json\" | \"toml\";\n /** Transform function - receives parsed content, returns transformed content */\n transform: (\n content: ConfigObject,\n ctx: MutationOptions\n ) => { content: ConfigObject | null; changed: boolean };\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nfunction merge(options: MergeOptions): ConfigMergeMutation {\n return {\n kind: \"configMerge\",\n target: options.target,\n value: options.value,\n format: options.format,\n pruneByPrefix: options.pruneByPrefix,\n label: options.label\n };\n}\n\nfunction prune(options: PruneOptions): ConfigPruneMutation {\n return {\n kind: \"configPrune\",\n target: options.target,\n shape: options.shape,\n format: options.format,\n onlyIf: options.onlyIf,\n label: options.label\n };\n}\n\nfunction transform(options: TransformOptions): ConfigTransformMutation {\n return {\n kind: \"configTransform\",\n target: options.target,\n format: options.format,\n transform: options.transform,\n label: options.label\n };\n}\n\nexport const configMutation = {\n merge,\n prune,\n transform\n};\n", "import type {\n EnsureDirectoryMutation,\n RemoveDirectoryMutation,\n RemoveFileMutation,\n ChmodMutation,\n BackupMutation,\n ValueResolver\n} from \"../types.js\";\n\nexport interface EnsureDirectoryOptions {\n /** Directory path (must start with ~) */\n path: ValueResolver<string>;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface RemoveOptions {\n /** Target file path (must start with ~) */\n target: ValueResolver<string>;\n /** Only remove if file is empty/whitespace */\n whenEmpty?: boolean;\n /** Only remove if content matches regex */\n whenContentMatches?: RegExp;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface RemoveDirectoryOptions {\n /** Directory path (must start with ~) */\n path: ValueResolver<string>;\n /** Remove directory even when not empty */\n force?: boolean;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface ChmodOptions {\n /** Target file path (must start with ~) */\n target: ValueResolver<string>;\n /** File permission mode (e.g., 0o755) */\n mode: number;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface BackupOptions {\n /** Target file path to backup (must start with ~) */\n target: ValueResolver<string>;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nfunction ensureDirectory(options: EnsureDirectoryOptions): EnsureDirectoryMutation {\n return {\n kind: \"ensureDirectory\",\n path: options.path,\n label: options.label\n };\n}\n\nfunction remove(options: RemoveOptions): RemoveFileMutation {\n return {\n kind: \"removeFile\",\n target: options.target,\n whenEmpty: options.whenEmpty,\n whenContentMatches: options.whenContentMatches,\n label: options.label\n };\n}\n\nfunction removeDirectory(\n options: RemoveDirectoryOptions\n): RemoveDirectoryMutation {\n return {\n kind: \"removeDirectory\",\n path: options.path,\n force: options.force,\n label: options.label\n };\n}\n\nfunction chmod(options: ChmodOptions): ChmodMutation {\n return {\n kind: \"chmod\",\n target: options.target,\n mode: options.mode,\n label: options.label\n };\n}\n\nfunction backup(options: BackupOptions): BackupMutation {\n return {\n kind: \"backup\",\n target: options.target,\n label: options.label\n };\n}\n\nexport const fileMutation = {\n ensureDirectory,\n remove,\n removeDirectory,\n chmod,\n backup\n};\n", "import Mustache from \"mustache\";\nimport type {\n Mutation,\n MutationContext,\n MutationOutcome,\n MutationDetails,\n ConfigObject,\n MutationOptions,\n ValueResolver,\n FileSystem\n} from \"../types.js\";\nimport { getConfigFormat, detectFormat } from \"../formats/index.js\";\nimport { resolvePath } from \"./path-utils.js\";\nimport {\n isNotFound,\n readFileIfExists,\n pathExists,\n createTimestamp\n} from \"../fs-utils.js\";\n\n// ============================================================================\n// Helper Functions\n// ============================================================================\n\nfunction resolveValue<T>(\n resolver: ValueResolver<T>,\n options: MutationOptions\n): T {\n if (typeof resolver === \"function\") {\n return (resolver as (ctx: MutationOptions) => T)(options);\n }\n return resolver;\n}\n\nfunction createInvalidDocumentBackupPath(targetPath: string): string {\n const ext = targetPath.includes(\".\") ? targetPath.split(\".\").pop() : \"bak\";\n return `${targetPath}.invalid-${createTimestamp()}.${ext}`;\n}\n\nasync function backupInvalidDocument(\n fs: FileSystem,\n targetPath: string,\n content: string\n): Promise<void> {\n const backupPath = createInvalidDocumentBackupPath(targetPath);\n await fs.writeFile(backupPath, content, { encoding: \"utf8\" });\n}\n\nfunction describeMutation(kind: string, targetPath?: string): string {\n const displayPath = targetPath ?? \"target\";\n switch (kind) {\n case \"ensureDirectory\":\n return `Create ${displayPath}`;\n case \"removeDirectory\":\n return `Remove directory ${displayPath}`;\n case \"backup\":\n return `Backup ${displayPath}`;\n case \"templateWrite\":\n return `Write ${displayPath}`;\n case \"chmod\":\n return `Set permissions on ${displayPath}`;\n case \"removeFile\":\n return `Remove ${displayPath}`;\n case \"configMerge\":\n case \"configPrune\":\n case \"configTransform\":\n case \"templateMergeToml\":\n case \"templateMergeJson\":\n return `Update ${displayPath}`;\n default:\n return \"Operation\";\n }\n}\n\nfunction pruneKeysByPrefix(\n table: ConfigObject,\n prefix: string\n): ConfigObject {\n const result: ConfigObject = {};\n for (const [key, value] of Object.entries(table)) {\n if (!key.startsWith(prefix)) {\n result[key] = value;\n }\n }\n return result;\n}\n\nfunction isConfigObject(value: unknown): value is ConfigObject {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction mergeWithPruneByPrefix(\n base: ConfigObject,\n patch: ConfigObject,\n pruneByPrefix?: Record<string, string>\n): ConfigObject {\n const result: ConfigObject = { ...base };\n const prefixMap = pruneByPrefix ?? {};\n\n for (const [key, value] of Object.entries(patch)) {\n const current = result[key];\n const prefix = prefixMap[key];\n\n if (isConfigObject(current) && isConfigObject(value)) {\n if (prefix) {\n const pruned = pruneKeysByPrefix(current, prefix);\n result[key] = { ...pruned, ...value };\n } else {\n result[key] = mergeWithPruneByPrefix(\n current,\n value as ConfigObject,\n prefixMap\n );\n }\n continue;\n }\n result[key] = value;\n }\n return result;\n}\n\n// ============================================================================\n// Apply Mutation\n// ============================================================================\n\nexport async function applyMutation(\n mutation: Mutation,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n switch (mutation.kind) {\n case \"ensureDirectory\":\n return applyEnsureDirectory(mutation, context, options);\n case \"removeDirectory\":\n return applyRemoveDirectory(mutation, context, options);\n case \"removeFile\":\n return applyRemoveFile(mutation, context, options);\n case \"chmod\":\n return applyChmod(mutation, context, options);\n case \"backup\":\n return applyBackup(mutation, context, options);\n case \"configMerge\":\n return applyConfigMerge(mutation, context, options);\n case \"configPrune\":\n return applyConfigPrune(mutation, context, options);\n case \"configTransform\":\n return applyConfigTransform(mutation, context, options);\n case \"templateWrite\":\n return applyTemplateWrite(mutation, context, options);\n case \"templateMergeToml\":\n return applyTemplateMerge(mutation, context, options, \"toml\");\n case \"templateMergeJson\":\n return applyTemplateMerge(mutation, context, options, \"json\");\n default: {\n const never: never = mutation;\n throw new Error(`Unknown mutation kind: ${(never as Mutation).kind}`);\n }\n }\n}\n\n// ============================================================================\n// File Mutation Handlers\n// ============================================================================\n\nasync function applyEnsureDirectory(\n mutation: Extract<Mutation, { kind: \"ensureDirectory\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.path, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const existed = await pathExists(context.fs, targetPath);\n\n if (!context.dryRun) {\n await context.fs.mkdir(targetPath, { recursive: true });\n }\n\n return {\n outcome: {\n changed: !existed,\n effect: \"mkdir\",\n detail: existed ? \"noop\" : \"create\"\n },\n details\n };\n}\n\nasync function applyRemoveDirectory(\n mutation: Extract<Mutation, { kind: \"removeDirectory\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.path, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const existed = await pathExists(context.fs, targetPath);\n if (!existed) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (typeof context.fs.rm !== \"function\") {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (mutation.force) {\n if (!context.dryRun) {\n await context.fs.rm(targetPath, { recursive: true, force: true });\n }\n return {\n outcome: { changed: true, effect: \"delete\", detail: \"delete\" },\n details\n };\n }\n\n const entries = await context.fs.readdir(targetPath);\n if (entries.length > 0) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (!context.dryRun) {\n await context.fs.rm(targetPath, { recursive: true, force: true });\n }\n\n return {\n outcome: { changed: true, effect: \"delete\", detail: \"delete\" },\n details\n };\n}\n\nasync function applyRemoveFile(\n mutation: Extract<Mutation, { kind: \"removeFile\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n try {\n const content = await context.fs.readFile(targetPath, \"utf8\");\n const trimmed = content.trim();\n\n // Check whenContentMatches guard\n if (mutation.whenContentMatches && !mutation.whenContentMatches.test(trimmed)) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n // Check whenEmpty guard\n if (mutation.whenEmpty && trimmed.length > 0) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (!context.dryRun) {\n await context.fs.unlink(targetPath);\n }\n\n return {\n outcome: { changed: true, effect: \"delete\", detail: \"delete\" },\n details\n };\n } catch (error) {\n if (isNotFound(error)) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n throw error;\n }\n}\n\nasync function applyChmod(\n mutation: Extract<Mutation, { kind: \"chmod\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n if (typeof context.fs.chmod !== \"function\") {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n try {\n const stat = await context.fs.stat(targetPath);\n const currentMode = typeof stat.mode === \"number\" ? stat.mode & 0o777 : null;\n\n if (currentMode === mutation.mode) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (!context.dryRun) {\n await context.fs.chmod(targetPath, mutation.mode);\n }\n\n return {\n outcome: { changed: true, effect: \"chmod\", detail: \"update\" },\n details\n };\n } catch (error) {\n if (isNotFound(error)) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n throw error;\n }\n}\n\nasync function applyBackup(\n mutation: Extract<Mutation, { kind: \"backup\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const content = await readFileIfExists(context.fs, targetPath);\n if (content === null) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (!context.dryRun) {\n const backupPath = `${targetPath}.backup-${createTimestamp()}`;\n await context.fs.writeFile(backupPath, content, { encoding: \"utf8\" });\n }\n\n return {\n outcome: { changed: true, effect: \"copy\", detail: \"backup\" },\n details\n };\n}\n\n// ============================================================================\n// Config Mutation Handlers\n// ============================================================================\n\nasync function applyConfigMerge(\n mutation: Extract<Mutation, { kind: \"configMerge\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const formatName = mutation.format ?? detectFormat(rawPath);\n if (!formatName) {\n throw new Error(\n `Cannot detect config format for \"${rawPath}\". Provide explicit format option.`\n );\n }\n const format = getConfigFormat(formatName);\n\n const rawContent = await readFileIfExists(context.fs, targetPath);\n let current: ConfigObject;\n try {\n current = rawContent === null ? {} : format.parse(rawContent);\n } catch {\n // Invalid file - backup and start fresh\n if (rawContent !== null) {\n await backupInvalidDocument(context.fs, targetPath, rawContent);\n }\n current = {};\n }\n\n const value = resolveValue(mutation.value, options);\n\n // Use mergeWithPruneByPrefix for TOML files with pruneByPrefix option\n let merged: ConfigObject;\n if (mutation.pruneByPrefix) {\n merged = mergeWithPruneByPrefix(current, value, mutation.pruneByPrefix);\n } else {\n merged = format.merge(current, value);\n }\n\n const serialized = format.serialize(merged);\n const changed = serialized !== rawContent;\n\n if (changed && !context.dryRun) {\n await context.fs.writeFile(targetPath, serialized, { encoding: \"utf8\" });\n }\n\n return {\n outcome: {\n changed,\n effect: changed ? \"write\" : \"none\",\n detail: changed ? (rawContent === null ? \"create\" : \"update\") : \"noop\"\n },\n details\n };\n}\n\nasync function applyConfigPrune(\n mutation: Extract<Mutation, { kind: \"configPrune\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const rawContent = await readFileIfExists(context.fs, targetPath);\n if (rawContent === null) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n const formatName = mutation.format ?? detectFormat(rawPath);\n if (!formatName) {\n throw new Error(\n `Cannot detect config format for \"${rawPath}\". Provide explicit format option.`\n );\n }\n const format = getConfigFormat(formatName);\n\n let current: ConfigObject;\n try {\n current = format.parse(rawContent);\n } catch {\n // Invalid file - can't prune, leave as-is\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n // Check onlyIf guard\n if (mutation.onlyIf && !mutation.onlyIf(current, options)) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n const shape = resolveValue(mutation.shape, options);\n const { changed, result } = format.prune(current, shape);\n\n if (!changed) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n // Delete file if empty\n if (Object.keys(result).length === 0) {\n if (!context.dryRun) {\n await context.fs.unlink(targetPath);\n }\n return {\n outcome: { changed: true, effect: \"delete\", detail: \"delete\" },\n details\n };\n }\n\n const serialized = format.serialize(result);\n if (!context.dryRun) {\n await context.fs.writeFile(targetPath, serialized, { encoding: \"utf8\" });\n }\n\n return {\n outcome: { changed: true, effect: \"write\", detail: \"update\" },\n details\n };\n}\n\nasync function applyConfigTransform(\n mutation: Extract<Mutation, { kind: \"configTransform\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const formatName = mutation.format ?? detectFormat(rawPath);\n if (!formatName) {\n throw new Error(\n `Cannot detect config format for \"${rawPath}\". Provide explicit format option.`\n );\n }\n const format = getConfigFormat(formatName);\n\n const rawContent = await readFileIfExists(context.fs, targetPath);\n let current: ConfigObject;\n try {\n current = rawContent === null ? {} : format.parse(rawContent);\n } catch {\n if (rawContent !== null) {\n await backupInvalidDocument(context.fs, targetPath, rawContent);\n }\n current = {};\n }\n\n const { content: transformed, changed } = mutation.transform(current, options);\n\n if (!changed) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n // Delete file if null\n if (transformed === null) {\n if (rawContent === null) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n if (!context.dryRun) {\n await context.fs.unlink(targetPath);\n }\n return {\n outcome: { changed: true, effect: \"delete\", detail: \"delete\" },\n details\n };\n }\n\n const serialized = format.serialize(transformed);\n if (!context.dryRun) {\n await context.fs.writeFile(targetPath, serialized, { encoding: \"utf8\" });\n }\n\n return {\n outcome: {\n changed: true,\n effect: \"write\",\n detail: rawContent === null ? \"create\" : \"update\"\n },\n details\n };\n}\n\n// ============================================================================\n// Template Mutation Handlers\n// ============================================================================\n\nasync function applyTemplateWrite(\n mutation: Extract<Mutation, { kind: \"templateWrite\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n if (!context.templates) {\n throw new Error(\n \"Template mutations require a templates loader. \" +\n \"Provide templates function to runMutations context.\"\n );\n }\n\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const template = await context.templates(mutation.templateId);\n const templateContext = mutation.context\n ? resolveValue(mutation.context, options)\n : {};\n const rendered = Mustache.render(template, templateContext);\n\n const existed = await pathExists(context.fs, targetPath);\n\n if (!context.dryRun) {\n await context.fs.writeFile(targetPath, rendered, { encoding: \"utf8\" });\n }\n\n return {\n outcome: {\n changed: true,\n effect: \"write\",\n detail: existed ? \"update\" : \"create\"\n },\n details\n };\n}\n\nasync function applyTemplateMerge(\n mutation: Extract<Mutation, { kind: \"templateMergeToml\" | \"templateMergeJson\" }>,\n context: MutationContext,\n options: MutationOptions,\n formatName: \"toml\" | \"json\"\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n if (!context.templates) {\n throw new Error(\n \"Template mutations require a templates loader. \" +\n \"Provide templates function to runMutations context.\"\n );\n }\n\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const format = getConfigFormat(formatName);\n\n // Load and render template\n const template = await context.templates(mutation.templateId);\n const templateContext = mutation.context\n ? resolveValue(mutation.context, options)\n : {};\n const rendered = Mustache.render(template, templateContext);\n\n // Parse rendered template\n let templateDoc: ConfigObject;\n try {\n templateDoc = format.parse(rendered);\n } catch (error) {\n throw new Error(\n `Failed to parse rendered template \"${mutation.templateId}\" as ${formatName.toUpperCase()}: ${error}`,\n { cause: error }\n );\n }\n\n // Read and parse existing file\n const rawContent = await readFileIfExists(context.fs, targetPath);\n let current: ConfigObject;\n try {\n current = rawContent === null ? {} : format.parse(rawContent);\n } catch {\n if (rawContent !== null) {\n await backupInvalidDocument(context.fs, targetPath, rawContent);\n }\n current = {};\n }\n\n // Merge\n const merged = format.merge(current, templateDoc);\n const serialized = format.serialize(merged);\n const changed = serialized !== rawContent;\n\n if (changed && !context.dryRun) {\n await context.fs.writeFile(targetPath, serialized, { encoding: \"utf8\" });\n }\n\n return {\n outcome: {\n changed,\n effect: changed ? \"write\" : \"none\",\n detail: changed ? (rawContent === null ? \"create\" : \"update\") : \"noop\"\n },\n details\n };\n}\n", "import * as jsonc from \"jsonc-parser\";\nimport type { ConfigFormat, ConfigObject, ConfigValue } from \"../types.js\";\n\nfunction isConfigObject(value: unknown): value is ConfigObject {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction detectIndent(content: string): string {\n const match = content.match(/^[\\t ]+/m);\n if (match) {\n return match[0];\n }\n return \" \";\n}\n\nfunction parse(content: string): ConfigObject {\n if (!content || content.trim() === \"\") {\n return {};\n }\n const errors: jsonc.ParseError[] = [];\n const parsed = jsonc.parse(content, errors, {\n allowTrailingComma: true,\n disallowComments: false\n });\n if (errors.length > 0) {\n throw new Error(`JSON parse error: ${jsonc.printParseErrorCode(errors[0].error)}`);\n }\n if (parsed === null || parsed === undefined) {\n return {};\n }\n if (!isConfigObject(parsed)) {\n throw new Error(\"Expected JSON object.\");\n }\n return parsed;\n}\n\nfunction serialize(obj: ConfigObject): string {\n return `${JSON.stringify(obj, null, 2)}\\n`;\n}\n\nfunction merge(base: ConfigObject, patch: ConfigObject): ConfigObject {\n const result: ConfigObject = { ...base };\n for (const [key, value] of Object.entries(patch)) {\n if (value === undefined) {\n continue;\n }\n const existing = result[key];\n if (isConfigObject(existing) && isConfigObject(value)) {\n result[key] = merge(existing, value);\n continue;\n }\n result[key] = value as ConfigValue;\n }\n return result;\n}\n\nfunction prune(\n obj: ConfigObject,\n shape: ConfigObject\n): { changed: boolean; result: ConfigObject } {\n let changed = false;\n const result: ConfigObject = { ...obj };\n\n for (const [key, pattern] of Object.entries(shape)) {\n if (!(key in result)) {\n continue;\n }\n\n const current = result[key];\n\n // Empty object pattern means \"delete this key entirely\"\n if (isConfigObject(pattern) && Object.keys(pattern).length === 0) {\n delete result[key];\n changed = true;\n continue;\n }\n\n // Non-empty object pattern with object current: recurse\n if (isConfigObject(pattern) && isConfigObject(current)) {\n const { changed: childChanged, result: childResult } = prune(\n current,\n pattern\n );\n if (childChanged) {\n changed = true;\n }\n if (Object.keys(childResult).length === 0) {\n delete result[key];\n } else {\n result[key] = childResult;\n }\n continue;\n }\n\n delete result[key];\n changed = true;\n }\n\n return { changed, result };\n}\n\n/**\n * Modify JSON content at a specific path while preserving comments and formatting.\n * Uses jsonc-parser's modify() for targeted updates.\n *\n * @param content - The original JSON content (may include comments)\n * @param path - JSON path array, e.g. [\"mcpServers\", \"my-server\"]\n * @param value - The value to set (or undefined to remove)\n * @returns The modified JSON content with comments preserved\n */\nfunction modifyAtPath(\n content: string,\n path: (string | number)[],\n value: ConfigValue | undefined\n): string {\n const indent = detectIndent(content);\n const formattingOptions: jsonc.FormattingOptions = {\n tabSize: indent === \"\\t\" ? 1 : indent.length,\n insertSpaces: indent !== \"\\t\",\n eol: \"\\n\"\n };\n\n const edits = jsonc.modify(content, path, value, { formattingOptions });\n let result = jsonc.applyEdits(content, edits);\n\n if (!result.endsWith(\"\\n\")) {\n result += \"\\n\";\n }\n\n return result;\n}\n\n/**\n * Merge a patch into JSON content while preserving comments and formatting.\n * Uses jsonc.modify() for each top-level key to preserve existing comments.\n *\n * @param content - The original JSON content (may include comments)\n * @param patch - Object with values to merge\n * @returns The modified JSON content with comments preserved\n */\nfunction mergePreservingComments(\n content: string,\n patch: ConfigObject\n): string {\n let result = content || \"{}\";\n\n for (const [key, value] of Object.entries(patch)) {\n if (value === undefined) {\n continue;\n }\n result = modifyAtPath(result, [key], value);\n }\n\n return result;\n}\n\n/**\n * Remove a key from JSON content while preserving comments and formatting.\n *\n * @param content - The original JSON content\n * @param path - JSON path array to the key to remove\n * @returns The modified JSON content with comments preserved\n */\nfunction removeAtPath(content: string, path: (string | number)[]): string {\n return modifyAtPath(content, path, undefined);\n}\n\nexport { detectIndent, modifyAtPath, mergePreservingComments, removeAtPath };\n\nexport const jsonFormat: ConfigFormat = {\n parse,\n serialize,\n merge,\n prune\n};\n", "import { parse as parseToml, stringify as stringifyToml } from \"smol-toml\";\nimport type { ConfigFormat, ConfigObject, ConfigValue } from \"../types.js\";\n\nfunction isConfigObject(value: unknown): value is ConfigObject {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction parse(content: string): ConfigObject {\n if (!content || content.trim() === \"\") {\n return {};\n }\n const parsed = parseToml(content);\n if (!isConfigObject(parsed)) {\n throw new Error(\"Expected TOML document to be a table.\");\n }\n return parsed as ConfigObject;\n}\n\nfunction serialize(obj: ConfigObject): string {\n const serialized = stringifyToml(obj);\n return serialized.endsWith(\"\\n\") ? serialized : `${serialized}\\n`;\n}\n\nfunction merge(base: ConfigObject, patch: ConfigObject): ConfigObject {\n const result: ConfigObject = { ...base };\n for (const [key, value] of Object.entries(patch)) {\n if (value === undefined) {\n continue;\n }\n const existing = result[key];\n if (isConfigObject(existing) && isConfigObject(value)) {\n result[key] = merge(existing, value);\n continue;\n }\n result[key] = value as ConfigValue;\n }\n return result;\n}\n\nfunction prune(\n obj: ConfigObject,\n shape: ConfigObject\n): { changed: boolean; result: ConfigObject } {\n let changed = false;\n const result: ConfigObject = { ...obj };\n\n for (const [key, pattern] of Object.entries(shape)) {\n if (!(key in result)) {\n continue;\n }\n\n const current = result[key];\n\n // Empty object pattern means \"delete this key entirely\"\n if (isConfigObject(pattern) && Object.keys(pattern).length === 0) {\n delete result[key];\n changed = true;\n continue;\n }\n\n // Non-empty object pattern with object current: recurse\n if (isConfigObject(pattern) && isConfigObject(current)) {\n const { changed: childChanged, result: childResult } = prune(\n current,\n pattern\n );\n if (childChanged) {\n changed = true;\n }\n if (Object.keys(childResult).length === 0) {\n delete result[key];\n } else {\n result[key] = childResult;\n }\n continue;\n }\n\n delete result[key];\n changed = true;\n }\n\n return { changed, result };\n}\n\nexport const tomlFormat: ConfigFormat = {\n parse,\n serialize,\n merge,\n prune\n};\n", "import type { ConfigFormat } from \"../types.js\";\nimport { jsonFormat } from \"./json.js\";\nimport { tomlFormat } from \"./toml.js\";\n\nexport type FormatName = \"json\" | \"toml\";\n\nconst formatRegistry: Record<FormatName, ConfigFormat> = {\n json: jsonFormat,\n toml: tomlFormat\n};\n\nconst extensionMap: Record<string, FormatName> = {\n \".json\": \"json\",\n \".toml\": \"toml\"\n};\n\n/**\n * Get a format handler by path (auto-detect from extension) or explicit format name.\n */\nexport function getConfigFormat(pathOrFormat: string): ConfigFormat {\n // Check if it's an explicit format name\n if (pathOrFormat in formatRegistry) {\n return formatRegistry[pathOrFormat as FormatName];\n }\n\n // Try to detect from extension\n const ext = getExtension(pathOrFormat);\n const formatName = extensionMap[ext];\n\n if (!formatName) {\n throw new Error(\n `Unsupported config format. Cannot detect format from \"${pathOrFormat}\". ` +\n `Supported extensions: ${Object.keys(extensionMap).join(\", \")}. ` +\n `Supported format names: ${Object.keys(formatRegistry).join(\", \")}.`\n );\n }\n\n return formatRegistry[formatName];\n}\n\n/**\n * Detect format name from a file path.\n */\nexport function detectFormat(path: string): FormatName | undefined {\n const ext = getExtension(path);\n return extensionMap[ext];\n}\n\nfunction getExtension(path: string): string {\n const lastDot = path.lastIndexOf(\".\");\n if (lastDot === -1) {\n return \"\";\n }\n return path.slice(lastDot).toLowerCase();\n}\n\nexport { jsonFormat } from \"./json.js\";\nexport { tomlFormat } from \"./toml.js\";\n", "import path from \"node:path\";\nimport type { PathMapper } from \"../types.js\";\n\n/**\n * Expand ~ shortcut to the provided home directory.\n */\nexport function expandHome(targetPath: string, homeDir: string): string {\n if (!targetPath?.startsWith(\"~\")) {\n return targetPath;\n }\n\n // Handle ~./ -> ~/.\n if (targetPath.startsWith(\"~./\")) {\n targetPath = `~/.${targetPath.slice(3)}`;\n }\n\n let remainder = targetPath.slice(1);\n\n // Remove leading slash or backslash\n if (remainder.startsWith(\"/\") || remainder.startsWith(\"\\\\\")) {\n remainder = remainder.slice(1);\n } else if (remainder.startsWith(\".\")) {\n // Handle ~/.\n remainder = remainder.slice(1);\n if (remainder.startsWith(\"/\") || remainder.startsWith(\"\\\\\")) {\n remainder = remainder.slice(1);\n }\n }\n\n return remainder.length === 0 ? homeDir : path.join(homeDir, remainder);\n}\n\n/**\n * Validate that a path is home-relative (starts with ~).\n * Throws if the path is not home-relative.\n */\nexport function validateHomePath(targetPath: string): void {\n if (typeof targetPath !== \"string\" || targetPath.length === 0) {\n throw new Error(\"Target path must be a non-empty string.\");\n }\n\n if (!targetPath.startsWith(\"~\")) {\n throw new Error(\n `All target paths must be home-relative (start with ~). Received: \"${targetPath}\"`\n );\n }\n}\n\n/**\n * Resolve a path with optional path mapping for isolated configurations.\n * 1. Validates the path starts with ~\n * 2. Expands ~ to home directory\n * 3. If pathMapper is provided, maps the directory portion and reconstructs the path\n */\nexport function resolvePath(\n rawPath: string,\n homeDir: string,\n pathMapper?: PathMapper\n): string {\n validateHomePath(rawPath);\n const expanded = expandHome(rawPath, homeDir);\n\n if (!pathMapper) {\n return expanded;\n }\n\n // Map the directory portion\n const rawDirectory = path.dirname(expanded);\n const mappedDirectory = pathMapper.mapTargetDirectory({\n targetDirectory: rawDirectory\n });\n const filename = path.basename(expanded);\n\n return filename.length === 0 ? mappedDirectory : path.join(mappedDirectory, filename);\n}\n", "import type { FileSystem } from \"./types.js\";\n\n/**\n * Check if an error is a \"file not found\" (ENOENT) error.\n */\nexport function isNotFound(error: unknown): boolean {\n return (\n typeof error === \"object\" &&\n error !== null &&\n \"code\" in error &&\n (error as { code?: string }).code === \"ENOENT\"\n );\n}\n\n/**\n * Read a file if it exists, returning null if not found.\n */\nexport async function readFileIfExists(\n fs: FileSystem,\n target: string\n): Promise<string | null> {\n try {\n return await fs.readFile(target, \"utf8\");\n } catch (error) {\n if (isNotFound(error)) {\n return null;\n }\n throw error;\n }\n}\n\n/**\n * Check if a path exists (file or directory).\n */\nexport async function pathExists(\n fs: FileSystem,\n target: string\n): Promise<boolean> {\n try {\n await fs.stat(target);\n return true;\n } catch (error) {\n if (isNotFound(error)) {\n return false;\n }\n throw error;\n }\n}\n\n/**\n * Create an ISO timestamp safe for use in filenames.\n * Replaces colons and dots with dashes.\n */\nexport function createTimestamp(): string {\n return new Date().toISOString().replaceAll(\":\", \"-\").replaceAll(\".\", \"-\");\n}\n", "import type {\n Mutation,\n MutationContext,\n MutationResult,\n MutationOptions\n} from \"../types.js\";\nimport { applyMutation } from \"./apply-mutation.js\";\n\n/**\n * Execute an array of mutations in order.\n *\n * All dependencies must be injected - no defaults, no globals.\n */\nexport async function runMutations(\n mutations: Mutation[],\n context: MutationContext,\n options?: MutationOptions\n): Promise<MutationResult> {\n const effects: MutationResult[\"effects\"] = [];\n let anyChanged = false;\n const resolverOptions = options ?? {};\n\n for (const mutation of mutations) {\n const { outcome } = await executeMutation(\n mutation,\n context,\n resolverOptions\n );\n effects.push(outcome);\n if (outcome.changed) {\n anyChanged = true;\n }\n }\n\n return {\n changed: anyChanged,\n effects\n };\n}\n\nasync function executeMutation(\n mutation: Mutation,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationResult[\"effects\"][number]; details: { kind: string; label: string; targetPath?: string } }> {\n // Call onStart observer\n context.observers?.onStart?.({\n kind: mutation.kind,\n label: mutation.label ?? mutation.kind,\n targetPath: undefined // Will be resolved during apply\n });\n\n try {\n const { outcome, details } = await applyMutation(mutation, context, options);\n\n // Call onComplete observer\n context.observers?.onComplete?.(details, outcome);\n\n return { outcome, details };\n } catch (error) {\n // Call onError observer\n context.observers?.onError?.(\n {\n kind: mutation.kind,\n label: mutation.label ?? mutation.kind,\n targetPath: undefined\n },\n error\n );\n\n // Re-throw the error\n throw error;\n }\n}\n", "import Mustache from \"mustache\";\n\nexport type TemplateVariables = Record<string, string | number | boolean | string[]>;\n\n// Disable HTML escaping - we're rendering prompts, not HTML\nconst originalEscape = Mustache.escape;\n\n/**\n * Render a mustache template with the given variables.\n * Arrays are automatically joined with newlines.\n * HTML escaping is disabled.\n */\nexport function renderTemplate(\n template: string,\n variables: TemplateVariables\n): string {\n // Pre-process variables to handle arrays\n const processed: Record<string, string | number | boolean> = {};\n for (const [key, value] of Object.entries(variables)) {\n if (Array.isArray(value)) {\n processed[key] = value.join(\"\\n\");\n } else {\n processed[key] = value;\n }\n }\n\n // Temporarily disable HTML escaping\n Mustache.escape = (text: string) => text;\n try {\n return Mustache.render(template, processed);\n } finally {\n Mustache.escape = originalEscape;\n }\n}\n", "export const FRONTIER_MODELS = [\n \"anthropic/claude-opus-4.6\",\n \"anthropic/claude-sonnet-4.6\",\n \"openai/gpt-5.3-codex\",\n \"openai/gpt-5.4\",\n \"google/gemini-3.1-pro\"\n] as const;\n\nexport const DEFAULT_FRONTIER_MODEL = \"anthropic/claude-sonnet-4.6\";\n\nexport const DEFAULT_TEXT_MODEL = \"anthropic/claude-sonnet-4.6\";\nexport const DEFAULT_IMAGE_BOT = \"google/nano-banana-pro\";\nexport const DEFAULT_AUDIO_BOT = \"elevenlabs/elevenlabs-v3\";\nexport const DEFAULT_VIDEO_BOT = \"google/veo-3.1\";\n\nexport const CLAUDE_CODE_VARIANTS = {\n haiku: \"anthropic/claude-haiku-4.5\",\n sonnet: \"anthropic/claude-sonnet-4.6\",\n opus: \"anthropic/claude-opus-4.6\"\n} as const;\n\nexport const DEFAULT_CLAUDE_CODE_MODEL = CLAUDE_CODE_VARIANTS.sonnet;\n\n/**\n * Extracts the model ID from a namespaced model slug (lowercase).\n * e.g., \"anthropic/claude-sonnet-4.6\" -> \"claude-sonnet-4.6\"\n */\nexport function stripModelNamespace(model: string): string {\n const slashIndex = model.indexOf(\"/\");\n const id = slashIndex === -1 ? model : model.slice(slashIndex + 1);\n return id.toLowerCase();\n}\n\nexport const CODEX_MODELS = [\n \"openai/gpt-5.4\",\n \"openai/gpt-5.3-codex\",\n \"openai/gpt-5.3-codex-spark\",\n \"openai/gpt-5.2-codex\",\n \"openai/gpt-5.2\",\n \"openai/gpt-5.2-chat\",\n \"openai/gpt-5.2-pro\",\n \"openai/gpt-5.1\",\n \"openai/gpt-5.1-codex-mini\",\n \"anthropic/claude-opus-4.6\"\n] as const;\nexport const DEFAULT_CODEX_MODEL = CODEX_MODELS[0];\n\nexport const KIMI_MODELS = [\n \"novitaai/kimi-k2.5\",\n \"novitaai/kimi-k2-thinking\",\n \"novitaai/kimi-k2.5-fw\",\n] as const;\nexport const DEFAULT_KIMI_MODEL = KIMI_MODELS[0];\n\nexport const DEFAULT_REASONING = \"medium\";\nexport const PROVIDER_NAME = \"poe\";\nexport const FEEDBACK_URL = \"https://github.com/poe-platform/poe-code/issues\";\n", "import type {\n CommandRunner,\n CommandCheckContext,\n CommandCheck\n} from \"../utils/command-checks.js\";\n\nexport interface InstallContext {\n isDryRun: boolean;\n runCommand: CommandRunner;\n logger: (message: string) => void;\n platform: NodeJS.Platform;\n}\n\nexport interface InstallCommand {\n id: string;\n command: string;\n args: string[];\n platforms?: NodeJS.Platform[];\n}\n\nexport interface ServiceInstallDefinition {\n id: string;\n summary: string;\n check: CommandCheck;\n steps: InstallCommand[];\n postChecks?: CommandCheck[];\n successMessage?: string;\n}\n\nexport async function runServiceInstall(\n definition: ServiceInstallDefinition,\n context: InstallContext\n): Promise<boolean> {\n const checkContext: CommandCheckContext = {\n isDryRun: context.isDryRun,\n runCommand: context.runCommand\n };\n\n let needsInstall = false;\n try {\n await definition.check.run(checkContext);\n context.logger(`${definition.summary} already installed.`);\n } catch (error) {\n const detail =\n error instanceof Error ? error.message : String(error);\n context.logger(`${definition.summary} not detected: ${detail}`);\n needsInstall = true;\n }\n\n if (!needsInstall) {\n return false;\n }\n\n if (context.isDryRun) {\n logInstallDryRun(definition, context);\n return true;\n }\n\n const platformSteps = filterStepsByPlatform(definition.steps, context.platform);\n for (const step of platformSteps) {\n await runInstallStep(step, context);\n }\n\n await definition.check.run(checkContext);\n\n if (definition.postChecks) {\n for (const postCheck of definition.postChecks) {\n await postCheck.run(checkContext);\n }\n }\n\n context.logger(\n definition.successMessage ?? `${definition.summary} installed.`\n );\n return true;\n}\n\nfunction describeInstallCommand(step: InstallCommand): string {\n return `[${step.id}] ${formatCommand(step.command, step.args)}`;\n}\n\nfunction formatCommand(command: string, args: string[]): string {\n return [command, ...args.map(quoteIfNeeded)].join(\" \");\n}\n\nfunction quoteIfNeeded(value: string): string {\n if (value.length === 0) {\n return '\"\"';\n }\n if (value.includes(\" \") || value.includes(\"\\t\") || value.includes(\"\\n\")) {\n return `\"${value.replaceAll('\"', '\\\\\"')}\"`;\n }\n return value;\n}\n\nfunction filterStepsByPlatform(\n steps: InstallCommand[],\n platform: NodeJS.Platform\n): InstallCommand[] {\n return steps.filter(\n (step) => !step.platforms || step.platforms.includes(platform)\n );\n}\n\nfunction logInstallDryRun(\n definition: ServiceInstallDefinition,\n context: InstallContext\n): void {\n context.logger(`Dry run: would install ${definition.summary}.`);\n const platformSteps = filterStepsByPlatform(definition.steps, context.platform);\n for (const step of platformSteps) {\n context.logger(`Dry run: ${describeInstallCommand(step)}`);\n }\n}\n\nasync function runInstallStep(\n step: InstallCommand,\n context: InstallContext\n): Promise<void> {\n context.logger(`Running ${describeInstallCommand(step)}`);\n const result = await context.runCommand(step.command, step.args);\n if (result.exitCode !== 0) {\n const stderr = result.stderr.trim();\n const suffix = stderr.length > 0 ? `: ${stderr}` : \"\";\n throw new Error(\n `${describeInstallCommand(step)} failed with exit code ${result.exitCode}${suffix}`\n );\n }\n}\n", "import type {\n ProviderService,\n ProviderContext,\n ProviderBranding,\n ProviderConfigurePrompts,\n ProviderIsolatedEnv\n} from \"../cli/service-registry.js\";\nimport {\n runMutations,\n type Mutation,\n type MutationObservers\n} from \"@poe-code/config-mutations\";\nimport {\n runServiceInstall,\n type ServiceInstallDefinition\n} from \"../services/service-install.js\";\n// Template imports are lazy to avoid breaking tsc output when imported\n// by generate-bin-wrappers.mjs (Node.js can't resolve .hbs as ESM modules)\nconst templateImports: Record<string, () => Promise<{ default: string }>> = {\n \"py-poe-spawn/env.hbs\": () => import(\"../templates/py-poe-spawn/env.hbs\"),\n \"py-poe-spawn/main.py.hbs\": () => import(\"../templates/py-poe-spawn/main.py.hbs\"),\n \"py-poe-spawn/requirements.txt.hbs\": () => import(\"../templates/py-poe-spawn/requirements.txt.hbs\"),\n \"codex/config.toml.hbs\": () => import(\"../templates/codex/config.toml.hbs\"),\n};\n\nasync function loadTemplate(templateId: string): Promise<string> {\n const loader = templateImports[templateId];\n if (!loader) {\n throw new Error(`Template not found: ${templateId}`);\n }\n const module = await loader();\n return module.default;\n}\n\ninterface ManifestVersionDefinition {\n configure: Mutation[];\n unconfigure?: Mutation[];\n}\n\nexport interface ServiceRunOptions {\n observers?: MutationObservers;\n}\n\ninterface CreateProviderOptions<\n ConfigureOptions,\n UnconfigureOptions,\n SpawnOptions\n> {\n name: string;\n aliases?: string[];\n label: string;\n id: string;\n summary: string;\n branding?: ProviderBranding;\n disabled?: boolean;\n supportsStdinPrompt?: boolean;\n supportsMcpSpawn?: boolean;\n configurePrompts?: ProviderConfigurePrompts;\n postConfigureMessages?: string[];\n isolatedEnv?: ProviderIsolatedEnv;\n manifest: ManifestVersionDefinition;\n install?: ServiceInstallDefinition;\n test?: ProviderService<ConfigureOptions, UnconfigureOptions, SpawnOptions>[\"test\"];\n spawn?: ProviderService<\n ConfigureOptions,\n UnconfigureOptions,\n SpawnOptions\n >[\"spawn\"];\n}\n\nexport function createProvider<\n ConfigureOptions = any,\n UnconfigureOptions = ConfigureOptions,\n SpawnOptions = any\n>(\n opts: CreateProviderOptions<ConfigureOptions, UnconfigureOptions, SpawnOptions>\n): ProviderService<ConfigureOptions, UnconfigureOptions, SpawnOptions> {\n const provider: ProviderService<\n ConfigureOptions,\n UnconfigureOptions,\n SpawnOptions\n > = {\n id: opts.id,\n summary: opts.summary,\n name: opts.name,\n aliases: opts.aliases,\n label: opts.label,\n branding: opts.branding,\n disabled: opts.disabled,\n supportsStdinPrompt: opts.supportsStdinPrompt,\n supportsMcpSpawn: opts.supportsMcpSpawn,\n configurePrompts: opts.configurePrompts,\n postConfigureMessages: opts.postConfigureMessages,\n isolatedEnv: opts.isolatedEnv,\n async configure(context, runOptions) {\n await runMutations(opts.manifest.configure, {\n fs: context.fs,\n homeDir: context.env.homeDir,\n observers: runOptions?.observers,\n templates: loadTemplate,\n pathMapper: context.pathMapper\n }, context.options as Record<string, unknown>);\n context.command.flushDryRun({ emitIfEmpty: false });\n },\n async unconfigure(context, runOptions) {\n if (!opts.manifest.unconfigure) {\n return false;\n }\n const result = await runMutations(opts.manifest.unconfigure, {\n fs: context.fs,\n homeDir: context.env.homeDir,\n observers: runOptions?.observers,\n templates: loadTemplate,\n pathMapper: context.pathMapper\n }, context.options as Record<string, unknown>);\n context.command.flushDryRun({ emitIfEmpty: false });\n return result.changed;\n }\n };\n\n if (opts.install) {\n provider.install = createInstallRunner(opts.install);\n }\n\n if (opts.test) {\n provider.test = opts.test;\n }\n\n if (opts.spawn) {\n provider.spawn = opts.spawn;\n }\n\n return provider;\n}\n\nfunction createInstallRunner(definition: ServiceInstallDefinition) {\n return async (context: ProviderContext): Promise<void> => {\n await runServiceInstall(definition, {\n isDryRun: context.logger.context.dryRun,\n runCommand: context.command.runCommand,\n logger: (message) => context.logger.verbose(message),\n platform: context.env.platform\n });\n };\n}\n", "import {\n createBinaryExistsCheck,\n createSpawnHealthCheck\n} from \"../utils/command-checks.js\";\nimport {\n configMutation,\n fileMutation,\n type ConfigObject\n} from \"@poe-code/config-mutations\";\nimport { type ServiceInstallDefinition } from \"../services/service-install.js\";\nimport { KIMI_MODELS, DEFAULT_KIMI_MODEL, PROVIDER_NAME, stripModelNamespace } from \"../cli/constants.js\";\nimport { createProvider } from \"./create-provider.js\";\nimport type {\n ProviderSpawnOptions,\n ModelConfigureOptions,\n EmptyProviderOptions\n} from \"./spawn-options.js\";\nimport { kimiAgent } from \"@poe-code/agent-defs\";\n\nexport const KIMI_INSTALL_DEFINITION: ServiceInstallDefinition = {\n id: \"kimi\",\n summary: \"Kimi CLI\",\n check: createBinaryExistsCheck(\n \"kimi\",\n \"kimi-cli-binary\",\n \"Kimi CLI binary must exist\"\n ),\n steps: [\n {\n id: \"install-kimi-cli-uv\",\n command: \"uv\",\n args: [\"tool\", \"install\", \"--python\", \"3.13\", \"kimi-cli\"]\n }\n ],\n successMessage: \"Installed Kimi CLI via uv.\"\n};\n\nfunction providerModel(model: string): string {\n const stripped = stripModelNamespace(model);\n return `${PROVIDER_NAME}/${stripped}`;\n}\n\nfunction buildKimiArgs(prompt: string, extraArgs?: string[]): string[] {\n return [\"--quiet\", \"-p\", prompt, ...(extraArgs ?? [])];\n}\n\nexport const kimiService = createProvider<\n ModelConfigureOptions,\n EmptyProviderOptions,\n ProviderSpawnOptions\n>({\n ...kimiAgent,\n disabled: false,\n supportsStdinPrompt: false,\n configurePrompts: {\n model: {\n label: \"Kimi default model\",\n defaultValue: DEFAULT_KIMI_MODEL,\n choices: KIMI_MODELS.map((id) => ({\n title: id,\n value: id\n }))\n }\n },\n isolatedEnv: {\n // Use \"kimi-cli\" to avoid stripAgentHome stripping \".kimi\" from paths\n agentBinary: \"kimi-cli\",\n configProbe: { kind: \"isolatedFile\", relativePath: \".kimi/config.toml\" },\n env: {\n HOME: { kind: \"isolatedDir\" }\n }\n },\n test(context) {\n return context.runCheck(\n createSpawnHealthCheck(\"kimi\", {\n model: context.model,\n expectedOutput: \"KIMI_OK\"\n })\n );\n },\n manifest: {\n configure: [\n fileMutation.ensureDirectory({ path: \"~/.kimi\" }),\n fileMutation.ensureDirectory({ path: \"~/.kimi/credentials\" }),\n configMutation.merge({\n target: \"~/.kimi/credentials/kimi-code.json\",\n value: () => ({\n access_token: \"poe-managed\",\n token_type: \"Bearer\",\n expires_at: Math.floor(Date.now() / 1000) + 86400 * 365 * 10\n })\n }),\n configMutation.merge({\n target: \"~/.kimi/config.toml\",\n pruneByPrefix: { models: `${PROVIDER_NAME}/` },\n value: (ctx) => {\n const { model, apiKey, env } = (ctx ?? {}) as {\n model?: string;\n apiKey?: string;\n env: { poeApiBaseUrl: string };\n };\n const selectedModel = model ?? DEFAULT_KIMI_MODEL;\n\n const models: ConfigObject = {};\n for (const m of KIMI_MODELS) {\n models[providerModel(m)] = {\n provider: PROVIDER_NAME,\n model: stripModelNamespace(m),\n max_context_size: 256000\n };\n }\n\n return {\n default_model: providerModel(selectedModel),\n default_thinking: true,\n models,\n providers: {\n [PROVIDER_NAME]: {\n type: \"openai_legacy\",\n base_url: env.poeApiBaseUrl,\n api_key: apiKey ?? \"\"\n }\n }\n };\n }\n })\n ],\n unconfigure: [\n configMutation.transform({\n target: \"~/.kimi/config.toml\",\n transform: (document) => {\n const providers = document.providers as ConfigObject | undefined;\n if (!providers || typeof providers !== \"object\") {\n return { changed: false, content: document };\n }\n if (!(PROVIDER_NAME in providers)) {\n return { changed: false, content: document };\n }\n const { [PROVIDER_NAME]: ignoredProvider, ...rest } = providers;\n void ignoredProvider;\n const updatedProviders = rest as ConfigObject;\n if (Object.keys(updatedProviders).length === 0) {\n const { providers: ignoredProviders, ...docWithoutProviders } = document;\n void ignoredProviders;\n return { changed: true, content: docWithoutProviders };\n }\n return { changed: true, content: { ...document, providers: updatedProviders } };\n }\n })\n ]\n },\n install: KIMI_INSTALL_DEFINITION,\n spawn(context, options) {\n const args = buildKimiArgs(options.prompt, options.args);\n if (options.cwd) {\n return context.command.runCommand(\"kimi\", args, {\n cwd: options.cwd\n });\n }\n return context.command.runCommand(\"kimi\", args);\n }\n});\n\nexport const provider = kimiService;\n"],
|
|
5
|
-
"mappings": "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",
|
|
4
|
+
"sourcesContent": ["'use strict';\n\nconst ESC = '\\x1B';\nconst CSI = `${ESC}[`;\nconst beep = '\\u0007';\n\nconst cursor = {\n to(x, y) {\n if (!y) return `${CSI}${x + 1}G`;\n return `${CSI}${y + 1};${x + 1}H`;\n },\n move(x, y) {\n let ret = '';\n\n if (x < 0) ret += `${CSI}${-x}D`;\n else if (x > 0) ret += `${CSI}${x}C`;\n\n if (y < 0) ret += `${CSI}${-y}A`;\n else if (y > 0) ret += `${CSI}${y}B`;\n\n return ret;\n },\n up: (count = 1) => `${CSI}${count}A`,\n down: (count = 1) => `${CSI}${count}B`,\n forward: (count = 1) => `${CSI}${count}C`,\n backward: (count = 1) => `${CSI}${count}D`,\n nextLine: (count = 1) => `${CSI}E`.repeat(count),\n prevLine: (count = 1) => `${CSI}F`.repeat(count),\n left: `${CSI}G`,\n hide: `${CSI}?25l`,\n show: `${CSI}?25h`,\n save: `${ESC}7`,\n restore: `${ESC}8`\n}\n\nconst scroll = {\n up: (count = 1) => `${CSI}S`.repeat(count),\n down: (count = 1) => `${CSI}T`.repeat(count)\n}\n\nconst erase = {\n screen: `${CSI}2J`,\n up: (count = 1) => `${CSI}1J`.repeat(count),\n down: (count = 1) => `${CSI}J`.repeat(count),\n line: `${CSI}2K`,\n lineEnd: `${CSI}K`,\n lineStart: `${CSI}1K`,\n lines(count) {\n let clear = '';\n for (let i = 0; i < count; i++)\n clear += this.line + (i < count - 1 ? cursor.up() : '');\n if (count)\n clear += cursor.left;\n return clear;\n }\n}\n\nmodule.exports = { cursor, scroll, erase, beep };\n", "POE_API_KEY={{apiKey}}\nPOE_BASE_URL=https://api.poe.com/v1\nMODEL={{model}}\n", "import os\nfrom openai import OpenAI\nfrom dotenv import load_dotenv\n\nload_dotenv()\n\nclient = OpenAI(\n api_key=os.getenv(\"POE_API_KEY\"),\n base_url=os.getenv(\"POE_BASE_URL\")\n)\n\nresponse = client.chat.completions.create(\n model=os.getenv(\"MODEL\", \"{{model}}\"),\n messages=[{\"role\": \"user\", \"content\": \"Tell me a joke\"}]\n)\n\nprint(response.choices[0].message.content)\n", "openai>=1.0.0\npython-dotenv>=1.0.0\n", "model_provider = \"poe\"\n\n[profiles.\"{{{profileName}}}\"]\nmodel = \"{{{model}}}\"\nmodel_provider = \"poe\"\nmodel_reasoning_effort = \"{{reasoningEffort}}\"\nmodel_verbosity = \"medium\"\n\n[model_providers.poe]\nname = \"poe\"\nbase_url = \"{{{baseUrl}}}\"\nwire_api = \"responses\"\nexperimental_bearer_token = \"{{apiKey}}\"\nrequires_openai_auth = false\nsupports_websockets = false\n", "import { spawn } from \"node:child_process\";\n\nexport interface CommandRunnerResult {\n stdout: string;\n stderr: string;\n exitCode: number;\n}\n\nexport interface CommandRunnerOptions {\n cwd?: string;\n env?: Record<string, string | undefined>;\n stdin?: string | Buffer;\n}\n\nexport type CommandRunner = (\n command: string,\n args: string[],\n options?: CommandRunnerOptions\n) => Promise<CommandRunnerResult>;\n\nexport function runCommand(\n command: string,\n args: string[],\n options?: CommandRunnerOptions\n): Promise<CommandRunnerResult> {\n return new Promise((resolve) => {\n const hasStdin = options?.stdin != null;\n const child = spawn(command, args, {\n stdio: [hasStdin ? \"pipe\" : \"ignore\", \"pipe\", \"pipe\"],\n cwd: options?.cwd,\n env: options?.env\n ? {\n ...(process.env as Record<string, string | undefined>),\n ...options.env\n }\n : undefined\n });\n let stdout = \"\";\n let stderr = \"\";\n\n if (hasStdin && child.stdin) {\n child.stdin.on(\"error\", () => {});\n child.stdin.end(options!.stdin);\n }\n\n child.stdout?.setEncoding(\"utf8\");\n child.stdout?.on(\"data\", (chunk: string | Buffer) => {\n stdout += chunk.toString();\n });\n\n child.stderr?.setEncoding(\"utf8\");\n child.stderr?.on(\"data\", (chunk: string | Buffer) => {\n stderr += chunk.toString();\n });\n\n child.on(\"error\", (error: NodeJS.ErrnoException) => {\n const exitCode =\n typeof error.code === \"number\"\n ? error.code\n : typeof error.errno === \"number\"\n ? error.errno\n : 127;\n const message =\n error instanceof Error ? error.message : String(error ?? \"error\");\n resolve({\n stdout,\n stderr: stderr ? `${stderr}${message}` : message,\n exitCode\n });\n });\n\n child.on(\"close\", (code) => {\n resolve({\n stdout,\n stderr,\n exitCode: code ?? 0\n });\n });\n });\n}\n", "import type { AgentDefinition } from \"../types.js\";\n\nexport const claudeCodeAgent: AgentDefinition = {\n id: \"claude-code\",\n name: \"claude-code\",\n label: \"Claude Code\",\n summary: \"Configure Claude Code to route through Poe.\",\n aliases: [\"claude\"],\n binaryName: \"claude\",\n configPath: \"~/.claude/settings.json\",\n branding: {\n colors: {\n dark: \"#C15F3C\",\n light: \"#C15F3C\"\n }\n }\n};\n", "import type { AgentDefinition } from \"../types.js\";\n\nexport const claudeDesktopAgent: AgentDefinition = {\n id: \"claude-desktop\",\n name: \"claude-desktop\",\n label: \"Claude Desktop\",\n summary: \"Anthropic's official desktop application for Claude\",\n configPath: \"~/.claude/settings.json\",\n branding: {\n colors: {\n dark: \"#D97757\",\n light: \"#D97757\"\n }\n }\n};\n", "import type { AgentDefinition } from \"../types.js\";\n\nexport const codexAgent: AgentDefinition = {\n id: \"codex\",\n name: \"codex\",\n label: \"Codex\",\n summary: \"Configure Codex to use Poe as the model provider.\",\n binaryName: \"codex\",\n configPath: \"~/.codex/config.toml\",\n branding: {\n colors: {\n dark: \"#D5D9DF\",\n light: \"#7A7F86\"\n }\n }\n};\n", "import type { AgentDefinition } from \"../types.js\";\n\nexport const openCodeAgent: AgentDefinition = {\n id: \"opencode\",\n name: \"opencode\",\n label: \"OpenCode CLI\",\n summary: \"Configure OpenCode CLI to use the Poe API.\",\n binaryName: \"opencode\",\n configPath: \"~/.config/opencode/config.json\",\n branding: {\n colors: {\n dark: \"#4A4F55\",\n light: \"#2F3338\"\n }\n }\n};\n", "import type { AgentDefinition } from \"../types.js\";\n\nexport const kimiAgent: AgentDefinition = {\n id: \"kimi\",\n name: \"kimi\",\n label: \"Kimi\",\n summary: \"Configure Kimi CLI to use Poe API\",\n aliases: [\"kimi-cli\"],\n binaryName: \"kimi\",\n configPath: \"~/.kimi/config.toml\",\n branding: {\n colors: {\n dark: \"#7B68EE\",\n light: \"#6A5ACD\"\n }\n }\n};\n", "import type { AgentDefinition } from \"./types.js\";\nimport {\n claudeCodeAgent,\n claudeDesktopAgent,\n codexAgent,\n openCodeAgent,\n kimiAgent\n} from \"./agents/index.js\";\n\nexport const allAgents: AgentDefinition[] = [\n claudeCodeAgent,\n claudeDesktopAgent,\n codexAgent,\n openCodeAgent,\n kimiAgent\n];\n\nconst lookup = new Map<string, string>();\n\nfor (const agent of allAgents) {\n const values = [agent.id, agent.name, ...(agent.aliases ?? [])];\n for (const value of values) {\n const normalized = value.toLowerCase();\n if (!lookup.has(normalized)) {\n lookup.set(normalized, agent.id);\n }\n }\n}\n\nexport function resolveAgentId(input: string): string | undefined {\n if (!input) {\n return undefined;\n }\n return lookup.get(input.toLowerCase());\n}\n", "import type { McpSpawnConfig } from \"../types.js\";\n\ninterface JsonMcpServer {\n command: string;\n args?: string[];\n env?: Record<string, string>;\n}\n\nfunction toJsonMcpServers(servers: McpSpawnConfig): Record<string, JsonMcpServer> {\n const out: Record<string, JsonMcpServer> = {};\n\n for (const [name, server] of Object.entries(servers)) {\n const mapped: JsonMcpServer = { command: server.command };\n if (server.args && server.args.length > 0) {\n mapped.args = server.args;\n }\n if (server.env && Object.keys(server.env).length > 0) {\n mapped.env = server.env;\n }\n out[name] = mapped;\n }\n\n return out;\n}\n\nfunction toTomlString(value: string): string {\n return JSON.stringify(value);\n}\n\nfunction toTomlArray(values: string[]): string {\n const serialized = values.map((value) => toTomlString(value));\n return `[${serialized.join(\", \")}]`;\n}\n\nfunction toTomlInlineTable(values: Record<string, string>): string {\n const parts: string[] = [];\n for (const [key, value] of Object.entries(values)) {\n parts.push(`${JSON.stringify(key)}=${toTomlString(value)}`);\n }\n return `{${parts.join(\", \")}}`;\n}\n\nexport function serializeJsonMcpArgs(servers: McpSpawnConfig): string[] {\n return [\"--mcp-config\", JSON.stringify({ mcpServers: toJsonMcpServers(servers) })];\n}\n\nexport function serializeOpenCodeMcpEnv(servers: McpSpawnConfig): Record<string, string> {\n const mcp: Record<\n string,\n { type: \"local\"; command: string[]; environment?: Record<string, string> }\n > = {};\n for (const [name, server] of Object.entries(servers)) {\n const entry: {\n type: \"local\";\n command: string[];\n environment?: Record<string, string>;\n } = { type: \"local\", command: [server.command, ...(server.args ?? [])] };\n if (server.env && Object.keys(server.env).length > 0) {\n entry.environment = server.env;\n }\n mcp[name] = entry;\n }\n return { OPENCODE_CONFIG_CONTENT: JSON.stringify({ mcp }) };\n}\n\nexport function serializeCodexMcpArgs(servers: McpSpawnConfig): string[] {\n const args: string[] = [];\n\n for (const [name, server] of Object.entries(servers)) {\n const prefix = `mcp_servers.${name}`;\n args.push(\"-c\", `${prefix}.command=${toTomlString(server.command)}`);\n\n if (server.args && server.args.length > 0) {\n args.push(\"-c\", `${prefix}.args=${toTomlArray(server.args)}`);\n }\n\n if (server.env && Object.keys(server.env).length > 0) {\n args.push(\"-c\", `${prefix}.env=${toTomlInlineTable(server.env)}`);\n }\n }\n\n return args;\n}\n", "import type { CliSpawnConfig } from \"../types.js\";\nimport { serializeJsonMcpArgs } from \"./mcp.js\";\n\nexport const claudeCodeSpawnConfig: CliSpawnConfig = {\n kind: \"cli\",\n agentId: \"claude-code\",\n // ACP adapter support: yes (adapter: \"claude\")\n adapter: \"claude\",\n promptFlag: \"-p\",\n modelFlag: \"--model\",\n modelStripProviderPrefix: true,\n modelTransform: (model) => model.replaceAll(\".\", \"-\"),\n defaultArgs: [\n \"--output-format\",\n \"stream-json\",\n \"--verbose\"\n ],\n mcpArgs: serializeJsonMcpArgs,\n modes: {\n yolo: [\"--dangerously-skip-permissions\"],\n edit: [\"--permission-mode\", \"acceptEdits\", \"--allowedTools\", \"Bash,Read,Write,Edit,Glob,Grep,NotebookEdit\"],\n read: [\"--permission-mode\", \"plan\"]\n },\n stdinMode: {\n omitPrompt: true,\n extraArgs: [\"--input-format\", \"text\"]\n },\n interactive: {\n defaultArgs: []\n },\n resumeCommand: (threadId) => [\"--resume\", threadId]\n};\n", "import type { CliSpawnConfig } from \"../types.js\";\nimport { serializeCodexMcpArgs } from \"./mcp.js\";\n\nexport const codexSpawnConfig: CliSpawnConfig = {\n kind: \"cli\",\n agentId: \"codex\",\n // ACP adapter support: yes (adapter: \"codex\")\n adapter: \"codex\",\n promptFlag: \"exec\",\n modelFlag: \"--model\",\n modelStripProviderPrefix: true,\n defaultArgs: [\"--skip-git-repo-check\", \"--json\"],\n mcpArgs: serializeCodexMcpArgs,\n mcpArgsBeforeCommand: true,\n modes: {\n yolo: [\"-s\", \"danger-full-access\"],\n edit: [\"-s\", \"workspace-write\"],\n read: [\"-s\", \"read-only\"]\n },\n stdinMode: {\n omitPrompt: true,\n extraArgs: [\"-\"]\n },\n interactive: {\n defaultArgs: [\"-a\", \"never\"]\n },\n resumeCommand: (threadId, cwd) => [\"resume\", \"-C\", cwd, threadId]\n};\n", "import type { AcpSpawnConfig, CliSpawnConfig } from \"../types.js\";\nimport { serializeOpenCodeMcpEnv } from \"./mcp.js\";\n\n/**\n * OpenCode JSON output format (empirically observed)\n *\n * OpenCode can emit \"raw JSON events\" when running a prompt via:\n * - `opencode run \"<prompt>\" --format json ...`\n *\n * Key observations (OpenCode CLI v1.1.47):\n * - Output is **NDJSON / line-delimited JSON**: one JSON object per stdout line.\n * - Each line is an event object with a top-level `type` string (NOT ACP's `{ event: ... }`).\n * - Common top-level fields:\n * - `type`: `\"step_start\" | \"text\" | \"tool_use\" | \"step_finish\" | ...`\n * - `timestamp`: number (ms since epoch)\n * - `sessionID`: string (e.g. `\"ses_...\"`)\n * - `part`: object with event-specific payload\n *\n * `text` events:\n * ```ts\n * {\n * type: \"text\",\n * sessionID: \"ses_...\",\n * part: {\n * type: \"text\",\n * messageID: \"msg_...\",\n * text: \"Hello ...\",\n * time: { start: 1770000000000, end: 1770000000000 }\n * }\n * }\n * ```\n *\n * Tool calls (`tool_use`):\n * - Represented as a single event with `part.type: \"tool\"` and `state.status`.\n * - `state.input` includes tool arguments; `state.output` is the tool result string.\n * ```ts\n * {\n * type: \"tool_use\",\n * sessionID: \"ses_...\",\n * part: {\n * type: \"tool\",\n * callID: \"call_...\",\n * tool: \"bash\",\n * state: {\n * status: \"completed\",\n * input: { command: \"echo hello\", description: \"...\" },\n * output: \"hello\\n\"\n * }\n * }\n * }\n * ```\n *\n * Step boundaries:\n * - `step_start` and `step_finish` wrap a single model/tool turn.\n * - `step_finish.part.tokens` contains token accounting:\n * `{ input, output, reasoning, cache: { read, write } }`\n *\n * Negative cases / gotchas:\n * - Some failures (e.g. invalid `--model` / unknown provider) can print a non-JSON stack trace\n * before any JSON events are emitted, even with `--format json`.\n * - If `--format json` is ever removed upstream, OpenCode will need a text-mode fallback\n * (no streaming event adapter).\n */\nexport const openCodeSpawnConfig: CliSpawnConfig = {\n kind: \"cli\",\n agentId: \"opencode\",\n // ACP adapter support: yes (adapter: \"opencode\").\n // OpenCode's `--format json` emits NDJSON events with `{ type, sessionID, part }`\n // (no `{ event, ... }` field), so it needs the OpenCode adapter (not \"native\").\n adapter: \"opencode\",\n promptFlag: \"run\",\n modelFlag: \"--model\",\n modelStripProviderPrefix: false,\n modelTransform: (model) => {\n return model.startsWith(\"poe/\") ? model : `poe/${model}`;\n },\n defaultArgs: [\"--format\", \"json\"],\n modes: {\n yolo: [],\n edit: [],\n read: [\"--agent\", \"plan\"]\n },\n interactive: {\n defaultArgs: [],\n promptFlag: \"--prompt\"\n },\n resumeCommand: (threadId, cwd) => [cwd, \"--session\", threadId],\n mcpEnv: serializeOpenCodeMcpEnv\n};\n\nexport const openCodeAcpSpawnConfig: AcpSpawnConfig = {\n kind: \"acp\",\n agentId: \"opencode\",\n acpArgs: [\"acp\"],\n skipAuth: true,\n mcpEnv: serializeOpenCodeMcpEnv,\n};\n", "import type { AcpSpawnConfig, CliSpawnConfig } from \"../types.js\";\nimport { serializeJsonMcpArgs } from \"./mcp.js\";\n\nexport const kimiSpawnConfig: CliSpawnConfig = {\n kind: \"cli\",\n agentId: \"kimi\",\n // ACP adapter support: yes (adapter: \"kimi\").\n // Kimi's `--output-format stream-json` emits OpenAI-style `{ role, content }` JSON\n // (no `{ event, ... }` field), so it needs the Kimi adapter (not \"native\").\n adapter: \"kimi\",\n promptFlag: \"-p\",\n modelStripProviderPrefix: true,\n defaultArgs: [\"--print\", \"--output-format\", \"stream-json\"],\n mcpArgs: serializeJsonMcpArgs,\n modes: {\n yolo: [\"--yolo\"],\n edit: [],\n read: []\n },\n stdinMode: {\n omitPrompt: true,\n extraArgs: [\"--input-format\", \"stream-json\"]\n },\n interactive: {\n defaultArgs: [],\n promptFlag: \"-p\"\n },\n resumeCommand: (threadId, cwd) => [\"--session\", threadId, \"--work-dir\", cwd]\n};\n\nexport const kimiAcpSpawnConfig: AcpSpawnConfig = {\n kind: \"acp\",\n agentId: \"kimi\",\n acpArgs: [\"acp\"],\n};\n", "import { resolveAgentId } from \"@poe-code/agent-defs\";\nimport type { AcpSpawnConfig, SpawnConfig } from \"../types.js\";\nimport { claudeCodeSpawnConfig } from \"./claude-code.js\";\nimport { codexSpawnConfig } from \"./codex.js\";\nimport { openCodeSpawnConfig, openCodeAcpSpawnConfig } from \"./opencode.js\";\nimport { kimiSpawnConfig, kimiAcpSpawnConfig } from \"./kimi.js\";\n\n// ACP adapter support (spawn streaming):\n// - Supported (has `adapter`): claude-code, codex, opencode, kimi\nexport const allSpawnConfigs: readonly SpawnConfig[] = [\n claudeCodeSpawnConfig,\n codexSpawnConfig,\n openCodeSpawnConfig,\n kimiSpawnConfig\n];\n\nconst lookup = new Map<string, SpawnConfig>();\n\nfor (const config of allSpawnConfigs) {\n lookup.set(config.agentId, config);\n}\n\nconst acpLookup = new Map<string, AcpSpawnConfig>();\nacpLookup.set(openCodeAcpSpawnConfig.agentId, openCodeAcpSpawnConfig);\nacpLookup.set(kimiAcpSpawnConfig.agentId, kimiAcpSpawnConfig);\n\nexport function getSpawnConfig(input: string): SpawnConfig | undefined {\n const resolvedId = resolveAgentId(input);\n if (!resolvedId) {\n return undefined;\n }\n return lookup.get(resolvedId);\n}\n\nexport function getAcpSpawnConfig(input: string): AcpSpawnConfig | undefined {\n const resolvedId = resolveAgentId(input);\n if (!resolvedId) {\n return undefined;\n }\n return acpLookup.get(resolvedId);\n}\n\nexport function supportsMcpAtSpawn(input: string): boolean {\n const config = getSpawnConfig(input);\n return (\n !!config &&\n config.kind === \"cli\" &&\n (typeof config.mcpArgs === \"function\" || typeof config.mcpEnv === \"function\")\n );\n}\n\nexport function listMcpSupportedAgents(): string[] {\n const supported: string[] = [];\n\n for (const config of allSpawnConfigs) {\n if (\n config.kind !== \"cli\" ||\n (typeof config.mcpArgs !== \"function\" && typeof config.mcpEnv !== \"function\")\n ) {\n continue;\n }\n supported.push(config.agentId);\n }\n\n return supported;\n}\n", "import { spawn as spawnChildProcess } from \"node:child_process\";\nimport { resolveConfig } from \"./configs/resolve-config.js\";\nimport { getMcpArgs } from \"./mcp-args.js\";\nimport { stripModelNamespace } from \"./model-utils.js\";\nimport type {\n CliSpawnConfig,\n McpSpawnConfig,\n SpawnContext,\n SpawnMode,\n SpawnOptions,\n SpawnResult,\n StdinMode\n} from \"./types.js\";\n\nfunction createAbortError(): Error {\n const error = new Error(\"Agent spawn aborted\");\n error.name = \"AbortError\";\n return error;\n}\n\nfunction createActivityTimeoutError(timeoutMs: number): Error {\n const error = new Error(\n `Agent spawn timed out after ${timeoutMs / 1000}s of inactivity`\n );\n error.name = \"ActivityTimeoutError\";\n return error;\n}\n\nexport function isActivityTimeoutError(error: unknown): boolean {\n return error instanceof Error && error.name === \"ActivityTimeoutError\";\n}\n\nexport interface BuildSpawnArgsOptions {\n prompt: string;\n model?: string;\n mode?: SpawnMode;\n args?: string[];\n mcpServers?: McpSpawnConfig;\n useStdin?: boolean;\n}\n\nexport interface BuildSpawnArgsResult {\n binaryName: string;\n args: string[];\n}\n\nfunction resolveCliConfig(agentId: string) {\n const resolved = resolveConfig(agentId);\n\n if (!resolved.spawnConfig) {\n throw new Error(`Agent \"${resolved.agentId}\" has no spawn config.`);\n }\n\n if (resolved.spawnConfig.kind !== \"cli\") {\n throw new Error(`Agent \"${resolved.agentId}\" does not support CLI spawn.`);\n }\n\n if (!resolved.binaryName) {\n throw new Error(`Agent \"${resolved.agentId}\" has no binaryName.`);\n }\n\n return {\n agentId: resolved.agentId,\n binaryName: resolved.binaryName,\n spawnConfig: resolved.spawnConfig\n };\n}\n\nfunction buildCliArgs(\n config: CliSpawnConfig,\n options: BuildSpawnArgsOptions,\n stdinMode?: StdinMode\n): string[] {\n const mcpArgs = getMcpArgs(config, options.mcpServers);\n\n const args: string[] = [];\n\n if (config.mcpArgsBeforeCommand) {\n args.push(...mcpArgs);\n }\n\n if (stdinMode) {\n args.push(\n config.promptFlag,\n ...(stdinMode.omitPrompt ? [] : [options.prompt]),\n ...stdinMode.extraArgs\n );\n } else {\n args.push(config.promptFlag, options.prompt);\n }\n\n if (options.model && config.modelFlag) {\n let model = config.modelStripProviderPrefix\n ? stripModelNamespace(options.model)\n : options.model;\n if (config.modelTransform) model = config.modelTransform(model);\n args.push(config.modelFlag, model);\n }\n\n args.push(...config.defaultArgs);\n\n if (!config.mcpArgsBeforeCommand) {\n args.push(...mcpArgs);\n }\n\n args.push(...config.modes[options.mode ?? \"yolo\"]);\n\n if (options.args && options.args.length > 0) {\n args.push(...options.args);\n }\n\n return args;\n}\n\nexport function buildSpawnArgs(\n agentId: string,\n options: BuildSpawnArgsOptions\n): BuildSpawnArgsResult {\n const { binaryName, spawnConfig } = resolveCliConfig(agentId);\n const stdinMode =\n options.useStdin && spawnConfig.stdinMode ? spawnConfig.stdinMode : undefined;\n return { binaryName, args: buildCliArgs(spawnConfig, options, stdinMode) };\n}\n\nexport async function spawn(\n agentId: string,\n options: SpawnOptions,\n context?: SpawnContext\n): Promise<SpawnResult> {\n if (options.signal?.aborted) {\n throw createAbortError();\n }\n\n const { agentId: resolvedId, binaryName, spawnConfig } = resolveCliConfig(agentId);\n\n const stdinMode =\n options.useStdin && spawnConfig.stdinMode ? spawnConfig.stdinMode : undefined;\n\n const spawnArgs = buildCliArgs(spawnConfig, options, stdinMode);\n\n if (context?.dryRun) {\n const rendered = [binaryName, ...spawnArgs].join(\" \");\n context.logger?.dryRun(rendered);\n return { stdout: \"\", stderr: \"\", exitCode: 0 };\n }\n\n const child = spawnChildProcess(binaryName, spawnArgs, {\n cwd: options.cwd,\n stdio: [stdinMode ? \"pipe\" : \"inherit\", \"pipe\", \"pipe\"]\n });\n\n if (!child.stdout || !child.stderr) {\n throw new Error(`Failed to spawn \"${resolvedId}\": missing stdio pipes.`);\n }\n\n const stdoutStream = child.stdout;\n const stderrStream = child.stderr;\n\n if (stdinMode) {\n if (!child.stdin) {\n throw new Error(`Failed to spawn \"${resolvedId}\": missing stdin pipe.`);\n }\n child.stdin.setDefaultEncoding(\"utf8\");\n child.stdin.write(options.prompt);\n child.stdin.end();\n }\n\n return new Promise<SpawnResult>((resolve, reject) => {\n let stdout = \"\";\n let stderr = \"\";\n let aborted = false;\n let timedOut = false;\n\n const onAbort = () => {\n aborted = true;\n child.kill(\"SIGTERM\");\n };\n\n options.signal?.addEventListener(\"abort\", onAbort, { once: true });\n\n let activityTimer: ReturnType<typeof setTimeout> | undefined;\n const resetActivityTimer = options.activityTimeoutMs\n ? () => {\n if (activityTimer) clearTimeout(activityTimer);\n activityTimer = setTimeout(() => {\n timedOut = true;\n child.kill(\"SIGTERM\");\n }, options.activityTimeoutMs);\n }\n : undefined;\n\n resetActivityTimer?.();\n\n const cleanup = () => {\n options.signal?.removeEventListener(\"abort\", onAbort);\n if (activityTimer) clearTimeout(activityTimer);\n };\n\n stdoutStream.setEncoding(\"utf8\");\n stdoutStream.on(\"data\", (chunk) => {\n stdout += chunk;\n resetActivityTimer?.();\n if (options.tee?.stdout) options.tee.stdout.write(chunk);\n });\n\n stderrStream.setEncoding(\"utf8\");\n stderrStream.on(\"data\", (chunk) => {\n stderr += chunk;\n resetActivityTimer?.();\n if (options.tee?.stderr) options.tee.stderr.write(chunk);\n });\n\n child.on(\"error\", (error) => {\n cleanup();\n if (aborted) {\n reject(createAbortError());\n return;\n }\n reject(error);\n });\n\n child.on(\"close\", (code) => {\n cleanup();\n if (aborted) {\n reject(createAbortError());\n return;\n }\n if (timedOut) {\n reject(createActivityTimeoutError(options.activityTimeoutMs!));\n return;\n }\n resolve({\n stdout,\n stderr,\n exitCode: code ?? 1\n });\n });\n });\n}\n", "import { allAgents, resolveAgentId } from \"@poe-code/agent-defs\";\nimport type { SpawnConfig } from \"../types.js\";\nimport { getSpawnConfig } from \"./index.js\";\n\nexport interface ResolvedSpawnConfig {\n agentId: string;\n binaryName?: string;\n spawnConfig?: SpawnConfig;\n}\n\nexport function resolveConfig(agentId: string): ResolvedSpawnConfig {\n const resolvedAgentId = resolveAgentId(agentId);\n if (!resolvedAgentId) {\n throw new Error(`Unknown agent \"${agentId}\".`);\n }\n\n const agentDefinition = allAgents.find((agent) => agent.id === resolvedAgentId);\n if (!agentDefinition) {\n throw new Error(`Unknown agent \"${agentId}\".`);\n }\n\n const spawnConfig = getSpawnConfig(resolvedAgentId);\n const binaryName = agentDefinition.binaryName;\n\n return { agentId: resolvedAgentId, binaryName, spawnConfig };\n}\n", "import { listMcpSupportedAgents } from \"./configs/index.js\";\nimport type { CliSpawnConfig, McpSpawnConfig } from \"./types.js\";\n\nexport function hasMcpServers(servers?: McpSpawnConfig): servers is McpSpawnConfig {\n if (!servers) {\n return false;\n }\n return Object.keys(servers).length > 0;\n}\n\nexport function getMcpArgs(\n config: CliSpawnConfig,\n servers?: McpSpawnConfig\n): string[] {\n if (!hasMcpServers(servers)) {\n return [];\n }\n if (!config.mcpArgs && !config.mcpEnv) {\n throw new Error(formatUnsupportedMcpSpawnMessage(config.agentId));\n }\n if (!config.mcpArgs) {\n return [];\n }\n return config.mcpArgs(servers);\n}\n\nexport function getMcpEnv(\n config: CliSpawnConfig,\n servers?: McpSpawnConfig\n): Record<string, string> {\n if (!hasMcpServers(servers) || !config.mcpEnv) {\n return {};\n }\n return config.mcpEnv(servers);\n}\n\nexport function formatUnsupportedMcpSpawnMessage(agentId: string): string {\n const supported = listMcpSupportedAgents();\n const supportedText = supported.length > 0 ? supported.join(\", \") : \"(none)\";\n return (\n `Agent \"${agentId}\" does not support MCP servers at spawn time.\\n` +\n `Agents with spawn-time MCP support: ${supportedText}`\n );\n}\n", "/**\n * Strips the provider namespace prefix from a model identifier.\n * e.g., \"anthropic/claude-opus-4.6\" \u2192 \"claude-opus-4.6\"\n *\n * CLI binaries (claude, codex, opencode, kimi) only accept bare model IDs.\n * Passing a namespaced model like \"anthropic/claude-opus-4.6\" causes the\n * binary to fail with \"model not found\". This function MUST be called\n * before passing any model to a CLI binary via spawn args.\n */\nexport function stripModelNamespace(model: string): string {\n const slashIndex = model.indexOf(\"/\");\n return slashIndex === -1 ? model : model.slice(slashIndex + 1);\n}\n", "import { spawn as spawnChildProcess } from \"node:child_process\";\nimport { resolveConfig } from \"./configs/resolve-config.js\";\nimport { getMcpArgs } from \"./mcp-args.js\";\nimport { stripModelNamespace } from \"./model-utils.js\";\nimport type { SpawnOptions, SpawnResult } from \"./types.js\";\n\nexport async function spawnInteractive(\n agentId: string,\n options: SpawnOptions\n): Promise<SpawnResult> {\n const resolved = resolveConfig(agentId);\n const spawnConfig = resolved.spawnConfig;\n\n if (!spawnConfig) {\n throw new Error(`Agent \"${resolved.agentId}\" has no spawn config.`);\n }\n\n if (spawnConfig.kind !== \"cli\") {\n throw new Error(`Agent \"${resolved.agentId}\" does not support CLI spawn.`);\n }\n\n if (!resolved.binaryName) {\n throw new Error(`Agent \"${resolved.agentId}\" has no binaryName.`);\n }\n\n if (!spawnConfig.interactive) {\n throw new Error(`Agent \"${resolved.agentId}\" does not support interactive mode.`);\n }\n\n const { interactive } = spawnConfig;\n\n const args: string[] = [];\n\n if (options.prompt) {\n if (interactive.promptFlag) {\n args.push(interactive.promptFlag, options.prompt);\n } else {\n args.push(options.prompt);\n }\n }\n\n if (options.model && spawnConfig.modelFlag) {\n let model = spawnConfig.modelStripProviderPrefix\n ? stripModelNamespace(options.model)\n : options.model;\n if (spawnConfig.modelTransform) model = spawnConfig.modelTransform(model);\n args.push(spawnConfig.modelFlag, model);\n }\n\n args.push(...interactive.defaultArgs);\n args.push(...getMcpArgs(spawnConfig, options.mcpServers));\n\n const mode = options.mode ?? \"yolo\";\n args.push(...spawnConfig.modes[mode]);\n\n if (options.args && options.args.length > 0) {\n args.push(...options.args);\n }\n\n const child = spawnChildProcess(resolved.binaryName, args, {\n cwd: options.cwd,\n stdio: \"inherit\"\n });\n\n return new Promise<SpawnResult>((resolve, reject) => {\n child.on(\"error\", (error) => {\n reject(error);\n });\n\n child.on(\"close\", (code) => {\n resolve({\n stdout: \"\",\n stderr: \"\",\n exitCode: code ?? 1\n });\n });\n });\n}\n", "import chalk from \"chalk\";\n\nexport const brand = \"#a200ff\";\n\nexport const dark = {\n header: (text: string) => chalk.magentaBright.bold(text),\n divider: (text: string) => chalk.dim(text),\n prompt: (text: string) => chalk.cyan(text),\n number: (text: string) => chalk.cyanBright(text),\n intro: (text: string) => chalk.bgMagenta.white(` Poe - ${text} `),\n resolvedSymbol: chalk.magenta(\"\u25C7\"),\n errorSymbol: chalk.red(\"\u25A0\"),\n accent: (text: string) => chalk.cyan(text),\n muted: (text: string) => chalk.dim(text),\n success: (text: string) => chalk.green(text),\n warning: (text: string) => chalk.yellow(text),\n error: (text: string) => chalk.red(text),\n info: (text: string) => chalk.magenta(text),\n badge: (text: string) => chalk.bgYellow.black(` ${text} `)\n};\n\nexport const light = {\n header: (text: string) => chalk.hex(\"#a200ff\").bold(text),\n divider: (text: string) => chalk.hex(\"#666666\")(text),\n prompt: (text: string) => chalk.hex(\"#006699\").bold(text),\n number: (text: string) => chalk.hex(\"#0077cc\").bold(text),\n intro: (text: string) => chalk.bgHex(\"#a200ff\").white(` Poe - ${text} `),\n resolvedSymbol: chalk.hex(\"#a200ff\")(\"\u25C7\"),\n errorSymbol: chalk.hex(\"#cc0000\")(\"\u25A0\"),\n accent: (text: string) => chalk.hex(\"#006699\").bold(text),\n muted: (text: string) => chalk.hex(\"#666666\")(text),\n success: (text: string) => chalk.hex(\"#008800\")(text),\n warning: (text: string) => chalk.hex(\"#cc6600\")(text),\n error: (text: string) => chalk.hex(\"#cc0000\")(text),\n info: (text: string) => chalk.hex(\"#a200ff\")(text),\n badge: (text: string) => chalk.bgHex(\"#cc6600\").white(` ${text} `)\n};\n\nexport type ThemeName = \"dark\" | \"light\";\nexport type ThemePalette = typeof dark;\n", "import chalk from \"chalk\";\n\nexport const typography = {\n bold: (text: string) => chalk.bold(text),\n dim: (text: string) => chalk.dim(text),\n italic: (text: string) => chalk.italic(text),\n underline: (text: string) => chalk.underline(text),\n strikethrough: (text: string) => chalk.strikethrough(text)\n} as const;\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { getTheme } from \"../internal/theme-detect.js\";\nimport { typography } from \"../tokens/typography.js\";\n\nexport const text = {\n intro(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `**${content}**`;\n return getTheme().intro(content);\n },\n heading(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `## ${content}`;\n return getTheme().header(content);\n },\n section(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `**${content}**`;\n return typography.bold(content);\n },\n command(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `\\`${content}\\``;\n return getTheme().accent(content);\n },\n argument(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `<${content}>`;\n return getTheme().muted(content);\n },\n option(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `\\`${content}\\``;\n return chalk.yellow(content);\n },\n example(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `\\`${content}\\``;\n return getTheme().muted(content);\n },\n usageCommand(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `\\`${content}\\``;\n return chalk.green(content);\n },\n link(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `[${content}](${content})`;\n return getTheme().accent(content);\n },\n muted(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `*${content}*`;\n return getTheme().muted(content);\n },\n badge(content: string): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return content;\n if (format === \"markdown\") return `[${content}]`;\n return getTheme().badge(content);\n },\n selectLabel(label: string, detail?: string): string {\n if (!detail) {\n return label;\n }\n const format = resolveOutputFormat();\n if (format !== \"terminal\") {\n return `${label} \u2014 ${detail}`;\n }\n return `${label} ${typography.dim(\"\u2014\")} ${typography.dim(detail)}`;\n }\n} as const;\n", "import { AsyncLocalStorage } from \"node:async_hooks\";\n\nexport type OutputFormat = \"terminal\" | \"markdown\" | \"json\";\n\nconst VALID_FORMATS = new Set<OutputFormat>([\"terminal\", \"markdown\", \"json\"]);\nconst formatStorage = new AsyncLocalStorage<OutputFormat>();\n\nlet cached: OutputFormat | undefined;\n\nexport function resolveOutputFormat(\n env: { OUTPUT_FORMAT?: string } = process.env as { OUTPUT_FORMAT?: string }\n): OutputFormat {\n const scoped = formatStorage.getStore();\n if (scoped) {\n return scoped;\n }\n if (cached) {\n return cached;\n }\n const raw = env.OUTPUT_FORMAT?.toLowerCase();\n cached = VALID_FORMATS.has(raw as OutputFormat) ? (raw as OutputFormat) : \"terminal\";\n return cached;\n}\n\nexport function withOutputFormat<T>(format: OutputFormat, fn: () => T): T {\n return formatStorage.run(format, fn);\n}\n\nexport function resetOutputFormatCache(): void {\n cached = undefined;\n}\n", "import { dark, light, type ThemeName, type ThemePalette } from \"../tokens/colors.js\";\n\nexport interface ThemeEnv {\n POE_CODE_THEME?: string;\n POE_THEME?: string;\n APPLE_INTERFACE_STYLE?: string;\n VSCODE_COLOR_THEME_KIND?: string;\n COLORFGBG?: string;\n}\n\nfunction detectThemeFromEnv(env: ThemeEnv): ThemeName | undefined {\n const apple = env.APPLE_INTERFACE_STYLE;\n if (typeof apple === \"string\") {\n return apple.toLowerCase() === \"dark\" ? \"dark\" : \"light\";\n }\n\n const vscodeKind = env.VSCODE_COLOR_THEME_KIND;\n if (typeof vscodeKind === \"string\") {\n const normalized = vscodeKind.toLowerCase();\n if (normalized.includes(\"light\")) {\n return \"light\";\n }\n if (normalized.includes(\"dark\")) {\n return \"dark\";\n }\n }\n\n const colorFGBG = env.COLORFGBG;\n if (typeof colorFGBG === \"string\") {\n const parts = colorFGBG.split(\";\").map((part) => Number.parseInt(part, 10));\n const background = parts.at(-1);\n if (Number.isFinite(background)) {\n return background! >= 8 ? \"light\" : \"dark\";\n }\n }\n\n return undefined;\n}\n\nexport function resolveThemeName(env: ThemeEnv = process.env as ThemeEnv): ThemeName {\n const raw = (env.POE_CODE_THEME ?? env.POE_THEME)?.toLowerCase();\n if (raw === \"light\" || raw === \"dark\") {\n return raw;\n }\n const detected = detectThemeFromEnv(env);\n if (detected) {\n return detected;\n }\n return \"dark\";\n}\n\nlet cachedTheme: ThemePalette | undefined;\n\nexport function getTheme(env?: ThemeEnv): ThemePalette {\n if (cachedTheme) {\n return cachedTheme;\n }\n const themeName = resolveThemeName(env);\n cachedTheme = themeName === \"light\" ? light : dark;\n return cachedTheme;\n}\n\nexport function resetThemeCache(): void {\n cachedTheme = undefined;\n}\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { getTheme } from \"../internal/theme-detect.js\";\n\nexport const symbols = {\n get info(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"info\";\n if (format === \"markdown\") return \"(i)\";\n return chalk.magenta(\"\u25CF\");\n },\n get success(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"success\";\n if (format === \"markdown\") return \"[ok]\";\n return chalk.magenta(\"\u25C6\");\n },\n get resolved(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"resolved\";\n if (format === \"markdown\") return \">\";\n return getTheme().resolvedSymbol;\n },\n get errorResolved(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"error\";\n if (format === \"markdown\") return \"[!]\";\n return getTheme().errorSymbol;\n },\n get bar(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"\";\n if (format === \"markdown\") return \"|\";\n return \"\u2502\";\n },\n cornerTopRight: \"\u256E\",\n cornerBottomRight: \"\u256F\",\n get warning(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"warning\";\n if (format === \"markdown\") return \"[!]\";\n return \"\u25B2\";\n },\n get active(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"active\";\n if (format === \"markdown\") return \"[x]\";\n return \"\u25C6\";\n },\n get inactive(): string {\n const format = resolveOutputFormat();\n if (format === \"json\") return \"inactive\";\n if (format === \"markdown\") return \"[ ]\";\n return \"\u25CB\";\n }\n} as const;\n", "import chalk from \"chalk\";\nimport { log } from \"../prompts/primitives/log.js\";\nimport { symbols } from \"./symbols.js\";\n\nexport interface LoggerOutput {\n info(message: string): void;\n success(message: string): void;\n warn(message: string): void;\n error(message: string): void;\n resolved(label: string, value: string): void;\n errorResolved(label: string, value: string): void;\n message(message: string, symbol?: string): void;\n}\n\nexport function createLogger(emitter?: (message: string) => void): LoggerOutput {\n const emit = (\n level: \"info\" | \"success\" | \"warn\" | \"error\",\n message: string\n ): void => {\n if (emitter) {\n emitter(message);\n return;\n }\n if (level === \"success\") {\n log.success(message);\n return;\n }\n if (level === \"warn\") {\n log.warn(message);\n return;\n }\n if (level === \"error\") {\n log.error(message);\n return;\n }\n log.info(message);\n };\n\n return {\n info(message: string): void {\n emit(\"info\", message);\n },\n success(message: string): void {\n emit(\"success\", message);\n },\n warn(message: string): void {\n emit(\"warn\", message);\n },\n error(message: string): void {\n emit(\"error\", message);\n },\n resolved(label: string, value: string): void {\n if (emitter) {\n emitter(`${label}: ${value}`);\n return;\n }\n log.message(`${label}\\n ${value}`, { symbol: symbols.resolved });\n },\n errorResolved(label: string, value: string): void {\n if (emitter) {\n emitter(`${label}: ${value}`);\n return;\n }\n log.message(`${label}\\n ${value}`, { symbol: symbols.errorResolved });\n },\n message(message: string, symbol?: string): void {\n if (emitter) {\n emitter(message);\n return;\n }\n log.message(message, { symbol: symbol ?? chalk.gray(\"\u2502\") });\n }\n };\n}\n\nexport const logger = createLogger();\n", "import chalk from \"chalk\";\nimport { symbols } from \"../../components/symbols.js\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\nimport { stripAnsi } from \"../../internal/strip-ansi.js\";\n\nexport interface LogMessageOptions {\n symbol?: string;\n secondarySymbol?: string;\n spacing?: number;\n withGuide?: boolean;\n}\n\nfunction writeTerminalMessage(\n msg: string,\n {\n symbol = chalk.gray(\"\u2502\"),\n secondarySymbol = chalk.gray(\"\u2502\"),\n spacing = 1,\n withGuide = true\n }: LogMessageOptions = {}\n): void {\n const lines: string[] = [];\n const showGuide = withGuide !== false;\n const contentLines = msg.split(\"\\n\");\n const prefix = showGuide ? `${symbol} ` : \"\";\n const continuationPrefix = showGuide ? `${secondarySymbol} ` : \"\";\n const emptyGuide = showGuide ? secondarySymbol : \"\";\n\n for (let index = 0; index < spacing; index += 1) {\n lines.push(emptyGuide);\n }\n\n if (contentLines.length === 0) {\n process.stdout.write(\"\\n\");\n return;\n }\n\n const [firstLine = \"\", ...continuationLines] = contentLines;\n if (firstLine.length > 0) {\n lines.push(`${prefix}${firstLine}`);\n } else {\n lines.push(showGuide ? symbol : \"\");\n }\n\n for (const line of continuationLines) {\n if (line.length > 0) {\n lines.push(`${continuationPrefix}${line}`);\n continue;\n }\n lines.push(emptyGuide);\n }\n\n process.stdout.write(`${lines.join(\"\\n\")}\\n`);\n}\n\nexport function message(msg: string, options?: LogMessageOptions): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`- ${stripAnsi(msg)}\\n`);\n return;\n }\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ level: \"message\", message: stripAnsi(msg) })}\\n`\n );\n return;\n }\n\n writeTerminalMessage(msg, options);\n}\n\nexport function info(msg: string): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`- **info:** ${stripAnsi(msg)}\\n`);\n return;\n }\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ level: \"info\", message: stripAnsi(msg) })}\\n`\n );\n return;\n }\n\n message(msg, { symbol: symbols.info });\n}\n\nexport function success(msg: string): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`- **success:** ${stripAnsi(msg)}\\n`);\n return;\n }\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ level: \"success\", message: stripAnsi(msg) })}\\n`\n );\n return;\n }\n\n message(msg, { symbol: symbols.success });\n}\n\nexport function warn(msg: string): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`- **warning:** ${stripAnsi(msg)}\\n`);\n return;\n }\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ level: \"warn\", message: stripAnsi(msg) })}\\n`\n );\n return;\n }\n\n message(msg, { symbol: chalk.yellow(\"\u25B2\") });\n}\n\nexport function error(msg: string): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`- **error:** ${stripAnsi(msg)}\\n`);\n return;\n }\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ level: \"error\", message: stripAnsi(msg) })}\\n`\n );\n return;\n }\n\n message(msg, { symbol: chalk.red(\"\u25A0\") });\n}\n\nexport const log = {\n info,\n success,\n message,\n warn,\n error\n};\n", "export function stripAnsi(value: string): string {\n return value.replace(/\\u001b\\[[0-9;]*m/g, \"\");\n}\n", "import { Table } from \"console-table-printer\";\nimport type { ThemePalette } from \"../tokens/colors.js\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { stripAnsi } from \"../internal/strip-ansi.js\";\n\nexport interface TableColumn {\n name: string;\n title: string;\n alignment: \"left\" | \"right\";\n maxLen: number;\n}\n\nexport interface RenderTableOptions {\n theme: ThemePalette;\n columns: TableColumn[];\n rows: Record<string, string>[];\n}\n\nfunction renderTableTerminal(options: RenderTableOptions): string {\n const { theme, columns, rows } = options;\n\n const table = new Table({\n style: {\n headerTop: {\n left: theme.muted(\"\u250C\"),\n mid: theme.muted(\"\u252C\"),\n right: theme.muted(\"\u2510\"),\n other: theme.muted(\"\u2500\")\n },\n headerBottom: {\n left: theme.muted(\"\u251C\"),\n mid: theme.muted(\"\u253C\"),\n right: theme.muted(\"\u2524\"),\n other: theme.muted(\"\u2500\")\n },\n tableBottom: {\n left: theme.muted(\"\u2514\"),\n mid: theme.muted(\"\u2534\"),\n right: theme.muted(\"\u2518\"),\n other: theme.muted(\"\u2500\")\n },\n vertical: theme.muted(\"\u2502\"),\n rowSeparator: {\n left: theme.muted(\"\u251C\"),\n mid: theme.muted(\"\u253C\"),\n right: theme.muted(\"\u2524\"),\n other: theme.muted(\"\u2500\")\n }\n },\n columns: columns.map((col) => ({\n name: col.name,\n title: theme.header(col.title),\n alignment: col.alignment,\n maxLen: col.maxLen\n }))\n });\n\n for (const row of rows) {\n table.addRow(row);\n }\n\n return table.render();\n}\n\nfunction renderTableMarkdown(options: RenderTableOptions): string {\n const { columns, rows } = options;\n\n const header = `| ${columns.map((c) => c.title).join(\" | \")} |`;\n const separator = `| ${columns.map((c) => (c.alignment === \"right\" ? \"---:\" : \":---\")).join(\" | \")} |`;\n\n const dataRows = rows.map(\n (row) =>\n `| ${columns.map((c) => stripAnsi(row[c.name] ?? \"\").replace(/\\|/g, \"\\\\|\")).join(\" | \")} |`\n );\n\n return [header, separator, ...dataRows].join(\"\\n\");\n}\n\nfunction renderTableJson(options: RenderTableOptions): string {\n const { columns, rows } = options;\n\n const cleaned = rows.map((row) => {\n const obj: Record<string, string> = {};\n for (const col of columns) {\n obj[col.name] = stripAnsi(row[col.name] ?? \"\");\n }\n return obj;\n });\n\n return JSON.stringify(cleaned, null, 2);\n}\n\nexport function renderTable(options: RenderTableOptions): string {\n const format = resolveOutputFormat();\n switch (format) {\n case \"markdown\":\n return renderTableMarkdown(options);\n case \"json\":\n return renderTableJson(options);\n default:\n return renderTableTerminal(options);\n }\n}\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { renderMarkdown } from \"../terminal-markdown/index.js\";\n\nfunction truncate(text: string, maxLength: number): string {\n if (text.length <= maxLength) return text;\n if (maxLength <= 3) return text.slice(0, maxLength);\n return `${text.slice(0, maxLength - 3)}...`;\n}\n\nconst KIND_COLORS: Record<string, (text: string) => string> = {\n exec: (text) => chalk.yellow(text),\n edit: (text) => chalk.magenta(text),\n read: (text) => chalk.cyan(text),\n search: (text) => chalk.blue(text),\n think: (text) => chalk.dim(text),\n other: (text) => chalk.dim(text)\n};\n\nfunction colorForKind(kind: string): (text: string) => string {\n return KIND_COLORS[kind] ?? ((text) => chalk.dim(text));\n}\n\nfunction writeLine(line: string): void {\n process.stdout.write(`${line}\\n`);\n}\n\nconst AGENT_PREFIX = `${chalk.green.bold(\"\u2713\")} agent: `;\n\nfunction formatCost(costUsd: number): string {\n return new Intl.NumberFormat(\"en-US\", {\n style: \"currency\",\n currency: \"USD\",\n minimumFractionDigits: 2,\n maximumFractionDigits: 6\n }).format(costUsd);\n}\n\nexport function renderAgentMessage(text: string): void {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n writeLine(`- **agent:** ${text}`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(JSON.stringify({ event: \"agent_message\", text }));\n return;\n }\n\n const rendered = renderMarkdown(text).trimEnd();\n writeLine(`${AGENT_PREFIX}${rendered}`);\n}\n\nexport function renderToolStart(kind: string, title: string): void {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n writeLine(`- *\u2192 ${kind}: ${title}*`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(JSON.stringify({ event: \"tool_start\", kind, title }));\n return;\n }\n\n const color = colorForKind(kind);\n writeLine(color(` \u2192 ${kind}: ${title}`));\n}\n\nexport function renderToolComplete(kind: string): void {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n writeLine(`- *\u2713 ${kind}*`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(JSON.stringify({ event: \"tool_complete\", kind }));\n return;\n }\n\n const color = colorForKind(kind);\n writeLine(color(` \u2713 ${kind}`));\n}\n\nexport function renderReasoning(text: string): void {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n writeLine(`- *thinking:* ${truncate(text, 80)}`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(JSON.stringify({ event: \"reasoning\", text }));\n return;\n }\n\n writeLine(chalk.dim(` \u2713 ${truncate(text, 80)}`));\n}\n\nexport function renderUsage(tokens: {\n input: number;\n output: number;\n cached?: number;\n costUsd?: number;\n}): void {\n const format = resolveOutputFormat();\n const cached =\n typeof tokens.cached === \"number\" && tokens.cached > 0 ? ` (${tokens.cached} cached)` : \"\";\n\n let cost = \"\";\n if (typeof tokens.costUsd === \"number\") {\n cost = ` (${formatCost(tokens.costUsd)})`;\n }\n\n if (format === \"markdown\") {\n writeLine(`- **tokens:** ${tokens.input} in \u2192 ${tokens.output} out${cost}`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(\n JSON.stringify({\n event: \"usage\",\n inputTokens: tokens.input,\n outputTokens: tokens.output,\n cachedTokens: tokens.cached ?? 0,\n costUsd: tokens.costUsd ?? 0\n })\n );\n return;\n }\n\n process.stdout.write(\"\\n\");\n writeLine(chalk.green(`\u2713 tokens: ${tokens.input} in${cached} \u2192 ${tokens.output} out${cost}`));\n}\n\nexport function renderError(message: string): void {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n writeLine(`- **error:** ${message}`);\n return;\n }\n\n if (format === \"json\") {\n writeLine(JSON.stringify({ event: \"error\", message }));\n return;\n }\n\n writeLine(chalk.red(`\u2717 ${message}`));\n}\n", "import chalk from \"chalk\";\nimport * as clack from \"@clack/prompts\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { stripAnsi } from \"../internal/strip-ansi.js\";\nimport { cancel, isCancel } from \"./primitives/cancel.js\";\nimport { intro } from \"./primitives/intro.js\";\nimport { log } from \"./primitives/log.js\";\nimport { note } from \"./primitives/note.js\";\nimport { outro } from \"./primitives/outro.js\";\nimport { spinner } from \"./primitives/spinner.js\";\n\nexport { isCancel, cancel, log };\nexport { intro, outro, note, spinner };\n\nexport function introPlain(title: string): void {\n const format = resolveOutputFormat();\n if (format === \"markdown\") {\n process.stdout.write(`# ${stripAnsi(title)}\\n\\n`);\n return;\n }\n if (format === \"json\") {\n return;\n }\n process.stdout.write(`${chalk.gray(\"\u250C\")} ${title}\\n`);\n}\n\nexport interface SelectOptions<Value> {\n message: string;\n options: Array<{ value: Value; label: string; hint?: string }>;\n initialValue?: Value;\n}\n\nexport async function select<Value>(\n opts: SelectOptions<Value>\n): Promise<Value | symbol> {\n return clack.select(opts as Parameters<typeof clack.select<Value>>[0]);\n}\n\nexport type MultiselectOptions<Value> = Parameters<typeof clack.multiselect<Value>>[0];\n\n/**\n * Prompts the user to select one or more values from a list.\n *\n * Returns the selected values as an array, or a cancellation symbol if the\n * user cancels. Use `isCancel` to check for cancellation.\n *\n * @example\n * const result = await multiselect({\n * message: \"Pick workflows to run\",\n * options: [{ label: \"Fix Vulnerabilities\", value: \"fix-vulnerabilities\" }],\n * required: true\n * });\n * if (!isCancel(result)) {\n * // result is Value[]\n * }\n */\nexport async function multiselect<Value>(\n opts: MultiselectOptions<Value>\n): Promise<Value[] | symbol> {\n return clack.multiselect(opts);\n}\n\nexport type TextOptions = Parameters<typeof clack.text>[0];\n\nexport async function text(opts: TextOptions): Promise<string | symbol> {\n return clack.text(opts as Parameters<typeof clack.text>[0]);\n}\n\nexport interface ConfirmOptions {\n message: string;\n initialValue?: boolean;\n}\n\nexport async function confirm(opts: ConfirmOptions): Promise<boolean | symbol> {\n return clack.confirm(opts as Parameters<typeof clack.confirm>[0]);\n}\n\nexport class PromptCancelledError extends Error {\n constructor(message = \"Operation cancelled.\") {\n super(message);\n this.name = \"PromptCancelledError\";\n if (Error.captureStackTrace) {\n Error.captureStackTrace(this, this.constructor);\n }\n }\n}\n\nexport async function confirmOrCancel(opts: ConfirmOptions): Promise<boolean> {\n const result = await confirm(opts);\n if (isCancel(result)) {\n cancel(\"Operation cancelled.\");\n throw new PromptCancelledError();\n }\n return result === true;\n}\n\nexport interface PasswordOptions {\n message: string;\n validate?: (value: string) => string | undefined;\n}\n\nexport async function password(opts: PasswordOptions): Promise<string | symbol> {\n return clack.password(opts as Parameters<typeof clack.password>[0]);\n}\n\nexport type SpinnerOptions = {\n start: (message?: string) => void;\n stop: (message?: string, code?: number) => void;\n message: (message?: string) => void;\n};\n\nexport interface WithSpinnerOptions<T> {\n message: string;\n fn: () => Promise<T>;\n stopMessage?: (result: T) => string;\n subtext?: (result: T) => string | undefined;\n}\n\nfunction formatElapsed(ms: number): string {\n const totalSeconds = Math.floor(ms / 1000);\n if (totalSeconds < 60) {\n return `${totalSeconds}s`;\n }\n const minutes = Math.floor(totalSeconds / 60);\n const seconds = totalSeconds % 60;\n return `${minutes}m ${seconds}s`;\n}\n\nexport async function withSpinner<T>(options: WithSpinnerOptions<T>): Promise<T> {\n const { message, fn, stopMessage, subtext } = options;\n\n if (resolveOutputFormat() === \"json\") {\n const result = await fn();\n const sub = subtext ? subtext(result) : undefined;\n if (sub) {\n process.stdout.write(sub + \"\\n\");\n }\n return result;\n }\n\n const noSpinner = process.env.POE_NO_SPINNER === \"1\";\n const isTTY = process.stdout.isTTY;\n\n if (noSpinner || !isTTY) {\n const result = await fn();\n const msg = stopMessage ? stopMessage(result) : undefined;\n if (msg) {\n process.stdout.write(`\\x1b[32m\u25C6\\x1b[0m ${msg}\\n`);\n }\n const sub = subtext ? subtext(result) : undefined;\n if (sub) {\n for (const line of sub.split(\"\\n\")) {\n process.stdout.write(`\\x1b[90m\u2502\\x1b[0m ${line}\\n`);\n }\n }\n return result;\n }\n\n const s = spinner();\n const start = Date.now();\n s.start(message);\n\n const timer = setInterval(() => {\n s.message(`${message} [${formatElapsed(Date.now() - start)}]`);\n }, 1000);\n\n try {\n const result = await fn();\n clearInterval(timer);\n const elapsed = formatElapsed(Date.now() - start);\n const msg = stopMessage ? stopMessage(result) : undefined;\n s.stop(msg ? `${msg} [${elapsed}]` : `Done [${elapsed}]`);\n\n const sub = subtext ? subtext(result) : undefined;\n if (sub) {\n for (const line of sub.split(\"\\n\")) {\n process.stdout.write(`\\x1b[90m\u2502\\x1b[0m ${line}\\n`);\n }\n }\n\n return result;\n } catch (error) {\n clearInterval(timer);\n s.stop(\"\", 1);\n throw error;\n }\n}\n", "import chalk from \"chalk\";\nexport { isCancel } from \"@clack/core\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\n\nexport function cancel(msg = \"\"): void {\n if (resolveOutputFormat() !== \"terminal\") {\n return;\n }\n\n process.stdout.write(`${chalk.gray(\"\u2514\")} ${chalk.red(msg)}\\n\\n`);\n}\n", "/* MAIN */\n//URL: https://github.com/sindresorhus/get-east-asian-width/blob/main/lookup.js\n//LICENSE: https://github.com/sindresorhus/get-east-asian-width/blob/main/license\n//TODO: Replace these with some unicode property classes, if the ones we need exist\nconst isAmbiguous = (x) => {\n return x === 0xA1 || x === 0xA4 || x === 0xA7 || x === 0xA8 || x === 0xAA || x === 0xAD || x === 0xAE || x >= 0xB0 && x <= 0xB4 || x >= 0xB6 && x <= 0xBA || x >= 0xBC && x <= 0xBF || x === 0xC6 || x === 0xD0 || x === 0xD7 || x === 0xD8 || x >= 0xDE && x <= 0xE1 || x === 0xE6 || x >= 0xE8 && x <= 0xEA || x === 0xEC || x === 0xED || x === 0xF0 || x === 0xF2 || x === 0xF3 || x >= 0xF7 && x <= 0xFA || x === 0xFC || x === 0xFE || x === 0x101 || x === 0x111 || x === 0x113 || x === 0x11B || x === 0x126 || x === 0x127 || x === 0x12B || x >= 0x131 && x <= 0x133 || x === 0x138 || x >= 0x13F && x <= 0x142 || x === 0x144 || x >= 0x148 && x <= 0x14B || x === 0x14D || x === 0x152 || x === 0x153 || x === 0x166 || x === 0x167 || x === 0x16B || x === 0x1CE || x === 0x1D0 || x === 0x1D2 || x === 0x1D4 || x === 0x1D6 || x === 0x1D8 || x === 0x1DA || x === 0x1DC || x === 0x251 || x === 0x261 || x === 0x2C4 || x === 0x2C7 || x >= 0x2C9 && x <= 0x2CB || x === 0x2CD || x === 0x2D0 || x >= 0x2D8 && x <= 0x2DB || x === 0x2DD || x === 0x2DF || x >= 0x300 && x <= 0x36F || x >= 0x391 && x <= 0x3A1 || x >= 0x3A3 && x <= 0x3A9 || x >= 0x3B1 && x <= 0x3C1 || x >= 0x3C3 && x <= 0x3C9 || x === 0x401 || x >= 0x410 && x <= 0x44F || x === 0x451 || x === 0x2010 || x >= 0x2013 && x <= 0x2016 || x === 0x2018 || x === 0x2019 || x === 0x201C || x === 0x201D || x >= 0x2020 && x <= 0x2022 || x >= 0x2024 && x <= 0x2027 || x === 0x2030 || x === 0x2032 || x === 0x2033 || x === 0x2035 || x === 0x203B || x === 0x203E || x === 0x2074 || x === 0x207F || x >= 0x2081 && x <= 0x2084 || x === 0x20AC || x === 0x2103 || x === 0x2105 || x === 0x2109 || x === 0x2113 || x === 0x2116 || x === 0x2121 || x === 0x2122 || x === 0x2126 || x === 0x212B || x === 0x2153 || x === 0x2154 || x >= 0x215B && x <= 0x215E || x >= 0x2160 && x <= 0x216B || x >= 0x2170 && x <= 0x2179 || x === 0x2189 || x >= 0x2190 && x <= 0x2199 || x === 0x21B8 || x === 0x21B9 || x === 0x21D2 || x === 0x21D4 || x === 0x21E7 || x === 0x2200 || x === 0x2202 || x === 0x2203 || x === 0x2207 || x === 0x2208 || x === 0x220B || x === 0x220F || x === 0x2211 || x === 0x2215 || x === 0x221A || x >= 0x221D && x <= 0x2220 || x === 0x2223 || x === 0x2225 || x >= 0x2227 && x <= 0x222C || x === 0x222E || x >= 0x2234 && x <= 0x2237 || x === 0x223C || x === 0x223D || x === 0x2248 || x === 0x224C || x === 0x2252 || x === 0x2260 || x === 0x2261 || x >= 0x2264 && x <= 0x2267 || x === 0x226A || x === 0x226B || x === 0x226E || x === 0x226F || x === 0x2282 || x === 0x2283 || x === 0x2286 || x === 0x2287 || x === 0x2295 || x === 0x2299 || x === 0x22A5 || x === 0x22BF || x === 0x2312 || x >= 0x2460 && x <= 0x24E9 || x >= 0x24EB && x <= 0x254B || x >= 0x2550 && x <= 0x2573 || x >= 0x2580 && x <= 0x258F || x >= 0x2592 && x <= 0x2595 || x === 0x25A0 || x === 0x25A1 || x >= 0x25A3 && x <= 0x25A9 || x === 0x25B2 || x === 0x25B3 || x === 0x25B6 || x === 0x25B7 || x === 0x25BC || x === 0x25BD || x === 0x25C0 || x === 0x25C1 || x >= 0x25C6 && x <= 0x25C8 || x === 0x25CB || x >= 0x25CE && x <= 0x25D1 || x >= 0x25E2 && x <= 0x25E5 || x === 0x25EF || x === 0x2605 || x === 0x2606 || x === 0x2609 || x === 0x260E || x === 0x260F || x === 0x261C || x === 0x261E || x === 0x2640 || x === 0x2642 || x === 0x2660 || x === 0x2661 || x >= 0x2663 && x <= 0x2665 || x >= 0x2667 && x <= 0x266A || x === 0x266C || x === 0x266D || x === 0x266F || x === 0x269E || x === 0x269F || x === 0x26BF || x >= 0x26C6 && x <= 0x26CD || x >= 0x26CF && x <= 0x26D3 || x >= 0x26D5 && x <= 0x26E1 || x === 0x26E3 || x === 0x26E8 || x === 0x26E9 || x >= 0x26EB && x <= 0x26F1 || x === 0x26F4 || x >= 0x26F6 && x <= 0x26F9 || x === 0x26FB || x === 0x26FC || x === 0x26FE || x === 0x26FF || x === 0x273D || x >= 0x2776 && x <= 0x277F || x >= 0x2B56 && x <= 0x2B59 || x >= 0x3248 && x <= 0x324F || x >= 0xE000 && x <= 0xF8FF || x >= 0xFE00 && x <= 0xFE0F || x === 0xFFFD || x >= 0x1F100 && x <= 0x1F10A || x >= 0x1F110 && x <= 0x1F12D || x >= 0x1F130 && x <= 0x1F169 || x >= 0x1F170 && x <= 0x1F18D || x === 0x1F18F || x === 0x1F190 || x >= 0x1F19B && x <= 0x1F1AC || x >= 0xE0100 && x <= 0xE01EF || x >= 0xF0000 && x <= 0xFFFFD || x >= 0x100000 && x <= 0x10FFFD;\n};\nconst isFullWidth = (x) => {\n return x === 0x3000 || x >= 0xFF01 && x <= 0xFF60 || x >= 0xFFE0 && x <= 0xFFE6;\n};\nconst isWide = (x) => {\n return x >= 0x1100 && x <= 0x115F || x === 0x231A || x === 0x231B || x === 0x2329 || x === 0x232A || x >= 0x23E9 && x <= 0x23EC || x === 0x23F0 || x === 0x23F3 || x === 0x25FD || x === 0x25FE || x === 0x2614 || x === 0x2615 || x >= 0x2648 && x <= 0x2653 || x === 0x267F || x === 0x2693 || x === 0x26A1 || x === 0x26AA || x === 0x26AB || x === 0x26BD || x === 0x26BE || x === 0x26C4 || x === 0x26C5 || x === 0x26CE || x === 0x26D4 || x === 0x26EA || x === 0x26F2 || x === 0x26F3 || x === 0x26F5 || x === 0x26FA || x === 0x26FD || x === 0x2705 || x === 0x270A || x === 0x270B || x === 0x2728 || x === 0x274C || x === 0x274E || x >= 0x2753 && x <= 0x2755 || x === 0x2757 || x >= 0x2795 && x <= 0x2797 || x === 0x27B0 || x === 0x27BF || x === 0x2B1B || x === 0x2B1C || x === 0x2B50 || x === 0x2B55 || x >= 0x2E80 && x <= 0x2E99 || x >= 0x2E9B && x <= 0x2EF3 || x >= 0x2F00 && x <= 0x2FD5 || x >= 0x2FF0 && x <= 0x2FFF || x >= 0x3001 && x <= 0x303E || x >= 0x3041 && x <= 0x3096 || x >= 0x3099 && x <= 0x30FF || x >= 0x3105 && x <= 0x312F || x >= 0x3131 && x <= 0x318E || x >= 0x3190 && x <= 0x31E3 || x >= 0x31EF && x <= 0x321E || x >= 0x3220 && x <= 0x3247 || x >= 0x3250 && x <= 0x4DBF || x >= 0x4E00 && x <= 0xA48C || x >= 0xA490 && x <= 0xA4C6 || x >= 0xA960 && x <= 0xA97C || x >= 0xAC00 && x <= 0xD7A3 || x >= 0xF900 && x <= 0xFAFF || x >= 0xFE10 && x <= 0xFE19 || x >= 0xFE30 && x <= 0xFE52 || x >= 0xFE54 && x <= 0xFE66 || x >= 0xFE68 && x <= 0xFE6B || x >= 0x16FE0 && x <= 0x16FE4 || x === 0x16FF0 || x === 0x16FF1 || x >= 0x17000 && x <= 0x187F7 || x >= 0x18800 && x <= 0x18CD5 || x >= 0x18D00 && x <= 0x18D08 || x >= 0x1AFF0 && x <= 0x1AFF3 || x >= 0x1AFF5 && x <= 0x1AFFB || x === 0x1AFFD || x === 0x1AFFE || x >= 0x1B000 && x <= 0x1B122 || x === 0x1B132 || x >= 0x1B150 && x <= 0x1B152 || x === 0x1B155 || x >= 0x1B164 && x <= 0x1B167 || x >= 0x1B170 && x <= 0x1B2FB || x === 0x1F004 || x === 0x1F0CF || x === 0x1F18E || x >= 0x1F191 && x <= 0x1F19A || x >= 0x1F200 && x <= 0x1F202 || x >= 0x1F210 && x <= 0x1F23B || x >= 0x1F240 && x <= 0x1F248 || x === 0x1F250 || x === 0x1F251 || x >= 0x1F260 && x <= 0x1F265 || x >= 0x1F300 && x <= 0x1F320 || x >= 0x1F32D && x <= 0x1F335 || x >= 0x1F337 && x <= 0x1F37C || x >= 0x1F37E && x <= 0x1F393 || x >= 0x1F3A0 && x <= 0x1F3CA || x >= 0x1F3CF && x <= 0x1F3D3 || x >= 0x1F3E0 && x <= 0x1F3F0 || x === 0x1F3F4 || x >= 0x1F3F8 && x <= 0x1F43E || x === 0x1F440 || x >= 0x1F442 && x <= 0x1F4FC || x >= 0x1F4FF && x <= 0x1F53D || x >= 0x1F54B && x <= 0x1F54E || x >= 0x1F550 && x <= 0x1F567 || x === 0x1F57A || x === 0x1F595 || x === 0x1F596 || x === 0x1F5A4 || x >= 0x1F5FB && x <= 0x1F64F || x >= 0x1F680 && x <= 0x1F6C5 || x === 0x1F6CC || x >= 0x1F6D0 && x <= 0x1F6D2 || x >= 0x1F6D5 && x <= 0x1F6D7 || x >= 0x1F6DC && x <= 0x1F6DF || x === 0x1F6EB || x === 0x1F6EC || x >= 0x1F6F4 && x <= 0x1F6FC || x >= 0x1F7E0 && x <= 0x1F7EB || x === 0x1F7F0 || x >= 0x1F90C && x <= 0x1F93A || x >= 0x1F93C && x <= 0x1F945 || x >= 0x1F947 && x <= 0x1F9FF || x >= 0x1FA70 && x <= 0x1FA7C || x >= 0x1FA80 && x <= 0x1FA88 || x >= 0x1FA90 && x <= 0x1FABD || x >= 0x1FABF && x <= 0x1FAC5 || x >= 0x1FACE && x <= 0x1FADB || x >= 0x1FAE0 && x <= 0x1FAE8 || x >= 0x1FAF0 && x <= 0x1FAF8 || x >= 0x20000 && x <= 0x2FFFD || x >= 0x30000 && x <= 0x3FFFD;\n};\n/* EXPORT */\nexport { isAmbiguous, isFullWidth, isWide };\n", "/* IMPORT */\nimport { isAmbiguous, isFullWidth, isWide } from './utils.js';\n/* HELPERS */\nconst ANSI_RE = /[\\u001b\\u009b][[()#;?]*(?:[0-9]{1,4}(?:;[0-9]{0,4})*)?[0-9A-ORZcf-nqry=><]/y;\nconst CONTROL_RE = /[\\x00-\\x08\\x0A-\\x1F\\x7F-\\x9F]{1,1000}/y;\nconst TAB_RE = /\\t{1,1000}/y;\nconst EMOJI_RE = /[\\u{1F1E6}-\\u{1F1FF}]{2}|\\u{1F3F4}[\\u{E0061}-\\u{E007A}]{2}[\\u{E0030}-\\u{E0039}\\u{E0061}-\\u{E007A}]{1,3}\\u{E007F}|(?:\\p{Emoji}\\uFE0F\\u20E3?|\\p{Emoji_Modifier_Base}\\p{Emoji_Modifier}?|\\p{Emoji_Presentation})(?:\\u200D(?:\\p{Emoji_Modifier_Base}\\p{Emoji_Modifier}?|\\p{Emoji_Presentation}|\\p{Emoji}\\uFE0F\\u20E3?))*/yu;\nconst LATIN_RE = /(?:[\\x20-\\x7E\\xA0-\\xFF](?!\\uFE0F)){1,1000}/y;\nconst MODIFIER_RE = /\\p{M}+/gu;\nconst NO_TRUNCATION = { limit: Infinity, ellipsis: '' };\n/* MAIN */\n//TODO: Optimize matching non-latin letters\nconst getStringTruncatedWidth = (input, truncationOptions = {}, widthOptions = {}) => {\n /* CONSTANTS */\n const LIMIT = truncationOptions.limit ?? Infinity;\n const ELLIPSIS = truncationOptions.ellipsis ?? '';\n const ELLIPSIS_WIDTH = truncationOptions?.ellipsisWidth ?? (ELLIPSIS ? getStringTruncatedWidth(ELLIPSIS, NO_TRUNCATION, widthOptions).width : 0);\n const ANSI_WIDTH = widthOptions.ansiWidth ?? 0;\n const CONTROL_WIDTH = widthOptions.controlWidth ?? 0;\n const TAB_WIDTH = widthOptions.tabWidth ?? 8;\n const AMBIGUOUS_WIDTH = widthOptions.ambiguousWidth ?? 1;\n const EMOJI_WIDTH = widthOptions.emojiWidth ?? 2;\n const FULL_WIDTH_WIDTH = widthOptions.fullWidthWidth ?? 2;\n const REGULAR_WIDTH = widthOptions.regularWidth ?? 1;\n const WIDE_WIDTH = widthOptions.wideWidth ?? 2;\n /* STATE */\n let indexPrev = 0;\n let index = 0;\n let length = input.length;\n let lengthExtra = 0;\n let truncationEnabled = false;\n let truncationIndex = length;\n let truncationLimit = Math.max(0, LIMIT - ELLIPSIS_WIDTH);\n let unmatchedStart = 0;\n let unmatchedEnd = 0;\n let width = 0;\n let widthExtra = 0;\n /* PARSE LOOP */\n outer: while (true) {\n /* UNMATCHED */\n if ((unmatchedEnd > unmatchedStart) || (index >= length && index > indexPrev)) {\n const unmatched = input.slice(unmatchedStart, unmatchedEnd) || input.slice(indexPrev, index);\n lengthExtra = 0;\n for (const char of unmatched.replaceAll(MODIFIER_RE, '')) {\n const codePoint = char.codePointAt(0) || 0;\n if (isFullWidth(codePoint)) {\n widthExtra = FULL_WIDTH_WIDTH;\n }\n else if (isWide(codePoint)) {\n widthExtra = WIDE_WIDTH;\n }\n else if (AMBIGUOUS_WIDTH !== REGULAR_WIDTH && isAmbiguous(codePoint)) {\n widthExtra = AMBIGUOUS_WIDTH;\n }\n else {\n widthExtra = REGULAR_WIDTH;\n }\n if ((width + widthExtra) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, Math.max(unmatchedStart, indexPrev) + lengthExtra);\n }\n if ((width + widthExtra) > LIMIT) {\n truncationEnabled = true;\n break outer;\n }\n lengthExtra += char.length;\n width += widthExtra;\n }\n unmatchedStart = unmatchedEnd = 0;\n }\n /* EXITING */\n if (index >= length)\n break;\n /* LATIN */\n LATIN_RE.lastIndex = index;\n if (LATIN_RE.test(input)) {\n lengthExtra = LATIN_RE.lastIndex - index;\n widthExtra = lengthExtra * REGULAR_WIDTH;\n if ((width + widthExtra) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, index + Math.floor((truncationLimit - width) / REGULAR_WIDTH));\n }\n if ((width + widthExtra) > LIMIT) {\n truncationEnabled = true;\n break;\n }\n width += widthExtra;\n unmatchedStart = indexPrev;\n unmatchedEnd = index;\n index = indexPrev = LATIN_RE.lastIndex;\n continue;\n }\n /* ANSI */\n ANSI_RE.lastIndex = index;\n if (ANSI_RE.test(input)) {\n if ((width + ANSI_WIDTH) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, index);\n }\n if ((width + ANSI_WIDTH) > LIMIT) {\n truncationEnabled = true;\n break;\n }\n width += ANSI_WIDTH;\n unmatchedStart = indexPrev;\n unmatchedEnd = index;\n index = indexPrev = ANSI_RE.lastIndex;\n continue;\n }\n /* CONTROL */\n CONTROL_RE.lastIndex = index;\n if (CONTROL_RE.test(input)) {\n lengthExtra = CONTROL_RE.lastIndex - index;\n widthExtra = lengthExtra * CONTROL_WIDTH;\n if ((width + widthExtra) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, index + Math.floor((truncationLimit - width) / CONTROL_WIDTH));\n }\n if ((width + widthExtra) > LIMIT) {\n truncationEnabled = true;\n break;\n }\n width += widthExtra;\n unmatchedStart = indexPrev;\n unmatchedEnd = index;\n index = indexPrev = CONTROL_RE.lastIndex;\n continue;\n }\n /* TAB */\n TAB_RE.lastIndex = index;\n if (TAB_RE.test(input)) {\n lengthExtra = TAB_RE.lastIndex - index;\n widthExtra = lengthExtra * TAB_WIDTH;\n if ((width + widthExtra) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, index + Math.floor((truncationLimit - width) / TAB_WIDTH));\n }\n if ((width + widthExtra) > LIMIT) {\n truncationEnabled = true;\n break;\n }\n width += widthExtra;\n unmatchedStart = indexPrev;\n unmatchedEnd = index;\n index = indexPrev = TAB_RE.lastIndex;\n continue;\n }\n /* EMOJI */\n EMOJI_RE.lastIndex = index;\n if (EMOJI_RE.test(input)) {\n if ((width + EMOJI_WIDTH) > truncationLimit) {\n truncationIndex = Math.min(truncationIndex, index);\n }\n if ((width + EMOJI_WIDTH) > LIMIT) {\n truncationEnabled = true;\n break;\n }\n width += EMOJI_WIDTH;\n unmatchedStart = indexPrev;\n unmatchedEnd = index;\n index = indexPrev = EMOJI_RE.lastIndex;\n continue;\n }\n /* UNMATCHED INDEX */\n index += 1;\n }\n /* RETURN */\n return {\n width: truncationEnabled ? truncationLimit : width,\n index: truncationEnabled ? truncationIndex : length,\n truncated: truncationEnabled,\n ellipsed: truncationEnabled && LIMIT >= ELLIPSIS_WIDTH\n };\n};\n/* EXPORT */\nexport default getStringTruncatedWidth;\n", "/* IMPORT */\nimport fastStringTruncatedWidth from 'fast-string-truncated-width';\n/* HELPERS */\nconst NO_TRUNCATION = {\n limit: Infinity,\n ellipsis: '',\n ellipsisWidth: 0,\n};\n/* MAIN */\nconst fastStringWidth = (input, options = {}) => {\n return fastStringTruncatedWidth(input, NO_TRUNCATION, options).width;\n};\n/* EXPORT */\nexport default fastStringWidth;\n", "import stringWidth from 'fast-string-width';\nconst ESC = '\\x1B';\nconst CSI = '\\x9B';\nconst END_CODE = 39;\nconst ANSI_ESCAPE_BELL = '\\u0007';\nconst ANSI_CSI = '[';\nconst ANSI_OSC = ']';\nconst ANSI_SGR_TERMINATOR = 'm';\nconst ANSI_ESCAPE_LINK = `${ANSI_OSC}8;;`;\nconst GROUP_REGEX = new RegExp(`(?:\\\\${ANSI_CSI}(?<code>\\\\d+)m|\\\\${ANSI_ESCAPE_LINK}(?<uri>.*)${ANSI_ESCAPE_BELL})`, 'y');\nconst getClosingCode = (openingCode) => {\n if (openingCode >= 30 && openingCode <= 37)\n return 39;\n if (openingCode >= 90 && openingCode <= 97)\n return 39;\n if (openingCode >= 40 && openingCode <= 47)\n return 49;\n if (openingCode >= 100 && openingCode <= 107)\n return 49;\n if (openingCode === 1 || openingCode === 2)\n return 22;\n if (openingCode === 3)\n return 23;\n if (openingCode === 4)\n return 24;\n if (openingCode === 7)\n return 27;\n if (openingCode === 8)\n return 28;\n if (openingCode === 9)\n return 29;\n if (openingCode === 0)\n return 0;\n return undefined;\n};\nconst wrapAnsiCode = (code) => `${ESC}${ANSI_CSI}${code}${ANSI_SGR_TERMINATOR}`;\nconst wrapAnsiHyperlink = (url) => `${ESC}${ANSI_ESCAPE_LINK}${url}${ANSI_ESCAPE_BELL}`;\nconst wordLengths = (words) => words.map((character) => stringWidth(character));\nconst wrapWord = (rows, word, columns) => {\n const characters = word[Symbol.iterator]();\n let isInsideEscape = false;\n let isInsideLinkEscape = false;\n let lastRow = rows.at(-1);\n let visible = lastRow === undefined ? 0 : stringWidth(lastRow);\n let currentCharacter = characters.next();\n let nextCharacter = characters.next();\n let rawCharacterIndex = 0;\n while (!currentCharacter.done) {\n const character = currentCharacter.value;\n const characterLength = stringWidth(character);\n if (visible + characterLength <= columns) {\n rows[rows.length - 1] += character;\n }\n else {\n rows.push(character);\n visible = 0;\n }\n if (character === ESC || character === CSI) {\n isInsideEscape = true;\n isInsideLinkEscape = word.startsWith(ANSI_ESCAPE_LINK, rawCharacterIndex + 1);\n }\n if (isInsideEscape) {\n if (isInsideLinkEscape) {\n if (character === ANSI_ESCAPE_BELL) {\n isInsideEscape = false;\n isInsideLinkEscape = false;\n }\n }\n else if (character === ANSI_SGR_TERMINATOR) {\n isInsideEscape = false;\n }\n }\n else {\n visible += characterLength;\n if (visible === columns && !nextCharacter.done) {\n rows.push('');\n visible = 0;\n }\n }\n currentCharacter = nextCharacter;\n nextCharacter = characters.next();\n rawCharacterIndex += character.length;\n }\n lastRow = rows.at(-1);\n if (!visible &&\n lastRow !== undefined &&\n lastRow.length > 0 &&\n rows.length > 1) {\n rows[rows.length - 2] += rows.pop();\n }\n};\nconst stringVisibleTrimSpacesRight = (string) => {\n const words = string.split(' ');\n let last = words.length;\n while (last > 0) {\n if (stringWidth(words[last - 1]) > 0) {\n break;\n }\n last--;\n }\n if (last === words.length) {\n return string;\n }\n return words.slice(0, last).join(' ') + words.slice(last).join('');\n};\nconst exec = (string, columns, options = {}) => {\n if (options.trim !== false && string.trim() === '') {\n return '';\n }\n let returnValue = '';\n let escapeCode;\n let escapeUrl;\n const words = string.split(' ');\n const lengths = wordLengths(words);\n let rows = [''];\n for (const [index, word] of words.entries()) {\n if (options.trim !== false) {\n rows[rows.length - 1] = (rows.at(-1) ?? '').trimStart();\n }\n let rowLength = stringWidth(rows.at(-1) ?? '');\n if (index !== 0) {\n if (rowLength >= columns &&\n (options.wordWrap === false || options.trim === false)) {\n rows.push('');\n rowLength = 0;\n }\n if (rowLength > 0 || options.trim === false) {\n rows[rows.length - 1] += ' ';\n rowLength++;\n }\n }\n if (options.hard && lengths[index] > columns) {\n const remainingColumns = columns - rowLength;\n const breaksStartingThisLine = 1 + Math.floor((lengths[index] - remainingColumns - 1) / columns);\n const breaksStartingNextLine = Math.floor((lengths[index] - 1) / columns);\n if (breaksStartingNextLine < breaksStartingThisLine) {\n rows.push('');\n }\n wrapWord(rows, word, columns);\n continue;\n }\n if (rowLength + lengths[index] > columns &&\n rowLength > 0 &&\n lengths[index] > 0) {\n if (options.wordWrap === false && rowLength < columns) {\n wrapWord(rows, word, columns);\n continue;\n }\n rows.push('');\n }\n if (rowLength + lengths[index] > columns && options.wordWrap === false) {\n wrapWord(rows, word, columns);\n continue;\n }\n rows[rows.length - 1] += word;\n }\n if (options.trim !== false) {\n rows = rows.map((row) => stringVisibleTrimSpacesRight(row));\n }\n const preString = rows.join('\\n');\n const pre = preString[Symbol.iterator]();\n let currentPre = pre.next();\n let nextPre = pre.next();\n // We need to keep a separate index as `String#slice()` works on Unicode code units, while `pre` is an array of codepoints.\n let preStringIndex = 0;\n while (!currentPre.done) {\n const character = currentPre.value;\n const nextCharacter = nextPre.value;\n returnValue += character;\n if (character === ESC || character === CSI) {\n GROUP_REGEX.lastIndex = preStringIndex + 1;\n const groupsResult = GROUP_REGEX.exec(preString);\n const groups = groupsResult?.groups;\n if (groups?.code !== undefined) {\n const code = Number.parseFloat(groups.code);\n escapeCode = code === END_CODE ? undefined : code;\n }\n else if (groups?.uri !== undefined) {\n escapeUrl = groups.uri.length === 0 ? undefined : groups.uri;\n }\n }\n const closingCode = escapeCode ? getClosingCode(escapeCode) : undefined;\n if (nextCharacter === '\\n') {\n if (escapeUrl) {\n returnValue += wrapAnsiHyperlink('');\n }\n if (escapeCode && closingCode) {\n returnValue += wrapAnsiCode(closingCode);\n }\n }\n else if (character === '\\n') {\n if (escapeCode && closingCode) {\n returnValue += wrapAnsiCode(escapeCode);\n }\n if (escapeUrl) {\n returnValue += wrapAnsiHyperlink(escapeUrl);\n }\n }\n preStringIndex += character.length;\n currentPre = nextPre;\n nextPre = pre.next();\n }\n return returnValue;\n};\nexport function wrapAnsi(string, columns, options) {\n return String(string)\n .normalize()\n .replaceAll('\\r\\n', '\\n')\n .split('\\n')\n .map((line) => exec(line, columns, options))\n .join('\\n');\n}\n//# sourceMappingURL=main.js.map", "const actions = ['up', 'down', 'left', 'right', 'space', 'enter', 'cancel'] as const;\nexport type Action = (typeof actions)[number];\n\n/** Global settings for Clack programs, stored in memory */\ninterface InternalClackSettings {\n\tactions: Set<Action>;\n\taliases: Map<string, Action>;\n\tmessages: {\n\t\tcancel: string;\n\t\terror: string;\n\t};\n\twithGuide: boolean;\n}\n\nexport const settings: InternalClackSettings = {\n\tactions: new Set(actions),\n\taliases: new Map<string, Action>([\n\t\t// vim support\n\t\t['k', 'up'],\n\t\t['j', 'down'],\n\t\t['h', 'left'],\n\t\t['l', 'right'],\n\t\t['\\x03', 'cancel'],\n\t\t// opinionated defaults!\n\t\t['escape', 'cancel'],\n\t]),\n\tmessages: {\n\t\tcancel: 'Canceled',\n\t\terror: 'Something went wrong',\n\t},\n\twithGuide: true,\n};\n\nexport interface ClackSettings {\n\t/**\n\t * Set custom global aliases for the default actions.\n\t * This will not overwrite existing aliases, it will only add new ones!\n\t *\n\t * @param aliases - An object that maps aliases to actions\n\t * @default { k: 'up', j: 'down', h: 'left', l: 'right', '\\x03': 'cancel', 'escape': 'cancel' }\n\t */\n\taliases?: Record<string, Action>;\n\n\t/**\n\t * Custom messages for prompts\n\t */\n\tmessages?: {\n\t\t/**\n\t\t * Custom message to display when a spinner is cancelled\n\t\t * @default \"Canceled\"\n\t\t */\n\t\tcancel?: string;\n\t\t/**\n\t\t * Custom message to display when a spinner encounters an error\n\t\t * @default \"Something went wrong\"\n\t\t */\n\t\terror?: string;\n\t};\n\n\twithGuide?: boolean;\n}\n\nexport function updateSettings(updates: ClackSettings) {\n\t// Handle each property in the updates\n\tif (updates.aliases !== undefined) {\n\t\tconst aliases = updates.aliases;\n\t\tfor (const alias in aliases) {\n\t\t\tif (!Object.hasOwn(aliases, alias)) continue;\n\n\t\t\tconst action = aliases[alias];\n\t\t\tif (!settings.actions.has(action)) continue;\n\n\t\t\tif (!settings.aliases.has(alias)) {\n\t\t\t\tsettings.aliases.set(alias, action);\n\t\t\t}\n\t\t}\n\t}\n\n\tif (updates.messages !== undefined) {\n\t\tconst messages = updates.messages;\n\t\tif (messages.cancel !== undefined) {\n\t\t\tsettings.messages.cancel = messages.cancel;\n\t\t}\n\t\tif (messages.error !== undefined) {\n\t\t\tsettings.messages.error = messages.error;\n\t\t}\n\t}\n\n\tif (updates.withGuide !== undefined) {\n\t\tsettings.withGuide = updates.withGuide !== false;\n\t}\n}\n\n/**\n * Check if a key is an alias for a default action\n * @param key - The raw key which might match to an action\n * @param action - The action to match\n * @returns boolean\n */\nexport function isActionKey(key: string | Array<string | undefined>, action: Action) {\n\tif (typeof key === 'string') {\n\t\treturn settings.aliases.get(key) === action;\n\t}\n\n\tfor (const value of key) {\n\t\tif (value === undefined) continue;\n\t\tif (isActionKey(value, action)) {\n\t\t\treturn true;\n\t\t}\n\t}\n\treturn false;\n}\n", "export function diffLines(a: string, b: string) {\n\tif (a === b) return;\n\n\tconst aLines = a.split('\\n');\n\tconst bLines = b.split('\\n');\n\tconst numLines = Math.max(aLines.length, bLines.length);\n\tconst diff: number[] = [];\n\n\tfor (let i = 0; i < numLines; i++) {\n\t\tif (aLines[i] !== bLines[i]) diff.push(i);\n\t}\n\n\treturn {\n\t\tlines: diff,\n\t\tnumLinesBefore: aLines.length,\n\t\tnumLinesAfter: bLines.length,\n\t\tnumLines,\n\t};\n}\n", "import { stdin, stdout } from 'node:process';\nimport type { Key } from 'node:readline';\nimport * as readline from 'node:readline';\nimport type { Readable, Writable } from 'node:stream';\nimport { ReadStream } from 'node:tty';\nimport { wrapAnsi } from 'fast-wrap-ansi';\nimport { cursor } from 'sisteransi';\nimport { isActionKey } from './settings.js';\n\nexport * from './settings.js';\nexport * from './string.js';\n\nconst isWindows = globalThis.process.platform.startsWith('win');\n\nexport const CANCEL_SYMBOL = Symbol('clack:cancel');\n\nexport function isCancel(value: unknown): value is symbol {\n\treturn value === CANCEL_SYMBOL;\n}\n\nexport function setRawMode(input: Readable, value: boolean) {\n\tconst i = input as typeof stdin;\n\n\tif (i.isTTY) i.setRawMode(value);\n}\n\ninterface BlockOptions {\n\tinput?: Readable;\n\toutput?: Writable;\n\toverwrite?: boolean;\n\thideCursor?: boolean;\n}\n\nexport function block({\n\tinput = stdin,\n\toutput = stdout,\n\toverwrite = true,\n\thideCursor = true,\n}: BlockOptions = {}) {\n\tconst rl = readline.createInterface({\n\t\tinput,\n\t\toutput,\n\t\tprompt: '',\n\t\ttabSize: 1,\n\t});\n\treadline.emitKeypressEvents(input, rl);\n\n\tif (input instanceof ReadStream && input.isTTY) {\n\t\tinput.setRawMode(true);\n\t}\n\n\tconst clear = (data: Buffer, { name, sequence }: Key) => {\n\t\tconst str = String(data);\n\t\tif (isActionKey([str, name, sequence], 'cancel')) {\n\t\t\tif (hideCursor) output.write(cursor.show);\n\t\t\tprocess.exit(0);\n\t\t\treturn;\n\t\t}\n\t\tif (!overwrite) return;\n\t\tconst dx = name === 'return' ? 0 : -1;\n\t\tconst dy = name === 'return' ? -1 : 0;\n\n\t\treadline.moveCursor(output, dx, dy, () => {\n\t\t\treadline.clearLine(output, 1, () => {\n\t\t\t\tinput.once('keypress', clear);\n\t\t\t});\n\t\t});\n\t};\n\tif (hideCursor) output.write(cursor.hide);\n\tinput.once('keypress', clear);\n\n\treturn () => {\n\t\tinput.off('keypress', clear);\n\t\tif (hideCursor) output.write(cursor.show);\n\n\t\t// Prevent Windows specific issues: https://github.com/bombshell-dev/clack/issues/176\n\t\tif (input instanceof ReadStream && input.isTTY && !isWindows) {\n\t\t\tinput.setRawMode(false);\n\t\t}\n\n\t\t// @ts-expect-error fix for https://github.com/nodejs/node/issues/31762#issuecomment-1441223907\n\t\trl.terminal = false;\n\t\trl.close();\n\t};\n}\n\nexport const getColumns = (output: Writable): number => {\n\tif ('columns' in output && typeof output.columns === 'number') {\n\t\treturn output.columns;\n\t}\n\treturn 80;\n};\n\nexport const getRows = (output: Writable): number => {\n\tif ('rows' in output && typeof output.rows === 'number') {\n\t\treturn output.rows;\n\t}\n\treturn 20;\n};\n\nexport function wrapTextWithPrefix(\n\toutput: Writable | undefined,\n\ttext: string,\n\tprefix: string,\n\tstartPrefix: string = prefix\n): string {\n\tconst columns = getColumns(output ?? stdout);\n\tconst wrapped = wrapAnsi(text, columns - prefix.length, {\n\t\thard: true,\n\t\ttrim: false,\n\t});\n\tconst lines = wrapped\n\t\t.split('\\n')\n\t\t.map((line, index) => {\n\t\t\treturn `${index === 0 ? startPrefix : prefix}${line}`;\n\t\t})\n\t\t.join('\\n');\n\treturn lines;\n}\n", "import { stdin, stdout } from 'node:process';\nimport readline, { type Key, type ReadLine } from 'node:readline';\nimport type { Readable, Writable } from 'node:stream';\nimport { wrapAnsi } from 'fast-wrap-ansi';\nimport { cursor, erase } from 'sisteransi';\nimport type { ClackEvents, ClackState } from '../types.js';\nimport type { Action } from '../utils/index.js';\nimport {\n\tCANCEL_SYMBOL,\n\tdiffLines,\n\tgetRows,\n\tisActionKey,\n\tsetRawMode,\n\tsettings,\n} from '../utils/index.js';\n\nexport interface PromptOptions<TValue, Self extends Prompt<TValue>> {\n\trender(this: Omit<Self, 'prompt'>): string | undefined;\n\tinitialValue?: any;\n\tinitialUserInput?: string;\n\tvalidate?: ((value: TValue | undefined) => string | Error | undefined) | undefined;\n\tinput?: Readable;\n\toutput?: Writable;\n\tdebug?: boolean;\n\tsignal?: AbortSignal;\n}\n\nexport default class Prompt<TValue> {\n\tprotected input: Readable;\n\tprotected output: Writable;\n\tprivate _abortSignal?: AbortSignal;\n\n\tprivate rl: ReadLine | undefined;\n\tprivate opts: Omit<PromptOptions<TValue, Prompt<TValue>>, 'render' | 'input' | 'output'>;\n\tprivate _render: (context: Omit<Prompt<TValue>, 'prompt'>) => string | undefined;\n\tprivate _track = false;\n\tprivate _prevFrame = '';\n\tprivate _subscribers = new Map<string, { cb: (...args: any) => any; once?: boolean }[]>();\n\tprotected _cursor = 0;\n\n\tpublic state: ClackState = 'initial';\n\tpublic error = '';\n\tpublic value: TValue | undefined;\n\tpublic userInput = '';\n\n\tconstructor(options: PromptOptions<TValue, Prompt<TValue>>, trackValue = true) {\n\t\tconst { input = stdin, output = stdout, render, signal, ...opts } = options;\n\n\t\tthis.opts = opts;\n\t\tthis.onKeypress = this.onKeypress.bind(this);\n\t\tthis.close = this.close.bind(this);\n\t\tthis.render = this.render.bind(this);\n\t\tthis._render = render.bind(this);\n\t\tthis._track = trackValue;\n\t\tthis._abortSignal = signal;\n\n\t\tthis.input = input;\n\t\tthis.output = output;\n\t}\n\n\t/**\n\t * Unsubscribe all listeners\n\t */\n\tprotected unsubscribe() {\n\t\tthis._subscribers.clear();\n\t}\n\n\t/**\n\t * Set a subscriber with opts\n\t * @param event - The event name\n\t */\n\tprivate setSubscriber<T extends keyof ClackEvents<TValue>>(\n\t\tevent: T,\n\t\topts: { cb: ClackEvents<TValue>[T]; once?: boolean }\n\t) {\n\t\tconst params = this._subscribers.get(event) ?? [];\n\t\tparams.push(opts);\n\t\tthis._subscribers.set(event, params);\n\t}\n\n\t/**\n\t * Subscribe to an event\n\t * @param event - The event name\n\t * @param cb - The callback\n\t */\n\tpublic on<T extends keyof ClackEvents<TValue>>(event: T, cb: ClackEvents<TValue>[T]) {\n\t\tthis.setSubscriber(event, { cb });\n\t}\n\n\t/**\n\t * Subscribe to an event once\n\t * @param event - The event name\n\t * @param cb - The callback\n\t */\n\tpublic once<T extends keyof ClackEvents<TValue>>(event: T, cb: ClackEvents<TValue>[T]) {\n\t\tthis.setSubscriber(event, { cb, once: true });\n\t}\n\n\t/**\n\t * Emit an event with data\n\t * @param event - The event name\n\t * @param data - The data to pass to the callback\n\t */\n\tpublic emit<T extends keyof ClackEvents<TValue>>(\n\t\tevent: T,\n\t\t...data: Parameters<ClackEvents<TValue>[T]>\n\t) {\n\t\tconst cbs = this._subscribers.get(event) ?? [];\n\t\tconst cleanup: (() => void)[] = [];\n\n\t\tfor (const subscriber of cbs) {\n\t\t\tsubscriber.cb(...data);\n\n\t\t\tif (subscriber.once) {\n\t\t\t\tcleanup.push(() => cbs.splice(cbs.indexOf(subscriber), 1));\n\t\t\t}\n\t\t}\n\n\t\tfor (const cb of cleanup) {\n\t\t\tcb();\n\t\t}\n\t}\n\n\tpublic prompt() {\n\t\treturn new Promise<TValue | symbol | undefined>((resolve) => {\n\t\t\tif (this._abortSignal) {\n\t\t\t\tif (this._abortSignal.aborted) {\n\t\t\t\t\tthis.state = 'cancel';\n\n\t\t\t\t\tthis.close();\n\t\t\t\t\treturn resolve(CANCEL_SYMBOL);\n\t\t\t\t}\n\n\t\t\t\tthis._abortSignal.addEventListener(\n\t\t\t\t\t'abort',\n\t\t\t\t\t() => {\n\t\t\t\t\t\tthis.state = 'cancel';\n\t\t\t\t\t\tthis.close();\n\t\t\t\t\t},\n\t\t\t\t\t{ once: true }\n\t\t\t\t);\n\t\t\t}\n\n\t\t\tthis.rl = readline.createInterface({\n\t\t\t\tinput: this.input,\n\t\t\t\ttabSize: 2,\n\t\t\t\tprompt: '',\n\t\t\t\tescapeCodeTimeout: 50,\n\t\t\t\tterminal: true,\n\t\t\t});\n\t\t\tthis.rl.prompt();\n\n\t\t\tif (this.opts.initialUserInput !== undefined) {\n\t\t\t\tthis._setUserInput(this.opts.initialUserInput, true);\n\t\t\t}\n\n\t\t\tthis.input.on('keypress', this.onKeypress);\n\t\t\tsetRawMode(this.input, true);\n\t\t\tthis.output.on('resize', this.render);\n\n\t\t\tthis.render();\n\n\t\t\tthis.once('submit', () => {\n\t\t\t\tthis.output.write(cursor.show);\n\t\t\t\tthis.output.off('resize', this.render);\n\t\t\t\tsetRawMode(this.input, false);\n\t\t\t\tresolve(this.value);\n\t\t\t});\n\t\t\tthis.once('cancel', () => {\n\t\t\t\tthis.output.write(cursor.show);\n\t\t\t\tthis.output.off('resize', this.render);\n\t\t\t\tsetRawMode(this.input, false);\n\t\t\t\tresolve(CANCEL_SYMBOL);\n\t\t\t});\n\t\t});\n\t}\n\n\tprotected _isActionKey(char: string | undefined, _key: Key): boolean {\n\t\treturn char === '\\t';\n\t}\n\n\tprotected _setValue(value: TValue | undefined): void {\n\t\tthis.value = value;\n\t\tthis.emit('value', this.value);\n\t}\n\n\tprotected _setUserInput(value: string | undefined, write?: boolean): void {\n\t\tthis.userInput = value ?? '';\n\t\tthis.emit('userInput', this.userInput);\n\t\tif (write && this._track && this.rl) {\n\t\t\tthis.rl.write(this.userInput);\n\t\t\tthis._cursor = this.rl.cursor;\n\t\t}\n\t}\n\n\tprotected _clearUserInput(): void {\n\t\tthis.rl?.write(null, { ctrl: true, name: 'u' });\n\t\tthis._setUserInput('');\n\t}\n\n\tprivate onKeypress(char: string | undefined, key: Key) {\n\t\tif (this._track && key.name !== 'return') {\n\t\t\tif (key.name && this._isActionKey(char, key)) {\n\t\t\t\tthis.rl?.write(null, { ctrl: true, name: 'h' });\n\t\t\t}\n\t\t\tthis._cursor = this.rl?.cursor ?? 0;\n\t\t\tthis._setUserInput(this.rl?.line);\n\t\t}\n\n\t\tif (this.state === 'error') {\n\t\t\tthis.state = 'active';\n\t\t}\n\t\tif (key?.name) {\n\t\t\tif (!this._track && settings.aliases.has(key.name)) {\n\t\t\t\tthis.emit('cursor', settings.aliases.get(key.name));\n\t\t\t}\n\t\t\tif (settings.actions.has(key.name as Action)) {\n\t\t\t\tthis.emit('cursor', key.name as Action);\n\t\t\t}\n\t\t}\n\t\tif (char && (char.toLowerCase() === 'y' || char.toLowerCase() === 'n')) {\n\t\t\tthis.emit('confirm', char.toLowerCase() === 'y');\n\t\t}\n\n\t\t// Call the key event handler and emit the key event\n\t\tthis.emit('key', char?.toLowerCase(), key);\n\n\t\tif (key?.name === 'return') {\n\t\t\tif (this.opts.validate) {\n\t\t\t\tconst problem = this.opts.validate(this.value);\n\t\t\t\tif (problem) {\n\t\t\t\t\tthis.error = problem instanceof Error ? problem.message : problem;\n\t\t\t\t\tthis.state = 'error';\n\t\t\t\t\tthis.rl?.write(this.userInput);\n\t\t\t\t}\n\t\t\t}\n\t\t\tif (this.state !== 'error') {\n\t\t\t\tthis.state = 'submit';\n\t\t\t}\n\t\t}\n\n\t\tif (isActionKey([char, key?.name, key?.sequence], 'cancel')) {\n\t\t\tthis.state = 'cancel';\n\t\t}\n\n\t\tif (this.state === 'submit' || this.state === 'cancel') {\n\t\t\tthis.emit('finalize');\n\t\t}\n\t\tthis.render();\n\t\tif (this.state === 'submit' || this.state === 'cancel') {\n\t\t\tthis.close();\n\t\t}\n\t}\n\n\tprotected close() {\n\t\tthis.input.unpipe();\n\t\tthis.input.removeListener('keypress', this.onKeypress);\n\t\tthis.output.write('\\n');\n\t\tsetRawMode(this.input, false);\n\t\tthis.rl?.close();\n\t\tthis.rl = undefined;\n\t\tthis.emit(`${this.state}`, this.value);\n\t\tthis.unsubscribe();\n\t}\n\n\tprivate restoreCursor() {\n\t\tconst lines =\n\t\t\twrapAnsi(this._prevFrame, process.stdout.columns, { hard: true, trim: false }).split('\\n')\n\t\t\t\t.length - 1;\n\t\tthis.output.write(cursor.move(-999, lines * -1));\n\t}\n\n\tprivate render() {\n\t\tconst frame = wrapAnsi(this._render(this) ?? '', process.stdout.columns, {\n\t\t\thard: true,\n\t\t\ttrim: false,\n\t\t});\n\t\tif (frame === this._prevFrame) return;\n\n\t\tif (this.state === 'initial') {\n\t\t\tthis.output.write(cursor.hide);\n\t\t} else {\n\t\t\tconst diff = diffLines(this._prevFrame, frame);\n\t\t\tconst rows = getRows(this.output);\n\t\t\tthis.restoreCursor();\n\t\t\tif (diff) {\n\t\t\t\tconst diffOffsetAfter = Math.max(0, diff.numLinesAfter - rows);\n\t\t\t\tconst diffOffsetBefore = Math.max(0, diff.numLinesBefore - rows);\n\t\t\t\tlet diffLine = diff.lines.find((line) => line >= diffOffsetAfter);\n\n\t\t\t\tif (diffLine === undefined) {\n\t\t\t\t\tthis._prevFrame = frame;\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\t// If a single line has changed, only update that line\n\t\t\t\tif (diff.lines.length === 1) {\n\t\t\t\t\tthis.output.write(cursor.move(0, diffLine - diffOffsetBefore));\n\t\t\t\t\tthis.output.write(erase.lines(1));\n\t\t\t\t\tconst lines = frame.split('\\n');\n\t\t\t\t\tthis.output.write(lines[diffLine]);\n\t\t\t\t\tthis._prevFrame = frame;\n\t\t\t\t\tthis.output.write(cursor.move(0, lines.length - diffLine - 1));\n\t\t\t\t\treturn;\n\t\t\t\t\t// If many lines have changed, rerender everything past the first line\n\t\t\t\t} else if (diff.lines.length > 1) {\n\t\t\t\t\tif (diffOffsetAfter < diffOffsetBefore) {\n\t\t\t\t\t\tdiffLine = diffOffsetAfter;\n\t\t\t\t\t} else {\n\t\t\t\t\t\tconst adjustedDiffLine = diffLine - diffOffsetBefore;\n\t\t\t\t\t\tif (adjustedDiffLine > 0) {\n\t\t\t\t\t\t\tthis.output.write(cursor.move(0, adjustedDiffLine));\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\tthis.output.write(erase.down());\n\t\t\t\t\tconst lines = frame.split('\\n');\n\t\t\t\t\tconst newLines = lines.slice(diffLine);\n\t\t\t\t\tthis.output.write(newLines.join('\\n'));\n\t\t\t\t\tthis._prevFrame = frame;\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tthis.output.write(erase.down());\n\t\t}\n\n\t\tthis.output.write(frame);\n\t\tif (this.state === 'initial') {\n\t\t\tthis.state = 'active';\n\t\t}\n\t\tthis._prevFrame = frame;\n\t}\n}\n", "import type { Key } from 'node:readline';\nimport color from 'picocolors';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\ninterface OptionLike {\n\tvalue: unknown;\n\tlabel?: string;\n}\n\ntype FilterFunction<T extends OptionLike> = (search: string, opt: T) => boolean;\n\nfunction getCursorForValue<T extends OptionLike>(\n\tselected: T['value'] | undefined,\n\titems: T[]\n): number {\n\tif (selected === undefined) {\n\t\treturn 0;\n\t}\n\n\tconst currLength = items.length;\n\n\t// If filtering changed the available options, update cursor\n\tif (currLength === 0) {\n\t\treturn 0;\n\t}\n\n\t// Try to maintain the same selected item\n\tconst index = items.findIndex((item) => item.value === selected);\n\treturn index !== -1 ? index : 0;\n}\n\nfunction defaultFilter<T extends OptionLike>(input: string, option: T): boolean {\n\tconst label = option.label ?? String(option.value);\n\treturn label.toLowerCase().includes(input.toLowerCase());\n}\n\nfunction normalisedValue<T>(multiple: boolean, values: T[] | undefined): T | T[] | undefined {\n\tif (!values) {\n\t\treturn undefined;\n\t}\n\tif (multiple) {\n\t\treturn values;\n\t}\n\treturn values[0];\n}\n\nexport interface AutocompleteOptions<T extends OptionLike>\n\textends PromptOptions<T['value'] | T['value'][], AutocompletePrompt<T>> {\n\toptions: T[] | ((this: AutocompletePrompt<T>) => T[]);\n\tfilter?: FilterFunction<T>;\n\tmultiple?: boolean;\n}\n\nexport default class AutocompletePrompt<T extends OptionLike> extends Prompt<\n\tT['value'] | T['value'][]\n> {\n\tfilteredOptions: T[];\n\tmultiple: boolean;\n\tisNavigating = false;\n\tselectedValues: Array<T['value']> = [];\n\n\tfocusedValue: T['value'] | undefined;\n\t#cursor = 0;\n\t#lastUserInput = '';\n\t#filterFn: FilterFunction<T>;\n\t#options: T[] | (() => T[]);\n\n\tget cursor(): number {\n\t\treturn this.#cursor;\n\t}\n\n\tget userInputWithCursor() {\n\t\tif (!this.userInput) {\n\t\t\treturn color.inverse(color.hidden('_'));\n\t\t}\n\t\tif (this._cursor >= this.userInput.length) {\n\t\t\treturn `${this.userInput}█`;\n\t\t}\n\t\tconst s1 = this.userInput.slice(0, this._cursor);\n\t\tconst [s2, ...s3] = this.userInput.slice(this._cursor);\n\t\treturn `${s1}${color.inverse(s2)}${s3.join('')}`;\n\t}\n\n\tget options(): T[] {\n\t\tif (typeof this.#options === 'function') {\n\t\t\treturn this.#options();\n\t\t}\n\t\treturn this.#options;\n\t}\n\n\tconstructor(opts: AutocompleteOptions<T>) {\n\t\tsuper(opts);\n\n\t\tthis.#options = opts.options;\n\t\tconst options = this.options;\n\t\tthis.filteredOptions = [...options];\n\t\tthis.multiple = opts.multiple === true;\n\t\tthis.#filterFn = opts.filter ?? defaultFilter;\n\t\tlet initialValues: unknown[] | undefined;\n\t\tif (opts.initialValue && Array.isArray(opts.initialValue)) {\n\t\t\tif (this.multiple) {\n\t\t\t\tinitialValues = opts.initialValue;\n\t\t\t} else {\n\t\t\t\tinitialValues = opts.initialValue.slice(0, 1);\n\t\t\t}\n\t\t} else {\n\t\t\tif (!this.multiple && this.options.length > 0) {\n\t\t\t\tinitialValues = [this.options[0].value];\n\t\t\t}\n\t\t}\n\n\t\tif (initialValues) {\n\t\t\tfor (const selectedValue of initialValues) {\n\t\t\t\tconst selectedIndex = options.findIndex((opt) => opt.value === selectedValue);\n\t\t\t\tif (selectedIndex !== -1) {\n\t\t\t\t\tthis.toggleSelected(selectedValue);\n\t\t\t\t\tthis.#cursor = selectedIndex;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tthis.focusedValue = this.options[this.#cursor]?.value;\n\n\t\tthis.on('key', (char, key) => this.#onKey(char, key));\n\t\tthis.on('userInput', (value) => this.#onUserInputChanged(value));\n\t}\n\n\tprotected override _isActionKey(char: string | undefined, key: Key): boolean {\n\t\treturn (\n\t\t\tchar === '\\t' ||\n\t\t\t(this.multiple &&\n\t\t\t\tthis.isNavigating &&\n\t\t\t\tkey.name === 'space' &&\n\t\t\t\tchar !== undefined &&\n\t\t\t\tchar !== '')\n\t\t);\n\t}\n\n\t#onKey(_char: string | undefined, key: Key): void {\n\t\tconst isUpKey = key.name === 'up';\n\t\tconst isDownKey = key.name === 'down';\n\t\tconst isReturnKey = key.name === 'return';\n\n\t\t// Start navigation mode with up/down arrows\n\t\tif (isUpKey || isDownKey) {\n\t\t\tthis.#cursor = Math.max(\n\t\t\t\t0,\n\t\t\t\tMath.min(this.#cursor + (isUpKey ? -1 : 1), this.filteredOptions.length - 1)\n\t\t\t);\n\t\t\tthis.focusedValue = this.filteredOptions[this.#cursor]?.value;\n\t\t\tif (!this.multiple) {\n\t\t\t\tthis.selectedValues = [this.focusedValue];\n\t\t\t}\n\t\t\tthis.isNavigating = true;\n\t\t} else if (isReturnKey) {\n\t\t\tthis.value = normalisedValue(this.multiple, this.selectedValues);\n\t\t} else {\n\t\t\tif (this.multiple) {\n\t\t\t\tif (\n\t\t\t\t\tthis.focusedValue !== undefined &&\n\t\t\t\t\t(key.name === 'tab' || (this.isNavigating && key.name === 'space'))\n\t\t\t\t) {\n\t\t\t\t\tthis.toggleSelected(this.focusedValue);\n\t\t\t\t} else {\n\t\t\t\t\tthis.isNavigating = false;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tif (this.focusedValue) {\n\t\t\t\t\tthis.selectedValues = [this.focusedValue];\n\t\t\t\t}\n\t\t\t\tthis.isNavigating = false;\n\t\t\t}\n\t\t}\n\t}\n\n\tdeselectAll() {\n\t\tthis.selectedValues = [];\n\t}\n\n\ttoggleSelected(value: T['value']) {\n\t\tif (this.filteredOptions.length === 0) {\n\t\t\treturn;\n\t\t}\n\n\t\tif (this.multiple) {\n\t\t\tif (this.selectedValues.includes(value)) {\n\t\t\t\tthis.selectedValues = this.selectedValues.filter((v) => v !== value);\n\t\t\t} else {\n\t\t\t\tthis.selectedValues = [...this.selectedValues, value];\n\t\t\t}\n\t\t} else {\n\t\t\tthis.selectedValues = [value];\n\t\t}\n\t}\n\n\t#onUserInputChanged(value: string): void {\n\t\tif (value !== this.#lastUserInput) {\n\t\t\tthis.#lastUserInput = value;\n\n\t\t\tconst options = this.options;\n\n\t\t\tif (value) {\n\t\t\t\tthis.filteredOptions = options.filter((opt) => this.#filterFn(value, opt));\n\t\t\t} else {\n\t\t\t\tthis.filteredOptions = [...options];\n\t\t\t}\n\t\t\tthis.#cursor = getCursorForValue(this.focusedValue, this.filteredOptions);\n\t\t\tthis.focusedValue = this.filteredOptions[this.#cursor]?.value;\n\t\t\tif (!this.multiple) {\n\t\t\t\tif (this.focusedValue !== undefined) {\n\t\t\t\t\tthis.toggleSelected(this.focusedValue);\n\t\t\t\t} else {\n\t\t\t\t\tthis.deselectAll();\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n}\n", "import { cursor } from 'sisteransi';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface ConfirmOptions extends PromptOptions<boolean, ConfirmPrompt> {\n\tactive: string;\n\tinactive: string;\n\tinitialValue?: boolean;\n}\n\nexport default class ConfirmPrompt extends Prompt<boolean> {\n\tget cursor() {\n\t\treturn this.value ? 0 : 1;\n\t}\n\n\tprivate get _value() {\n\t\treturn this.cursor === 0;\n\t}\n\n\tconstructor(opts: ConfirmOptions) {\n\t\tsuper(opts, false);\n\t\tthis.value = !!opts.initialValue;\n\n\t\tthis.on('userInput', () => {\n\t\t\tthis.value = this._value;\n\t\t});\n\n\t\tthis.on('confirm', (confirm) => {\n\t\t\tthis.output.write(cursor.move(0, -1));\n\t\t\tthis.value = confirm;\n\t\t\tthis.state = 'submit';\n\t\t\tthis.close();\n\t\t});\n\n\t\tthis.on('cursor', () => {\n\t\t\tthis.value = !this.value;\n\t\t});\n\t}\n}\n", "import Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface GroupMultiSelectOptions<T extends { value: any }>\n\textends PromptOptions<T['value'][], GroupMultiSelectPrompt<T>> {\n\toptions: Record<string, T[]>;\n\tinitialValues?: T['value'][];\n\trequired?: boolean;\n\tcursorAt?: T['value'];\n\tselectableGroups?: boolean;\n}\nexport default class GroupMultiSelectPrompt<T extends { value: any }> extends Prompt<T['value'][]> {\n\toptions: (T & { group: string | boolean })[];\n\tcursor = 0;\n\t#selectableGroups: boolean;\n\n\tgetGroupItems(group: string): T[] {\n\t\treturn this.options.filter((o) => o.group === group);\n\t}\n\n\tisGroupSelected(group: string) {\n\t\tconst items = this.getGroupItems(group);\n\t\tconst value = this.value;\n\t\tif (value === undefined) {\n\t\t\treturn false;\n\t\t}\n\t\treturn items.every((i) => value.includes(i.value));\n\t}\n\n\tprivate toggleValue() {\n\t\tconst item = this.options[this.cursor];\n\t\tif (this.value === undefined) {\n\t\t\tthis.value = [];\n\t\t}\n\t\tif (item.group === true) {\n\t\t\tconst group = item.value;\n\t\t\tconst groupedItems = this.getGroupItems(group);\n\t\t\tif (this.isGroupSelected(group)) {\n\t\t\t\tthis.value = this.value.filter(\n\t\t\t\t\t(v: string) => groupedItems.findIndex((i) => i.value === v) === -1\n\t\t\t\t);\n\t\t\t} else {\n\t\t\t\tthis.value = [...this.value, ...groupedItems.map((i) => i.value)];\n\t\t\t}\n\t\t\tthis.value = Array.from(new Set(this.value));\n\t\t} else {\n\t\t\tconst selected = this.value.includes(item.value);\n\t\t\tthis.value = selected\n\t\t\t\t? this.value.filter((v: T['value']) => v !== item.value)\n\t\t\t\t: [...this.value, item.value];\n\t\t}\n\t}\n\n\tconstructor(opts: GroupMultiSelectOptions<T>) {\n\t\tsuper(opts, false);\n\t\tconst { options } = opts;\n\t\tthis.#selectableGroups = opts.selectableGroups !== false;\n\t\tthis.options = Object.entries(options).flatMap(([key, option]) => [\n\t\t\t{ value: key, group: true, label: key },\n\t\t\t...option.map((opt) => ({ ...opt, group: key })),\n\t\t]) as any;\n\t\tthis.value = [...(opts.initialValues ?? [])];\n\t\tthis.cursor = Math.max(\n\t\t\tthis.options.findIndex(({ value }) => value === opts.cursorAt),\n\t\t\tthis.#selectableGroups ? 0 : 1\n\t\t);\n\n\t\tthis.on('cursor', (key) => {\n\t\t\tswitch (key) {\n\t\t\t\tcase 'left':\n\t\t\t\tcase 'up': {\n\t\t\t\t\tthis.cursor = this.cursor === 0 ? this.options.length - 1 : this.cursor - 1;\n\t\t\t\t\tconst currentIsGroup = this.options[this.cursor]?.group === true;\n\t\t\t\t\tif (!this.#selectableGroups && currentIsGroup) {\n\t\t\t\t\t\tthis.cursor = this.cursor === 0 ? this.options.length - 1 : this.cursor - 1;\n\t\t\t\t\t}\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t\tcase 'down':\n\t\t\t\tcase 'right': {\n\t\t\t\t\tthis.cursor = this.cursor === this.options.length - 1 ? 0 : this.cursor + 1;\n\t\t\t\t\tconst currentIsGroup = this.options[this.cursor]?.group === true;\n\t\t\t\t\tif (!this.#selectableGroups && currentIsGroup) {\n\t\t\t\t\t\tthis.cursor = this.cursor === this.options.length - 1 ? 0 : this.cursor + 1;\n\t\t\t\t\t}\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t\tcase 'space':\n\t\t\t\t\tthis.toggleValue();\n\t\t\t\t\tbreak;\n\t\t\t}\n\t\t});\n\t}\n}\n", "export function findCursor<T extends { disabled?: boolean }>(\n\tcursor: number,\n\tdelta: number,\n\toptions: T[]\n) {\n\tconst newCursor = cursor + delta;\n\tconst maxCursor = Math.max(options.length - 1, 0);\n\tconst clampedCursor = newCursor < 0 ? maxCursor : newCursor > maxCursor ? 0 : newCursor;\n\tconst newOption = options[clampedCursor];\n\tif (newOption.disabled) {\n\t\treturn findCursor(clampedCursor, delta < 0 ? -1 : 1, options);\n\t}\n\treturn clampedCursor;\n}\n", "import { findCursor } from '../utils/cursor.js';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\ninterface OptionLike {\n\tvalue: any;\n\tdisabled?: boolean;\n}\n\nexport interface MultiSelectOptions<T extends OptionLike>\n\textends PromptOptions<T['value'][], MultiSelectPrompt<T>> {\n\toptions: T[];\n\tinitialValues?: T['value'][];\n\trequired?: boolean;\n\tcursorAt?: T['value'];\n}\nexport default class MultiSelectPrompt<T extends OptionLike> extends Prompt<T['value'][]> {\n\toptions: T[];\n\tcursor = 0;\n\n\tprivate get _value(): T['value'] {\n\t\treturn this.options[this.cursor].value;\n\t}\n\n\tprivate get _enabledOptions(): T[] {\n\t\treturn this.options.filter((option) => option.disabled !== true);\n\t}\n\n\tprivate toggleAll() {\n\t\tconst enabledOptions = this._enabledOptions;\n\t\tconst allSelected = this.value !== undefined && this.value.length === enabledOptions.length;\n\t\tthis.value = allSelected ? [] : enabledOptions.map((v) => v.value);\n\t}\n\n\tprivate toggleInvert() {\n\t\tconst value = this.value;\n\t\tif (!value) {\n\t\t\treturn;\n\t\t}\n\t\tconst notSelected = this._enabledOptions.filter((v) => !value.includes(v.value));\n\t\tthis.value = notSelected.map((v) => v.value);\n\t}\n\n\tprivate toggleValue() {\n\t\tif (this.value === undefined) {\n\t\t\tthis.value = [];\n\t\t}\n\t\tconst selected = this.value.includes(this._value);\n\t\tthis.value = selected\n\t\t\t? this.value.filter((value) => value !== this._value)\n\t\t\t: [...this.value, this._value];\n\t}\n\n\tconstructor(opts: MultiSelectOptions<T>) {\n\t\tsuper(opts, false);\n\n\t\tthis.options = opts.options;\n\t\tthis.value = [...(opts.initialValues ?? [])];\n\t\tconst cursor = Math.max(\n\t\t\tthis.options.findIndex(({ value }) => value === opts.cursorAt),\n\t\t\t0\n\t\t);\n\t\tthis.cursor = this.options[cursor].disabled ? findCursor<T>(cursor, 1, this.options) : cursor;\n\t\tthis.on('key', (char) => {\n\t\t\tif (char === 'a') {\n\t\t\t\tthis.toggleAll();\n\t\t\t}\n\t\t\tif (char === 'i') {\n\t\t\t\tthis.toggleInvert();\n\t\t\t}\n\t\t});\n\n\t\tthis.on('cursor', (key) => {\n\t\t\tswitch (key) {\n\t\t\t\tcase 'left':\n\t\t\t\tcase 'up':\n\t\t\t\t\tthis.cursor = findCursor<T>(this.cursor, -1, this.options);\n\t\t\t\t\tbreak;\n\t\t\t\tcase 'down':\n\t\t\t\tcase 'right':\n\t\t\t\t\tthis.cursor = findCursor<T>(this.cursor, 1, this.options);\n\t\t\t\t\tbreak;\n\t\t\t\tcase 'space':\n\t\t\t\t\tthis.toggleValue();\n\t\t\t\t\tbreak;\n\t\t\t}\n\t\t});\n\t}\n}\n", "import color from 'picocolors';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface PasswordOptions extends PromptOptions<string, PasswordPrompt> {\n\tmask?: string;\n}\nexport default class PasswordPrompt extends Prompt<string> {\n\tprivate _mask = '•';\n\tget cursor() {\n\t\treturn this._cursor;\n\t}\n\tget masked() {\n\t\treturn this.userInput.replaceAll(/./g, this._mask);\n\t}\n\tget userInputWithCursor() {\n\t\tif (this.state === 'submit' || this.state === 'cancel') {\n\t\t\treturn this.masked;\n\t\t}\n\t\tconst userInput = this.userInput;\n\t\tif (this.cursor >= userInput.length) {\n\t\t\treturn `${this.masked}${color.inverse(color.hidden('_'))}`;\n\t\t}\n\t\tconst masked = this.masked;\n\t\tconst s1 = masked.slice(0, this.cursor);\n\t\tconst s2 = masked.slice(this.cursor);\n\t\treturn `${s1}${color.inverse(s2[0])}${s2.slice(1)}`;\n\t}\n\tclear() {\n\t\tthis._clearUserInput();\n\t}\n\tconstructor({ mask, ...opts }: PasswordOptions) {\n\t\tsuper(opts);\n\t\tthis._mask = mask ?? '•';\n\t\tthis.on('userInput', (input) => {\n\t\t\tthis._setValue(input);\n\t\t});\n\t}\n}\n", "import { findCursor } from '../utils/cursor.js';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface SelectOptions<T extends { value: any; disabled?: boolean }>\n\textends PromptOptions<T['value'], SelectPrompt<T>> {\n\toptions: T[];\n\tinitialValue?: T['value'];\n}\nexport default class SelectPrompt<T extends { value: any; disabled?: boolean }> extends Prompt<\n\tT['value']\n> {\n\toptions: T[];\n\tcursor = 0;\n\n\tprivate get _selectedValue() {\n\t\treturn this.options[this.cursor];\n\t}\n\n\tprivate changeValue() {\n\t\tthis.value = this._selectedValue.value;\n\t}\n\n\tconstructor(opts: SelectOptions<T>) {\n\t\tsuper(opts, false);\n\n\t\tthis.options = opts.options;\n\n\t\tconst initialCursor = this.options.findIndex(({ value }) => value === opts.initialValue);\n\t\tconst cursor = initialCursor === -1 ? 0 : initialCursor;\n\t\tthis.cursor = this.options[cursor].disabled ? findCursor<T>(cursor, 1, this.options) : cursor;\n\t\tthis.changeValue();\n\n\t\tthis.on('cursor', (key) => {\n\t\t\tswitch (key) {\n\t\t\t\tcase 'left':\n\t\t\t\tcase 'up':\n\t\t\t\t\tthis.cursor = findCursor<T>(this.cursor, -1, this.options);\n\t\t\t\t\tbreak;\n\t\t\t\tcase 'down':\n\t\t\t\tcase 'right':\n\t\t\t\t\tthis.cursor = findCursor<T>(this.cursor, 1, this.options);\n\t\t\t\t\tbreak;\n\t\t\t}\n\t\t\tthis.changeValue();\n\t\t});\n\t}\n}\n", "import Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface SelectKeyOptions<T extends { value: string }>\n\textends PromptOptions<T['value'], SelectKeyPrompt<T>> {\n\toptions: T[];\n\tcaseSensitive?: boolean;\n}\nexport default class SelectKeyPrompt<T extends { value: string }> extends Prompt<T['value']> {\n\toptions: T[];\n\tcursor = 0;\n\n\tconstructor(opts: SelectKeyOptions<T>) {\n\t\tsuper(opts, false);\n\n\t\tthis.options = opts.options;\n\t\tconst caseSensitive = opts.caseSensitive === true;\n\t\tconst keys = this.options.map(({ value: [initial] }) => {\n\t\t\treturn caseSensitive ? initial : initial?.toLowerCase();\n\t\t});\n\t\tthis.cursor = Math.max(keys.indexOf(opts.initialValue), 0);\n\n\t\tthis.on('key', (key, keyInfo) => {\n\t\t\tif (!key) {\n\t\t\t\treturn;\n\t\t\t}\n\t\t\tconst casedKey = caseSensitive && keyInfo.shift ? key.toUpperCase() : key;\n\t\t\tif (!keys.includes(casedKey)) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst value = this.options.find(({ value: [initial] }) => {\n\t\t\t\treturn caseSensitive ? initial === casedKey : initial?.toLowerCase() === key;\n\t\t\t});\n\t\t\tif (value) {\n\t\t\t\tthis.value = value.value;\n\t\t\t\tthis.state = 'submit';\n\t\t\t\tthis.emit('submit');\n\t\t\t}\n\t\t});\n\t}\n}\n", "import color from 'picocolors';\nimport Prompt, { type PromptOptions } from './prompt.js';\n\nexport interface TextOptions extends PromptOptions<string, TextPrompt> {\n\tplaceholder?: string;\n\tdefaultValue?: string;\n}\n\nexport default class TextPrompt extends Prompt<string> {\n\tget userInputWithCursor() {\n\t\tif (this.state === 'submit') {\n\t\t\treturn this.userInput;\n\t\t}\n\t\tconst userInput = this.userInput;\n\t\tif (this.cursor >= userInput.length) {\n\t\t\treturn `${this.userInput}█`;\n\t\t}\n\t\tconst s1 = userInput.slice(0, this.cursor);\n\t\tconst [s2, ...s3] = userInput.slice(this.cursor);\n\t\treturn `${s1}${color.inverse(s2)}${s3.join('')}`;\n\t}\n\tget cursor() {\n\t\treturn this._cursor;\n\t}\n\tconstructor(opts: TextOptions) {\n\t\tsuper({\n\t\t\t...opts,\n\t\t\tinitialUserInput: opts.initialUserInput ?? opts.initialValue,\n\t\t});\n\n\t\tthis.on('userInput', (input) => {\n\t\t\tthis._setValue(input);\n\t\t});\n\t\tthis.on('finalize', () => {\n\t\t\tif (!this.value) {\n\t\t\t\tthis.value = opts.defaultValue;\n\t\t\t}\n\t\t\tif (this.value === undefined) {\n\t\t\t\tthis.value = '';\n\t\t\t}\n\t\t});\n\t}\n}\n", "import chalk from \"chalk\";\nimport { text } from \"../../components/text.js\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\nimport { stripAnsi } from \"../../internal/strip-ansi.js\";\n\nexport function intro(title: string): void {\n const format = resolveOutputFormat();\n if (format === \"json\") {\n return;\n }\n\n if (format === \"markdown\") {\n process.stdout.write(`# ${stripAnsi(title)}\\n\\n`);\n return;\n }\n\n process.stdout.write(`${chalk.gray(\"\u250C\")} ${text.intro(title)}\\n`);\n}\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\nimport { stripAnsi } from \"../../internal/strip-ansi.js\";\n\nfunction getVisibleWidth(value: string): number {\n return stripAnsi(value).length;\n}\n\nfunction renderTerminalNote(message: string, title?: string): string {\n const contentLines = [\"\", ...message.split(\"\\n\"), \"\"];\n const visibleTitle = stripAnsi(title ?? \"\");\n const contentWidth = Math.max(\n visibleTitle.length,\n ...contentLines.map((line) => getVisibleWidth(line))\n ) + 2;\n const titleLine = `${chalk.green(\"\u25C7\")} ${chalk.reset(title ?? \"\")} ${chalk.gray(\n `${\"\u2500\".repeat(Math.max(contentWidth - visibleTitle.length - 1, 1))}\u256E`\n )}`;\n const content = contentLines.map((line) => {\n const padding = \" \".repeat(contentWidth - getVisibleWidth(line));\n return `${chalk.gray(\"\u2502\")} ${line}${padding}${chalk.gray(\"\u2502\")}`;\n });\n const bottom = chalk.gray(`\u251C${\"\u2500\".repeat(contentWidth + 2)}\u256F`);\n\n return [chalk.gray(\"\u2502\"), titleLine, ...content, bottom].join(\"\\n\");\n}\n\nexport function note(message: string, title?: string): void {\n const format = resolveOutputFormat();\n const strippedMessage = stripAnsi(message);\n const strippedTitle = stripAnsi(title ?? \"\");\n\n if (format === \"markdown\") {\n const lines = strippedMessage.split(\"\\n\");\n const heading = strippedTitle ? `> **${strippedTitle}**\\n` : \"\";\n const body = lines.map((line) => `> ${line}`).join(\"\\n\");\n process.stdout.write(`${heading}${body}\\n`);\n return;\n }\n\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({\n type: \"note\",\n title: strippedTitle,\n message: strippedMessage\n })}\\n`\n );\n return;\n }\n\n process.stdout.write(`${renderTerminalNote(message, title)}\\n`);\n}\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\nimport { stripAnsi } from \"../../internal/strip-ansi.js\";\n\nexport function outro(message: string): void {\n const format = resolveOutputFormat();\n const stripped = stripAnsi(message);\n\n if (format === \"markdown\") {\n process.stdout.write(`---\\n${stripped}\\n`);\n return;\n }\n\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ type: \"outro\", message: stripped })}\\n`\n );\n return;\n }\n\n process.stdout.write(`${chalk.gray(\"\u2502\")}\\n${chalk.gray(\"\u2514\")} ${message}\\n\\n`);\n}\n", "import chalk from \"chalk\";\nimport { resolveOutputFormat } from \"../../internal/output-format.js\";\nimport { stripAnsi } from \"../../internal/strip-ansi.js\";\nimport { SPINNER_FRAMES } from \"../../static/spinner.js\";\n\nexport interface SpinnerOptions {\n start: (message?: string) => void;\n stop: (message?: string, code?: number) => void;\n message: (message?: string) => void;\n}\n\nfunction writeTerminalFrame(frame: string, message: string): void {\n process.stdout.write(`\\r\\x1b[K${frame} ${message}`);\n}\n\nexport function spinner(): SpinnerOptions {\n let currentMessage = \"\";\n let frameIndex = 0;\n let timer: ReturnType<typeof setInterval> | undefined;\n let fallback = false;\n\n const format = resolveOutputFormat();\n\n const renderFrame = (): void => {\n writeTerminalFrame(SPINNER_FRAMES[frameIndex % SPINNER_FRAMES.length], currentMessage);\n };\n\n const clearTimer = (): void => {\n if (!timer) {\n return;\n }\n clearInterval(timer);\n timer = undefined;\n };\n\n return {\n start(message = \"\"): void {\n currentMessage = stripAnsi(message);\n\n if (format === \"json\") {\n return;\n }\n\n if (format === \"markdown\") {\n process.stdout.write(`- ${currentMessage}...\\n`);\n return;\n }\n\n fallback = process.env.POE_NO_SPINNER === \"1\" || !process.stdout.isTTY;\n if (fallback) {\n process.stdout.write(`${currentMessage}\\n`);\n return;\n }\n\n frameIndex = 0;\n renderFrame();\n timer = setInterval(() => {\n frameIndex += 1;\n renderFrame();\n }, 16);\n },\n\n message(message = \"\"): void {\n currentMessage = stripAnsi(message);\n\n if (format !== \"terminal\" || fallback || !timer) {\n return;\n }\n\n renderFrame();\n },\n\n stop(message = currentMessage, code?: number): void {\n currentMessage = stripAnsi(message);\n\n if (format === \"json\") {\n process.stdout.write(\n `${JSON.stringify({ type: \"spinner\", state: \"stopped\", message: currentMessage })}\\n`\n );\n return;\n }\n\n if (format === \"markdown\") {\n process.stdout.write(`- ${currentMessage}\\n`);\n return;\n }\n\n clearTimer();\n\n const symbol = code === undefined || code === 0\n ? chalk.green(\"\u25C6\")\n : chalk.red(\"\u25A0\");\n\n if (fallback) {\n process.stdout.write(`${symbol} ${currentMessage}\\n`);\n return;\n }\n\n process.stdout.write(`\\r\\x1b[K${symbol} ${currentMessage}\\n`);\n }\n };\n}\n", "import chalk from \"chalk\";\nimport { symbols } from \"../components/symbols.js\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\n\nexport const SPINNER_FRAMES = [\"\u25D2\", \"\u25D0\", \"\u25D3\", \"\u25D1\"] as const;\n\nexport interface SpinnerFrameOptions {\n frame?: number;\n message: string;\n timer?: string;\n}\n\nexport function renderSpinnerFrame(options: SpinnerFrameOptions): string {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n return `- ${options.message}${options.timer ? ` [${options.timer}]` : \"\"}...\\n`;\n }\n\n if (format === \"json\") {\n return `${JSON.stringify({\n type: \"spinner\",\n state: \"running\",\n message: options.message,\n ...(options.timer ? { timer: options.timer } : {})\n })}\\n`;\n }\n\n const frame = options.frame ?? 0;\n const spinnerChar = chalk.magenta(SPINNER_FRAMES[frame % SPINNER_FRAMES.length]);\n const timerSuffix = options.timer ? chalk.dim(` [${options.timer}]`) : \"\";\n const bar = chalk.gray(symbols.bar);\n\n return `${spinnerChar} ${options.message}${timerSuffix}\\n${bar}`;\n}\n\nexport interface SpinnerStoppedOptions {\n message: string;\n code?: number;\n timer?: string;\n subtext?: string;\n}\n\nexport function renderSpinnerStopped(options: SpinnerStoppedOptions): string {\n const format = resolveOutputFormat();\n\n if (format === \"markdown\") {\n return `- ${options.message}${options.timer ? ` [${options.timer}]` : \"\"}\\n`;\n }\n\n if (format === \"json\") {\n return `${JSON.stringify({\n type: \"spinner\",\n state: \"stopped\",\n message: options.message,\n ...(options.timer ? { timer: options.timer } : {})\n })}\\n`;\n }\n\n const code = options.code ?? 0;\n const symbol = code === 0 ? chalk.green(\"\u25C6\") : chalk.red(\"\u25A0\");\n const timerSuffix = options.timer ? chalk.dim(` [${options.timer}]`) : \"\";\n const bar = chalk.gray(symbols.bar);\n\n let output = `${symbol} ${options.message}${timerSuffix}`;\n if (options.subtext) {\n output += `\\n${bar} ${chalk.dim(options.subtext)}`;\n }\n return output;\n}\n", "import chalk from \"chalk\";\nimport { symbols } from \"../components/symbols.js\";\nimport { resolveOutputFormat } from \"../internal/output-format.js\";\nimport { getTheme } from \"../internal/theme-detect.js\";\n\nexport interface MenuOption {\n label: string;\n value: string;\n hint?: string;\n}\n\nexport interface RenderMenuOptions {\n message: string;\n options: MenuOption[];\n selectedIndex?: number;\n}\n\nexport function renderMenu(opts: RenderMenuOptions): string {\n const format = resolveOutputFormat();\n const selectedIndex = opts.selectedIndex ?? 0;\n\n if (format === \"markdown\") {\n return [\n `**${opts.message}**`,\n ...opts.options.map(\n (option, index) => `- [${index === selectedIndex ? \"x\" : \" \"}] ${option.label}`\n )\n ].join(\"\\n\");\n }\n\n if (format === \"json\") {\n return JSON.stringify({\n type: \"menu\",\n message: opts.message,\n options: opts.options,\n selected: selectedIndex\n });\n }\n\n const theme = getTheme();\n const bar = chalk.gray(symbols.bar);\n const lines: string[] = [];\n\n lines.push(`${chalk.cyan(symbols.active)} ${opts.message}`);\n lines.push(bar);\n\n opts.options.forEach((option, index) => {\n const isSelected = index === selectedIndex;\n const prefix = isSelected ? chalk.cyan(symbols.active) : chalk.gray(symbols.inactive);\n const label = isSelected ? theme.accent(option.label) : option.label;\n const hint = option.hint ? chalk.dim(` (${option.hint})`) : \"\";\n lines.push(`${bar} ${prefix} ${label}${hint}`);\n });\n\n lines.push(`${bar}`);\n return lines.join(\"\\n\");\n}\n", "import { spawn as spawnChildProcess } from \"node:child_process\";\nimport { getAdapter } from \"../adapters/index.js\";\nimport type { AcpEvent } from \"./types.js\";\nimport { readLines } from \"./line-reader.js\";\nimport { resolveConfig } from \"../configs/resolve-config.js\";\nimport { getMcpArgs, getMcpEnv } from \"../mcp-args.js\";\nimport { stripModelNamespace } from \"../model-utils.js\";\nimport type { SpawnOptions, SpawnResult } from \"../types.js\";\n\nfunction createAbortError(): Error {\n const error = new Error(\"Agent spawn aborted\");\n error.name = \"AbortError\";\n return error;\n}\n\nfunction createActivityTimeoutError(timeoutMs: number): Error {\n const error = new Error(\n `Agent spawn timed out after ${timeoutMs / 1000}s of inactivity`\n );\n error.name = \"ActivityTimeoutError\";\n return error;\n}\n\nexport interface SpawnStreamingOptions extends SpawnOptions {\n agentId: string;\n}\n\nexport interface SpawnStreamingResult {\n events: AsyncIterable<AcpEvent>;\n done: Promise<SpawnResult>;\n}\n\nfunction isAcpEvent(value: unknown): value is AcpEvent {\n return !!value && typeof value === \"object\" && \"event\" in value;\n}\n\nexport function spawnStreaming(options: SpawnStreamingOptions): SpawnStreamingResult {\n if (options.signal?.aborted) {\n throw createAbortError();\n }\n\n const { agentId, binaryName, spawnConfig } = resolveConfig(options.agentId);\n\n if (spawnConfig === undefined) {\n throw new Error(`Agent \"${agentId}\" has no spawn config.`);\n }\n\n if (spawnConfig.kind !== \"cli\") {\n throw new Error(`Agent \"${agentId}\" does not support CLI spawn.`);\n }\n\n if (!binaryName) {\n throw new Error(`Agent \"${agentId}\" has no binaryName.`);\n }\n\n const mcpArgs = getMcpArgs(spawnConfig, options.mcpServers);\n const mcpEnvVars = getMcpEnv(spawnConfig, options.mcpServers);\n const args: string[] = [];\n\n if (spawnConfig.mcpArgsBeforeCommand) {\n args.push(...mcpArgs);\n }\n\n args.push(spawnConfig.promptFlag);\n\n const useStdin = !!options.useStdin && !!spawnConfig.stdinMode;\n if (!useStdin || !spawnConfig.stdinMode?.omitPrompt) {\n args.push(options.prompt);\n }\n\n if (options.model && spawnConfig.modelFlag) {\n let model = spawnConfig.modelStripProviderPrefix\n ? stripModelNamespace(options.model)\n : options.model;\n if (spawnConfig.modelTransform) model = spawnConfig.modelTransform(model);\n args.push(spawnConfig.modelFlag, model);\n }\n\n args.push(...spawnConfig.defaultArgs);\n\n if (!spawnConfig.mcpArgsBeforeCommand) {\n args.push(...mcpArgs);\n }\n\n const mode = options.mode ?? \"yolo\";\n args.push(...spawnConfig.modes[mode]);\n\n if (useStdin) {\n args.push(...spawnConfig.stdinMode!.extraArgs);\n }\n\n if (options.args && options.args.length > 0) {\n args.push(...options.args);\n }\n\n const child = spawnChildProcess(binaryName, args, {\n cwd: options.cwd,\n stdio: [\"pipe\", \"pipe\", \"pipe\"],\n env: Object.keys(mcpEnvVars).length > 0\n ? { ...process.env, ...mcpEnvVars }\n : undefined\n });\n let aborted = false;\n let timedOut = false;\n const onAbort = () => {\n aborted = true;\n child.kill(\"SIGTERM\");\n };\n options.signal?.addEventListener(\"abort\", onAbort, { once: true });\n\n let activityTimer: ReturnType<typeof setTimeout> | undefined;\n const resetActivityTimer = options.activityTimeoutMs\n ? () => {\n if (activityTimer) clearTimeout(activityTimer);\n activityTimer = setTimeout(() => {\n timedOut = true;\n child.kill(\"SIGTERM\");\n }, options.activityTimeoutMs);\n }\n : undefined;\n\n resetActivityTimer?.();\n\n const result: SpawnResult = { stdout: \"\", stderr: \"\", exitCode: 1 };\n child.stderr.setEncoding(\"utf8\");\n child.stderr.on(\"data\", (chunk) => {\n result.stderr += chunk;\n resetActivityTimer?.();\n });\n\n if (useStdin) {\n child.stdin.write(options.prompt);\n }\n child.stdin.end();\n\n const adapter = getAdapter(spawnConfig.adapter);\n\n const events: AsyncIterable<AcpEvent> = (async function* () {\n for await (const output of adapter(readLines(child.stdout))) {\n if (!isAcpEvent(output)) continue;\n resetActivityTimer?.();\n yield output;\n }\n })();\n\n const done = new Promise<SpawnResult>((resolve, reject) => {\n child.on(\"error\", (error) => {\n options.signal?.removeEventListener(\"abort\", onAbort);\n if (activityTimer) clearTimeout(activityTimer);\n if (aborted) {\n reject(createAbortError());\n return;\n }\n reject(error);\n });\n\n child.on(\"close\", (code) => {\n options.signal?.removeEventListener(\"abort\", onAbort);\n if (activityTimer) clearTimeout(activityTimer);\n if (aborted) {\n reject(createAbortError());\n return;\n }\n if (timedOut) {\n reject(createActivityTimeoutError(options.activityTimeoutMs!));\n return;\n }\n result.exitCode = code ?? 1;\n resolve(result);\n });\n });\n\n return { events, done };\n}\n", "import { isAbsolute } from \"node:path\";\nimport {\n AcpTransport,\n type AcpTransportClosedEvent,\n type AcpTransportOptions,\n} from \"./acp-transport.js\";\nimport type { JsonRpcRequestOptions } from \"./jsonrpc-message-layer.js\";\nimport {\n ACP_ERROR_CODE_INVALID_PARAMS,\n ACP_ERROR_CODE_RESOURCE_NOT_FOUND,\n AcpError,\n type AgentCapabilities,\n type AuthenticateResponse,\n type AuthMethod,\n type CancelNotification,\n type ClientCapabilities,\n type ContentBlock,\n type CreateTerminalRequest,\n type CreateTerminalResponse,\n type EnvVariable,\n type Implementation,\n type InitializeResponse,\n type KillTerminalCommandRequest,\n type KillTerminalCommandResponse,\n type LoadSessionResponse,\n type McpServer,\n type NewSessionResponse,\n type PermissionOption,\n type PromptResponse,\n type RequestId,\n type SessionConfigId,\n type SessionConfigOption,\n type SessionConfigValueId,\n type RequestPermissionOutcome,\n type RequestPermissionRequest,\n type RequestPermissionResponse,\n type ReadTextFileRequest,\n type ReadTextFileResponse,\n type ProtocolVersion,\n type SessionModeId,\n type SessionNotification,\n type SessionId,\n type SetSessionModeResponse,\n type SessionUpdateNotification,\n type ToolCallUpdate,\n type TerminalOutputRequest,\n type TerminalOutputResponse,\n type WaitForTerminalExitRequest,\n type WaitForTerminalExitResponse,\n type WriteTextFileRequest,\n type WriteTextFileResponse,\n type ReleaseTerminalRequest,\n type ReleaseTerminalResponse,\n} from \"./types.js\";\n\nexport type AcpClientState = \"uninitialized\" | \"initialized\" | \"ready\";\ntype ExtensionMethod = `_${string}`;\n\nexport interface PromptTurn extends AsyncIterable<SessionUpdateNotification> {\n response: Promise<PromptResponse>;\n}\n\ninterface AsyncQueue<T> extends AsyncIterable<T>, AsyncIterator<T> {\n push(value: T): void;\n complete(): void;\n fail(error: Error): void;\n}\n\nexport interface AcpClientFsHandler {\n readTextFile?: (args: {\n sessionId: SessionId;\n path: string;\n line?: number | null;\n limit?: number | null;\n }) => string | Promise<string>;\n writeTextFile?: (args: {\n sessionId: SessionId;\n path: string;\n content: string;\n }) => void | Promise<void>;\n}\n\nexport interface AcpClientTerminalHandler {\n create: (args: {\n sessionId: SessionId;\n command: string;\n args?: string[];\n cwd?: string | null;\n env?: EnvVariable[];\n outputByteLimit?: number | null;\n }) => string | Promise<string>;\n output: (args: {\n sessionId: SessionId;\n terminalId: string;\n }) => TerminalOutputResponse | Promise<TerminalOutputResponse>;\n waitForExit: (args: {\n sessionId: SessionId;\n terminalId: string;\n }) => WaitForTerminalExitResponse | Promise<WaitForTerminalExitResponse>;\n kill: (args: { sessionId: SessionId; terminalId: string }) => void | Promise<void>;\n release: (args: {\n sessionId: SessionId;\n terminalId: string;\n }) => void | Promise<void>;\n}\n\ntype AcpClientPermissionHandler = (args: {\n toolCall: ToolCallUpdate;\n options: PermissionOption[];\n}) => RequestPermissionOutcome | Promise<RequestPermissionOutcome>;\n\nexport interface AcpClientHandlers {\n permission?: AcpClientPermissionHandler;\n fs?: AcpClientFsHandler;\n terminal?: AcpClientTerminalHandler;\n}\n\ntype AcpClientTransport = Pick<\n AcpTransport,\n \"sendRequest\" | \"sendNotification\" | \"onRequest\" | \"onNotification\"\n> &\n Partial<Pick<AcpTransport, \"dispose\" | \"closed\">>;\n\ninterface AcpClientSharedOptions {\n protocolVersion?: ProtocolVersion;\n clientCapabilities?: ClientCapabilities;\n clientInfo?: Implementation | null;\n handlers?: AcpClientHandlers;\n permissionHandler?: AcpClientPermissionHandler;\n fsHandler?: AcpClientFsHandler;\n terminalHandler?: AcpClientTerminalHandler;\n skipAuth?: boolean;\n /**\n * Automatically approve all permission requests (selects the first\n * \"allow_always\" or \"allow_once\" option). Ignored when a custom\n * `permissionHandler` is provided.\n */\n autoApprove?: boolean;\n}\n\nexport interface AcpClientProcessOptions extends AcpClientSharedOptions {\n command: string;\n args?: readonly string[];\n cwd?: string;\n env?: NodeJS.ProcessEnv;\n requestTimeoutMs?: number;\n firstRequestId?: number;\n spawn?: AcpTransportOptions[\"spawn\"];\n}\n\nexport interface AcpClientInjectedTransportOptions extends AcpClientSharedOptions {\n transport: AcpClientTransport;\n}\n\nexport type AcpClientOptions = AcpClientProcessOptions | AcpClientInjectedTransportOptions;\n\nfunction toError(reason: unknown): Error {\n return reason instanceof Error ? reason : new Error(String(reason));\n}\n\nfunction invalidParams(message: string): AcpError {\n return new AcpError(\n ACP_ERROR_CODE_INVALID_PARAMS,\n `Invalid params: ${message}`\n );\n}\n\nfunction resourceNotFound(resource: string): AcpError {\n return new AcpError(\n ACP_ERROR_CODE_RESOURCE_NOT_FOUND,\n `Resource not found: ${resource}`\n );\n}\n\nfunction assertAbsolutePath(path: string): void {\n if (!isAbsolute(path)) {\n throw invalidParams('\"path\" must be an absolute path');\n }\n}\n\nfunction assertOneBasedLineNumber(line: number | null | undefined): void {\n if (line === null || line === undefined) {\n return;\n }\n\n if (!Number.isInteger(line) || line < 1) {\n throw invalidParams('\"line\" must be a 1-based integer');\n }\n}\n\nfunction assertExtensionMethod(method: string): asserts method is ExtensionMethod {\n if (!method.startsWith(\"_\")) {\n throw new Error('Extension method must start with \"_\"');\n }\n}\n\nfunction isInjectedTransportOptions(\n options: AcpClientOptions\n): options is AcpClientInjectedTransportOptions {\n return \"transport\" in options;\n}\n\nfunction createAsyncQueue<T>(): AsyncQueue<T> {\n const values: T[] = [];\n const waiters: Array<{\n resolve: (result: IteratorResult<T>) => void;\n reject: (error: Error) => void;\n }> = [];\n let closed = false;\n let failure: Error | null = null;\n\n const resolveOne = (value: T): boolean => {\n const waiter = waiters.shift();\n if (!waiter) {\n return false;\n }\n\n waiter.resolve({ done: false, value });\n return true;\n };\n\n const iterator: AsyncQueue<T> = {\n push(value: T): void {\n if (closed || failure) {\n return;\n }\n\n if (!resolveOne(value)) {\n values.push(value);\n }\n },\n complete(): void {\n if (closed || failure) {\n return;\n }\n\n closed = true;\n while (waiters.length > 0) {\n waiters.shift()?.resolve({ done: true, value: undefined });\n }\n },\n fail(error: Error): void {\n if (closed || failure) {\n return;\n }\n\n failure = error;\n while (waiters.length > 0) {\n waiters.shift()?.reject(error);\n }\n },\n async next(): Promise<IteratorResult<T>> {\n if (values.length > 0) {\n const value = values.shift() as T;\n return { done: false, value };\n }\n\n if (failure) {\n throw failure;\n }\n\n if (closed) {\n return { done: true, value: undefined };\n }\n\n return new Promise<IteratorResult<T>>((resolve, reject) => {\n waiters.push({ resolve, reject });\n });\n },\n async return(): Promise<IteratorResult<T>> {\n iterator.complete();\n return { done: true, value: undefined };\n },\n async throw(error: unknown): Promise<IteratorResult<T>> {\n const normalizedError = toError(error);\n iterator.fail(normalizedError);\n throw normalizedError;\n },\n [Symbol.asyncIterator](): AsyncIterator<T> {\n return iterator;\n },\n };\n\n return iterator;\n}\n\nexport class AcpClient {\n private readonly transport: AcpClientTransport;\n private readonly clientProtocolVersion: ProtocolVersion;\n private clientCapabilities?: ClientCapabilities;\n private readonly clientInfo?: Implementation | null;\n private readonly skipAuth: boolean;\n private readonly permissionHandler?: AcpClientPermissionHandler;\n private readonly fsHandler?: AcpClientFsHandler;\n private readonly terminalHandler?: AcpClientTerminalHandler;\n private readonly activePromptUpdates = new Map<\n SessionId,\n AsyncQueue<SessionUpdateNotification>\n >();\n private readonly trackedTerminalIds = new Map<SessionId, Set<string>>();\n private hasRegisteredFsReadHandler = false;\n private hasRegisteredFsWriteHandler = false;\n private hasRegisteredTerminalHandlers = false;\n private disposed = false;\n\n private lifecycleState: AcpClientState = \"uninitialized\";\n private negotiatedVersion: ProtocolVersion | null = null;\n private availableAuthMethods: AuthMethod[] = [];\n private negotiatedAgentCapabilities: AgentCapabilities | undefined;\n private negotiatedAgentInfo: Implementation | null | undefined;\n\n constructor(options: AcpClientOptions) {\n this.transport = isInjectedTransportOptions(options)\n ? options.transport\n : new AcpTransport({\n command: options.command,\n args: options.args,\n cwd: options.cwd,\n env: options.env,\n requestTimeoutMs: options.requestTimeoutMs,\n firstRequestId: options.firstRequestId,\n spawn: options.spawn,\n });\n this.clientProtocolVersion = options.protocolVersion ?? 1;\n this.clientCapabilities = options.clientCapabilities;\n this.clientInfo = options.clientInfo;\n this.skipAuth = options.skipAuth ?? false;\n this.permissionHandler = options.handlers?.permission ?? options.permissionHandler;\n this.fsHandler = options.handlers?.fs ?? options.fsHandler;\n this.terminalHandler = options.handlers?.terminal ?? options.terminalHandler;\n\n const autoApprove = options.autoApprove === true && !this.permissionHandler;\n\n this.transport.onRequest(\n \"session/request_permission\",\n async (params: RequestPermissionRequest): Promise<RequestPermissionResponse> => {\n if (this.permissionHandler) {\n const outcome = await this.permissionHandler({\n toolCall: params.toolCall,\n options: params.options,\n });\n return { outcome };\n }\n\n if (autoApprove) {\n const allow =\n params.options.find((o: PermissionOption) => o.kind === \"allow_always\") ??\n params.options.find((o: PermissionOption) => o.kind === \"allow_once\");\n if (allow) {\n return { outcome: { outcome: \"selected\", optionId: allow.optionId } };\n }\n }\n\n return { outcome: { outcome: \"cancelled\" } };\n }\n );\n\n this.registerCapabilityHandlers(this.clientCapabilities);\n\n this.transport.onNotification(\"session/update\", (params: SessionNotification) => {\n this.handleSessionUpdateNotification(params);\n });\n }\n\n get state(): AcpClientState {\n return this.lifecycleState;\n }\n\n get negotiatedProtocolVersion(): ProtocolVersion | null {\n return this.negotiatedVersion;\n }\n\n get authMethods(): AuthMethod[] {\n return [...this.availableAuthMethods];\n }\n\n get agentCapabilities(): AgentCapabilities | undefined {\n return this.negotiatedAgentCapabilities;\n }\n\n get agentInfo(): Implementation | null | undefined {\n return this.negotiatedAgentInfo;\n }\n\n get closed(): Promise<AcpTransportClosedEvent> | undefined {\n return this.transport.closed;\n }\n\n async initialize(clientCapabilities?: ClientCapabilities): Promise<InitializeResponse> {\n if (this.lifecycleState !== \"uninitialized\") {\n throw new Error(\"initialize() can only be called once.\");\n }\n\n if (clientCapabilities !== undefined) {\n this.clientCapabilities = clientCapabilities;\n this.registerCapabilityHandlers(clientCapabilities);\n }\n\n const response = await this.transport.sendRequest(\"initialize\", {\n protocolVersion: this.clientProtocolVersion,\n clientInfo: this.clientInfo,\n clientCapabilities: this.clientCapabilities,\n });\n\n const negotiatedProtocolVersion = Math.min(\n this.clientProtocolVersion,\n response.protocolVersion\n );\n\n this.negotiatedVersion = negotiatedProtocolVersion;\n this.negotiatedAgentCapabilities = response.agentCapabilities;\n this.negotiatedAgentInfo = response.agentInfo;\n this.availableAuthMethods = response.authMethods ? [...response.authMethods] : [];\n\n const requiresAuth = this.availableAuthMethods.length > 0 && !this.skipAuth;\n this.lifecycleState = requiresAuth ? \"initialized\" : \"ready\";\n\n return {\n protocolVersion: negotiatedProtocolVersion,\n ...(this.negotiatedAgentCapabilities !== undefined\n ? { agentCapabilities: this.negotiatedAgentCapabilities }\n : {}),\n ...(this.negotiatedAgentInfo !== undefined ? { agentInfo: this.negotiatedAgentInfo } : {}),\n ...(this.availableAuthMethods.length > 0 ? { authMethods: this.authMethods } : {}),\n };\n }\n\n async authenticate(methodId: string): Promise<AuthenticateResponse> {\n if (this.lifecycleState === \"uninitialized\") {\n throw new Error(\"Cannot authenticate before initialize().\");\n }\n\n if (this.lifecycleState === \"ready\") {\n throw new Error(\"Authentication is not required for this agent.\");\n }\n\n if (!this.availableAuthMethods.some((authMethod) => authMethod.id === methodId)) {\n throw new Error(`Unknown auth method \"${methodId}\".`);\n }\n\n const response = await this.transport.sendRequest(\"authenticate\", {\n methodId,\n });\n\n this.lifecycleState = \"ready\";\n return response;\n }\n\n async newSession(cwd: string, mcpServers: McpServer[]): Promise<NewSessionResponse> {\n this.assertReady(\"session/new\");\n this.assertMcpServerCapabilitySupport(mcpServers);\n\n return this.transport.sendRequest(\"session/new\", {\n cwd,\n mcpServers,\n });\n }\n\n async loadSession(\n sessionId: SessionId,\n cwd: string,\n mcpServers: McpServer[]\n ): Promise<LoadSessionResponse> {\n this.assertReady(\"session/load\");\n if (this.negotiatedAgentCapabilities?.loadSession !== true) {\n throw new Error(\n 'Cannot call \"session/load\" because the agent does not support session loading.'\n );\n }\n this.assertMcpServerCapabilitySupport(mcpServers);\n\n return this.transport.sendRequest(\"session/load\", {\n sessionId,\n cwd,\n mcpServers,\n });\n }\n\n async cancelSession(sessionId: SessionId): Promise<void> {\n this.assertReady(\"session/cancel\");\n const payload: CancelNotification = { sessionId };\n this.transport.sendNotification(\"session/cancel\", payload);\n }\n\n async setMode(\n sessionId: SessionId,\n modeId: SessionModeId\n ): Promise<SetSessionModeResponse> {\n this.assertReady(\"session/set_mode\");\n return this.transport.sendRequest(\"session/set_mode\", {\n sessionId,\n modeId,\n });\n }\n\n async setConfigOption(\n sessionId: SessionId,\n configId: SessionConfigId,\n value: SessionConfigValueId\n ): Promise<SessionConfigOption[]> {\n this.assertReady(\"session/set_config_option\");\n const response = await this.transport.sendRequest(\"session/set_config_option\", {\n sessionId,\n configId,\n value,\n });\n\n return response.configOptions;\n }\n\n prompt(sessionId: SessionId, content: ContentBlock[]): PromptTurn {\n this.assertReady(\"session/prompt\");\n this.assertPromptContentCapabilitySupport(content);\n\n if (this.activePromptUpdates.has(sessionId)) {\n throw new Error(\n `Cannot call \"session/prompt\" while another prompt is in progress for session \"${sessionId}\".`\n );\n }\n\n const updates = createAsyncQueue<SessionUpdateNotification>();\n this.activePromptUpdates.set(sessionId, updates);\n\n let requestPromise: Promise<PromptResponse>;\n try {\n requestPromise = this.transport.sendRequest(\"session/prompt\", {\n sessionId,\n prompt: content,\n });\n } catch (error) {\n const normalizedError = toError(error);\n this.activePromptUpdates.delete(sessionId);\n updates.fail(normalizedError);\n throw normalizedError;\n }\n\n const response = requestPromise\n .then((promptResponse) => {\n this.activePromptUpdates.delete(sessionId);\n updates.complete();\n return promptResponse;\n })\n .catch((error) => {\n const normalizedError = toError(error);\n this.activePromptUpdates.delete(sessionId);\n updates.fail(normalizedError);\n throw normalizedError;\n });\n\n return {\n response,\n [Symbol.asyncIterator](): AsyncIterator<SessionUpdateNotification> {\n return updates;\n },\n };\n }\n\n async sendExtRequest<TResult = unknown>(\n method: ExtensionMethod,\n params?: unknown,\n options?: JsonRpcRequestOptions\n ): Promise<TResult>;\n async sendExtRequest<TResult = unknown>(\n method: string,\n params?: unknown,\n options: JsonRpcRequestOptions = {}\n ): Promise<TResult> {\n assertExtensionMethod(method);\n return this.transport.sendRequest(method, params, options) as Promise<TResult>;\n }\n\n async sendExtNotification(method: ExtensionMethod, params?: unknown): Promise<void>;\n async sendExtNotification(method: string, params?: unknown): Promise<void> {\n assertExtensionMethod(method);\n this.transport.sendNotification(method, params);\n }\n\n onExtRequest<TMethod extends ExtensionMethod>(\n method: TMethod,\n handler: (\n params: unknown,\n context: { id: RequestId; method: TMethod }\n ) => unknown | Promise<unknown>\n ): void;\n onExtRequest(\n method: string,\n handler: (params: unknown, context: { id: RequestId; method: string }) => unknown\n ): void {\n assertExtensionMethod(method);\n this.transport.onRequest(method, handler);\n }\n\n onExtNotification<TMethod extends ExtensionMethod>(\n method: TMethod,\n handler: (params: unknown, context: { method: TMethod }) => void | Promise<void>\n ): void;\n onExtNotification(\n method: string,\n handler: (params: unknown, context: { method: string }) => void | Promise<void>\n ): void {\n assertExtensionMethod(method);\n this.transport.onNotification(method, handler);\n }\n\n async dispose(): Promise<void> {\n if (this.disposed) {\n if (this.transport.closed) {\n await this.transport.closed;\n }\n return;\n }\n\n this.disposed = true;\n const disposeReason = new Error(\"ACP client disposed\");\n for (const updates of this.activePromptUpdates.values()) {\n updates.fail(disposeReason);\n }\n this.activePromptUpdates.clear();\n\n if (typeof this.transport.dispose === \"function\") {\n this.transport.dispose(disposeReason);\n }\n\n if (this.transport.closed) {\n await this.transport.closed;\n }\n }\n\n assertReady(operation: string): void {\n if (this.lifecycleState === \"ready\") {\n return;\n }\n\n if (this.lifecycleState === \"uninitialized\") {\n throw new Error(`Cannot call \"${operation}\" before initialize().`);\n }\n\n throw new Error(`Cannot call \"${operation}\" before authentication completes.`);\n }\n\n private registerCapabilityHandlers(capabilities: ClientCapabilities | undefined): void {\n if (\n !this.hasRegisteredFsReadHandler &&\n capabilities?.fs?.readTextFile === true &&\n this.fsHandler?.readTextFile\n ) {\n const readTextFile = this.fsHandler.readTextFile;\n this.transport.onRequest(\n \"fs/read_text_file\",\n async (params: ReadTextFileRequest): Promise<ReadTextFileResponse> => {\n assertAbsolutePath(params.path);\n assertOneBasedLineNumber(params.line);\n\n const content = await readTextFile({\n sessionId: params.sessionId,\n path: params.path,\n line: params.line,\n limit: params.limit,\n });\n return { content };\n }\n );\n this.hasRegisteredFsReadHandler = true;\n }\n\n if (\n !this.hasRegisteredFsWriteHandler &&\n capabilities?.fs?.writeTextFile === true &&\n this.fsHandler?.writeTextFile\n ) {\n const writeTextFile = this.fsHandler.writeTextFile;\n this.transport.onRequest(\n \"fs/write_text_file\",\n async (params: WriteTextFileRequest): Promise<WriteTextFileResponse> => {\n assertAbsolutePath(params.path);\n\n await writeTextFile({\n sessionId: params.sessionId,\n path: params.path,\n content: params.content,\n });\n\n return {};\n }\n );\n this.hasRegisteredFsWriteHandler = true;\n }\n\n if (\n !this.hasRegisteredTerminalHandlers &&\n capabilities?.terminal === true &&\n this.terminalHandler\n ) {\n const terminalHandler = this.terminalHandler;\n this.transport.onRequest(\n \"terminal/create\",\n async (params: CreateTerminalRequest): Promise<CreateTerminalResponse> => {\n const terminalId = await terminalHandler.create({\n sessionId: params.sessionId,\n command: params.command,\n args: params.args,\n cwd: params.cwd,\n env: params.env,\n outputByteLimit: params.outputByteLimit,\n });\n this.trackTerminal(params.sessionId, terminalId);\n\n return { terminalId };\n }\n );\n\n this.transport.onRequest(\n \"terminal/output\",\n async (params: TerminalOutputRequest): Promise<TerminalOutputResponse> => {\n this.assertKnownTerminal(params.sessionId, params.terminalId);\n\n return terminalHandler.output({\n sessionId: params.sessionId,\n terminalId: params.terminalId,\n });\n }\n );\n\n this.transport.onRequest(\n \"terminal/wait_for_exit\",\n async (params: WaitForTerminalExitRequest): Promise<WaitForTerminalExitResponse> => {\n this.assertKnownTerminal(params.sessionId, params.terminalId);\n\n return terminalHandler.waitForExit({\n sessionId: params.sessionId,\n terminalId: params.terminalId,\n });\n }\n );\n\n this.transport.onRequest(\n \"terminal/kill\",\n async (params: KillTerminalCommandRequest): Promise<KillTerminalCommandResponse> => {\n this.assertKnownTerminal(params.sessionId, params.terminalId);\n\n await terminalHandler.kill({\n sessionId: params.sessionId,\n terminalId: params.terminalId,\n });\n\n return {};\n }\n );\n\n this.transport.onRequest(\n \"terminal/release\",\n async (params: ReleaseTerminalRequest): Promise<ReleaseTerminalResponse> => {\n this.assertKnownTerminal(params.sessionId, params.terminalId);\n\n await terminalHandler.release({\n sessionId: params.sessionId,\n terminalId: params.terminalId,\n });\n this.untrackTerminal(params.sessionId, params.terminalId);\n\n return {};\n }\n );\n this.hasRegisteredTerminalHandlers = true;\n }\n }\n\n private assertMcpServerCapabilitySupport(mcpServers: McpServer[]): void {\n const mcpCapabilities = this.negotiatedAgentCapabilities?.mcpCapabilities;\n\n for (const mcpServer of mcpServers) {\n if (!(\"type\" in mcpServer)) {\n continue;\n }\n\n if (mcpServer.type === \"http\" && mcpCapabilities?.http !== true) {\n throw new Error('Agent does not support MCP server type \"http\".');\n }\n\n if (mcpServer.type === \"sse\" && mcpCapabilities?.sse !== true) {\n throw new Error('Agent does not support MCP server type \"sse\".');\n }\n }\n }\n\n private handleSessionUpdateNotification(notification: SessionNotification): void {\n const activePrompt = this.activePromptUpdates.get(notification.sessionId);\n if (!activePrompt) {\n return;\n }\n\n activePrompt.push({\n jsonrpc: \"2.0\",\n method: \"session/update\",\n params: notification,\n });\n }\n\n private trackTerminal(sessionId: SessionId, terminalId: string): void {\n const sessionTerminals = this.trackedTerminalIds.get(sessionId);\n if (sessionTerminals) {\n sessionTerminals.add(terminalId);\n return;\n }\n\n this.trackedTerminalIds.set(sessionId, new Set([terminalId]));\n }\n\n private assertKnownTerminal(sessionId: SessionId, terminalId: string): void {\n const sessionTerminals = this.trackedTerminalIds.get(sessionId);\n if (sessionTerminals?.has(terminalId) === true) {\n return;\n }\n\n throw resourceNotFound(`terminal \"${terminalId}\"`);\n }\n\n private untrackTerminal(sessionId: SessionId, terminalId: string): void {\n const sessionTerminals = this.trackedTerminalIds.get(sessionId);\n if (!sessionTerminals) {\n return;\n }\n\n sessionTerminals.delete(terminalId);\n if (sessionTerminals.size === 0) {\n this.trackedTerminalIds.delete(sessionId);\n }\n }\n\n private assertPromptContentCapabilitySupport(content: ContentBlock[]): void {\n const promptCapabilities = this.negotiatedAgentCapabilities?.promptCapabilities;\n\n for (const block of content) {\n if (block.type === \"image\" && promptCapabilities?.image !== true) {\n throw new Error('Agent does not support prompt content type \"image\".');\n }\n\n if (block.type === \"audio\" && promptCapabilities?.audio !== true) {\n throw new Error('Agent does not support prompt content type \"audio\".');\n }\n\n if (block.type === \"resource\" && promptCapabilities?.embeddedContext !== true) {\n throw new Error('Agent does not support prompt content type \"resource\".');\n }\n }\n }\n}\n", "import {\n spawn as spawnChildProcess,\n type ChildProcessWithoutNullStreams,\n type SpawnOptionsWithoutStdio,\n} from \"node:child_process\";\nimport {\n JsonRpcMessageLayer,\n type JsonRpcNotificationHandler,\n type JsonRpcRequestHandler,\n type JsonRpcRequestOptions,\n} from \"./jsonrpc-message-layer.js\";\nimport type {\n AuthenticateRequest,\n AuthenticateResponse,\n CancelNotification,\n CreateTerminalRequest,\n CreateTerminalResponse,\n InitializeRequest,\n InitializeResponse,\n KillTerminalCommandRequest,\n KillTerminalCommandResponse,\n LoadSessionRequest,\n LoadSessionResponse,\n NewSessionRequest,\n NewSessionResponse,\n PromptRequest,\n PromptResponse,\n ReadTextFileRequest,\n ReadTextFileResponse,\n ReleaseTerminalRequest,\n ReleaseTerminalResponse,\n RequestId,\n RequestPermissionRequest,\n RequestPermissionResponse,\n SessionNotification,\n SetSessionConfigOptionRequest,\n SetSessionConfigOptionResponse,\n SetSessionModeRequest,\n SetSessionModeResponse,\n TerminalOutputRequest,\n TerminalOutputResponse,\n WaitForTerminalExitRequest,\n WaitForTerminalExitResponse,\n WriteTextFileRequest,\n WriteTextFileResponse,\n} from \"./types.js\";\n\ninterface AcpRequestShape<TParams, TResult> {\n params: TParams;\n result: TResult;\n}\n\nexport interface AcpAgentRequestMap {\n initialize: AcpRequestShape<InitializeRequest, InitializeResponse>;\n authenticate: AcpRequestShape<AuthenticateRequest, AuthenticateResponse>;\n \"session/new\": AcpRequestShape<NewSessionRequest, NewSessionResponse>;\n \"session/load\": AcpRequestShape<LoadSessionRequest, LoadSessionResponse>;\n \"session/prompt\": AcpRequestShape<PromptRequest, PromptResponse>;\n \"session/set_mode\": AcpRequestShape<SetSessionModeRequest, SetSessionModeResponse>;\n \"session/set_config_option\": AcpRequestShape<\n SetSessionConfigOptionRequest,\n SetSessionConfigOptionResponse\n >;\n}\n\nexport interface AcpAgentNotificationMap {\n \"session/cancel\": CancelNotification;\n}\n\nexport interface AcpClientRequestMap {\n \"session/request_permission\": AcpRequestShape<\n RequestPermissionRequest,\n RequestPermissionResponse\n >;\n \"fs/read_text_file\": AcpRequestShape<ReadTextFileRequest, ReadTextFileResponse>;\n \"fs/write_text_file\": AcpRequestShape<WriteTextFileRequest, WriteTextFileResponse>;\n \"terminal/create\": AcpRequestShape<CreateTerminalRequest, CreateTerminalResponse>;\n \"terminal/output\": AcpRequestShape<TerminalOutputRequest, TerminalOutputResponse>;\n \"terminal/wait_for_exit\": AcpRequestShape<\n WaitForTerminalExitRequest,\n WaitForTerminalExitResponse\n >;\n \"terminal/kill\": AcpRequestShape<\n KillTerminalCommandRequest,\n KillTerminalCommandResponse\n >;\n \"terminal/release\": AcpRequestShape<ReleaseTerminalRequest, ReleaseTerminalResponse>;\n}\n\nexport interface AcpClientNotificationMap {\n \"session/update\": SessionNotification;\n}\n\ntype ExtensionMethod = `_${string}`;\n\nfunction assertExtensionMethod(method: string): asserts method is ExtensionMethod {\n if (!method.startsWith(\"_\")) {\n throw new Error('Extension method must start with \"_\"');\n }\n}\n\ntype SpawnFunction = (\n command: string,\n args?: ReadonlyArray<string>,\n options?: SpawnOptionsWithoutStdio\n) => ChildProcessWithoutNullStreams;\n\nexport interface AcpTransportOptions {\n command: string;\n args?: readonly string[];\n cwd?: string;\n env?: NodeJS.ProcessEnv;\n requestTimeoutMs?: number;\n firstRequestId?: number;\n spawn?: SpawnFunction;\n}\n\nexport interface AcpTransportClosedEvent {\n code: number | null;\n signal: NodeJS.Signals | null;\n reason: Error;\n stderr: string;\n}\n\nexport class AcpTransport {\n readonly closed: Promise<AcpTransportClosedEvent>;\n\n private readonly command: string;\n private readonly child: ChildProcessWithoutNullStreams;\n private readonly layer: JsonRpcMessageLayer;\n private readonly stderrChunks: string[] = [];\n private resolveClosed: ((value: AcpTransportClosedEvent) => void) | null = null;\n private closeEvent: AcpTransportClosedEvent | null = null;\n private closeReason: Error | null = null;\n\n constructor(options: AcpTransportOptions) {\n const {\n command,\n args = [],\n cwd,\n env,\n requestTimeoutMs,\n firstRequestId,\n spawn = spawnChildProcess,\n } = options;\n\n this.command = command;\n this.closed = new Promise<AcpTransportClosedEvent>((resolve) => {\n this.resolveClosed = resolve;\n });\n\n this.child = spawn(command, [...args], {\n cwd,\n env,\n stdio: [\"pipe\", \"pipe\", \"pipe\"],\n });\n\n this.child.stderr.setEncoding(\"utf8\");\n this.child.stderr.on(\"data\", (chunk) => {\n this.stderrChunks.push(String(chunk));\n });\n\n this.child.stdin.on(\"error\", (error) => {\n const reason = error instanceof Error ? error : new Error(String(error));\n this.close(reason, this.child.exitCode ?? null, this.child.signalCode ?? null);\n });\n\n this.layer = new JsonRpcMessageLayer({\n input: this.child.stdout,\n output: this.child.stdin,\n requestTimeoutMs,\n firstRequestId,\n });\n\n this.child.once(\"error\", (error) => {\n const reason = error instanceof Error ? error : new Error(String(error));\n this.close(reason, this.child.exitCode ?? null, this.child.signalCode ?? null);\n });\n\n this.child.once(\"close\", (code, signal) => {\n const reason =\n this.closeReason ??\n new Error(\n `ACP transport closed (command \"${this.command}\", code: ${code ?? \"null\"}${\n signal ? `, signal: ${signal}` : \"\"\n })`\n );\n this.close(reason, code ?? null, signal ?? null);\n });\n }\n\n sendRequest<TMethod extends keyof AcpAgentRequestMap>(\n method: TMethod,\n params: AcpAgentRequestMap[TMethod][\"params\"],\n options?: JsonRpcRequestOptions\n ): Promise<AcpAgentRequestMap[TMethod][\"result\"]>;\n sendRequest<TResult = unknown>(\n method: string,\n params?: unknown,\n options?: JsonRpcRequestOptions\n ): Promise<TResult>;\n sendRequest(\n method: string,\n params?: unknown,\n options: JsonRpcRequestOptions = {}\n ): Promise<unknown> {\n return this.layer.sendRequest(method, params, options);\n }\n\n sendExtRequest<TResult = unknown>(\n method: ExtensionMethod,\n params?: unknown,\n options?: JsonRpcRequestOptions\n ): Promise<TResult>;\n sendExtRequest<TResult = unknown>(\n method: string,\n params?: unknown,\n options: JsonRpcRequestOptions = {}\n ): Promise<TResult> {\n assertExtensionMethod(method);\n return this.layer.sendRequest(method, params, options) as Promise<TResult>;\n }\n\n sendNotification<TMethod extends keyof AcpAgentNotificationMap>(\n method: TMethod,\n params: AcpAgentNotificationMap[TMethod]\n ): void;\n sendNotification(method: string, params?: unknown): void;\n sendNotification(method: string, params?: unknown): void {\n this.layer.sendNotification(method, params);\n }\n\n sendExtNotification(method: ExtensionMethod, params?: unknown): void;\n sendExtNotification(method: string, params?: unknown): void;\n sendExtNotification(method: string, params?: unknown): void {\n assertExtensionMethod(method);\n this.layer.sendNotification(method, params);\n }\n\n onRequest<TMethod extends keyof AcpClientRequestMap>(\n method: TMethod,\n handler: (\n params: AcpClientRequestMap[TMethod][\"params\"],\n context: { id: RequestId; method: TMethod }\n ) =>\n | AcpClientRequestMap[TMethod][\"result\"]\n | Promise<AcpClientRequestMap[TMethod][\"result\"]>\n ): void;\n onRequest(method: string, handler: JsonRpcRequestHandler): void;\n onRequest(method: string, handler: JsonRpcRequestHandler): void {\n this.layer.onRequest(method, handler);\n }\n\n onExtRequest<TMethod extends ExtensionMethod>(\n method: TMethod,\n handler: (\n params: unknown,\n context: { id: RequestId; method: TMethod }\n ) => unknown | Promise<unknown>\n ): void;\n onExtRequest(method: string, handler: JsonRpcRequestHandler): void;\n onExtRequest(method: string, handler: JsonRpcRequestHandler): void {\n assertExtensionMethod(method);\n this.layer.onRequest(method, handler);\n }\n\n onNotification<TMethod extends keyof AcpClientNotificationMap>(\n method: TMethod,\n handler: (\n params: AcpClientNotificationMap[TMethod],\n context: { method: TMethod }\n ) => void | Promise<void>\n ): void;\n onNotification(method: string, handler: JsonRpcNotificationHandler): void;\n onNotification(method: string, handler: JsonRpcNotificationHandler): void {\n this.layer.onNotification(method, handler);\n }\n\n onExtNotification<TMethod extends ExtensionMethod>(\n method: TMethod,\n handler: (params: unknown, context: { method: TMethod }) => void | Promise<void>\n ): void;\n onExtNotification(method: string, handler: JsonRpcNotificationHandler): void;\n onExtNotification(method: string, handler: JsonRpcNotificationHandler): void {\n assertExtensionMethod(method);\n this.layer.onNotification(method, handler);\n }\n\n getStderrOutput(): string {\n return this.stderrChunks.join(\"\");\n }\n\n pendingRequestCount(): number {\n return this.layer.pendingRequestCount();\n }\n\n dispose(reason: Error = new Error(\"ACP transport disposed\")): void {\n if (this.closeEvent !== null) {\n return;\n }\n\n this.closeReason = reason;\n this.layer.dispose(reason);\n\n if (!this.child.stdin.destroyed && !this.child.stdin.writableEnded) {\n this.child.stdin.end();\n }\n\n if (this.child.exitCode !== null || this.child.signalCode !== null) {\n this.close(reason, this.child.exitCode, this.child.signalCode);\n return;\n }\n\n const killed = this.child.kill();\n if (!killed) {\n this.close(reason, this.child.exitCode, this.child.signalCode);\n }\n }\n\n private close(reason: Error, code: number | null, signal: NodeJS.Signals | null): void {\n if (this.closeEvent !== null) {\n return;\n }\n\n this.layer.dispose(reason);\n this.closeEvent = {\n code,\n signal,\n reason,\n stderr: this.getStderrOutput(),\n };\n this.resolveClosed?.(this.closeEvent);\n this.resolveClosed = null;\n }\n}\n", "import * as fsPromises from \"node:fs/promises\";\nimport { homedir } from \"node:os\";\nimport { join } from \"node:path\";\nimport {\n extractToolCallSummariesFromSessionUpdateStream,\n extractUsageFromSessionUpdateStream,\n type ToolCallSummary,\n} from \"./stream-helpers.js\";\nimport type { Cost, SessionUpdate, SessionUpdateNotification, UsageUpdate } from \"./types.js\";\n\ntype SessionUpdateStreamItem = SessionUpdateNotification | SessionUpdate;\n\nexport type RunExitStatus = \"success\" | \"failed\";\n\nexport interface RunReportUsage {\n used: number;\n size: number;\n updates: number;\n cost?: Cost | null;\n}\n\nexport interface RunReportError {\n message: string;\n toolCallId?: string;\n}\n\nexport interface RunReport {\n runId: string;\n startTime: string;\n endTime: string;\n exitStatus: RunExitStatus;\n toolCalls: ToolCallSummary[];\n usage: RunReportUsage;\n errors: RunReportError[];\n}\n\nexport interface GenerateRunReportOptions {\n runId?: string;\n startTime?: string | Date;\n endTime?: string | Date;\n exitStatus?: RunExitStatus;\n errors?: string[];\n now?: () => Date;\n}\n\nexport type RunReportFileSystem = {\n mkdir(path: string, options?: { recursive?: boolean }): Promise<void>;\n writeFile(\n path: string,\n data: string,\n options?: { encoding?: BufferEncoding },\n ): Promise<void>;\n};\n\nexport interface SaveRunReportOptions {\n fs?: RunReportFileSystem;\n homeDir?: string;\n now?: () => Date;\n}\n\nexport interface SavedRunReportPaths {\n reportsDir: string;\n jsonPath: string;\n summaryPath: string;\n}\n\nexport async function generateRunReportFromSessionUpdateStream(\n stream: AsyncIterable<SessionUpdateStreamItem> | Iterable<SessionUpdateStreamItem>,\n options: GenerateRunReportOptions = {},\n): Promise<RunReport> {\n const now = options.now ?? (() => new Date());\n\n const bufferedEntries: SessionUpdateStreamItem[] = [];\n let runIdFromStream: string | undefined;\n\n for await (const entry of stream) {\n bufferedEntries.push(entry);\n if (runIdFromStream) {\n continue;\n }\n\n if (isSessionUpdateNotification(entry)) {\n const sessionId = toNonEmptyString(entry.params.sessionId);\n if (sessionId) {\n runIdFromStream = sessionId;\n }\n }\n }\n\n const runId = toNonEmptyString(options.runId) ?? runIdFromStream;\n if (!runId) {\n throw new Error(\"Run id is required via options.runId or session/update stream items\");\n }\n\n const startTime = normalizeTime(options.startTime, now);\n const endTime = normalizeTime(options.endTime, now);\n\n const toolCalls = await extractToolCallSummariesFromSessionUpdateStream(bufferedEntries);\n const usageUpdates = await extractUsageFromSessionUpdateStream(bufferedEntries);\n\n const usage = summarizeUsage(usageUpdates);\n const errors = collectErrors(toolCalls, options.errors);\n const exitStatus = options.exitStatus ?? (errors.length > 0 ? \"failed\" : \"success\");\n\n return {\n runId,\n startTime,\n endTime,\n exitStatus,\n toolCalls,\n usage,\n errors,\n };\n}\n\nexport function formatRunReportSummary(report: RunReport): string {\n const lines = [\n `Run ID: ${report.runId}`,\n `Start time: ${report.startTime}`,\n `End time: ${report.endTime}`,\n `Duration: ${toDuration(report.startTime, report.endTime)}`,\n `Exit status: ${report.exitStatus}`,\n `Tool count: ${report.toolCalls.length}`,\n `Token usage: ${report.usage.used}/${report.usage.size}`,\n `Error count: ${report.errors.length}`,\n ];\n\n if (report.usage.cost) {\n lines.push(`Cost: ${report.usage.cost.amount} ${report.usage.cost.currency}`);\n }\n\n return lines.join(\"\\n\");\n}\n\nexport async function saveRunReport(\n report: RunReport,\n options: SaveRunReportOptions = {},\n): Promise<SavedRunReportPaths> {\n const fs = options.fs ?? fsPromises;\n const now = options.now ?? (() => new Date());\n\n const reportsDir = join(options.homeDir ?? homedir(), \".poe-code\", \"reports\");\n await fs.mkdir(reportsDir, { recursive: true });\n\n const timestamp = toTimestampForFileName(now());\n const safeRunId = toSafeFileSegment(report.runId);\n const baseFileName = `${timestamp}-${safeRunId}`;\n\n const jsonPath = join(reportsDir, `${baseFileName}.json`);\n const summaryPath = join(reportsDir, `${baseFileName}.txt`);\n\n await fs.writeFile(jsonPath, JSON.stringify(report, null, 2), { encoding: \"utf8\" });\n await fs.writeFile(summaryPath, formatRunReportSummary(report), { encoding: \"utf8\" });\n\n return {\n reportsDir,\n jsonPath,\n summaryPath,\n };\n}\n\nfunction isSessionUpdateNotification(entry: SessionUpdateStreamItem): entry is SessionUpdateNotification {\n return (\n typeof (entry as SessionUpdateNotification).jsonrpc === \"string\" &&\n (entry as SessionUpdateNotification).method === \"session/update\"\n );\n}\n\nfunction normalizeTime(value: string | Date | undefined, now: () => Date): string {\n if (value instanceof Date) {\n return value.toISOString();\n }\n\n if (typeof value === \"string\" && value.length > 0) {\n const parsed = new Date(value);\n if (!Number.isNaN(parsed.getTime())) {\n return parsed.toISOString();\n }\n }\n\n return now().toISOString();\n}\n\nfunction summarizeUsage(updates: readonly UsageUpdate[]): RunReportUsage {\n let used = 0;\n let size = 0;\n let cost: Cost | null | undefined;\n\n for (const update of updates) {\n used += update.used;\n size += update.size;\n\n if (update.cost !== undefined) {\n cost = update.cost;\n }\n }\n\n const usage: RunReportUsage = {\n used,\n size,\n updates: updates.length,\n };\n\n if (cost !== undefined) {\n usage.cost = cost;\n }\n\n return usage;\n}\n\nfunction collectErrors(\n toolCalls: readonly ToolCallSummary[],\n additionalErrors: readonly string[] | undefined,\n): RunReportError[] {\n const errors: RunReportError[] = [];\n\n for (const toolCall of toolCalls) {\n if (toolCall.status !== \"failed\") {\n continue;\n }\n\n errors.push({\n toolCallId: toolCall.toolCallId,\n message: toErrorMessage(toolCall),\n });\n }\n\n if (additionalErrors) {\n for (const message of additionalErrors) {\n const text = toNonEmptyString(message);\n if (text) {\n errors.push({ message: text });\n }\n }\n }\n\n return errors;\n}\n\nfunction toErrorMessage(toolCall: ToolCallSummary): string {\n if (typeof toolCall.rawOutput === \"string\" && toolCall.rawOutput.length > 0) {\n return toolCall.rawOutput;\n }\n\n if (toolCall.rawOutput instanceof Error && toolCall.rawOutput.message.length > 0) {\n return toolCall.rawOutput.message;\n }\n\n if (toolCall.rawOutput !== undefined && toolCall.rawOutput !== null) {\n const encoded = trySerialize(toolCall.rawOutput);\n if (encoded) {\n return encoded;\n }\n }\n\n return `${toolCall.title} failed`;\n}\n\nfunction trySerialize(value: unknown): string | undefined {\n try {\n const serialized = JSON.stringify(value);\n if (typeof serialized === \"string\" && serialized.length > 0) {\n return serialized;\n }\n } catch {\n return undefined;\n }\n\n return undefined;\n}\n\nfunction toDuration(startTime: string, endTime: string): string {\n const startMs = Date.parse(startTime);\n const endMs = Date.parse(endTime);\n\n if (Number.isNaN(startMs) || Number.isNaN(endMs) || endMs < startMs) {\n return \"unknown\";\n }\n\n const seconds = (endMs - startMs) / 1000;\n return `${Number(seconds.toFixed(3))}s`;\n}\n\nfunction toSafeFileSegment(value: string): string {\n let output = \"\";\n\n for (const char of value) {\n if (isAsciiLetterOrDigit(char) || char === \"-\" || char === \"_\") {\n output += char;\n continue;\n }\n\n output += \"-\";\n }\n\n return output.length > 0 ? output : \"run\";\n}\n\nfunction isAsciiLetterOrDigit(value: string): boolean {\n const code = value.charCodeAt(0);\n return (\n (code >= 48 && code <= 57) ||\n (code >= 65 && code <= 90) ||\n (code >= 97 && code <= 122)\n );\n}\n\nfunction toTimestampForFileName(value: Date): string {\n return [\n String(value.getUTCFullYear()),\n pad(value.getUTCMonth() + 1, 2),\n pad(value.getUTCDate(), 2),\n ].join(\"\")\n + \"-\"\n + [\n pad(value.getUTCHours(), 2),\n pad(value.getUTCMinutes(), 2),\n pad(value.getUTCSeconds(), 2),\n ].join(\"\")\n + \"-\"\n + pad(value.getUTCMilliseconds(), 3);\n}\n\nfunction pad(value: number, size: number): string {\n const text = String(value);\n if (text.length >= size) {\n return text;\n }\n\n return `${\"0\".repeat(size - text.length)}${text}`;\n}\n\nfunction toNonEmptyString(value: unknown): string | undefined {\n if (typeof value !== \"string\" || value.length === 0) {\n return undefined;\n }\n\n return value;\n}\n", "import path from \"node:path\";\nimport { homedir } from \"node:os\";\nimport { mkdir, open, type FileHandle } from \"node:fs/promises\";\nimport type { AcpEvent } from \"../types.js\";\nimport type { AcpMiddleware, SpawnContext } from \"../middleware.js\";\n\nfunction pad(value: number, width: number): string {\n return String(value).padStart(width, \"0\");\n}\n\nfunction formatTimestamp(date: Date): { day: string; time: string; milliseconds: string } {\n const day = `${date.getUTCFullYear()}${pad(date.getUTCMonth() + 1, 2)}${pad(date.getUTCDate(), 2)}`;\n const time = `${pad(date.getUTCHours(), 2)}${pad(date.getUTCMinutes(), 2)}${pad(date.getUTCSeconds(), 2)}`;\n const milliseconds = pad(date.getUTCMilliseconds(), 3);\n return { day, time, milliseconds };\n}\n\nfunction normalizeAgent(agent: string): string {\n let normalized = \"\";\n for (const char of agent) {\n const code = char.charCodeAt(0);\n const isLower = code >= 97 && code <= 122;\n const isUpper = code >= 65 && code <= 90;\n const isDigit = code >= 48 && code <= 57;\n\n if (isLower || isUpper || isDigit || char === \"-\" || char === \"_\") {\n normalized += char;\n } else {\n normalized += \"-\";\n }\n }\n\n return normalized.length > 0 ? normalized : \"agent\";\n}\n\nfunction resolveStartedAt(value: Date | undefined): Date {\n if (!(value instanceof Date) || Number.isNaN(value.getTime())) {\n return new Date();\n }\n\n return value;\n}\n\nfunction resolveLogFilePath(ctx: SpawnContext): string {\n const baseDir = ctx.logDir ?? path.join(homedir(), \".poe-code\", \"spawn-logs\");\n const startedAt = resolveStartedAt(ctx.startedAt);\n const { day, time, milliseconds } = formatTimestamp(startedAt);\n const fileName = `${day}-${time}-${milliseconds}-${normalizeAgent(ctx.agent)}.jsonl`;\n return path.join(baseDir, fileName);\n}\n\nclass SpawnLogWriter {\n private fileHandle: FileHandle | undefined;\n\n private isDisabled = false;\n\n private readonly logFilePath: string;\n\n private readonly logDirPath: string;\n\n constructor(ctx: SpawnContext) {\n this.logFilePath = resolveLogFilePath(ctx);\n this.logDirPath = path.dirname(this.logFilePath);\n }\n\n async writeEvent(event: AcpEvent): Promise<void> {\n if (this.isDisabled) {\n return;\n }\n\n try {\n await this.ensureOpen();\n if (!this.fileHandle) {\n return;\n }\n\n await this.fileHandle.appendFile(`${JSON.stringify(event)}\\n`, \"utf8\");\n } catch {\n this.isDisabled = true;\n await this.close();\n }\n }\n\n async close(): Promise<void> {\n if (!this.fileHandle) {\n return;\n }\n\n try {\n await this.fileHandle.close();\n } catch {\n // Ignore close errors to avoid disrupting event processing.\n } finally {\n this.fileHandle = undefined;\n }\n }\n\n private async ensureOpen(): Promise<void> {\n if (this.fileHandle || this.isDisabled) {\n return;\n }\n\n try {\n await mkdir(this.logDirPath, { recursive: true });\n this.fileHandle = await open(this.logFilePath, \"a\");\n } catch {\n this.isDisabled = true;\n }\n }\n}\n\nasync function writePreloadedEvents(writer: SpawnLogWriter, events: AcpEvent[]): Promise<void> {\n for (const event of events) {\n await writer.writeEvent(event);\n }\n}\n\nexport const spawnLog: AcpMiddleware = async (ctx, next) => {\n await next();\n\n const source = ctx.eventStream;\n const writer = new SpawnLogWriter(ctx);\n\n await writePreloadedEvents(writer, ctx.events);\n\n if (!source) {\n await writer.close();\n return;\n }\n\n ctx.eventStream = (async function* () {\n try {\n for await (const event of source) {\n await writer.writeEvent(event);\n yield event;\n }\n } finally {\n await writer.close();\n }\n })();\n};\n", "import type { CommandRunner, CommandRunnerResult } from \"@poe-code/agent-spawn\";\nimport { buildSpawnArgs } from \"@poe-code/agent-spawn\";\n\nexport type {\n CommandRunner,\n CommandRunnerOptions,\n CommandRunnerResult\n} from \"@poe-code/agent-spawn\";\n\nexport function formatCommandRunnerResult(\n result: CommandRunnerResult\n): string {\n const stdout =\n result.stdout.length > 0 ? result.stdout : \"<empty>\";\n const stderr =\n result.stderr.length > 0 ? result.stderr : \"<empty>\";\n return `stdout:\\n${stdout}\\nstderr:\\n${stderr}`;\n}\n\nexport interface RunAndMatchOutputOptions {\n command: string;\n args: string[];\n expectedOutput: string;\n skipOnDryRun?: boolean;\n}\n\nexport function describeCommandExpectation(\n command: string,\n args: string[],\n expectedOutput: string\n): string {\n return `${renderCommandLine(command, args)} (expecting \"${expectedOutput}\")`;\n}\n\nexport interface CommandExpectationCheckOptions\n extends RunAndMatchOutputOptions {\n id: string;\n}\n\nexport function createCommandExpectationCheck(\n options: CommandExpectationCheckOptions\n): CommandCheck {\n return {\n id: options.id,\n description: describeCommandExpectation(\n options.command,\n options.args,\n options.expectedOutput\n ),\n async run(context) {\n await runAndMatchOutput(context, options);\n }\n };\n}\n\nexport async function runAndMatchOutput(\n context: CommandCheckContext,\n options: RunAndMatchOutputOptions\n): Promise<void> {\n const rendered = renderCommandLine(options.command, options.args);\n if (options.skipOnDryRun !== false && context.isDryRun) {\n if (context.logDryRun) {\n context.logDryRun(\n `Dry run: ${rendered} (expecting \"${options.expectedOutput}\")`\n );\n }\n return;\n }\n\n const result = await context.runCommand(options.command, options.args);\n if (result.exitCode !== 0) {\n const detail = formatCommandRunnerResult(result);\n throw new Error(\n [`Command ${rendered} failed with exit code ${result.exitCode}.`, detail].join(\"\\n\")\n );\n }\n\n if (!stdoutMatchesExpected(result.stdout, options.expectedOutput)) {\n const detail = formatCommandRunnerResult(result);\n const received = result.stdout.trim();\n throw new Error(\n [\n `Command ${rendered} failed: expected \"${options.expectedOutput}\" but received \"${received}\".`,\n detail\n ].join(\"\\n\")\n );\n }\n}\n\nexport function stdoutMatchesExpected(stdout: string, expected: string): boolean {\n const trimmed = stdout.trim();\n if (trimmed === expected) {\n return true;\n }\n\n const lines = stdout\n .split(/\\r?\\n/)\n .map((line) => line.trim())\n .filter((line) => line.length > 0);\n\n if (lines.some((line) => line === expected)) {\n return true;\n }\n\n for (const line of lines) {\n if (line[0] !== \"{\") continue;\n try {\n const parsed = JSON.parse(line) as { type?: string; result?: string };\n if (parsed.type === \"result\" && parsed.result === expected) {\n return true;\n }\n } catch {\n continue;\n }\n }\n\n return false;\n}\n\nfunction renderCommandLine(command: string, args: string[]): string {\n return [command, ...args].map(quoteIfNeeded).join(\" \").trim();\n}\n\nfunction quoteIfNeeded(value: string): string {\n if (value.length === 0) {\n return '\"\"';\n }\n if (needsQuoting(value)) {\n return `\"${value.replaceAll('\"', '\\\\\"')}\"`;\n }\n return value;\n}\n\nfunction needsQuoting(value: string): boolean {\n return (\n value.includes(\" \") ||\n value.includes(\"\\t\") ||\n value.includes(\"\\n\")\n );\n}\n\nexport interface CommandCheckContext {\n isDryRun: boolean;\n runCommand: CommandRunner;\n logDryRun?: (message: string) => void;\n}\n\nexport interface CommandCheck {\n id: string;\n description?: string;\n run(context: CommandCheckContext): Promise<void>;\n}\n\nexport function createSpawnHealthCheck(\n agentId: string,\n options: { model?: string; expectedOutput: string }\n): CommandCheck {\n const prompt = `Output exactly: ${options.expectedOutput}`;\n const { binaryName, args } = buildSpawnArgs(agentId, {\n prompt,\n model: options.model,\n mode: \"yolo\"\n });\n return {\n id: `${agentId}-cli-health`,\n description: `spawn ${agentId} (expecting \"${options.expectedOutput}\")`,\n async run(context) {\n if (context.isDryRun) {\n context.logDryRun?.(\n `Dry run: ${[binaryName, ...args].join(\" \")} (expecting \"${options.expectedOutput}\")`\n );\n return;\n }\n\n const result = await context.runCommand(binaryName, args);\n\n if (result.exitCode !== 0) {\n throw new Error(\n `spawn ${agentId} failed with exit code ${result.exitCode}.\\n${formatCommandRunnerResult(result)}`\n );\n }\n\n if (!result.stdout.includes(options.expectedOutput)) {\n throw new Error(\n `spawn ${agentId}: expected \"${options.expectedOutput}\" in stdout.\\n${formatCommandRunnerResult(result)}`\n );\n }\n }\n };\n}\n\n/**\n * Creates a check that detects if a binary exists using multiple fallback methods.\n * This is useful in Docker/containerized environments where PATH may not be updated after npm install.\n *\n * @param binaryName - The name of the binary to check for (e.g., \"claude\", \"codex\")\n * @param id - Unique identifier for the check\n * @param description - Human-readable description of what's being checked\n * @returns A CommandCheck that verifies the binary using multiple detection methods\n */\nexport function createBinaryExistsCheck(\n binaryName: string,\n id: string,\n description: string\n): CommandCheck {\n return {\n id,\n description,\n async run({ runCommand }) {\n // Common installation paths for CLI tools\n const commonPaths = [\n `/usr/local/bin/${binaryName}`,\n `/usr/bin/${binaryName}`,\n `$HOME/.local/bin/${binaryName}`,\n `$HOME/.claude/local/bin/${binaryName}`\n ];\n\n const detectors: Array<{\n command: string;\n args: string[];\n validate: (result: CommandRunnerResult) => boolean;\n }> = [\n {\n command: \"which\",\n args: [binaryName],\n validate: (result) => result.exitCode === 0\n },\n {\n command: \"where\",\n args: [binaryName],\n validate: (result) =>\n result.exitCode === 0 && result.stdout.trim().length > 0\n },\n // Check common installation paths using shell expansion for $HOME\n {\n command: \"sh\",\n args: [\n \"-c\",\n commonPaths.map((p) => `test -f \"${p}\"`).join(\" || \")\n ],\n validate: (result) => result.exitCode === 0\n }\n ];\n\n for (const detector of detectors) {\n const result = await runCommand(detector.command, detector.args);\n if (detector.validate(result)) {\n return;\n }\n }\n\n throw new Error(`${binaryName} CLI binary not found on PATH.`);\n }\n };\n}\n", "import type {\n ConfigMergeMutation,\n ConfigPruneMutation,\n ConfigTransformMutation,\n ConfigObject,\n ValueResolver,\n MutationOptions\n} from \"../types.js\";\n\nexport interface MergeOptions {\n /** Target file path (must start with ~) */\n target: ValueResolver<string>;\n /** Value to merge into the config file */\n value: ValueResolver<ConfigObject>;\n /** Optional explicit format override */\n format?: \"json\" | \"toml\";\n /** Optional prune by prefix before merging (TOML) */\n pruneByPrefix?: Record<string, string>;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface PruneOptions {\n /** Target file path (must start with ~) */\n target: ValueResolver<string>;\n /** Shape to prune from the config file */\n shape: ValueResolver<ConfigObject>;\n /** Optional explicit format override */\n format?: \"json\" | \"toml\";\n /** Optional guard - only prune if predicate returns true */\n onlyIf?: (doc: ConfigObject, ctx: MutationOptions) => boolean;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface TransformOptions {\n /** Target file path (must start with ~) */\n target: ValueResolver<string>;\n /** Optional explicit format override */\n format?: \"json\" | \"toml\";\n /** Transform function - receives parsed content, returns transformed content */\n transform: (\n content: ConfigObject,\n ctx: MutationOptions\n ) => { content: ConfigObject | null; changed: boolean };\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nfunction merge(options: MergeOptions): ConfigMergeMutation {\n return {\n kind: \"configMerge\",\n target: options.target,\n value: options.value,\n format: options.format,\n pruneByPrefix: options.pruneByPrefix,\n label: options.label\n };\n}\n\nfunction prune(options: PruneOptions): ConfigPruneMutation {\n return {\n kind: \"configPrune\",\n target: options.target,\n shape: options.shape,\n format: options.format,\n onlyIf: options.onlyIf,\n label: options.label\n };\n}\n\nfunction transform(options: TransformOptions): ConfigTransformMutation {\n return {\n kind: \"configTransform\",\n target: options.target,\n format: options.format,\n transform: options.transform,\n label: options.label\n };\n}\n\nexport const configMutation = {\n merge,\n prune,\n transform\n};\n", "import type {\n EnsureDirectoryMutation,\n RemoveDirectoryMutation,\n RemoveFileMutation,\n ChmodMutation,\n BackupMutation,\n ValueResolver\n} from \"../types.js\";\n\nexport interface EnsureDirectoryOptions {\n /** Directory path (must start with ~) */\n path: ValueResolver<string>;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface RemoveOptions {\n /** Target file path (must start with ~) */\n target: ValueResolver<string>;\n /** Only remove if file is empty/whitespace */\n whenEmpty?: boolean;\n /** Only remove if content matches regex */\n whenContentMatches?: RegExp;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface RemoveDirectoryOptions {\n /** Directory path (must start with ~) */\n path: ValueResolver<string>;\n /** Remove directory even when not empty */\n force?: boolean;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface ChmodOptions {\n /** Target file path (must start with ~) */\n target: ValueResolver<string>;\n /** File permission mode (e.g., 0o755) */\n mode: number;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nexport interface BackupOptions {\n /** Target file path to backup (must start with ~) */\n target: ValueResolver<string>;\n /** Optional human-readable label for logging */\n label?: string;\n}\n\nfunction ensureDirectory(options: EnsureDirectoryOptions): EnsureDirectoryMutation {\n return {\n kind: \"ensureDirectory\",\n path: options.path,\n label: options.label\n };\n}\n\nfunction remove(options: RemoveOptions): RemoveFileMutation {\n return {\n kind: \"removeFile\",\n target: options.target,\n whenEmpty: options.whenEmpty,\n whenContentMatches: options.whenContentMatches,\n label: options.label\n };\n}\n\nfunction removeDirectory(\n options: RemoveDirectoryOptions\n): RemoveDirectoryMutation {\n return {\n kind: \"removeDirectory\",\n path: options.path,\n force: options.force,\n label: options.label\n };\n}\n\nfunction chmod(options: ChmodOptions): ChmodMutation {\n return {\n kind: \"chmod\",\n target: options.target,\n mode: options.mode,\n label: options.label\n };\n}\n\nfunction backup(options: BackupOptions): BackupMutation {\n return {\n kind: \"backup\",\n target: options.target,\n label: options.label\n };\n}\n\nexport const fileMutation = {\n ensureDirectory,\n remove,\n removeDirectory,\n chmod,\n backup\n};\n", "import Mustache from \"mustache\";\nimport type {\n Mutation,\n MutationContext,\n MutationOutcome,\n MutationDetails,\n ConfigObject,\n MutationOptions,\n ValueResolver,\n FileSystem\n} from \"../types.js\";\nimport { getConfigFormat, detectFormat } from \"../formats/index.js\";\nimport { resolvePath } from \"./path-utils.js\";\nimport {\n isNotFound,\n readFileIfExists,\n pathExists,\n createTimestamp\n} from \"../fs-utils.js\";\n\n// ============================================================================\n// Helper Functions\n// ============================================================================\n\nfunction resolveValue<T>(\n resolver: ValueResolver<T>,\n options: MutationOptions\n): T {\n if (typeof resolver === \"function\") {\n return (resolver as (ctx: MutationOptions) => T)(options);\n }\n return resolver;\n}\n\nfunction createInvalidDocumentBackupPath(targetPath: string): string {\n const ext = targetPath.includes(\".\") ? targetPath.split(\".\").pop() : \"bak\";\n return `${targetPath}.invalid-${createTimestamp()}.${ext}`;\n}\n\nasync function backupInvalidDocument(\n fs: FileSystem,\n targetPath: string,\n content: string\n): Promise<void> {\n const backupPath = createInvalidDocumentBackupPath(targetPath);\n await fs.writeFile(backupPath, content, { encoding: \"utf8\" });\n}\n\nfunction describeMutation(kind: string, targetPath?: string): string {\n const displayPath = targetPath ?? \"target\";\n switch (kind) {\n case \"ensureDirectory\":\n return `Create ${displayPath}`;\n case \"removeDirectory\":\n return `Remove directory ${displayPath}`;\n case \"backup\":\n return `Backup ${displayPath}`;\n case \"templateWrite\":\n return `Write ${displayPath}`;\n case \"chmod\":\n return `Set permissions on ${displayPath}`;\n case \"removeFile\":\n return `Remove ${displayPath}`;\n case \"configMerge\":\n case \"configPrune\":\n case \"configTransform\":\n case \"templateMergeToml\":\n case \"templateMergeJson\":\n return `Update ${displayPath}`;\n default:\n return \"Operation\";\n }\n}\n\nfunction pruneKeysByPrefix(\n table: ConfigObject,\n prefix: string\n): ConfigObject {\n const result: ConfigObject = {};\n for (const [key, value] of Object.entries(table)) {\n if (!key.startsWith(prefix)) {\n result[key] = value;\n }\n }\n return result;\n}\n\nfunction isConfigObject(value: unknown): value is ConfigObject {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction mergeWithPruneByPrefix(\n base: ConfigObject,\n patch: ConfigObject,\n pruneByPrefix?: Record<string, string>\n): ConfigObject {\n const result: ConfigObject = { ...base };\n const prefixMap = pruneByPrefix ?? {};\n\n for (const [key, value] of Object.entries(patch)) {\n const current = result[key];\n const prefix = prefixMap[key];\n\n if (isConfigObject(current) && isConfigObject(value)) {\n if (prefix) {\n const pruned = pruneKeysByPrefix(current, prefix);\n result[key] = { ...pruned, ...value };\n } else {\n result[key] = mergeWithPruneByPrefix(\n current,\n value as ConfigObject,\n prefixMap\n );\n }\n continue;\n }\n result[key] = value;\n }\n return result;\n}\n\n// ============================================================================\n// Apply Mutation\n// ============================================================================\n\nexport async function applyMutation(\n mutation: Mutation,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n switch (mutation.kind) {\n case \"ensureDirectory\":\n return applyEnsureDirectory(mutation, context, options);\n case \"removeDirectory\":\n return applyRemoveDirectory(mutation, context, options);\n case \"removeFile\":\n return applyRemoveFile(mutation, context, options);\n case \"chmod\":\n return applyChmod(mutation, context, options);\n case \"backup\":\n return applyBackup(mutation, context, options);\n case \"configMerge\":\n return applyConfigMerge(mutation, context, options);\n case \"configPrune\":\n return applyConfigPrune(mutation, context, options);\n case \"configTransform\":\n return applyConfigTransform(mutation, context, options);\n case \"templateWrite\":\n return applyTemplateWrite(mutation, context, options);\n case \"templateMergeToml\":\n return applyTemplateMerge(mutation, context, options, \"toml\");\n case \"templateMergeJson\":\n return applyTemplateMerge(mutation, context, options, \"json\");\n default: {\n const never: never = mutation;\n throw new Error(`Unknown mutation kind: ${(never as Mutation).kind}`);\n }\n }\n}\n\n// ============================================================================\n// File Mutation Handlers\n// ============================================================================\n\nasync function applyEnsureDirectory(\n mutation: Extract<Mutation, { kind: \"ensureDirectory\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.path, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const existed = await pathExists(context.fs, targetPath);\n\n if (!context.dryRun) {\n await context.fs.mkdir(targetPath, { recursive: true });\n }\n\n return {\n outcome: {\n changed: !existed,\n effect: \"mkdir\",\n detail: existed ? \"noop\" : \"create\"\n },\n details\n };\n}\n\nasync function applyRemoveDirectory(\n mutation: Extract<Mutation, { kind: \"removeDirectory\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.path, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const existed = await pathExists(context.fs, targetPath);\n if (!existed) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (typeof context.fs.rm !== \"function\") {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (mutation.force) {\n if (!context.dryRun) {\n await context.fs.rm(targetPath, { recursive: true, force: true });\n }\n return {\n outcome: { changed: true, effect: \"delete\", detail: \"delete\" },\n details\n };\n }\n\n const entries = await context.fs.readdir(targetPath);\n if (entries.length > 0) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (!context.dryRun) {\n await context.fs.rm(targetPath, { recursive: true, force: true });\n }\n\n return {\n outcome: { changed: true, effect: \"delete\", detail: \"delete\" },\n details\n };\n}\n\nasync function applyRemoveFile(\n mutation: Extract<Mutation, { kind: \"removeFile\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n try {\n const content = await context.fs.readFile(targetPath, \"utf8\");\n const trimmed = content.trim();\n\n // Check whenContentMatches guard\n if (mutation.whenContentMatches && !mutation.whenContentMatches.test(trimmed)) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n // Check whenEmpty guard\n if (mutation.whenEmpty && trimmed.length > 0) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (!context.dryRun) {\n await context.fs.unlink(targetPath);\n }\n\n return {\n outcome: { changed: true, effect: \"delete\", detail: \"delete\" },\n details\n };\n } catch (error) {\n if (isNotFound(error)) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n throw error;\n }\n}\n\nasync function applyChmod(\n mutation: Extract<Mutation, { kind: \"chmod\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n if (typeof context.fs.chmod !== \"function\") {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n try {\n const stat = await context.fs.stat(targetPath);\n const currentMode = typeof stat.mode === \"number\" ? stat.mode & 0o777 : null;\n\n if (currentMode === mutation.mode) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (!context.dryRun) {\n await context.fs.chmod(targetPath, mutation.mode);\n }\n\n return {\n outcome: { changed: true, effect: \"chmod\", detail: \"update\" },\n details\n };\n } catch (error) {\n if (isNotFound(error)) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n throw error;\n }\n}\n\nasync function applyBackup(\n mutation: Extract<Mutation, { kind: \"backup\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const content = await readFileIfExists(context.fs, targetPath);\n if (content === null) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n if (!context.dryRun) {\n const backupPath = `${targetPath}.backup-${createTimestamp()}`;\n await context.fs.writeFile(backupPath, content, { encoding: \"utf8\" });\n }\n\n return {\n outcome: { changed: true, effect: \"copy\", detail: \"backup\" },\n details\n };\n}\n\n// ============================================================================\n// Config Mutation Handlers\n// ============================================================================\n\nasync function applyConfigMerge(\n mutation: Extract<Mutation, { kind: \"configMerge\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const formatName = mutation.format ?? detectFormat(rawPath);\n if (!formatName) {\n throw new Error(\n `Cannot detect config format for \"${rawPath}\". Provide explicit format option.`\n );\n }\n const format = getConfigFormat(formatName);\n\n const rawContent = await readFileIfExists(context.fs, targetPath);\n let current: ConfigObject;\n try {\n current = rawContent === null ? {} : format.parse(rawContent);\n } catch {\n // Invalid file - backup and start fresh\n if (rawContent !== null) {\n await backupInvalidDocument(context.fs, targetPath, rawContent);\n }\n current = {};\n }\n\n const value = resolveValue(mutation.value, options);\n\n // Use mergeWithPruneByPrefix for TOML files with pruneByPrefix option\n let merged: ConfigObject;\n if (mutation.pruneByPrefix) {\n merged = mergeWithPruneByPrefix(current, value, mutation.pruneByPrefix);\n } else {\n merged = format.merge(current, value);\n }\n\n const serialized = format.serialize(merged);\n const changed = serialized !== rawContent;\n\n if (changed && !context.dryRun) {\n await context.fs.writeFile(targetPath, serialized, { encoding: \"utf8\" });\n }\n\n return {\n outcome: {\n changed,\n effect: changed ? \"write\" : \"none\",\n detail: changed ? (rawContent === null ? \"create\" : \"update\") : \"noop\"\n },\n details\n };\n}\n\nasync function applyConfigPrune(\n mutation: Extract<Mutation, { kind: \"configPrune\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const rawContent = await readFileIfExists(context.fs, targetPath);\n if (rawContent === null) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n const formatName = mutation.format ?? detectFormat(rawPath);\n if (!formatName) {\n throw new Error(\n `Cannot detect config format for \"${rawPath}\". Provide explicit format option.`\n );\n }\n const format = getConfigFormat(formatName);\n\n let current: ConfigObject;\n try {\n current = format.parse(rawContent);\n } catch {\n // Invalid file - can't prune, leave as-is\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n // Check onlyIf guard\n if (mutation.onlyIf && !mutation.onlyIf(current, options)) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n const shape = resolveValue(mutation.shape, options);\n const { changed, result } = format.prune(current, shape);\n\n if (!changed) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n // Delete file if empty\n if (Object.keys(result).length === 0) {\n if (!context.dryRun) {\n await context.fs.unlink(targetPath);\n }\n return {\n outcome: { changed: true, effect: \"delete\", detail: \"delete\" },\n details\n };\n }\n\n const serialized = format.serialize(result);\n if (!context.dryRun) {\n await context.fs.writeFile(targetPath, serialized, { encoding: \"utf8\" });\n }\n\n return {\n outcome: { changed: true, effect: \"write\", detail: \"update\" },\n details\n };\n}\n\nasync function applyConfigTransform(\n mutation: Extract<Mutation, { kind: \"configTransform\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const formatName = mutation.format ?? detectFormat(rawPath);\n if (!formatName) {\n throw new Error(\n `Cannot detect config format for \"${rawPath}\". Provide explicit format option.`\n );\n }\n const format = getConfigFormat(formatName);\n\n const rawContent = await readFileIfExists(context.fs, targetPath);\n let current: ConfigObject;\n try {\n current = rawContent === null ? {} : format.parse(rawContent);\n } catch {\n if (rawContent !== null) {\n await backupInvalidDocument(context.fs, targetPath, rawContent);\n }\n current = {};\n }\n\n const { content: transformed, changed } = mutation.transform(current, options);\n\n if (!changed) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n\n // Delete file if null\n if (transformed === null) {\n if (rawContent === null) {\n return {\n outcome: { changed: false, effect: \"none\", detail: \"noop\" },\n details\n };\n }\n if (!context.dryRun) {\n await context.fs.unlink(targetPath);\n }\n return {\n outcome: { changed: true, effect: \"delete\", detail: \"delete\" },\n details\n };\n }\n\n const serialized = format.serialize(transformed);\n if (!context.dryRun) {\n await context.fs.writeFile(targetPath, serialized, { encoding: \"utf8\" });\n }\n\n return {\n outcome: {\n changed: true,\n effect: \"write\",\n detail: rawContent === null ? \"create\" : \"update\"\n },\n details\n };\n}\n\n// ============================================================================\n// Template Mutation Handlers\n// ============================================================================\n\nasync function applyTemplateWrite(\n mutation: Extract<Mutation, { kind: \"templateWrite\" }>,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n if (!context.templates) {\n throw new Error(\n \"Template mutations require a templates loader. \" +\n \"Provide templates function to runMutations context.\"\n );\n }\n\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const template = await context.templates(mutation.templateId);\n const templateContext = mutation.context\n ? resolveValue(mutation.context, options)\n : {};\n const rendered = Mustache.render(template, templateContext);\n\n const existed = await pathExists(context.fs, targetPath);\n\n if (!context.dryRun) {\n await context.fs.writeFile(targetPath, rendered, { encoding: \"utf8\" });\n }\n\n return {\n outcome: {\n changed: true,\n effect: \"write\",\n detail: existed ? \"update\" : \"create\"\n },\n details\n };\n}\n\nasync function applyTemplateMerge(\n mutation: Extract<Mutation, { kind: \"templateMergeToml\" | \"templateMergeJson\" }>,\n context: MutationContext,\n options: MutationOptions,\n formatName: \"toml\" | \"json\"\n): Promise<{ outcome: MutationOutcome; details: MutationDetails }> {\n if (!context.templates) {\n throw new Error(\n \"Template mutations require a templates loader. \" +\n \"Provide templates function to runMutations context.\"\n );\n }\n\n const rawPath = resolveValue(mutation.target, options);\n const targetPath = resolvePath(rawPath, context.homeDir, context.pathMapper);\n\n const details: MutationDetails = {\n kind: mutation.kind,\n label: mutation.label ?? describeMutation(mutation.kind, targetPath),\n targetPath\n };\n\n const format = getConfigFormat(formatName);\n\n // Load and render template\n const template = await context.templates(mutation.templateId);\n const templateContext = mutation.context\n ? resolveValue(mutation.context, options)\n : {};\n const rendered = Mustache.render(template, templateContext);\n\n // Parse rendered template\n let templateDoc: ConfigObject;\n try {\n templateDoc = format.parse(rendered);\n } catch (error) {\n throw new Error(\n `Failed to parse rendered template \"${mutation.templateId}\" as ${formatName.toUpperCase()}: ${error}`,\n { cause: error }\n );\n }\n\n // Read and parse existing file\n const rawContent = await readFileIfExists(context.fs, targetPath);\n let current: ConfigObject;\n try {\n current = rawContent === null ? {} : format.parse(rawContent);\n } catch {\n if (rawContent !== null) {\n await backupInvalidDocument(context.fs, targetPath, rawContent);\n }\n current = {};\n }\n\n // Merge\n const merged = format.merge(current, templateDoc);\n const serialized = format.serialize(merged);\n const changed = serialized !== rawContent;\n\n if (changed && !context.dryRun) {\n await context.fs.writeFile(targetPath, serialized, { encoding: \"utf8\" });\n }\n\n return {\n outcome: {\n changed,\n effect: changed ? \"write\" : \"none\",\n detail: changed ? (rawContent === null ? \"create\" : \"update\") : \"noop\"\n },\n details\n };\n}\n", "import * as jsonc from \"jsonc-parser\";\nimport type { ConfigFormat, ConfigObject, ConfigValue } from \"../types.js\";\n\nfunction isConfigObject(value: unknown): value is ConfigObject {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction detectIndent(content: string): string {\n const match = content.match(/^[\\t ]+/m);\n if (match) {\n return match[0];\n }\n return \" \";\n}\n\nfunction parse(content: string): ConfigObject {\n if (!content || content.trim() === \"\") {\n return {};\n }\n const errors: jsonc.ParseError[] = [];\n const parsed = jsonc.parse(content, errors, {\n allowTrailingComma: true,\n disallowComments: false\n });\n if (errors.length > 0) {\n throw new Error(`JSON parse error: ${jsonc.printParseErrorCode(errors[0].error)}`);\n }\n if (parsed === null || parsed === undefined) {\n return {};\n }\n if (!isConfigObject(parsed)) {\n throw new Error(\"Expected JSON object.\");\n }\n return parsed;\n}\n\nfunction serialize(obj: ConfigObject): string {\n return `${JSON.stringify(obj, null, 2)}\\n`;\n}\n\nfunction merge(base: ConfigObject, patch: ConfigObject): ConfigObject {\n const result: ConfigObject = { ...base };\n for (const [key, value] of Object.entries(patch)) {\n if (value === undefined) {\n continue;\n }\n const existing = result[key];\n if (isConfigObject(existing) && isConfigObject(value)) {\n result[key] = merge(existing, value);\n continue;\n }\n result[key] = value as ConfigValue;\n }\n return result;\n}\n\nfunction prune(\n obj: ConfigObject,\n shape: ConfigObject\n): { changed: boolean; result: ConfigObject } {\n let changed = false;\n const result: ConfigObject = { ...obj };\n\n for (const [key, pattern] of Object.entries(shape)) {\n if (!(key in result)) {\n continue;\n }\n\n const current = result[key];\n\n // Empty object pattern means \"delete this key entirely\"\n if (isConfigObject(pattern) && Object.keys(pattern).length === 0) {\n delete result[key];\n changed = true;\n continue;\n }\n\n // Non-empty object pattern with object current: recurse\n if (isConfigObject(pattern) && isConfigObject(current)) {\n const { changed: childChanged, result: childResult } = prune(\n current,\n pattern\n );\n if (childChanged) {\n changed = true;\n }\n if (Object.keys(childResult).length === 0) {\n delete result[key];\n } else {\n result[key] = childResult;\n }\n continue;\n }\n\n delete result[key];\n changed = true;\n }\n\n return { changed, result };\n}\n\n/**\n * Modify JSON content at a specific path while preserving comments and formatting.\n * Uses jsonc-parser's modify() for targeted updates.\n *\n * @param content - The original JSON content (may include comments)\n * @param path - JSON path array, e.g. [\"mcpServers\", \"my-server\"]\n * @param value - The value to set (or undefined to remove)\n * @returns The modified JSON content with comments preserved\n */\nfunction modifyAtPath(\n content: string,\n path: (string | number)[],\n value: ConfigValue | undefined\n): string {\n const indent = detectIndent(content);\n const formattingOptions: jsonc.FormattingOptions = {\n tabSize: indent === \"\\t\" ? 1 : indent.length,\n insertSpaces: indent !== \"\\t\",\n eol: \"\\n\"\n };\n\n const edits = jsonc.modify(content, path, value, { formattingOptions });\n let result = jsonc.applyEdits(content, edits);\n\n if (!result.endsWith(\"\\n\")) {\n result += \"\\n\";\n }\n\n return result;\n}\n\n/**\n * Merge a patch into JSON content while preserving comments and formatting.\n * Uses jsonc.modify() for each top-level key to preserve existing comments.\n *\n * @param content - The original JSON content (may include comments)\n * @param patch - Object with values to merge\n * @returns The modified JSON content with comments preserved\n */\nfunction mergePreservingComments(\n content: string,\n patch: ConfigObject\n): string {\n let result = content || \"{}\";\n\n for (const [key, value] of Object.entries(patch)) {\n if (value === undefined) {\n continue;\n }\n result = modifyAtPath(result, [key], value);\n }\n\n return result;\n}\n\n/**\n * Remove a key from JSON content while preserving comments and formatting.\n *\n * @param content - The original JSON content\n * @param path - JSON path array to the key to remove\n * @returns The modified JSON content with comments preserved\n */\nfunction removeAtPath(content: string, path: (string | number)[]): string {\n return modifyAtPath(content, path, undefined);\n}\n\nexport { detectIndent, modifyAtPath, mergePreservingComments, removeAtPath };\n\nexport const jsonFormat: ConfigFormat = {\n parse,\n serialize,\n merge,\n prune\n};\n", "import { parse as parseToml, stringify as stringifyToml } from \"smol-toml\";\nimport type { ConfigFormat, ConfigObject, ConfigValue } from \"../types.js\";\n\nfunction isConfigObject(value: unknown): value is ConfigObject {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction parse(content: string): ConfigObject {\n if (!content || content.trim() === \"\") {\n return {};\n }\n const parsed = parseToml(content);\n if (!isConfigObject(parsed)) {\n throw new Error(\"Expected TOML document to be a table.\");\n }\n return parsed as ConfigObject;\n}\n\nfunction serialize(obj: ConfigObject): string {\n const serialized = stringifyToml(obj);\n return serialized.endsWith(\"\\n\") ? serialized : `${serialized}\\n`;\n}\n\nfunction merge(base: ConfigObject, patch: ConfigObject): ConfigObject {\n const result: ConfigObject = { ...base };\n for (const [key, value] of Object.entries(patch)) {\n if (value === undefined) {\n continue;\n }\n const existing = result[key];\n if (isConfigObject(existing) && isConfigObject(value)) {\n result[key] = merge(existing, value);\n continue;\n }\n result[key] = value as ConfigValue;\n }\n return result;\n}\n\nfunction prune(\n obj: ConfigObject,\n shape: ConfigObject\n): { changed: boolean; result: ConfigObject } {\n let changed = false;\n const result: ConfigObject = { ...obj };\n\n for (const [key, pattern] of Object.entries(shape)) {\n if (!(key in result)) {\n continue;\n }\n\n const current = result[key];\n\n // Empty object pattern means \"delete this key entirely\"\n if (isConfigObject(pattern) && Object.keys(pattern).length === 0) {\n delete result[key];\n changed = true;\n continue;\n }\n\n // Non-empty object pattern with object current: recurse\n if (isConfigObject(pattern) && isConfigObject(current)) {\n const { changed: childChanged, result: childResult } = prune(\n current,\n pattern\n );\n if (childChanged) {\n changed = true;\n }\n if (Object.keys(childResult).length === 0) {\n delete result[key];\n } else {\n result[key] = childResult;\n }\n continue;\n }\n\n delete result[key];\n changed = true;\n }\n\n return { changed, result };\n}\n\nexport const tomlFormat: ConfigFormat = {\n parse,\n serialize,\n merge,\n prune\n};\n", "import type { ConfigFormat } from \"../types.js\";\nimport { jsonFormat } from \"./json.js\";\nimport { tomlFormat } from \"./toml.js\";\n\nexport type FormatName = \"json\" | \"toml\";\n\nconst formatRegistry: Record<FormatName, ConfigFormat> = {\n json: jsonFormat,\n toml: tomlFormat\n};\n\nconst extensionMap: Record<string, FormatName> = {\n \".json\": \"json\",\n \".toml\": \"toml\"\n};\n\n/**\n * Get a format handler by path (auto-detect from extension) or explicit format name.\n */\nexport function getConfigFormat(pathOrFormat: string): ConfigFormat {\n // Check if it's an explicit format name\n if (pathOrFormat in formatRegistry) {\n return formatRegistry[pathOrFormat as FormatName];\n }\n\n // Try to detect from extension\n const ext = getExtension(pathOrFormat);\n const formatName = extensionMap[ext];\n\n if (!formatName) {\n throw new Error(\n `Unsupported config format. Cannot detect format from \"${pathOrFormat}\". ` +\n `Supported extensions: ${Object.keys(extensionMap).join(\", \")}. ` +\n `Supported format names: ${Object.keys(formatRegistry).join(\", \")}.`\n );\n }\n\n return formatRegistry[formatName];\n}\n\n/**\n * Detect format name from a file path.\n */\nexport function detectFormat(path: string): FormatName | undefined {\n const ext = getExtension(path);\n return extensionMap[ext];\n}\n\nfunction getExtension(path: string): string {\n const lastDot = path.lastIndexOf(\".\");\n if (lastDot === -1) {\n return \"\";\n }\n return path.slice(lastDot).toLowerCase();\n}\n\nexport { jsonFormat } from \"./json.js\";\nexport { tomlFormat } from \"./toml.js\";\n", "import path from \"node:path\";\nimport type { PathMapper } from \"../types.js\";\n\n/**\n * Expand ~ shortcut to the provided home directory.\n */\nexport function expandHome(targetPath: string, homeDir: string): string {\n if (!targetPath?.startsWith(\"~\")) {\n return targetPath;\n }\n\n // Handle ~./ -> ~/.\n if (targetPath.startsWith(\"~./\")) {\n targetPath = `~/.${targetPath.slice(3)}`;\n }\n\n let remainder = targetPath.slice(1);\n\n // Remove leading slash or backslash\n if (remainder.startsWith(\"/\") || remainder.startsWith(\"\\\\\")) {\n remainder = remainder.slice(1);\n } else if (remainder.startsWith(\".\")) {\n // Handle ~/.\n remainder = remainder.slice(1);\n if (remainder.startsWith(\"/\") || remainder.startsWith(\"\\\\\")) {\n remainder = remainder.slice(1);\n }\n }\n\n return remainder.length === 0 ? homeDir : path.join(homeDir, remainder);\n}\n\n/**\n * Validate that a path is home-relative (starts with ~).\n * Throws if the path is not home-relative.\n */\nexport function validateHomePath(targetPath: string): void {\n if (typeof targetPath !== \"string\" || targetPath.length === 0) {\n throw new Error(\"Target path must be a non-empty string.\");\n }\n\n if (!targetPath.startsWith(\"~\")) {\n throw new Error(\n `All target paths must be home-relative (start with ~). Received: \"${targetPath}\"`\n );\n }\n}\n\n/**\n * Resolve a path with optional path mapping for isolated configurations.\n * 1. Validates the path starts with ~\n * 2. Expands ~ to home directory\n * 3. If pathMapper is provided, maps the directory portion and reconstructs the path\n */\nexport function resolvePath(\n rawPath: string,\n homeDir: string,\n pathMapper?: PathMapper\n): string {\n validateHomePath(rawPath);\n const expanded = expandHome(rawPath, homeDir);\n\n if (!pathMapper) {\n return expanded;\n }\n\n // Map the directory portion\n const rawDirectory = path.dirname(expanded);\n const mappedDirectory = pathMapper.mapTargetDirectory({\n targetDirectory: rawDirectory\n });\n const filename = path.basename(expanded);\n\n return filename.length === 0 ? mappedDirectory : path.join(mappedDirectory, filename);\n}\n", "import type { FileSystem } from \"./types.js\";\n\n/**\n * Check if an error is a \"file not found\" (ENOENT) error.\n */\nexport function isNotFound(error: unknown): boolean {\n return (\n typeof error === \"object\" &&\n error !== null &&\n \"code\" in error &&\n (error as { code?: string }).code === \"ENOENT\"\n );\n}\n\n/**\n * Read a file if it exists, returning null if not found.\n */\nexport async function readFileIfExists(\n fs: FileSystem,\n target: string\n): Promise<string | null> {\n try {\n return await fs.readFile(target, \"utf8\");\n } catch (error) {\n if (isNotFound(error)) {\n return null;\n }\n throw error;\n }\n}\n\n/**\n * Check if a path exists (file or directory).\n */\nexport async function pathExists(\n fs: FileSystem,\n target: string\n): Promise<boolean> {\n try {\n await fs.stat(target);\n return true;\n } catch (error) {\n if (isNotFound(error)) {\n return false;\n }\n throw error;\n }\n}\n\n/**\n * Create an ISO timestamp safe for use in filenames.\n * Replaces colons and dots with dashes.\n */\nexport function createTimestamp(): string {\n return new Date().toISOString().replaceAll(\":\", \"-\").replaceAll(\".\", \"-\");\n}\n", "import type {\n Mutation,\n MutationContext,\n MutationResult,\n MutationOptions\n} from \"../types.js\";\nimport { applyMutation } from \"./apply-mutation.js\";\n\n/**\n * Execute an array of mutations in order.\n *\n * All dependencies must be injected - no defaults, no globals.\n */\nexport async function runMutations(\n mutations: Mutation[],\n context: MutationContext,\n options?: MutationOptions\n): Promise<MutationResult> {\n const effects: MutationResult[\"effects\"] = [];\n let anyChanged = false;\n const resolverOptions = options ?? {};\n\n for (const mutation of mutations) {\n const { outcome } = await executeMutation(\n mutation,\n context,\n resolverOptions\n );\n effects.push(outcome);\n if (outcome.changed) {\n anyChanged = true;\n }\n }\n\n return {\n changed: anyChanged,\n effects\n };\n}\n\nasync function executeMutation(\n mutation: Mutation,\n context: MutationContext,\n options: MutationOptions\n): Promise<{ outcome: MutationResult[\"effects\"][number]; details: { kind: string; label: string; targetPath?: string } }> {\n // Call onStart observer\n context.observers?.onStart?.({\n kind: mutation.kind,\n label: mutation.label ?? mutation.kind,\n targetPath: undefined // Will be resolved during apply\n });\n\n try {\n const { outcome, details } = await applyMutation(mutation, context, options);\n\n // Call onComplete observer\n context.observers?.onComplete?.(details, outcome);\n\n return { outcome, details };\n } catch (error) {\n // Call onError observer\n context.observers?.onError?.(\n {\n kind: mutation.kind,\n label: mutation.label ?? mutation.kind,\n targetPath: undefined\n },\n error\n );\n\n // Re-throw the error\n throw error;\n }\n}\n", "import Mustache from \"mustache\";\n\nexport type TemplateVariables = Record<string, string | number | boolean | string[]>;\n\n// Disable HTML escaping - we're rendering prompts, not HTML\nconst originalEscape = Mustache.escape;\n\n/**\n * Render a mustache template with the given variables.\n * Arrays are automatically joined with newlines.\n * HTML escaping is disabled.\n */\nexport function renderTemplate(\n template: string,\n variables: TemplateVariables\n): string {\n // Pre-process variables to handle arrays\n const processed: Record<string, string | number | boolean> = {};\n for (const [key, value] of Object.entries(variables)) {\n if (Array.isArray(value)) {\n processed[key] = value.join(\"\\n\");\n } else {\n processed[key] = value;\n }\n }\n\n // Temporarily disable HTML escaping\n Mustache.escape = (text: string) => text;\n try {\n return Mustache.render(template, processed);\n } finally {\n Mustache.escape = originalEscape;\n }\n}\n", "export const FRONTIER_MODELS = [\n \"anthropic/claude-opus-4.6\",\n \"anthropic/claude-sonnet-4.6\",\n \"openai/gpt-5.3-codex\",\n \"openai/gpt-5.4\",\n \"google/gemini-3.1-pro\"\n] as const;\n\nexport const DEFAULT_FRONTIER_MODEL = \"anthropic/claude-sonnet-4.6\";\n\nexport const DEFAULT_TEXT_MODEL = \"anthropic/claude-sonnet-4.6\";\nexport const DEFAULT_IMAGE_BOT = \"google/nano-banana-pro\";\nexport const DEFAULT_AUDIO_BOT = \"elevenlabs/elevenlabs-v3\";\nexport const DEFAULT_VIDEO_BOT = \"google/veo-3.1\";\n\nexport const CLAUDE_CODE_VARIANTS = {\n haiku: \"anthropic/claude-haiku-4.5\",\n sonnet: \"anthropic/claude-sonnet-4.6\",\n opus: \"anthropic/claude-opus-4.6\"\n} as const;\n\nexport const DEFAULT_CLAUDE_CODE_MODEL = CLAUDE_CODE_VARIANTS.sonnet;\n\n/**\n * Extracts the model ID from a namespaced model slug (lowercase).\n * e.g., \"anthropic/claude-sonnet-4.6\" -> \"claude-sonnet-4.6\"\n */\nexport function stripModelNamespace(model: string): string {\n const slashIndex = model.indexOf(\"/\");\n const id = slashIndex === -1 ? model : model.slice(slashIndex + 1);\n return id.toLowerCase();\n}\n\nexport const CODEX_MODELS = [\n \"openai/gpt-5.4\",\n \"openai/gpt-5.3-codex\",\n \"openai/gpt-5.3-codex-spark\",\n \"openai/gpt-5.2-codex\",\n \"openai/gpt-5.2\",\n \"openai/gpt-5.2-chat\",\n \"openai/gpt-5.2-pro\",\n \"openai/gpt-5.1\",\n \"openai/gpt-5.1-codex-mini\",\n \"anthropic/claude-opus-4.6\"\n] as const;\nexport const DEFAULT_CODEX_MODEL = CODEX_MODELS[0];\n\nexport const KIMI_MODELS = [\n \"novitaai/kimi-k2.5\",\n \"novitaai/kimi-k2-thinking\",\n \"novitaai/kimi-k2.5-fw\",\n] as const;\nexport const DEFAULT_KIMI_MODEL = KIMI_MODELS[0];\n\nexport const DEFAULT_REASONING = \"medium\";\nexport const PROVIDER_NAME = \"poe\";\nexport const FEEDBACK_URL = \"https://github.com/poe-platform/poe-code/issues\";\n", "import type {\n CommandRunner,\n CommandCheckContext,\n CommandCheck\n} from \"../utils/command-checks.js\";\n\nexport interface InstallContext {\n isDryRun: boolean;\n runCommand: CommandRunner;\n logger: (message: string) => void;\n platform: NodeJS.Platform;\n}\n\nexport interface InstallCommand {\n id: string;\n command: string;\n args: string[];\n platforms?: NodeJS.Platform[];\n}\n\nexport interface ServiceInstallDefinition {\n id: string;\n summary: string;\n check: CommandCheck;\n steps: InstallCommand[];\n postChecks?: CommandCheck[];\n successMessage?: string;\n}\n\nexport async function runServiceInstall(\n definition: ServiceInstallDefinition,\n context: InstallContext\n): Promise<boolean> {\n const checkContext: CommandCheckContext = {\n isDryRun: context.isDryRun,\n runCommand: context.runCommand\n };\n\n let needsInstall = false;\n try {\n await definition.check.run(checkContext);\n context.logger(`${definition.summary} already installed.`);\n } catch (error) {\n const detail =\n error instanceof Error ? error.message : String(error);\n context.logger(`${definition.summary} not detected: ${detail}`);\n needsInstall = true;\n }\n\n if (!needsInstall) {\n return false;\n }\n\n if (context.isDryRun) {\n logInstallDryRun(definition, context);\n return true;\n }\n\n const platformSteps = filterStepsByPlatform(definition.steps, context.platform);\n for (const step of platformSteps) {\n await runInstallStep(step, context);\n }\n\n await definition.check.run(checkContext);\n\n if (definition.postChecks) {\n for (const postCheck of definition.postChecks) {\n await postCheck.run(checkContext);\n }\n }\n\n context.logger(\n definition.successMessage ?? `${definition.summary} installed.`\n );\n return true;\n}\n\nfunction describeInstallCommand(step: InstallCommand): string {\n return `[${step.id}] ${formatCommand(step.command, step.args)}`;\n}\n\nfunction formatCommand(command: string, args: string[]): string {\n return [command, ...args.map(quoteIfNeeded)].join(\" \");\n}\n\nfunction quoteIfNeeded(value: string): string {\n if (value.length === 0) {\n return '\"\"';\n }\n if (value.includes(\" \") || value.includes(\"\\t\") || value.includes(\"\\n\")) {\n return `\"${value.replaceAll('\"', '\\\\\"')}\"`;\n }\n return value;\n}\n\nfunction filterStepsByPlatform(\n steps: InstallCommand[],\n platform: NodeJS.Platform\n): InstallCommand[] {\n return steps.filter(\n (step) => !step.platforms || step.platforms.includes(platform)\n );\n}\n\nfunction logInstallDryRun(\n definition: ServiceInstallDefinition,\n context: InstallContext\n): void {\n context.logger(`Dry run: would install ${definition.summary}.`);\n const platformSteps = filterStepsByPlatform(definition.steps, context.platform);\n for (const step of platformSteps) {\n context.logger(`Dry run: ${describeInstallCommand(step)}`);\n }\n}\n\nasync function runInstallStep(\n step: InstallCommand,\n context: InstallContext\n): Promise<void> {\n context.logger(`Running ${describeInstallCommand(step)}`);\n const result = await context.runCommand(step.command, step.args);\n if (result.exitCode !== 0) {\n const stderr = result.stderr.trim();\n const suffix = stderr.length > 0 ? `: ${stderr}` : \"\";\n throw new Error(\n `${describeInstallCommand(step)} failed with exit code ${result.exitCode}${suffix}`\n );\n }\n}\n", "import type {\n ProviderService,\n ProviderContext,\n ProviderBranding,\n ProviderConfigurePrompts,\n ProviderIsolatedEnv\n} from \"../cli/service-registry.js\";\nimport {\n runMutations,\n type Mutation,\n type MutationObservers\n} from \"@poe-code/config-mutations\";\nimport {\n runServiceInstall,\n type ServiceInstallDefinition\n} from \"../services/service-install.js\";\n// Template imports are lazy to avoid breaking tsc output when imported\n// by generate-bin-wrappers.mjs (Node.js can't resolve .hbs as ESM modules)\nconst templateImports: Record<string, () => Promise<{ default: string }>> = {\n \"py-poe-spawn/env.hbs\": () => import(\"../templates/py-poe-spawn/env.hbs\"),\n \"py-poe-spawn/main.py.hbs\": () => import(\"../templates/py-poe-spawn/main.py.hbs\"),\n \"py-poe-spawn/requirements.txt.hbs\": () => import(\"../templates/py-poe-spawn/requirements.txt.hbs\"),\n \"codex/config.toml.hbs\": () => import(\"../templates/codex/config.toml.hbs\"),\n};\n\nasync function loadTemplate(templateId: string): Promise<string> {\n const loader = templateImports[templateId];\n if (!loader) {\n throw new Error(`Template not found: ${templateId}`);\n }\n const module = await loader();\n return module.default;\n}\n\ninterface ManifestVersionDefinition {\n configure: Mutation[];\n unconfigure?: Mutation[];\n}\n\nexport interface ServiceRunOptions {\n observers?: MutationObservers;\n}\n\ninterface CreateProviderOptions<\n ConfigureOptions,\n UnconfigureOptions,\n SpawnOptions\n> {\n name: string;\n aliases?: string[];\n label: string;\n id: string;\n summary: string;\n branding?: ProviderBranding;\n disabled?: boolean;\n supportsStdinPrompt?: boolean;\n supportsMcpSpawn?: boolean;\n configurePrompts?: ProviderConfigurePrompts;\n postConfigureMessages?: string[];\n isolatedEnv?: ProviderIsolatedEnv;\n manifest: ManifestVersionDefinition;\n install?: ServiceInstallDefinition;\n test?: ProviderService<ConfigureOptions, UnconfigureOptions, SpawnOptions>[\"test\"];\n spawn?: ProviderService<\n ConfigureOptions,\n UnconfigureOptions,\n SpawnOptions\n >[\"spawn\"];\n}\n\nexport function createProvider<\n ConfigureOptions = any,\n UnconfigureOptions = ConfigureOptions,\n SpawnOptions = any\n>(\n opts: CreateProviderOptions<ConfigureOptions, UnconfigureOptions, SpawnOptions>\n): ProviderService<ConfigureOptions, UnconfigureOptions, SpawnOptions> {\n const provider: ProviderService<\n ConfigureOptions,\n UnconfigureOptions,\n SpawnOptions\n > = {\n id: opts.id,\n summary: opts.summary,\n name: opts.name,\n aliases: opts.aliases,\n label: opts.label,\n branding: opts.branding,\n disabled: opts.disabled,\n supportsStdinPrompt: opts.supportsStdinPrompt,\n supportsMcpSpawn: opts.supportsMcpSpawn,\n configurePrompts: opts.configurePrompts,\n postConfigureMessages: opts.postConfigureMessages,\n isolatedEnv: opts.isolatedEnv,\n async configure(context, runOptions) {\n await runMutations(opts.manifest.configure, {\n fs: context.fs,\n homeDir: context.env.homeDir,\n observers: runOptions?.observers,\n templates: loadTemplate,\n pathMapper: context.pathMapper\n }, context.options as Record<string, unknown>);\n context.command.flushDryRun({ emitIfEmpty: false });\n },\n async unconfigure(context, runOptions) {\n if (!opts.manifest.unconfigure) {\n return false;\n }\n const result = await runMutations(opts.manifest.unconfigure, {\n fs: context.fs,\n homeDir: context.env.homeDir,\n observers: runOptions?.observers,\n templates: loadTemplate,\n pathMapper: context.pathMapper\n }, context.options as Record<string, unknown>);\n context.command.flushDryRun({ emitIfEmpty: false });\n return result.changed;\n }\n };\n\n if (opts.install) {\n provider.install = createInstallRunner(opts.install);\n }\n\n if (opts.test) {\n provider.test = opts.test;\n }\n\n if (opts.spawn) {\n provider.spawn = opts.spawn;\n }\n\n return provider;\n}\n\nfunction createInstallRunner(definition: ServiceInstallDefinition) {\n return async (context: ProviderContext): Promise<void> => {\n await runServiceInstall(definition, {\n isDryRun: context.logger.context.dryRun,\n runCommand: context.command.runCommand,\n logger: (message) => context.logger.verbose(message),\n platform: context.env.platform\n });\n };\n}\n", "import {\n createBinaryExistsCheck,\n createSpawnHealthCheck\n} from \"../utils/command-checks.js\";\nimport {\n configMutation,\n fileMutation,\n type ConfigObject\n} from \"@poe-code/config-mutations\";\nimport { type ServiceInstallDefinition } from \"../services/service-install.js\";\nimport { KIMI_MODELS, DEFAULT_KIMI_MODEL, PROVIDER_NAME, stripModelNamespace } from \"../cli/constants.js\";\nimport { createProvider } from \"./create-provider.js\";\nimport type {\n ProviderSpawnOptions,\n ModelConfigureOptions,\n EmptyProviderOptions\n} from \"./spawn-options.js\";\nimport { kimiAgent } from \"@poe-code/agent-defs\";\n\nexport const KIMI_INSTALL_DEFINITION: ServiceInstallDefinition = {\n id: \"kimi\",\n summary: \"Kimi CLI\",\n check: createBinaryExistsCheck(\n \"kimi\",\n \"kimi-cli-binary\",\n \"Kimi CLI binary must exist\"\n ),\n steps: [\n {\n id: \"install-kimi-cli-uv\",\n command: \"uv\",\n args: [\"tool\", \"install\", \"--python\", \"3.13\", \"kimi-cli\"]\n }\n ],\n successMessage: \"Installed Kimi CLI via uv.\"\n};\n\nfunction providerModel(model: string): string {\n const stripped = stripModelNamespace(model);\n return `${PROVIDER_NAME}/${stripped}`;\n}\n\nfunction buildKimiArgs(prompt: string, extraArgs?: string[]): string[] {\n return [\"--quiet\", \"-p\", prompt, ...(extraArgs ?? [])];\n}\n\nexport const kimiService = createProvider<\n ModelConfigureOptions,\n EmptyProviderOptions,\n ProviderSpawnOptions\n>({\n ...kimiAgent,\n disabled: false,\n supportsStdinPrompt: false,\n configurePrompts: {\n model: {\n label: \"Kimi default model\",\n defaultValue: DEFAULT_KIMI_MODEL,\n choices: KIMI_MODELS.map((id) => ({\n title: id,\n value: id\n }))\n }\n },\n isolatedEnv: {\n // Use \"kimi-cli\" to avoid stripAgentHome stripping \".kimi\" from paths\n agentBinary: \"kimi-cli\",\n configProbe: { kind: \"isolatedFile\", relativePath: \".kimi/config.toml\" },\n env: {\n HOME: { kind: \"isolatedDir\" }\n }\n },\n test(context) {\n return context.runCheck(\n createSpawnHealthCheck(\"kimi\", {\n model: context.model,\n expectedOutput: \"KIMI_OK\"\n })\n );\n },\n manifest: {\n configure: [\n fileMutation.ensureDirectory({ path: \"~/.kimi\" }),\n fileMutation.ensureDirectory({ path: \"~/.kimi/credentials\" }),\n configMutation.merge({\n target: \"~/.kimi/credentials/kimi-code.json\",\n value: () => ({\n access_token: \"poe-managed\",\n token_type: \"Bearer\",\n expires_at: Math.floor(Date.now() / 1000) + 86400 * 365 * 10\n })\n }),\n configMutation.merge({\n target: \"~/.kimi/config.toml\",\n pruneByPrefix: { models: `${PROVIDER_NAME}/` },\n value: (ctx) => {\n const { model, apiKey, env } = (ctx ?? {}) as {\n model?: string;\n apiKey?: string;\n env: { poeApiBaseUrl: string };\n };\n const selectedModel = model ?? DEFAULT_KIMI_MODEL;\n\n const models: ConfigObject = {};\n for (const m of KIMI_MODELS) {\n models[providerModel(m)] = {\n provider: PROVIDER_NAME,\n model: stripModelNamespace(m),\n max_context_size: 256000\n };\n }\n\n return {\n default_model: providerModel(selectedModel),\n default_thinking: true,\n models,\n providers: {\n [PROVIDER_NAME]: {\n type: \"openai_legacy\",\n base_url: env.poeApiBaseUrl,\n api_key: apiKey ?? \"\"\n }\n }\n };\n }\n })\n ],\n unconfigure: [\n configMutation.transform({\n target: \"~/.kimi/config.toml\",\n transform: (document) => {\n const providers = document.providers as ConfigObject | undefined;\n if (!providers || typeof providers !== \"object\") {\n return { changed: false, content: document };\n }\n if (!(PROVIDER_NAME in providers)) {\n return { changed: false, content: document };\n }\n const { [PROVIDER_NAME]: ignoredProvider, ...rest } = providers;\n void ignoredProvider;\n const updatedProviders = rest as ConfigObject;\n if (Object.keys(updatedProviders).length === 0) {\n const { providers: ignoredProviders, ...docWithoutProviders } = document;\n void ignoredProviders;\n return { changed: true, content: docWithoutProviders };\n }\n return { changed: true, content: { ...document, providers: updatedProviders } };\n }\n })\n ]\n },\n install: KIMI_INSTALL_DEFINITION,\n spawn(context, options) {\n const args = buildKimiArgs(options.prompt, options.args);\n if (options.cwd) {\n return context.command.runCommand(\"kimi\", args, {\n cwd: options.cwd\n });\n }\n return context.command.runCommand(\"kimi\", args);\n }\n});\n\nexport const provider = kimiService;\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["lookup", "spawnChildProcess", "text", "chalk", "chalk", "chalk", "chalk", "chalk", "chalk", "chalk", "spacing", "message", "chalk", "chalk", "chalk", "chalk", "chalk", "EMOJI_RE", "MODIFIER_RE", "NO_TRUNCATION", "NO_TRUNCATION", "ANSI_ESCAPE_BELL", "ANSI_CSI", "ANSI_OSC", "ANSI_ESCAPE_LINK", "ANSI_OSC", "GROUP_REGEX", "ANSI_CSI", "ANSI_ESCAPE_BELL", "actions", "settings", "isWindows", "chalk", "chalk", "chalk", "chalk", "chalk", "chalk", "spawnChildProcess", "spawnChildProcess", "homedir", "runCommand", "parse", "merge", "prune", "parse", "merge", "prune", "isConfigObject", "parse", "serialize", "merge", "prune", "path", "path", "error", "isConfigObject", "error", "error", "Mustache", "stripModelNamespace", "error", "formatCommand", "provider", "message", "stripModelNamespace"]
|
|
7
7
|
}
|