agent-afk 3.23.1 → 3.24.0

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/telegram.mjs CHANGED
@@ -1,5 +1,5 @@
1
1
  #!/usr/bin/env node
2
- import{existsSync as oh,readFileSync as Br}from"fs";import{fileURLToPath as uc}from"url";import{dirname as pc,join as fc}from"path";var hc=[{name:"AFK_COMPACT_KEEP_LAST_TURNS",description:"Number of recent turns the compactor keeps verbatim during /compact. Default tuned in compact-handler.ts.",type:"number",required:!1,example:"6",category:"model"},{name:"AFK_COMPACT_MODEL",description:"Override the model used by the /compact summarizer. Falls back to a cheap default (haiku-class).",type:"string",required:!1,example:"claude-haiku-4-5",category:"model"},{name:"AFK_DEFAULT_SUBAGENT_MODEL",description:"Override the default model used when a subagent is dispatched without an explicit model.",type:"string",required:!1,example:"sonnet",category:"model"},{name:"AFK_DISABLE_PROMPT_CACHE",description:"Disable Anthropic prompt caching when set to 1/true/yes/on. Unset = caching enabled.",type:"boolean",required:!1,default:"0",example:"1",category:"model"},{name:"AFK_EFFORT",description:"Reasoning-effort hint for the Codex provider. Accepts low | medium | high.",type:"string",required:!1,example:"medium",category:"model"},{name:"AFK_MAX_BUDGET_USD",description:"Per-turn USD budget ceiling. Aborts the turn when projected spend would exceed this.",type:"number",required:!1,default:"5.00",example:"10.00",category:"model"},{name:"AFK_MAX_OUTPUT_TOKENS",description:"Cap on output tokens per turn. Falls back to provider default when unset.",type:"number",required:!1,example:"8192",category:"model"},{name:"AFK_MAX_TOKENS",description:"Cap on total tokens per turn (input + output). Default 4096.",type:"number",required:!1,default:"4096",example:"8192",category:"model"},{name:"AFK_MODEL",description:"Default model for agent turns. Accepts short aliases (opus, sonnet, haiku) or full model IDs.",type:"string",required:!1,default:"sonnet",example:"claude-opus-4-5",category:"model"},{name:"AFK_PROMPT_CACHE_TTL",description:"TTL for Anthropic prompt-cache blocks. Accepts 5m or 1h.",type:"string",required:!1,default:"1h",example:"1h",category:"model"},{name:"AFK_TASK_BUDGET",description:"Per-task token budget ceiling. Aborts when cumulative usage would exceed it.",type:"number",required:!1,default:"100000",example:"200000",category:"model"},{name:"AFK_TEMPERATURE",description:"Numeric temperature override for model sampling. Provider default if unset.",type:"number",required:!1,example:"0.7",category:"model"},{name:"AFK_THINKING",description:"Extended-thinking toggle. Accepts on | off | <budget-tokens>. On by default.",type:"string",required:!1,default:"on",example:"on",category:"model"},{name:"AFK_TIMEOUT_MS",description:"Per-turn timeout in milliseconds. Provider/SDK default if unset.",type:"number",required:!1,example:"120000",category:"model"},{name:"CLAUDE_MODEL",description:"Legacy alias for AFK_MODEL \u2014 supported for back-compat with pre-AFK_* deployments.",type:"string",required:!1,example:"sonnet",category:"model"},{name:"AFK_SYSTEM_PROMPT",description:"Raw system-prompt string. Tier-1 source (highest priority over afk.config.json and AFK.md).",type:"string",required:!1,example:"You are a helpful agent.",category:"model"},{name:"AFK_DUMP_PROMPT",description:"Write the resolved system prompt to a file at startup. Accepts a path or 1 for default location.",type:"string",required:!1,example:"/tmp/afk-prompt.txt",category:"debug"},{name:"ANTHROPIC_API_KEY",description:"Anthropic API key. Tier-1 credential \u2014 overrides keychain OAuth and CLAUDE_CODE_OAUTH_TOKEN.",type:"string",required:!1,category:"auth",secret:!0},{name:"CLAUDE_CODE_OAUTH_TOKEN",description:"Claude Code OAuth token. Tier-2 credential \u2014 used when ANTHROPIC_API_KEY is unset; falls back to keychain.",type:"string",required:!1,category:"auth",secret:!0},{name:"OPENAI_API_KEY",description:"OpenAI API key for the openai-compatible provider (gpt-*, o1*, o3*, o4* models).",type:"string",required:!1,category:"auth",secret:!0},{name:"CODEX_API_KEY",description:"Codex API key for the @openai/codex-sdk provider when AFK_MODEL=codex-*.",type:"string",required:!1,category:"auth",secret:!0},{name:"AFK_LOCAL_API_KEY",description:"Placeholder API key for local Anthropic-compatible servers (vllm-mlx, etc.). Set when AFK_LOCAL_BASE_URL is configured.",type:"string",required:!1,default:"local",example:"local",category:"auth",secret:!0},{name:"AFK_LOCAL_BASE_URL",description:"Base URL for a self-hosted Anthropic-compatible server. When set, routes traffic away from api.anthropic.com.",type:"string",required:!1,example:"http://127.0.0.1:8080",category:"model"},{name:"AFK_OPENAI_BASE_URL",description:"Base URL override for the OpenAI-compatible provider. Used for local shims (mlx_lm.server, Ollama, vLLM, LM Studio).",type:"string",required:!1,example:"http://127.0.0.1:8000/v1",category:"model"},{name:"TELEGRAM_BOT_TOKEN",description:"Telegram bot token from @BotFather. Required to run the Telegram bot surface.",type:"string",required:!1,category:"telegram",secret:!0},{name:"AFK_TELEGRAM_BOT_TOKEN",description:"Alternative env var name for the Telegram bot token, accepted by the setup wizard.",type:"string",required:!1,category:"telegram",secret:!0},{name:"AFK_TELEGRAM_ALLOWED_CHAT_IDS",description:"Comma-separated list of Telegram chat IDs allowed to interact with the bot. Required when the bot is running.",type:"string",required:!1,example:"123456789,987654321",category:"telegram"},{name:"TELEGRAM_DATA_DIR",description:"Override the directory where Telegram bot state is stored. Defaults to ~/.afk/state/telegram/.",type:"string",required:!1,category:"telegram"},{name:"TELEGRAM_VERBOSE",description:"Set to 1 to log per-message details from the Telegram bot \u2014 chat IDs, message text, latency.",type:"boolean",required:!1,example:"1",category:"telegram"},{name:"AFK_TELEGRAM_TRACE",description:"Set to 1 to dump raw bridge traffic between the agent and the Telegram bot \u2014 debugging only.",type:"boolean",required:!1,example:"1",category:"debug"},{name:"AFK_TELEGRAM_CWD",description:"Override the working directory used by the Telegram bot when spawning agent sessions.",type:"string",required:!1,category:"telegram"},{name:"AFK_HOME",description:"Override the AFK home directory. Default: ~/.afk/.",type:"string",required:!1,default:"~/.afk",example:"/opt/afk",category:"paths"},{name:"AFK_STATE_DIR",description:"Override the AFK state directory. Default: $AFK_HOME/state/.",type:"string",required:!1,category:"paths"},{name:"AFK_FRAMEWORK_DIR",description:"Override the AFK agent-framework directory used for telemetry and briefs. Default: $AFK_HOME/agent-framework/.",type:"string",required:!1,category:"paths"},{name:"AFK_EVAL_HARNESS_ROOT",description:"Root path for the forge evaluation harness. Used by the L1/L2 capability-gate checks.",type:"string",required:!1,category:"paths"},{name:"HOME",description:"Standard Unix home directory. Used as the fallback when AFK_HOME is unset.",type:"string",required:!1,category:"process"},{name:"PATH",description:"System PATH. Read for executable resolution (git, gh, etc.) in tool handlers.",type:"string",required:!1,category:"process"},{name:"AFK_DAEMON_CWD",description:"Working directory used by the daemon process for spawned agent sessions.",type:"string",required:!1,category:"daemon"},{name:"AFK_DAEMON_TASK",description:"Default task description for the daemon. Falls back to afk.config.json daemon.task.",type:"string",required:!1,category:"daemon"},{name:"AFK_DAEMON_TASK_ID",description:"Task identifier the daemon uses to scope its state directory and telemetry.",type:"string",required:!1,category:"daemon"},{name:"AFK_SESSIONSTART_COOLDOWN_MS",description:"Cooldown in milliseconds between SessionStart trigger fires in the daemon. Prevents thundering-herd on rapid restarts.",type:"number",required:!1,category:"daemon"},{name:"AFK_WORKTREE_AUTONAME",description:"Auto-rename worktree branches based on the first user message in interactive mode. 1 = on (default), 0 = off.",type:"boolean",required:!1,default:"1",example:"0",category:"worktree"},{name:"AFK_WORKTREE_BRANCH_PREFIX",description:"Branch-name prefix for AFK-managed worktrees. Default afk/. Set to empty string to drop the prefix.",type:"string",required:!1,default:"afk/",example:"wt/",category:"worktree"},{name:"AFK_WORKTREE_BOOT_PRUNE",description:"When set, the daemon prunes stale worktrees at boot in addition to the cron-driven sweep.",type:"boolean",required:!1,category:"worktree"},{name:"AFK_WORKTREE_PRUNE_DISABLE",description:"Disable the worktree prune job entirely. Useful for long-running tests.",type:"boolean",required:!1,category:"worktree"},{name:"AFK_WORKTREE_MAX_AGE_CLEAN",description:"Maximum age (in days) before a clean worktree is auto-pruned. Default 14.",type:"number",required:!1,default:"14",category:"worktree"},{name:"AFK_WORKTREE_MAX_AGE_DIRTY",description:"Maximum age (in days) before a dirty worktree is auto-pruned. Default 30.",type:"number",required:!1,default:"30",category:"worktree"},{name:"AFK_WORKTREE_SWEEP_ROOT",description:"Override the root directory under which AFK worktrees are tracked for pruning.",type:"string",required:!1,category:"worktree"},{name:"AFK_THREADS_ALLOWED_USERNAMES",description:"Comma-separated allowlist of Threads usernames the agent will respond to.",type:"string",required:!1,example:"alice,bob",category:"threads"},{name:"AFK_THREADS_DRY_RUN",description:"Set to 1 to log Threads replies without actually posting them.",type:"boolean",required:!1,example:"1",category:"threads"},{name:"AFK_THREADS_POLL_INTERVAL_MS",description:"Poll interval for the Threads inbox watcher in milliseconds.",type:"number",required:!1,category:"threads"},{name:"AFK_THREADS_REPLY_MODE",description:"Threads reply mode. Accepts post (default) or other modes defined in src/threads.ts.",type:"string",required:!1,default:"post",example:"post",category:"threads"},{name:"AFK_ALLOW_PROJECT_MCP",description:"Allow loading MCP configuration from <cwd>/.mcp.json. Disabled by default \u2014 opt-in to mitigate config-injection risks.",type:"boolean",required:!1,example:"1",category:"mcp"},{name:"AFK_AUTO_ROUTING",description:"Auto-route bare slash inputs to matching skills. Applies to interactive, chat, telegram, and daemon surfaces.",type:"boolean",required:!1,example:"true",category:"misc"},{name:"AFK_BANNER_PLAIN",description:"Suppress the ANSI-colored banner at REPL startup. Useful for non-TTY captures and CI logs.",type:"boolean",required:!1,example:"1",category:"misc"},{name:"AFK_SPINNER_TIPS",description:"Show rotating tips in the loading spinner during long calls. 1 = on, 0 = off.",type:"boolean",required:!1,category:"misc"},{name:"AFK_SHOW_DIFFS",description:"Show inline diffs in the tool-lane output for edit/write tool calls. 1 = on, 0 = off.",type:"boolean",required:!1,category:"misc"},{name:"AFK_SKILL_STREAM_VERBOSE",description:"Verbose streaming output when a skill is dispatched. Logs sub-agent setup, intermediate events, and final result.",type:"boolean",required:!1,category:"debug"},{name:"FORCE_COLOR",description:"Standard Node convention. Force-enable ANSI color output even when stdout is not a TTY.",type:"string",required:!1,example:"1",category:"process"},{name:"NO_COLOR",description:"Standard convention (https://no-color.org). When set to any non-empty value, disables ANSI color output.",type:"string",required:!1,example:"1",category:"process"},{name:"AFK_DEBUG",description:"Enable verbose debug logging across the codebase. Accepts 1 to enable.",type:"boolean",required:!1,example:"1",category:"debug"},{name:"AFK_DEBUG_CLIPBOARD",description:"Debug bracketed-paste and image-paste handling in the interactive REPL.",type:"boolean",required:!1,category:"debug"},{name:"AFK_TRACE_DISABLED",description:"Disable the agent trace subsystem entirely. Set to 1 to skip trace file writes.",type:"boolean",required:!1,example:"1",category:"debug"},{name:"DEBUG",description:"Standard Node `debug`-package convention. When set to 1, enables verbose logging in several modules alongside AFK_DEBUG.",type:"string",required:!1,category:"debug"},{name:"AGENT_AFK_ASCII",description:"Force the interactive REPL tool-lane renderer to ASCII-only glyphs instead of the default Unicode box-drawing set. Accepts 1/true/yes (case-insensitive). Useful for terminals whose font lacks \u2503\u251C\u2570\u251C glyphs.",type:"boolean",required:!1,example:"1",category:"debug"},{name:"AGENT_SURFACE",description:"Internal surface marker propagated to subprocesses. Identifies which AFK surface (cli, telegram, daemon, threads) spawned the process.",type:"string",required:!1,example:"cli",category:"process"},{name:"CI",description:"Standard CI-detection convention. Auto-set by GitHub Actions, CircleCI, etc. Used to switch off TTY-only UX.",type:"string",required:!1,example:"true",category:"process"},{name:"NODE_ENV",description:"Standard Node environment marker. test | development | production. Used by routing-telemetry.ts to suppress test-time writes.",type:"string",required:!1,example:"production",category:"process"},{name:"VITEST",description:"Set automatically by Vitest. Used at runtime to short-circuit code paths that should not fire in tests.",type:"string",required:!1,category:"process"},{name:"NO_UPDATE_NOTIFIER",description:"Disable the update-available notifier on CLI startup. Standard convention shared with many Node CLIs.",type:"boolean",required:!1,category:"process"},{name:"AFK_WRITE_DENYLIST",description:"Comma-separated list of additional path globs that the write_file tool refuses to write to.",type:"string",required:!1,example:"**/.env,**/secrets/**",category:"misc"},{name:"AFK_WRITE_DIFF",description:"Show a diff preview before each write_file tool call. Defaults provider-controlled when unset.",type:"boolean",required:!1,category:"misc"}],v={get AFK_COMPACT_KEEP_LAST_TURNS(){return process.env.AFK_COMPACT_KEEP_LAST_TURNS},get AFK_COMPACT_MODEL(){return process.env.AFK_COMPACT_MODEL},get AFK_DEFAULT_SUBAGENT_MODEL(){return process.env.AFK_DEFAULT_SUBAGENT_MODEL},get AFK_DISABLE_PROMPT_CACHE(){return process.env.AFK_DISABLE_PROMPT_CACHE},get AFK_EFFORT(){return process.env.AFK_EFFORT},get AFK_MAX_BUDGET_USD(){return process.env.AFK_MAX_BUDGET_USD},get AFK_MAX_OUTPUT_TOKENS(){return process.env.AFK_MAX_OUTPUT_TOKENS},get AFK_MAX_TOKENS(){return process.env.AFK_MAX_TOKENS},get AFK_MODEL(){return process.env.AFK_MODEL},get AFK_PROMPT_CACHE_TTL(){return process.env.AFK_PROMPT_CACHE_TTL},get AFK_TASK_BUDGET(){return process.env.AFK_TASK_BUDGET},get AFK_TEMPERATURE(){return process.env.AFK_TEMPERATURE},get AFK_THINKING(){return process.env.AFK_THINKING},get AFK_TIMEOUT_MS(){return process.env.AFK_TIMEOUT_MS},get CLAUDE_MODEL(){return process.env.CLAUDE_MODEL},get AFK_SYSTEM_PROMPT(){return process.env.AFK_SYSTEM_PROMPT},get AFK_DUMP_PROMPT(){return process.env.AFK_DUMP_PROMPT},get ANTHROPIC_API_KEY(){return process.env.ANTHROPIC_API_KEY},get CLAUDE_CODE_OAUTH_TOKEN(){return process.env.CLAUDE_CODE_OAUTH_TOKEN},get OPENAI_API_KEY(){return process.env.OPENAI_API_KEY},get CODEX_API_KEY(){return process.env.CODEX_API_KEY},get AFK_LOCAL_API_KEY(){return process.env.AFK_LOCAL_API_KEY},get AFK_LOCAL_BASE_URL(){return process.env.AFK_LOCAL_BASE_URL},get AFK_OPENAI_BASE_URL(){return process.env.AFK_OPENAI_BASE_URL},get TELEGRAM_BOT_TOKEN(){return process.env.TELEGRAM_BOT_TOKEN},get AFK_TELEGRAM_BOT_TOKEN(){return process.env.AFK_TELEGRAM_BOT_TOKEN},get AFK_TELEGRAM_ALLOWED_CHAT_IDS(){return process.env.AFK_TELEGRAM_ALLOWED_CHAT_IDS},get TELEGRAM_DATA_DIR(){return process.env.TELEGRAM_DATA_DIR},get TELEGRAM_VERBOSE(){return process.env.TELEGRAM_VERBOSE},get AFK_TELEGRAM_TRACE(){return process.env.AFK_TELEGRAM_TRACE},get AFK_TELEGRAM_CWD(){return process.env.AFK_TELEGRAM_CWD},get AFK_HOME(){return process.env.AFK_HOME},get AFK_STATE_DIR(){return process.env.AFK_STATE_DIR},get AFK_FRAMEWORK_DIR(){return process.env.AFK_FRAMEWORK_DIR},get AFK_EVAL_HARNESS_ROOT(){return process.env.AFK_EVAL_HARNESS_ROOT},get HOME(){return process.env.HOME},get PATH(){return process.env.PATH},get AFK_DAEMON_CWD(){return process.env.AFK_DAEMON_CWD},get AFK_DAEMON_TASK(){return process.env.AFK_DAEMON_TASK},get AFK_DAEMON_TASK_ID(){return process.env.AFK_DAEMON_TASK_ID},get AFK_SESSIONSTART_COOLDOWN_MS(){return process.env.AFK_SESSIONSTART_COOLDOWN_MS},get AFK_WORKTREE_AUTONAME(){return process.env.AFK_WORKTREE_AUTONAME},get AFK_WORKTREE_BRANCH_PREFIX(){return process.env.AFK_WORKTREE_BRANCH_PREFIX},get AFK_WORKTREE_BOOT_PRUNE(){return process.env.AFK_WORKTREE_BOOT_PRUNE},get AFK_WORKTREE_PRUNE_DISABLE(){return process.env.AFK_WORKTREE_PRUNE_DISABLE},get AFK_WORKTREE_MAX_AGE_CLEAN(){return process.env.AFK_WORKTREE_MAX_AGE_CLEAN},get AFK_WORKTREE_MAX_AGE_DIRTY(){return process.env.AFK_WORKTREE_MAX_AGE_DIRTY},get AFK_WORKTREE_SWEEP_ROOT(){return process.env.AFK_WORKTREE_SWEEP_ROOT},get AFK_THREADS_ALLOWED_USERNAMES(){return process.env.AFK_THREADS_ALLOWED_USERNAMES},get AFK_THREADS_DRY_RUN(){return process.env.AFK_THREADS_DRY_RUN},get AFK_THREADS_POLL_INTERVAL_MS(){return process.env.AFK_THREADS_POLL_INTERVAL_MS},get AFK_THREADS_REPLY_MODE(){return process.env.AFK_THREADS_REPLY_MODE},get AFK_ALLOW_PROJECT_MCP(){return process.env.AFK_ALLOW_PROJECT_MCP},get AFK_AUTO_ROUTING(){return process.env.AFK_AUTO_ROUTING},get AFK_BANNER_PLAIN(){return process.env.AFK_BANNER_PLAIN},get AFK_SPINNER_TIPS(){return process.env.AFK_SPINNER_TIPS},get AFK_SHOW_DIFFS(){return process.env.AFK_SHOW_DIFFS},get AFK_SKILL_STREAM_VERBOSE(){return process.env.AFK_SKILL_STREAM_VERBOSE},get FORCE_COLOR(){return process.env.FORCE_COLOR},get NO_COLOR(){return process.env.NO_COLOR},get AFK_DEBUG(){return process.env.AFK_DEBUG},get AFK_DEBUG_CLIPBOARD(){return process.env.AFK_DEBUG_CLIPBOARD},get AFK_TRACE_DISABLED(){return process.env.AFK_TRACE_DISABLED},get DEBUG(){return process.env.DEBUG},get AGENT_AFK_ASCII(){return process.env.AGENT_AFK_ASCII},get AGENT_SURFACE(){return process.env.AGENT_SURFACE},get CI(){return process.env.CI},get NODE_ENV(){return process.env.NODE_ENV},get VITEST(){return process.env.VITEST},get NO_UPDATE_NOTIFIER(){return process.env.NO_UPDATE_NOTIFIER},get AFK_WRITE_DENYLIST(){return process.env.AFK_WRITE_DENYLIST},get AFK_WRITE_DIFF(){return process.env.AFK_WRITE_DIFF}};(function(){for(let e of hc){if(!e.secret)continue;let n=Object.getOwnPropertyDescriptor(v,e.name);n&&Object.defineProperty(v,e.name,{...n,enumerable:!1})}})();function Wr(t,e){return!t||!e||t==="unknown"||e==="unknown"?{drift:!1}:t===e?{drift:!1}:{drift:!0,message:`[daemon] Version mismatch: running ${t} but installed is ${e}. Exiting.`}}import{Telegraf as hg}from"telegraf";import vc from"better-sqlite3";import{existsSync as nt,mkdirSync as Yr,readFileSync as Rt,writeFileSync as Xr,readdirSync as Ec,appendFileSync as _c,unlinkSync as Qr,copyFileSync as Ac}from"fs";import{join as te,basename as Zr,resolve as Pt,relative as Tc}from"path";import{join as N,dirname as yc,isAbsolute as bc}from"path";import{homedir as In}from"os";import{fileURLToPath as wc}from"url";function Y(){let t=v.AFK_HOME;if(t!==void 0&&t!==""){if(!bc(t)||t==="/")throw new Error(`AFK_HOME must be an absolute path that is not /, got: ${t}`);return t}return N(In(),".afk")}function xe(){return N(Y(),"agent-framework")}function Et(){return N(xe(),"forge-telemetry.jsonl")}function $e(){return N(xe(),"briefs")}function _t(){return N(xe(),"ceiling-ledger")}function At(){return N(Y(),"skills")}function Ue(){return N(Y(),"plugins")}function kc(){return N(process.cwd(),".afk")}function Cn(){return N(kc(),"plugins")}function Tt(){return N(Ue(),".index.json")}function Mn(){return N(Dn(),"schedules.json")}function On(){let t=wc(import.meta.url),e=yc(t);return N(e,"bundled-plugins")}function Dn(){return N(Y(),"config")}function Ze(){return N(Y(),"state")}function et(){return N(Ze(),"sessions")}function xt(){return N(Ze(),"memory")}function je(){return N(Ze(),"session-grants.jsonl")}function Kr(){return N(Y(),"farms")}function Gr(t){return N(Kr(),t)}function qr(t="default"){return N(Ze(),"daemon",`agent-afk@${t}`)}function tt(){return N(Dn(),"afk.env")}function zr(){return N(Dn(),"afk.config.json")}function Jr(){return N(In(),".afk.env")}function Vr(){return N(In(),".afk.config.json")}function Sc(){return v.AFK_DEBUG==="1"||v.DEBUG==="1"}function M(...t){Sc()&&console.log(...t)}var eo="HOT.md",xc="HOT.md.bak",to="memory.db",no="memory-wal.jsonl",It="procedures",Rc=5250,rt=2,Pc=`
2
+ import{existsSync as oh,readFileSync as Br}from"fs";import{fileURLToPath as uc}from"url";import{dirname as pc,join as fc}from"path";var hc=[{name:"AFK_COMPACT_KEEP_LAST_TURNS",description:"Number of recent turns the compactor keeps verbatim during /compact. Default tuned in compact-handler.ts.",type:"number",required:!1,example:"6",category:"model"},{name:"AFK_COMPACT_MODEL",description:"Override the model used by the /compact summarizer. Falls back to a cheap default (haiku-class).",type:"string",required:!1,example:"claude-haiku-4-5",category:"model"},{name:"AFK_DEFAULT_SUBAGENT_MODEL",description:"Override the default model used when a subagent is dispatched without an explicit model.",type:"string",required:!1,example:"sonnet",category:"model"},{name:"AFK_DISABLE_PROMPT_CACHE",description:"Disable Anthropic prompt caching when set to 1/true/yes/on. Unset = caching enabled.",type:"boolean",required:!1,default:"0",example:"1",category:"model"},{name:"AFK_EFFORT",description:"Reasoning-effort hint for the Codex provider. Accepts low | medium | high.",type:"string",required:!1,example:"medium",category:"model"},{name:"AFK_MAX_BUDGET_USD",description:"Per-turn USD budget ceiling. Aborts the turn when projected spend would exceed this.",type:"number",required:!1,default:"5.00",example:"10.00",category:"model"},{name:"AFK_MAX_OUTPUT_TOKENS",description:"Cap on output tokens per turn. Falls back to provider default when unset.",type:"number",required:!1,example:"8192",category:"model"},{name:"AFK_MAX_TOKENS",description:"Cap on total tokens per turn (input + output). Default 4096.",type:"number",required:!1,default:"4096",example:"8192",category:"model"},{name:"AFK_MODEL",description:"Default model for agent turns. Accepts short aliases (opus, sonnet, haiku) or full model IDs.",type:"string",required:!1,default:"sonnet",example:"claude-opus-4-5",category:"model"},{name:"AFK_PROMPT_CACHE_TTL",description:"TTL for Anthropic prompt-cache blocks. Accepts 5m or 1h.",type:"string",required:!1,default:"1h",example:"1h",category:"model"},{name:"AFK_TASK_BUDGET",description:"Per-task token budget ceiling. Aborts when cumulative usage would exceed it.",type:"number",required:!1,default:"100000",example:"200000",category:"model"},{name:"AFK_TEMPERATURE",description:"Numeric temperature override for model sampling. Provider default if unset.",type:"number",required:!1,example:"0.7",category:"model"},{name:"AFK_THINKING",description:"Extended-thinking toggle. Accepts on | off | <budget-tokens>. On by default.",type:"string",required:!1,default:"on",example:"on",category:"model"},{name:"AFK_TIMEOUT_MS",description:"Per-turn timeout in milliseconds. Provider/SDK default if unset.",type:"number",required:!1,example:"120000",category:"model"},{name:"CLAUDE_MODEL",description:"Legacy alias for AFK_MODEL \u2014 supported for back-compat with pre-AFK_* deployments.",type:"string",required:!1,example:"sonnet",category:"model"},{name:"AFK_SYSTEM_PROMPT",description:"Raw system-prompt string. Tier-1 source (highest priority over afk.config.json and AFK.md).",type:"string",required:!1,example:"You are a helpful agent.",category:"model"},{name:"AFK_DUMP_PROMPT",description:"Write the resolved system prompt to a file at startup. Accepts a path or 1 for default location.",type:"string",required:!1,example:"/tmp/afk-prompt.txt",category:"debug"},{name:"ANTHROPIC_API_KEY",description:"Anthropic API key. Tier-1 credential \u2014 overrides keychain OAuth and CLAUDE_CODE_OAUTH_TOKEN.",type:"string",required:!1,category:"auth",secret:!0},{name:"CLAUDE_CODE_OAUTH_TOKEN",description:"Claude Code OAuth token. Tier-2 credential \u2014 used when ANTHROPIC_API_KEY is unset; falls back to keychain.",type:"string",required:!1,category:"auth",secret:!0},{name:"OPENAI_API_KEY",description:"OpenAI API key for the openai-compatible provider (gpt-*, o1*, o3*, o4* models).",type:"string",required:!1,category:"auth",secret:!0},{name:"CODEX_API_KEY",description:"Codex API key for the @openai/codex-sdk provider when AFK_MODEL=codex-*.",type:"string",required:!1,category:"auth",secret:!0},{name:"AFK_LOCAL_API_KEY",description:"Placeholder API key for local Anthropic-compatible servers (vllm-mlx, etc.). Set when AFK_LOCAL_BASE_URL is configured.",type:"string",required:!1,default:"local",example:"local",category:"auth",secret:!0},{name:"AFK_LOCAL_BASE_URL",description:"Base URL for a self-hosted Anthropic-compatible server. When set, routes traffic away from api.anthropic.com.",type:"string",required:!1,example:"http://127.0.0.1:8080",category:"model"},{name:"AFK_OPENAI_BASE_URL",description:"Base URL override for the OpenAI-compatible provider. Used for local shims (mlx_lm.server, Ollama, vLLM, LM Studio).",type:"string",required:!1,example:"http://127.0.0.1:8000/v1",category:"model"},{name:"TELEGRAM_BOT_TOKEN",description:"Telegram bot token from @BotFather. Required to run the Telegram bot surface.",type:"string",required:!1,category:"telegram",secret:!0},{name:"AFK_TELEGRAM_BOT_TOKEN",description:"Alternative env var name for the Telegram bot token, accepted by the setup wizard.",type:"string",required:!1,category:"telegram",secret:!0},{name:"AFK_TELEGRAM_ALLOWED_CHAT_IDS",description:"Comma-separated list of Telegram chat IDs allowed to interact with the bot. Required when the bot is running.",type:"string",required:!1,example:"123456789,987654321",category:"telegram"},{name:"TELEGRAM_DATA_DIR",description:"Override the directory where Telegram bot state is stored. Defaults to ~/.afk/state/telegram/.",type:"string",required:!1,category:"telegram"},{name:"TELEGRAM_VERBOSE",description:"Set to 1 to log per-message details from the Telegram bot \u2014 chat IDs, message text, latency.",type:"boolean",required:!1,example:"1",category:"telegram"},{name:"AFK_TELEGRAM_TRACE",description:"Set to 1 to dump raw bridge traffic between the agent and the Telegram bot \u2014 debugging only.",type:"boolean",required:!1,example:"1",category:"debug"},{name:"AFK_TELEGRAM_CWD",description:"Override the working directory used by the Telegram bot when spawning agent sessions.",type:"string",required:!1,category:"telegram"},{name:"AFK_HOME",description:"Override the AFK home directory. Default: ~/.afk/.",type:"string",required:!1,default:"~/.afk",example:"/opt/afk",category:"paths"},{name:"AFK_STATE_DIR",description:"Override the AFK state directory. Default: $AFK_HOME/state/.",type:"string",required:!1,category:"paths"},{name:"AFK_FRAMEWORK_DIR",description:"Override the AFK agent-framework directory used for telemetry and briefs. Default: $AFK_HOME/agent-framework/.",type:"string",required:!1,category:"paths"},{name:"AFK_EVAL_HARNESS_ROOT",description:"Root path for the forge evaluation harness. Used by the L1/L2 capability-gate checks.",type:"string",required:!1,category:"paths"},{name:"HOME",description:"Standard Unix home directory. Used as the fallback when AFK_HOME is unset.",type:"string",required:!1,category:"process"},{name:"PATH",description:"System PATH. Read for executable resolution (git, gh, etc.) in tool handlers.",type:"string",required:!1,category:"process"},{name:"AFK_DAEMON_CWD",description:"Working directory used by the daemon process for spawned agent sessions.",type:"string",required:!1,category:"daemon"},{name:"AFK_DAEMON_TASK",description:"Default task description for the daemon. Falls back to afk.config.json daemon.task.",type:"string",required:!1,category:"daemon"},{name:"AFK_DAEMON_TASK_ID",description:"Task identifier the daemon uses to scope its state directory and telemetry.",type:"string",required:!1,category:"daemon"},{name:"AFK_SESSIONSTART_COOLDOWN_MS",description:"Cooldown in milliseconds between SessionStart trigger fires in the daemon. Prevents thundering-herd on rapid restarts.",type:"number",required:!1,category:"daemon"},{name:"AFK_WORKTREE_AUTONAME",description:"Auto-rename worktree branches based on the first user message in interactive mode. 1 = on (default), 0 = off.",type:"boolean",required:!1,default:"1",example:"0",category:"worktree"},{name:"AFK_WORKTREE_BRANCH_PREFIX",description:"Branch-name prefix for AFK-managed worktrees. Default afk/. Set to empty string to drop the prefix.",type:"string",required:!1,default:"afk/",example:"wt/",category:"worktree"},{name:"AFK_WORKTREE_BOOT_PRUNE",description:"When set, the daemon prunes stale worktrees at boot in addition to the cron-driven sweep.",type:"boolean",required:!1,category:"worktree"},{name:"AFK_WORKTREE_PRUNE_DISABLE",description:"Disable the worktree prune job entirely. Useful for long-running tests.",type:"boolean",required:!1,category:"worktree"},{name:"AFK_WORKTREE_MAX_AGE_CLEAN",description:"Maximum age (in days) before a clean worktree is auto-pruned. Default 14.",type:"number",required:!1,default:"14",category:"worktree"},{name:"AFK_WORKTREE_MAX_AGE_DIRTY",description:"Maximum age (in days) before a dirty worktree is auto-pruned. Default 30.",type:"number",required:!1,default:"30",category:"worktree"},{name:"AFK_WORKTREE_SWEEP_ROOT",description:"Override the root directory under which AFK worktrees are tracked for pruning.",type:"string",required:!1,category:"worktree"},{name:"AFK_THREADS_ALLOWED_USERNAMES",description:"Comma-separated allowlist of Threads usernames the agent will respond to.",type:"string",required:!1,example:"alice,bob",category:"threads"},{name:"AFK_THREADS_DRY_RUN",description:"Set to 1 to log Threads replies without actually posting them.",type:"boolean",required:!1,example:"1",category:"threads"},{name:"AFK_THREADS_POLL_INTERVAL_MS",description:"Poll interval for the Threads inbox watcher in milliseconds.",type:"number",required:!1,category:"threads"},{name:"AFK_THREADS_REPLY_MODE",description:"Threads reply mode. Accepts post (default) or other modes defined in src/threads.ts.",type:"string",required:!1,default:"post",example:"post",category:"threads"},{name:"AFK_ALLOW_PROJECT_MCP",description:"Allow loading MCP configuration from <cwd>/.mcp.json. Disabled by default \u2014 opt-in to mitigate config-injection risks.",type:"boolean",required:!1,example:"1",category:"mcp"},{name:"AFK_AUTO_ROUTING",description:"Auto-route bare slash inputs to matching skills. Applies to interactive, chat, telegram, and daemon surfaces.",type:"boolean",required:!1,example:"true",category:"misc"},{name:"AFK_BANNER_PLAIN",description:"Suppress the ANSI-colored banner at REPL startup. Useful for non-TTY captures and CI logs.",type:"boolean",required:!1,example:"1",category:"misc"},{name:"AFK_SPINNER_TIPS",description:"Show rotating tips in the loading spinner during long calls. 1 = on, 0 = off.",type:"boolean",required:!1,category:"misc"},{name:"AFK_SHOW_DIFFS",description:"Show inline diffs in the tool-lane output for edit/write tool calls. 1 = on, 0 = off.",type:"boolean",required:!1,category:"misc"},{name:"AFK_SKILL_STREAM_VERBOSE",description:"Verbose streaming output when a skill is dispatched. Logs sub-agent setup, intermediate events, and final result.",type:"boolean",required:!1,category:"debug"},{name:"FORCE_COLOR",description:"Standard Node convention. Force-enable ANSI color output even when stdout is not a TTY.",type:"string",required:!1,example:"1",category:"process"},{name:"NO_COLOR",description:"Standard convention (https://no-color.org). When set to any non-empty value, disables ANSI color output.",type:"string",required:!1,example:"1",category:"process"},{name:"AFK_DEBUG",description:"Enable verbose debug logging across the codebase. Accepts 1 to enable.",type:"boolean",required:!1,example:"1",category:"debug"},{name:"AFK_DEBUG_CLIPBOARD",description:"Debug bracketed-paste and image-paste handling in the interactive REPL.",type:"boolean",required:!1,category:"debug"},{name:"AFK_TRACE_DISABLED",description:"Disable the agent trace subsystem entirely. Set to 1 to skip trace file writes.",type:"boolean",required:!1,example:"1",category:"debug"},{name:"DEBUG",description:"Standard Node `debug`-package convention. When set to 1, enables verbose logging in several modules alongside AFK_DEBUG.",type:"string",required:!1,category:"debug"},{name:"AGENT_AFK_ASCII",description:"Force the interactive REPL tool-lane renderer to ASCII-only glyphs instead of the default Unicode box-drawing set. Accepts 1/true/yes (case-insensitive). Useful for terminals whose font lacks \u2503\u251C\u2570\u251C glyphs.",type:"boolean",required:!1,example:"1",category:"debug"},{name:"AGENT_SURFACE",description:"Internal surface marker propagated to subprocesses. Identifies which AFK surface (cli, telegram, daemon, threads) spawned the process.",type:"string",required:!1,example:"cli",category:"process"},{name:"CI",description:"Standard CI-detection convention. Auto-set by GitHub Actions, CircleCI, etc. Used to switch off TTY-only UX.",type:"string",required:!1,example:"true",category:"process"},{name:"NODE_ENV",description:"Standard Node environment marker. test | development | production. Used by routing-telemetry.ts to suppress test-time writes.",type:"string",required:!1,example:"production",category:"process"},{name:"VITEST",description:"Set automatically by Vitest. Used at runtime to short-circuit code paths that should not fire in tests.",type:"string",required:!1,category:"process"},{name:"NO_UPDATE_NOTIFIER",description:"Disable the update-available notifier on CLI startup. Standard convention shared with many Node CLIs.",type:"boolean",required:!1,category:"process"},{name:"AFK_WRITE_DENYLIST",description:"Comma-separated list of additional path globs that the write_file tool refuses to write to.",type:"string",required:!1,example:"**/.env,**/secrets/**",category:"misc"},{name:"AFK_WRITE_DIFF",description:"Show a diff preview before each write_file tool call. Defaults provider-controlled when unset.",type:"boolean",required:!1,category:"misc"},{name:"AFK_DEMO_CLEAN",description:"Explicit opt-in to capture-mode. When set to 1, suppresses high-frequency repaint drivers (spinner ticker, live thinking-preview) so recorded artifacts contain each state once instead of once per timer tick.",type:"boolean",required:!1,example:"1",category:"misc"},{name:"SCRIPT",description:"Set by script(1) on BSD/macOS/Linux to the typescript filename while a terminal session is being recorded. Presence of a non-empty value triggers capture-mode.",type:"string",required:!1,example:"/tmp/typescript",category:"process"},{name:"ASCIINEMA_REC",description:"Set to 1 by asciinema rec while a session is being recorded. Triggers capture-mode.",type:"boolean",required:!1,example:"1",category:"process"},{name:"SHELL",description:"Standard POSIX env var pointing to the user's login shell binary. Used by shell-init and worktree commands to auto-detect the correct shell syntax for emitted wrapper code.",type:"string",required:!1,example:"/bin/zsh",category:"process"},{name:"AFK_DIFF_LINES",description:"Maximum number of diff lines shown in the inline diff render during write_file tool calls. Set to 0 for no cap. Non-integer values are silently ignored and the default applies.",type:"number",required:!1,example:"50",category:"misc"},{name:"AFK_SHELL_WRAPPER",description:"Set to 1 or true by the optional afk shell wrapper function (installed via `afk shell-init`). Signals that the parent shell has the wrapper active so the post-exit cd can fire.",type:"boolean",required:!1,example:"1",category:"process"}],v={get AFK_COMPACT_KEEP_LAST_TURNS(){return process.env.AFK_COMPACT_KEEP_LAST_TURNS},get AFK_COMPACT_MODEL(){return process.env.AFK_COMPACT_MODEL},get AFK_DEFAULT_SUBAGENT_MODEL(){return process.env.AFK_DEFAULT_SUBAGENT_MODEL},get AFK_DISABLE_PROMPT_CACHE(){return process.env.AFK_DISABLE_PROMPT_CACHE},get AFK_EFFORT(){return process.env.AFK_EFFORT},get AFK_MAX_BUDGET_USD(){return process.env.AFK_MAX_BUDGET_USD},get AFK_MAX_OUTPUT_TOKENS(){return process.env.AFK_MAX_OUTPUT_TOKENS},get AFK_MAX_TOKENS(){return process.env.AFK_MAX_TOKENS},get AFK_MODEL(){return process.env.AFK_MODEL},get AFK_PROMPT_CACHE_TTL(){return process.env.AFK_PROMPT_CACHE_TTL},get AFK_TASK_BUDGET(){return process.env.AFK_TASK_BUDGET},get AFK_TEMPERATURE(){return process.env.AFK_TEMPERATURE},get AFK_THINKING(){return process.env.AFK_THINKING},get AFK_TIMEOUT_MS(){return process.env.AFK_TIMEOUT_MS},get CLAUDE_MODEL(){return process.env.CLAUDE_MODEL},get AFK_SYSTEM_PROMPT(){return process.env.AFK_SYSTEM_PROMPT},get AFK_DUMP_PROMPT(){return process.env.AFK_DUMP_PROMPT},get ANTHROPIC_API_KEY(){return process.env.ANTHROPIC_API_KEY},get CLAUDE_CODE_OAUTH_TOKEN(){return process.env.CLAUDE_CODE_OAUTH_TOKEN},get OPENAI_API_KEY(){return process.env.OPENAI_API_KEY},get CODEX_API_KEY(){return process.env.CODEX_API_KEY},get AFK_LOCAL_API_KEY(){return process.env.AFK_LOCAL_API_KEY},get AFK_LOCAL_BASE_URL(){return process.env.AFK_LOCAL_BASE_URL},get AFK_OPENAI_BASE_URL(){return process.env.AFK_OPENAI_BASE_URL},get TELEGRAM_BOT_TOKEN(){return process.env.TELEGRAM_BOT_TOKEN},get AFK_TELEGRAM_BOT_TOKEN(){return process.env.AFK_TELEGRAM_BOT_TOKEN},get AFK_TELEGRAM_ALLOWED_CHAT_IDS(){return process.env.AFK_TELEGRAM_ALLOWED_CHAT_IDS},get TELEGRAM_DATA_DIR(){return process.env.TELEGRAM_DATA_DIR},get TELEGRAM_VERBOSE(){return process.env.TELEGRAM_VERBOSE},get AFK_TELEGRAM_TRACE(){return process.env.AFK_TELEGRAM_TRACE},get AFK_TELEGRAM_CWD(){return process.env.AFK_TELEGRAM_CWD},get AFK_HOME(){return process.env.AFK_HOME},get AFK_STATE_DIR(){return process.env.AFK_STATE_DIR},get AFK_FRAMEWORK_DIR(){return process.env.AFK_FRAMEWORK_DIR},get AFK_EVAL_HARNESS_ROOT(){return process.env.AFK_EVAL_HARNESS_ROOT},get HOME(){return process.env.HOME},get PATH(){return process.env.PATH},get AFK_DAEMON_CWD(){return process.env.AFK_DAEMON_CWD},get AFK_DAEMON_TASK(){return process.env.AFK_DAEMON_TASK},get AFK_DAEMON_TASK_ID(){return process.env.AFK_DAEMON_TASK_ID},get AFK_SESSIONSTART_COOLDOWN_MS(){return process.env.AFK_SESSIONSTART_COOLDOWN_MS},get AFK_WORKTREE_AUTONAME(){return process.env.AFK_WORKTREE_AUTONAME},get AFK_WORKTREE_BRANCH_PREFIX(){return process.env.AFK_WORKTREE_BRANCH_PREFIX},get AFK_WORKTREE_BOOT_PRUNE(){return process.env.AFK_WORKTREE_BOOT_PRUNE},get AFK_WORKTREE_PRUNE_DISABLE(){return process.env.AFK_WORKTREE_PRUNE_DISABLE},get AFK_WORKTREE_MAX_AGE_CLEAN(){return process.env.AFK_WORKTREE_MAX_AGE_CLEAN},get AFK_WORKTREE_MAX_AGE_DIRTY(){return process.env.AFK_WORKTREE_MAX_AGE_DIRTY},get AFK_WORKTREE_SWEEP_ROOT(){return process.env.AFK_WORKTREE_SWEEP_ROOT},get AFK_THREADS_ALLOWED_USERNAMES(){return process.env.AFK_THREADS_ALLOWED_USERNAMES},get AFK_THREADS_DRY_RUN(){return process.env.AFK_THREADS_DRY_RUN},get AFK_THREADS_POLL_INTERVAL_MS(){return process.env.AFK_THREADS_POLL_INTERVAL_MS},get AFK_THREADS_REPLY_MODE(){return process.env.AFK_THREADS_REPLY_MODE},get AFK_ALLOW_PROJECT_MCP(){return process.env.AFK_ALLOW_PROJECT_MCP},get AFK_AUTO_ROUTING(){return process.env.AFK_AUTO_ROUTING},get AFK_BANNER_PLAIN(){return process.env.AFK_BANNER_PLAIN},get AFK_SPINNER_TIPS(){return process.env.AFK_SPINNER_TIPS},get AFK_SHOW_DIFFS(){return process.env.AFK_SHOW_DIFFS},get AFK_SKILL_STREAM_VERBOSE(){return process.env.AFK_SKILL_STREAM_VERBOSE},get FORCE_COLOR(){return process.env.FORCE_COLOR},get NO_COLOR(){return process.env.NO_COLOR},get AFK_DEBUG(){return process.env.AFK_DEBUG},get AFK_DEBUG_CLIPBOARD(){return process.env.AFK_DEBUG_CLIPBOARD},get AFK_TRACE_DISABLED(){return process.env.AFK_TRACE_DISABLED},get DEBUG(){return process.env.DEBUG},get AGENT_AFK_ASCII(){return process.env.AGENT_AFK_ASCII},get AGENT_SURFACE(){return process.env.AGENT_SURFACE},get CI(){return process.env.CI},get NODE_ENV(){return process.env.NODE_ENV},get VITEST(){return process.env.VITEST},get NO_UPDATE_NOTIFIER(){return process.env.NO_UPDATE_NOTIFIER},get AFK_WRITE_DENYLIST(){return process.env.AFK_WRITE_DENYLIST},get AFK_WRITE_DIFF(){return process.env.AFK_WRITE_DIFF},get AFK_DEMO_CLEAN(){return process.env.AFK_DEMO_CLEAN},get SCRIPT(){return process.env.SCRIPT},get ASCIINEMA_REC(){return process.env.ASCIINEMA_REC},get SHELL(){return process.env.SHELL},get AFK_DIFF_LINES(){return process.env.AFK_DIFF_LINES},get AFK_SHELL_WRAPPER(){return process.env.AFK_SHELL_WRAPPER}};(function(){for(let e of hc){if(!e.secret)continue;let n=Object.getOwnPropertyDescriptor(v,e.name);n&&Object.defineProperty(v,e.name,{...n,enumerable:!1})}})();function Wr(t,e){return!t||!e||t==="unknown"||e==="unknown"?{drift:!1}:t===e?{drift:!1}:{drift:!0,message:`[daemon] Version mismatch: running ${t} but installed is ${e}. Exiting.`}}import{Telegraf as hg}from"telegraf";import vc from"better-sqlite3";import{existsSync as nt,mkdirSync as Yr,readFileSync as Rt,writeFileSync as Xr,readdirSync as Ec,appendFileSync as _c,unlinkSync as Qr,copyFileSync as Ac}from"fs";import{join as te,basename as Zr,resolve as Pt,relative as Tc}from"path";import{join as N,dirname as yc,isAbsolute as bc}from"path";import{homedir as In}from"os";import{fileURLToPath as wc}from"url";function Y(){let t=v.AFK_HOME;if(t!==void 0&&t!==""){if(!bc(t)||t==="/")throw new Error(`AFK_HOME must be an absolute path that is not /, got: ${t}`);return t}return N(In(),".afk")}function xe(){return N(Y(),"agent-framework")}function Et(){return N(xe(),"forge-telemetry.jsonl")}function $e(){return N(xe(),"briefs")}function _t(){return N(xe(),"ceiling-ledger")}function At(){return N(Y(),"skills")}function Ue(){return N(Y(),"plugins")}function kc(){return N(process.cwd(),".afk")}function Cn(){return N(kc(),"plugins")}function Tt(){return N(Ue(),".index.json")}function Mn(){return N(Dn(),"schedules.json")}function On(){let t=wc(import.meta.url),e=yc(t);return N(e,"bundled-plugins")}function Dn(){return N(Y(),"config")}function Ze(){return N(Y(),"state")}function et(){return N(Ze(),"sessions")}function xt(){return N(Ze(),"memory")}function je(){return N(Ze(),"session-grants.jsonl")}function Kr(){return N(Y(),"farms")}function Gr(t){return N(Kr(),t)}function qr(t="default"){return N(Ze(),"daemon",`agent-afk@${t}`)}function tt(){return N(Dn(),"afk.env")}function zr(){return N(Dn(),"afk.config.json")}function Jr(){return N(In(),".afk.env")}function Vr(){return N(In(),".afk.config.json")}function Sc(){return v.AFK_DEBUG==="1"||v.DEBUG==="1"}function M(...t){Sc()&&console.log(...t)}var eo="HOT.md",xc="HOT.md.bak",to="memory.db",no="memory-wal.jsonl",It="procedures",Rc=5250,rt=2,Pc=`
3
3
  CREATE TABLE IF NOT EXISTS sessions (
4
4
  session_id TEXT PRIMARY KEY,
5
5
  surface TEXT NOT NULL,
@@ -1485,7 +1485,7 @@ SECURITY NOTE: upstream node output injected into downstream prompts is user-con
1485
1485
  --
1486
1486
  ${d.content}`,isError:!0}),i.push({call:l,result:d}),yield{type:"tool.output",toolUseId:l.id,content:d.content,...d.isError===!0?{isError:!0}:{},sessionId:this.initSessionId}}}this.priorTurns.push(Ti(e.assistantText,r));for(let c of xi(i))this.priorTurns.push(c)}async interrupt(){let e=this.abortController;if(e&&!e.signal.aborted){e.abort("interrupted");return}this.pendingAbortReason="interrupted"}async setModel(e){e!==void 0&&(this.currentModel=e)}async setPermissionMode(e){this.currentPermissionMode=Pi(e)}async supportedCommands(){try{return Le().map(n=>{let r={name:n.name,description:n.description};return n.argumentHint&&(r.argumentHint=n.argumentHint),r})}catch{return[]}}async supportedModels(){return[{value:"gpt-4o",displayName:"GPT-4o",description:"OpenAI flagship multimodal"},{value:"gpt-4o-mini",displayName:"GPT-4o mini",description:"Fast/cheap GPT-4o"},{value:"gpt-4.1",displayName:"GPT-4.1",description:"Long-context GPT-4"},{value:"gpt-4.1-mini",displayName:"GPT-4.1 mini",description:"Fast 4.1 variant"},{value:"o1",displayName:"o1",description:"Reasoning model"},{value:"o1-mini",displayName:"o1 mini",description:"Fast reasoning"},{value:"o3-mini",displayName:"o3 mini",description:"Newer reasoning, faster"}]}async supportedAgents(){return[]}async getContextUsage(){let e=this.lastUsage,n=ft(this.currentModel),r;if(e&&n>0){let o=mt(e);r=Math.min(100,Math.max(0,o/n*100))}return{tools:[],agents:[],isAutoCompactEnabled:!1,apiUsage:this.lastUsage,...r!==void 0?{percentage:r}:{},maxTokens:n}}async mcpServerStatus(){return this.opts.mcpManager?this.opts.mcpManager.getServerStates().map(e=>({name:e.serverName,status:e.status})):[]}async accountInfo(){return{authSource:this.opts.auth.source}}async rewindFiles(e,n){return{canRewind:!1,error:`${br} provider does not support file checkpoint rewind yet.`}}close(){this.closed=!0;let e=this.abortController;e&&!e.signal.aborted?e.abort("closed"):this.pendingAbortReason="closed",this.closeResolve?.(),M(`\u{1F7E2} ${br}: closed`)}};function zp(t){let e={apiKey:t.apiKey};return t.baseURL!==void 0&&(e.baseURL=t.baseURL),new Kp(e)}function Jp(t){if(!t||typeof t!="object")return"";let e=t,n=e.file_path??e.path??e.filePath;if(typeof n=="string")return" "+n;let r=e.command??e.cmd;if(typeof r=="string"){let s=r.split(`
1487
1487
  `)[0];return" "+(s.length>80?s.slice(0,77)+"\u2026":s)}let o=e.query??e.pattern??e.url??e.description;return typeof o=="string"?" "+o:""}function Ii(t,e,n={}){let r=hr(t.apiKey),o=t.resume??`openai-pending-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,s=typeof t.model=="string"?t.model:"gpt-4o-mini",i={auth:r,model:s,synthesizedSessionId:o,promptStream:e,config:t};return n.baseURL!==void 0&&(i.baseURL=n.baseURL),n.toolDispatcher!==void 0&&(i.toolDispatcher=n.toolDispatcher),n.mcpManager!==void 0&&(i.mcpManager=n.mcpManager),new dn(i)}var Qp="openai-compatible",ye=class{name=Qp;providerOpts;memoryStore;schemas;_sharedReadRoots;_sharedWriteRoots;_initialResolveBase;constructor(e={}){this.providerOpts=e,this.memoryStore=e.memoryStore??new X;let n=[...he];e.subagentExecutor&&n.push(Me),e.skillExecutor&&n.push(Oe),e.composeExecutor&&n.push(De),n.push(...we),this.schemas=n}query(e){let n=e.config,r=n.permissionMode??"default";this.ensureSharedRoots(n.cwd),n.readRoots&&this._sharedReadRoots&&this._sharedReadRoots.length<=1&&(this._sharedReadRoots.length=0,this._sharedReadRoots.push(...n.readRoots)),n.writeRoots&&this._sharedWriteRoots&&this._sharedWriteRoots.length<=1&&(this._sharedWriteRoots.length=0,this._sharedWriteRoots.push(...n.writeRoots));let o=this.providerOpts.tools??this.buildDispatcher(r,{...n.cwd!==void 0?{cwd:n.cwd}:{},...this._sharedReadRoots!==void 0?{readRoots:this._sharedReadRoots}:{},...this._sharedWriteRoots!==void 0?{writeRoots:this._sharedWriteRoots}:{},...n.sessionId!==void 0?{sessionId:n.sessionId}:{},...n.traceWriter!==void 0?{traceWriter:n.traceWriter}:{}}),s={};return this.providerOpts.baseURL!==void 0&&(s.baseURL=this.providerOpts.baseURL),s.toolDispatcher=o,this.providerOpts.mcpManager!==void 0&&(s.mcpManager=this.providerOpts.mcpManager),Ii(n,e.prompt,s)}buildDispatcher(e,n){let r=ln(e,n.cwd),o=ot(this.memoryStore,void 0,this.providerOpts.surface??"cli");for(let[c,a]of o)r.set(c,a);let s=this.providerOpts.mcpManager?this.providerOpts.mcpManager.getMcpTools():[];if(this.providerOpts.mcpManager)for(let[c,a]of this.providerOpts.mcpManager.getMcpHandlers())r.set(c,a);let i={handlers:r,schemas:[...this.schemas,...s]};return this.providerOpts.hookRegistry!==void 0&&(i.hookRegistry=this.providerOpts.hookRegistry),this.providerOpts.permissions!==void 0&&(i.permissions=this.providerOpts.permissions),this.providerOpts.subagentExecutor!==void 0&&(i.subagentExecutor=this.providerOpts.subagentExecutor),this.providerOpts.skillExecutor!==void 0&&(i.skillExecutor=this.providerOpts.skillExecutor),this.providerOpts.composeExecutor!==void 0&&(i.composeExecutor=this.providerOpts.composeExecutor),n.cwd!==void 0&&(i.cwd=n.cwd),n.readRoots!==void 0&&(i.readRoots=n.readRoots),n.writeRoots!==void 0&&(i.writeRoots=n.writeRoots),n.sessionId!==void 0&&(i.sessionId=n.sessionId),n.traceWriter!==void 0&&(i.traceWriter=n.traceWriter),new Fe(i)}ensureSharedRoots(e){if(!this._sharedReadRoots){let n=e?[e]:[];this._sharedReadRoots=n.slice(),this._sharedWriteRoots=n.slice(),e&&!this._initialResolveBase&&(this._initialResolveBase=e)}}addReadRoot(e,n="slash",r){this.ensureSharedRoots();let o=wr.resolve(e);this._sharedReadRoots.includes(o)||this._sharedReadRoots.push(o),this.appendProviderAuditLog({action:"grant-read",path:o,source:n,sessionId:r})}addWriteRoot(e,n="slash",r){this.ensureSharedRoots();let o=wr.resolve(e);this._sharedReadRoots.includes(o)||this._sharedReadRoots.push(o),this._sharedWriteRoots.includes(o)||this._sharedWriteRoots.push(o),this.appendProviderAuditLog({action:"grant-write",path:o,source:n,sessionId:r})}revokeRoot(e,n="slash",r){if(!this._sharedReadRoots)return;let o=wr.resolve(e);if(this._initialResolveBase&&o===this._initialResolveBase)return;let s=this._sharedReadRoots.indexOf(o);if(s!==-1&&this._sharedReadRoots.splice(s,1),this._sharedWriteRoots){let i=this._sharedWriteRoots.indexOf(o);i!==-1&&this._sharedWriteRoots.splice(i,1)}this.appendProviderAuditLog({action:"revoke",path:o,source:n,sessionId:r})}getGrants(){return{resolveBase:this._initialResolveBase,readRoots:this._sharedReadRoots?.slice()??[],writeRoots:this._sharedWriteRoots?.slice()??[]}}appendProviderAuditLog(e){try{let n=je();Yp(Xp(n),{recursive:!0});let r=JSON.stringify({timestamp:new Date().toISOString(),sessionId:e.sessionId??null,action:e.action,path:e.path,source:e.source});Vp(n,r+`
1488
- `)}catch{}}close(){this.memoryStore.close()}},Zp=new ye;import{readFileSync as Di,existsSync as Sr}from"fs";import{join as un}from"path";import{config as rf}from"dotenv";import{execFile as ef}from"node:child_process";import{promisify as tf}from"node:util";var Kv=tf(ef);var nf=/^[A-Za-z0-9_\-./]*$/,Ci=64;function Mi(t,e){if(t.length>Ci)throw new Error(`Invalid branch prefix from ${e}: length ${t.length} exceeds ${Ci}.`);if(!nf.test(t))throw new Error(`Invalid branch prefix from ${e}: '${t}' \u2014 only [A-Za-z0-9_-./] are allowed.`);if(t.startsWith("-"))throw new Error(`Invalid branch prefix from ${e}: '${t}' \u2014 must not start with '-' (would be parsed by git as a flag).`);return t}var gt={model:"sonnet",maxTokens:4096,temperature:1,updatePolicy:"notify"},Oi=!1;function ht(){return v.ANTHROPIC_API_KEY||v.CLAUDE_CODE_OAUTH_TOKEN||ie()}var kr;function of(){if(kr!==void 0)return kr;if(!Oi){let o=[un(process.cwd(),".env"),tt(),Jr()];for(let s of o)Sr(s)&&rf({path:s,override:!1});Oi=!0}let t={},e=ht();e!==void 0&&(t.apiKey=e);let n=v.AFK_MODEL??v.CLAUDE_MODEL;if(n){let o=n.toLowerCase();t.model=Jt(o)?o:n}let r=v.AFK_LOCAL_BASE_URL;if(r&&r.length>0&&(t.baseUrl=r,t.apiKey=v.AFK_LOCAL_API_KEY||"local"),v.AFK_MAX_TOKENS&&(t.maxTokens=parseInt(v.AFK_MAX_TOKENS,10)),v.AFK_TEMPERATURE&&(t.temperature=parseFloat(v.AFK_TEMPERATURE)),v.AFK_SYSTEM_PROMPT&&(t.systemPrompt=v.AFK_SYSTEM_PROMPT),v.AFK_AUTO_ROUTING){let o=v.AFK_AUTO_ROUTING.toLowerCase()==="true";t.autoRouting={interactive:o,chat:o,telegram:o,daemon:o}}return v.AFK_OPENAI_BASE_URL&&(t.openaiBaseUrl=v.AFK_OPENAI_BASE_URL),kr=t,t}var Ke,Ge;function sf(){if(Ke!==void 0)return Ke;let t=[un(process.cwd(),"afk.config.json"),zr(),Vr()];for(let e of t)if(Sr(e))try{let n=Di(e,"utf-8"),r=JSON.parse(n),o={};if(typeof r.model=="string"&&r.model.length>0){let s=r.model.toLowerCase();o.model=Jt(s)?s:r.model}if(typeof r.maxTokens=="number"&&(o.maxTokens=r.maxTokens),typeof r.temperature=="number"&&(o.temperature=r.temperature),r.systemPrompt&&(o.systemPrompt=r.systemPrompt),r.autoRouting&&typeof r.autoRouting=="object"){let s={};typeof r.autoRouting.interactive=="boolean"&&(s.interactive=r.autoRouting.interactive),typeof r.autoRouting.chat=="boolean"&&(s.chat=r.autoRouting.chat),typeof r.autoRouting.telegram=="boolean"&&(s.telegram=r.autoRouting.telegram),typeof r.autoRouting.daemon=="boolean"&&(s.daemon=r.autoRouting.daemon),o.autoRouting=s}if(r.daemon&&typeof r.daemon=="object"){let s={};typeof r.daemon.task=="string"&&(s.task=r.daemon.task),typeof r.daemon.taskId=="string"&&(s.taskId=r.daemon.taskId);let i=r.daemon.worktreePrune;i&&typeof i=="object"&&(s.worktreePrune={enabled:typeof i.enabled=="boolean"?i.enabled:!0,cron:typeof i.cron=="string"?i.cron:"0 4 * * *",maxAgeDaysClean:typeof i.maxAgeDaysClean=="number"?i.maxAgeDaysClean:14,maxAgeDaysDirty:typeof i.maxAgeDaysDirty=="number"?i.maxAgeDaysDirty:30,scope:typeof i.scope=="string"?i.scope:"all"}),o.daemon=s}if(r.updatePolicy&&["notify","auto","off"].includes(r.updatePolicy)&&(o.updatePolicy=r.updatePolicy),typeof r.autoResumeOnUsageLimit=="boolean"&&(o.autoResumeOnUsageLimit=r.autoResumeOnUsageLimit),typeof r.bgSummaries=="boolean"&&(o.bgSummaries=r.bgSummaries),typeof r.maxSummaryCallsPerSession=="number"&&(o.maxSummaryCallsPerSession=Math.min(500,Math.max(1,r.maxSummaryCallsPerSession))),r.interactive&&typeof r.interactive=="object"){let s={};typeof r.interactive.worktreeAutoname=="boolean"&&(s.worktreeAutoname=r.interactive.worktreeAutoname),typeof r.interactive.worktreeBranchPrefix=="string"&&(s.worktreeBranchPrefix=Mi(r.interactive.worktreeBranchPrefix,`${e}#/interactive/worktreeBranchPrefix`)),Object.keys(s).length>0&&(o.interactive=s)}return Ke={config:o,sourcePath:e},Ke}catch(n){console.error(`Warning: Failed to parse ${e}:`,n)}return Ke={config:{},sourcePath:void 0},Ke}function af(){if(Ge!==void 0)return Ge.value;let t=[un(process.cwd(),"AFK.md"),un(Y(),"AFK.md")];for(let e of t)if(Sr(e))try{let n=Di(e,"utf-8").trim();if(n.length>0)return Ge={value:{content:n,path:e}},Ge.value}catch{}return Ge={value:null},Ge.value}function vr(t){let e=of(),{config:n,sourcePath:r}=sf(),o={...gt,...e,...n,...t},s;if(e.systemPrompt!==void 0)s="env:AFK_SYSTEM_PROMPT";else if(n.systemPrompt!==void 0&&r!==void 0)s=`file:${r}`;else if(o.systemPrompt===void 0){let c=af();c!==null&&(o.systemPrompt=c.content,s=`afk-md:${c.path}`)}let i={model:o.model??gt.model,maxTokens:o.maxTokens??gt.maxTokens,temperature:o.temperature??gt.temperature,updatePolicy:o.updatePolicy??gt.updatePolicy,...o.apiKey!==void 0?{apiKey:o.apiKey}:{},...o.baseUrl!==void 0?{baseUrl:o.baseUrl}:{},...o.openaiBaseUrl!==void 0?{openaiBaseUrl:o.openaiBaseUrl}:{},...o.systemPrompt!==void 0?{systemPrompt:o.systemPrompt}:{},...s!==void 0?{systemPromptSource:s}:{},...o.autoRouting!==void 0?{autoRouting:o.autoRouting}:{},...o.daemon!==void 0?{daemon:o.daemon}:{},...o.bgSummaries!==void 0?{bgSummaries:o.bgSummaries}:{},...o.maxSummaryCallsPerSession!==void 0?{maxSummaryCallsPerSession:o.maxSummaryCallsPerSession}:{}};if(typeof i.model=="string"&&i.model.toLowerCase().startsWith("local-")&&(i.baseUrl===void 0||i.baseUrl.length===0))throw new Error(`Model '${i.model}' requires AFK_LOCAL_BASE_URL to be set (e.g. AFK_LOCAL_BASE_URL=http://127.0.0.1:8080). Point it at your local Anthropic-Messages-compatible server.`);return i}function J(){return ht()}function pn(t){let e=v.AFK_DEFAULT_SUBAGENT_MODEL;return e&&e.length>0?e:typeof t=="string"&&ee(t)==="openai-compatible"?t:"sonnet"}function cf(t){if(t===void 0)return;if(t==="max")return Number.POSITIVE_INFINITY;if(t===""||t==="NaN")throw new Error(`Invalid --max-output-tokens value: ${JSON.stringify(t)}. Expected a positive integer or 'max'.`);if(!/^\d+$/.test(t))throw new Error(`Invalid --max-output-tokens value: ${JSON.stringify(t)}. Expected a positive integer or 'max'.`);let e=Number(t);if(!Number.isFinite(e)||!Number.isInteger(e)||e<=0)throw new Error(`Invalid --max-output-tokens value: ${JSON.stringify(t)}. Must be a positive integer.`);return e}function Fi(){return cf(v.AFK_MAX_OUTPUT_TOKENS)}async function Li(t,e,n,r){let s=$("mint")["spec.md"];if(!s)throw new Error("mint skill missing spec.md prompt");let a=await(await new I(n!==void 0?{cwd:n}:{}).forkSubagent({parent:{sessionId:e},config:{model:"sonnet",systemPrompt:s,apiKey:J()},idPrefix:"mint-spec",...r?{parentId:r}:{}})).runToResult(`Create a detailed specification for: ${t}`);if(a.status!=="succeeded"||!a.message)throw new Error(`spec phase failed: ${H(a)}`);return a.message.content}async function Ni(t,e,n,r){let s=$("mint")["research.md"];if(!s)throw new Error("mint skill missing research.md prompt");let a=await(await new I(n!==void 0?{cwd:n}:{}).forkSubagent({parent:{sessionId:e},config:{model:"sonnet",systemPrompt:s,apiKey:J()},idPrefix:"mint-research",...r?{parentId:r}:{}})).runToResult(`Gather context and research for this specification:
1488
+ `)}catch{}}close(){this.memoryStore.close()}},Zp=new ye;import{readFileSync as Di,existsSync as Sr}from"fs";import{join as un}from"path";import{config as rf}from"dotenv";import{execFile as ef}from"node:child_process";import{promisify as tf}from"node:util";var qv=tf(ef);var nf=/^[A-Za-z0-9_\-./]*$/,Ci=64;function Mi(t,e){if(t.length>Ci)throw new Error(`Invalid branch prefix from ${e}: length ${t.length} exceeds ${Ci}.`);if(!nf.test(t))throw new Error(`Invalid branch prefix from ${e}: '${t}' \u2014 only [A-Za-z0-9_-./] are allowed.`);if(t.startsWith("-"))throw new Error(`Invalid branch prefix from ${e}: '${t}' \u2014 must not start with '-' (would be parsed by git as a flag).`);return t}var gt={model:"sonnet",maxTokens:4096,temperature:1,updatePolicy:"notify"},Oi=!1;function ht(){return v.ANTHROPIC_API_KEY||v.CLAUDE_CODE_OAUTH_TOKEN||ie()}var kr;function of(){if(kr!==void 0)return kr;if(!Oi){let o=[un(process.cwd(),".env"),tt(),Jr()];for(let s of o)Sr(s)&&rf({path:s,override:!1});Oi=!0}let t={},e=ht();e!==void 0&&(t.apiKey=e);let n=v.AFK_MODEL??v.CLAUDE_MODEL;if(n){let o=n.toLowerCase();t.model=Jt(o)?o:n}let r=v.AFK_LOCAL_BASE_URL;if(r&&r.length>0&&(t.baseUrl=r,t.apiKey=v.AFK_LOCAL_API_KEY||"local"),v.AFK_MAX_TOKENS&&(t.maxTokens=parseInt(v.AFK_MAX_TOKENS,10)),v.AFK_TEMPERATURE&&(t.temperature=parseFloat(v.AFK_TEMPERATURE)),v.AFK_SYSTEM_PROMPT&&(t.systemPrompt=v.AFK_SYSTEM_PROMPT),v.AFK_AUTO_ROUTING){let o=v.AFK_AUTO_ROUTING.toLowerCase()==="true";t.autoRouting={interactive:o,chat:o,telegram:o,daemon:o}}return v.AFK_OPENAI_BASE_URL&&(t.openaiBaseUrl=v.AFK_OPENAI_BASE_URL),kr=t,t}var Ke,Ge;function sf(){if(Ke!==void 0)return Ke;let t=[un(process.cwd(),"afk.config.json"),zr(),Vr()];for(let e of t)if(Sr(e))try{let n=Di(e,"utf-8"),r=JSON.parse(n),o={};if(typeof r.model=="string"&&r.model.length>0){let s=r.model.toLowerCase();o.model=Jt(s)?s:r.model}if(typeof r.maxTokens=="number"&&(o.maxTokens=r.maxTokens),typeof r.temperature=="number"&&(o.temperature=r.temperature),r.systemPrompt&&(o.systemPrompt=r.systemPrompt),r.autoRouting&&typeof r.autoRouting=="object"){let s={};typeof r.autoRouting.interactive=="boolean"&&(s.interactive=r.autoRouting.interactive),typeof r.autoRouting.chat=="boolean"&&(s.chat=r.autoRouting.chat),typeof r.autoRouting.telegram=="boolean"&&(s.telegram=r.autoRouting.telegram),typeof r.autoRouting.daemon=="boolean"&&(s.daemon=r.autoRouting.daemon),o.autoRouting=s}if(r.daemon&&typeof r.daemon=="object"){let s={};typeof r.daemon.task=="string"&&(s.task=r.daemon.task),typeof r.daemon.taskId=="string"&&(s.taskId=r.daemon.taskId);let i=r.daemon.worktreePrune;i&&typeof i=="object"&&(s.worktreePrune={enabled:typeof i.enabled=="boolean"?i.enabled:!0,cron:typeof i.cron=="string"?i.cron:"0 4 * * *",maxAgeDaysClean:typeof i.maxAgeDaysClean=="number"?i.maxAgeDaysClean:14,maxAgeDaysDirty:typeof i.maxAgeDaysDirty=="number"?i.maxAgeDaysDirty:30,scope:typeof i.scope=="string"?i.scope:"all"}),o.daemon=s}if(r.updatePolicy&&["notify","auto","off"].includes(r.updatePolicy)&&(o.updatePolicy=r.updatePolicy),typeof r.autoResumeOnUsageLimit=="boolean"&&(o.autoResumeOnUsageLimit=r.autoResumeOnUsageLimit),typeof r.bgSummaries=="boolean"&&(o.bgSummaries=r.bgSummaries),typeof r.maxSummaryCallsPerSession=="number"&&(o.maxSummaryCallsPerSession=Math.min(500,Math.max(1,r.maxSummaryCallsPerSession))),r.interactive&&typeof r.interactive=="object"){let s={};typeof r.interactive.worktreeAutoname=="boolean"&&(s.worktreeAutoname=r.interactive.worktreeAutoname),typeof r.interactive.worktreeBranchPrefix=="string"&&(s.worktreeBranchPrefix=Mi(r.interactive.worktreeBranchPrefix,`${e}#/interactive/worktreeBranchPrefix`)),Object.keys(s).length>0&&(o.interactive=s)}return Ke={config:o,sourcePath:e},Ke}catch(n){console.error(`Warning: Failed to parse ${e}:`,n)}return Ke={config:{},sourcePath:void 0},Ke}function af(){if(Ge!==void 0)return Ge.value;let t=[un(process.cwd(),"AFK.md"),un(Y(),"AFK.md")];for(let e of t)if(Sr(e))try{let n=Di(e,"utf-8").trim();if(n.length>0)return Ge={value:{content:n,path:e}},Ge.value}catch{}return Ge={value:null},Ge.value}function vr(t){let e=of(),{config:n,sourcePath:r}=sf(),o={...gt,...e,...n,...t},s;if(e.systemPrompt!==void 0)s="env:AFK_SYSTEM_PROMPT";else if(n.systemPrompt!==void 0&&r!==void 0)s=`file:${r}`;else if(o.systemPrompt===void 0){let c=af();c!==null&&(o.systemPrompt=c.content,s=`afk-md:${c.path}`)}let i={model:o.model??gt.model,maxTokens:o.maxTokens??gt.maxTokens,temperature:o.temperature??gt.temperature,updatePolicy:o.updatePolicy??gt.updatePolicy,...o.apiKey!==void 0?{apiKey:o.apiKey}:{},...o.baseUrl!==void 0?{baseUrl:o.baseUrl}:{},...o.openaiBaseUrl!==void 0?{openaiBaseUrl:o.openaiBaseUrl}:{},...o.systemPrompt!==void 0?{systemPrompt:o.systemPrompt}:{},...s!==void 0?{systemPromptSource:s}:{},...o.autoRouting!==void 0?{autoRouting:o.autoRouting}:{},...o.daemon!==void 0?{daemon:o.daemon}:{},...o.bgSummaries!==void 0?{bgSummaries:o.bgSummaries}:{},...o.maxSummaryCallsPerSession!==void 0?{maxSummaryCallsPerSession:o.maxSummaryCallsPerSession}:{}};if(typeof i.model=="string"&&i.model.toLowerCase().startsWith("local-")&&(i.baseUrl===void 0||i.baseUrl.length===0))throw new Error(`Model '${i.model}' requires AFK_LOCAL_BASE_URL to be set (e.g. AFK_LOCAL_BASE_URL=http://127.0.0.1:8080). Point it at your local Anthropic-Messages-compatible server.`);return i}function J(){return ht()}function pn(t){let e=v.AFK_DEFAULT_SUBAGENT_MODEL;return e&&e.length>0?e:typeof t=="string"&&ee(t)==="openai-compatible"?t:"sonnet"}function cf(t){if(t===void 0)return;if(t==="max")return Number.POSITIVE_INFINITY;if(t===""||t==="NaN")throw new Error(`Invalid --max-output-tokens value: ${JSON.stringify(t)}. Expected a positive integer or 'max'.`);if(!/^\d+$/.test(t))throw new Error(`Invalid --max-output-tokens value: ${JSON.stringify(t)}. Expected a positive integer or 'max'.`);let e=Number(t);if(!Number.isFinite(e)||!Number.isInteger(e)||e<=0)throw new Error(`Invalid --max-output-tokens value: ${JSON.stringify(t)}. Must be a positive integer.`);return e}function Fi(){return cf(v.AFK_MAX_OUTPUT_TOKENS)}async function Li(t,e,n,r){let s=$("mint")["spec.md"];if(!s)throw new Error("mint skill missing spec.md prompt");let a=await(await new I(n!==void 0?{cwd:n}:{}).forkSubagent({parent:{sessionId:e},config:{model:"sonnet",systemPrompt:s,apiKey:J()},idPrefix:"mint-spec",...r?{parentId:r}:{}})).runToResult(`Create a detailed specification for: ${t}`);if(a.status!=="succeeded"||!a.message)throw new Error(`spec phase failed: ${H(a)}`);return a.message.content}async function Ni(t,e,n,r){let s=$("mint")["research.md"];if(!s)throw new Error("mint skill missing research.md prompt");let a=await(await new I(n!==void 0?{cwd:n}:{}).forkSubagent({parent:{sessionId:e},config:{model:"sonnet",systemPrompt:s,apiKey:J()},idPrefix:"mint-research",...r?{parentId:r}:{}})).runToResult(`Gather context and research for this specification:
1489
1489
 
1490
1490
  ${t}`);if(a.status!=="succeeded"||!a.message)throw new Error(`research phase failed: ${H(a)}`);return a.message.content}async function $i(t,e,n,r,o){let i=$("mint")["plan.md"];if(!i)throw new Error("mint skill missing plan.md prompt");let a=await new I(r!==void 0?{cwd:r}:{}).forkSubagent({parent:{sessionId:n},config:{model:"sonnet",systemPrompt:i,apiKey:J()},idPrefix:"mint-plan",...o?{parentId:o}:{}}),l=`Specification:
1491
1491
  ${t}
@@ -1603,7 +1603,7 @@ Save reusable multi-step workflows the user teaches you or that you discover wor
1603
1603
  `),x=this.buildDispatcher(y,{cwd:R,readRoots:this._sharedReadRoots,writeRoots:this._sharedWriteRoots,...n.env!==void 0?{env:n.env}:{},sessionId:n.sessionId,traceWriter:n.traceWriter});return{userSystem:A,dispatcher:x}},C=Pm(n.effort,u);return new yn({client:a,authMode:r?"api-key":s,promptStream:e.prompt,toolDispatcher:f,...E!==void 0?{sessionId:E}:{},..._!==void 0?{initialMessages:_}:{},model:u,...n.permissionMode!==void 0?{permissionMode:n.permissionMode}:{},maxTokens:p,tools:m,userSystem:h,systemPrefix:l,tokenRefresher:S,...n.thinking!==void 0?{thinking:Rm(n.thinking,p,u)}:{},...C!==void 0?{effort:C}:{},...r?{baseUrl:n.baseUrl}:{},...n.traceWriter?{traceWriter:n.traceWriter}:{},...n.autoResumeOnUsageLimit!==void 0?{autoResumeOnUsageLimit:n.autoResumeOnUsageLimit}:{},...T!==void 0?{cwdDependentsFactory:T}:{},...this.mcpManager!==void 0?{mcpManager:this.mcpManager}:{},...wa(n.autoCompact)!==void 0?{autoCompactThreshold:wa(n.autoCompact)}:{}})}};function _m(t){if(t===void 0)return null;if(typeof t=="string")return t.length>0?t:null;if(typeof t=="object"&&t!==null&&"append"in t){let e=t.append;return e&&e.length>0?e:null}return null}var Am=.9;function wa(t){if(t===void 0||t===!1)return;if(t===!0)return Am;let e=t.threshold;if(!(typeof e!="number"||!Number.isFinite(e)||e<=0||e>=1))return e}function Tm(t,e){let n=t.maxOutputTokens;return typeof n=="number"&&Number.isFinite(n)&&n>0?Math.floor(n):hi(e)}function xm(t){if(!t||t.length===0)return;let e=[];for(let n of t)n.user.length>0&&e.push({role:"user",content:n.user}),n.assistant.length>0&&e.push({role:"assistant",content:n.assistant});return e.length>0?e:void 0}function Rm(t,e,n){switch(t.type){case"adaptive":return{type:"adaptive",display:"summarized"};case"disabled":return{type:"disabled"};case"enabled":{if(typeof n=="string"&&Em(n))return{type:"adaptive",display:"summarized"};let r=t.budgetTokens!==void 0&&Number.isFinite(t.budgetTokens)?Math.min(t.budgetTokens,e-1):e-1;return{type:"enabled",budget_tokens:Math.max(r,1024),display:"summarized"}}}}function Pm(t,e){if(t!==void 0)return t;let n=e.toLowerCase();if(/(claude-)?(opus|sonnet)-4-[67]/.test(n))return"max"}var Im=new ce;var Cm=new Set([...Object.keys(zt),"auto"]);function ee(t){if(!t)return"anthropic-direct";let e=t.trim().toLowerCase();return!e||Cm.has(e)||e.startsWith("claude-")||e.startsWith("claude_")?"anthropic-direct":e.startsWith("gpt-")||e.startsWith("gpt_")||e.startsWith("o1")||e.startsWith("o3")||e.startsWith("o4")||e.startsWith("codex-")||e.startsWith("codex_")||e==="codex"||e.includes("/")?"openai-compatible":"anthropic-direct"}function Jo(t){switch(ee(t)){case"openai-compatible":case"openai-codex":return new ye;default:return new ce}}async function Fr(t,e,n,r){let o=t.chat?.id;if(!o){await t.reply(j("Could not identify chat"));return}try{await e.resetSession(o),n.delete(o),await t.reply(Lt())}catch(s){r("Clear error:",s),await t.reply(j(s))}}async function Sa(t,e,n){let r=t.chat?.id;if(!r){await t.reply(j("Could not identify chat"));return}try{let o=await e.getSession(r);await t.sendChatAction("typing").catch(()=>{});let s=await o.compact();s.compacted?await t.reply(wo({before:s.messagesBefore,after:s.messagesAfter,...s.tokensSavedEstimate!==void 0?{tokensSavedEstimate:s.tokensSavedEstimate}:{}})):await t.reply(ko(s.reason??"unknown"))}catch(o){n("Compact error:",o),await t.reply(j(o))}}async function Lr(t,e,n){let r=t.chat?.id;if(!r){await t.reply(j("Could not identify chat"));return}let s=t.message.text.split(/\s+/).slice(1);if(s.length===0){let u=e.getModel(r);await t.reply(`Current model: ${u.toUpperCase()}
1604
1604
 
1605
1605
  Usage: /model [opus|opus_1m|sonnet|sonnet_1m|haiku] or org/model HF id`);return}let i=s[0];if(!i){await t.reply(j("Please specify a model: opus, opus_1m, sonnet, sonnet_1m, haiku, or an org/model id"));return}let c=["opus","opus_1m","sonnet","sonnet_1m","haiku"],a=i.toLowerCase(),l=c.includes(a),d=ee(a)==="openai-compatible";if(!l&&!d){await t.reply(j(`Invalid model: ${i}
1606
- Aliases: ${c.join(", ")}, or org/model HF id`));return}try{await e.switchModel(r,a),await t.reply(ho(a))}catch(u){n("Model switch error:",u),await t.reply(j(u))}}function Dm(t,e){let n=t.trim();return n==="~"?ka():n.startsWith("~/")?Dr(ka(),n.slice(2)):Om(n)?Dr(n):Dr(e,n)}async function va(t,e,n){let r=t.chat?.id;if(!r){await t.reply(j("Could not identify chat"));return}let s=t.message.text.split(/\s+/).slice(1).filter(l=>l.length>0);if(s.length===0){let l=e.getCwd(r);await t.reply(yo(l));return}let i=s[0];if(!i){await t.reply(j("Please specify a directory path"));return}let c=e.getCwd(r)??process.cwd(),a=Dm(i,c);try{if(!(await Mm.stat(a)).isDirectory()){await t.reply(j(`Not a directory: ${a}`));return}}catch(l){let d=l.code;d==="ENOENT"?await t.reply(j(`Directory does not exist: ${a}`)):d==="EACCES"?await t.reply(j(`Permission denied: ${a}`)):(n("cwd stat error:",l),await t.reply(j(l)));return}try{await e.setCwd(r,a),await t.reply(bo(a))}catch(l){n("Cwd switch error:",l),await t.reply(j(l))}}import{execFile as Qm,spawn as Zm}from"node:child_process";import{promisify as eg}from"node:util";import{execFile as Fm}from"node:child_process";import{randomBytes as Lm}from"node:crypto";import{promises as bn}from"node:fs";import{join as wn}from"node:path";import{promisify as Nm}from"node:util";var RT=Nm(Fm);var Ae=class extends Error{cause;code;constructor(e,n,r){super(e),this.name="WorktreeError",this.cause=n,this.code=r}};function $m(t,e=40){return t.toLowerCase().trim().replace(/[^a-z0-9]+/g,"-").replace(/^-+|-+$/g,"").slice(0,e).replace(/-+$/g,"")||"task"}function Um(){return Lm(4).toString("hex").slice(0,4)}function jm(t){let e=n=>String(n).padStart(2,"0");return`${t.getUTCFullYear()}${e(t.getUTCMonth()+1)}${e(t.getUTCDate())}T${e(t.getUTCHours())}${e(t.getUTCMinutes())}${e(t.getUTCSeconds())}`}function Ea(t,e={}){let n=(e.now??(()=>new Date))(),r=(e.randomSuffix??Um)();return`${jm(n)}-${$m(t,32)}-${r}`}function Hm(t){let e=t;return e.respawnedAt===void 0&&(e.respawnedAt=void 0),e.respawnedAs===void 0&&(e.respawnedAs=void 0),e.prUrl===void 0&&(e.prUrl=void 0),e.prCreatedAt===void 0&&(e.prCreatedAt=void 0),e}async function Je(t){let e=wn(Gr(t),"farm.json");try{let n=await bn.readFile(e,"utf8"),r=JSON.parse(n);if(r.schemaVersion!==1&&r.schemaVersion!==2&&r.schemaVersion!==3)throw new Ae(`unsupported farm manifest schema: ${r.schemaVersion} (expected 1, 2, or 3)`,void 0,"unsupported-schema");return Hm(r)}catch(n){if(n.code==="ENOENT")return null;throw n instanceof Ae?n:new Ae(`failed to load farm manifest ${e}`,n,"invalid")}}async function _a(t,e){let n=await Je(t);if(!n)throw new Ae(`farm not found: ${t}`);return n.human_decision=e,n.decidedAt=new Date().toISOString(),n.schemaVersion=3,await bn.writeFile(wn(n.farmDir,"farm.json"),JSON.stringify(n,null,2)+`
1606
+ Aliases: ${c.join(", ")}, or org/model HF id`));return}try{await e.switchModel(r,a),await t.reply(ho(a))}catch(u){n("Model switch error:",u),await t.reply(j(u))}}function Dm(t,e){let n=t.trim();return n==="~"?ka():n.startsWith("~/")?Dr(ka(),n.slice(2)):Om(n)?Dr(n):Dr(e,n)}async function va(t,e,n){let r=t.chat?.id;if(!r){await t.reply(j("Could not identify chat"));return}let s=t.message.text.split(/\s+/).slice(1).filter(l=>l.length>0);if(s.length===0){let l=e.getCwd(r);await t.reply(yo(l));return}let i=s[0];if(!i){await t.reply(j("Please specify a directory path"));return}let c=e.getCwd(r)??process.cwd(),a=Dm(i,c);try{if(!(await Mm.stat(a)).isDirectory()){await t.reply(j(`Not a directory: ${a}`));return}}catch(l){let d=l.code;d==="ENOENT"?await t.reply(j(`Directory does not exist: ${a}`)):d==="EACCES"?await t.reply(j(`Permission denied: ${a}`)):(n("cwd stat error:",l),await t.reply(j(l)));return}try{await e.setCwd(r,a),await t.reply(bo(a))}catch(l){n("Cwd switch error:",l),await t.reply(j(l))}}import{execFile as Qm,spawn as Zm}from"node:child_process";import{promisify as eg}from"node:util";import{execFile as Fm}from"node:child_process";import{randomBytes as Lm}from"node:crypto";import{promises as bn}from"node:fs";import{join as wn}from"node:path";import{promisify as Nm}from"node:util";var IT=Nm(Fm);var Ae=class extends Error{cause;code;constructor(e,n,r){super(e),this.name="WorktreeError",this.cause=n,this.code=r}};function $m(t,e=40){return t.toLowerCase().trim().replace(/[^a-z0-9]+/g,"-").replace(/^-+|-+$/g,"").slice(0,e).replace(/-+$/g,"")||"task"}function Um(){return Lm(4).toString("hex").slice(0,4)}function jm(t){let e=n=>String(n).padStart(2,"0");return`${t.getUTCFullYear()}${e(t.getUTCMonth()+1)}${e(t.getUTCDate())}T${e(t.getUTCHours())}${e(t.getUTCMinutes())}${e(t.getUTCSeconds())}`}function Ea(t,e={}){let n=(e.now??(()=>new Date))(),r=(e.randomSuffix??Um)();return`${jm(n)}-${$m(t,32)}-${r}`}function Hm(t){let e=t;return e.respawnedAt===void 0&&(e.respawnedAt=void 0),e.respawnedAs===void 0&&(e.respawnedAs=void 0),e.prUrl===void 0&&(e.prUrl=void 0),e.prCreatedAt===void 0&&(e.prCreatedAt=void 0),e}async function Je(t){let e=wn(Gr(t),"farm.json");try{let n=await bn.readFile(e,"utf8"),r=JSON.parse(n);if(r.schemaVersion!==1&&r.schemaVersion!==2&&r.schemaVersion!==3)throw new Ae(`unsupported farm manifest schema: ${r.schemaVersion} (expected 1, 2, or 3)`,void 0,"unsupported-schema");return Hm(r)}catch(n){if(n.code==="ENOENT")return null;throw n instanceof Ae?n:new Ae(`failed to load farm manifest ${e}`,n,"invalid")}}async function _a(t,e){let n=await Je(t);if(!n)throw new Ae(`farm not found: ${t}`);return n.human_decision=e,n.decidedAt=new Date().toISOString(),n.schemaVersion=3,await bn.writeFile(wn(n.farmDir,"farm.json"),JSON.stringify(n,null,2)+`
1607
1607
  `,"utf8"),n}async function Aa(t,e){let n=await Je(t);if(!n)throw new Ae(`farm not found: ${t}`);return n.respawnedAt=new Date().toISOString(),n.respawnedAs=e,n.schemaVersion=3,await bn.writeFile(wn(n.farmDir,"farm.json"),JSON.stringify(n,null,2)+`
1608
1608
  `,"utf8"),n}async function Ta(t,e){let n=await Je(t);if(!n)throw new Ae(`farm not found: ${t}`);return n.prUrl=e,n.prCreatedAt=new Date().toISOString(),n.schemaVersion=3,await bn.writeFile(wn(n.farmDir,"farm.json"),JSON.stringify(n,null,2)+`
1609
1609
  `,"utf8"),n}import{execFile as Bm}from"node:child_process";import{promisify as Wm}from"node:util";var Km=Wm(Bm),Gm=2e4,bt=class extends Error{kind;exitCode;stderr;constructor(e,n,r,o){super(e),this.name="GhError",this.kind=n,this.exitCode=r,this.stderr=o}},kn=null,yt=null;function qm(t,e,n){return n===!0?"timeout":e==="ENOENT"?"not-found":/already exists/i.test(t)?"already-exists":/authentication|please log in|HTTP 40[13]|bad credentials|token|scope/i.test(t)?"unauthed":/ETIMEDOUT|ECONNRESET|ECONNREFUSED/i.test(t+(e??""))?"network":"unknown"}function xa(t,e){return Km(t,e,{timeout:Gm,killSignal:"SIGTERM"}).then(n=>({stdout:n.stdout,stderr:n.stderr}))}async function Ra(t={}){let e=(t._now??(()=>Date.now()))(),n=t.ttlMs??6e4,r=t.execFn??xa,o=t.log??(()=>{});return kn&&kn.expiresAt>e?(o("[gh] checkGhReady cache hit"),kn.result):(o("[gh] checkGhReady cache miss \u2014 probing"),yt||(yt=(async()=>{try{await r("gh",["--version"])}catch(c){let a=c;return a.killed?{ok:!1,hint:"`gh` timed out \u2014 check connectivity"}:a.code==="ENOENT"?{ok:!1,hint:"`gh` CLI not found \u2014 install with: brew install gh"}:{ok:!1,hint:"`gh --version` failed unexpectedly \u2014 check gh installation"}}try{await r("gh",["auth","status"])}catch(c){let a=c;return a.killed||/ETIMEDOUT|ECONNRESET|ENOTFOUND/i.test(String(a.code??""))?{ok:!1,hint:"check network \u2014 cannot reach GitHub"}:{ok:!1,hint:"`gh` is not authenticated \u2014 run: gh auth login"}}let i={ok:!0};return n>0&&(kn={result:i,expiresAt:e+n}),i})().finally(()=>{yt=null}),yt))}async function Pa(t,e){let n=e??xa,r=["pr","create","--base",t.base,"--head",t.head,"--title",t.title,"--body",t.body];try{let{stdout:o}=await n("gh",r);return o.trim()}catch(o){let s=o,i=s.stderr??"",c=s.code,a=s.exitCode??1,l=qm(i,c,s.killed);throw new bt(`gh pr create failed (${l}): ${i.trim()}`,l,a,i)}}function Ia(t,e){let n;try{n=e?._store??new X}catch(r){return{skipped:!0,reason:r instanceof Error?r.message:String(r)}}try{let r={type:"farm-decision",taskSlug:t.taskSlug,decision:t.decision,decidedAt:t.decidedAt,via:t.via};return{factId:n.storeFact({category:"decision",content:JSON.stringify(r),source_surface:"afk"})}}catch(r){return{skipped:!0,reason:r instanceof Error?r.message:String(r)}}}import{promises as zm}from"node:fs";import{join as Jm}from"node:path";function Oa(t){let e=t.filter(r=>r.score!==null),n=t.filter(r=>r.score===null).map(r=>r.index).sort((r,o)=>r-o);return e.sort((r,o)=>{let s=r.score,i=o.score,c=Ca(s),a=Ca(i);if(c!==a)return a-c;let l=Ma(s.lint_ok),d=Ma(i.lint_ok);return l!==d?d-l:s.loc_delta!==i.loc_delta?s.loc_delta-i.loc_delta:r.index-o.index}),[...e.map(r=>r.index),...n]}function Ca(t){let e=t.pass+t.fail;return e===0?0:t.pass/e}function Ma(t){return t===!0?2:t===!1?1:0}async function Sn(t,e={}){if(t.branches.length===0)throw new Error(`resolveWinnerBranch: farm ${t.taskSlug} has no branches`);let n=e.loadScore??Vm,r=await Promise.all(t.branches.map(async i=>({index:i.index,score:await n(t.farmDir,i.index)}))),o=Oa(r),s=new Map(t.branches.map(i=>[i.index,i]));for(let i of o){let c=r.find(a=>a.index===i)?.score;if(c&&c.pass>0&&c.fail===0){let a=s.get(i);if(a)return{branch:a,source:"winner"}}}for(let i of o)if(r.find(a=>a.index===i)?.score){let a=s.get(i);if(a)return{branch:a,source:"top-scored"}}return{branch:t.branches[0],source:"fallback-first-branch"}}async function Vm(t,e){let n=Jm(t,"scores",`branch-${e}.json`);try{let r=await zm.readFile(n,"utf8");return JSON.parse(r)}catch(r){return r.code==="ENOENT",null}}var vn="afk:f:";var Ym=new Set(["p","d","r","x"]),Xm=/^[a-z0-9T][a-z0-9T-]{0,62}$/;function Da(t){if(!t||!t.startsWith(vn)||Buffer.byteLength(t,"utf8")>64)return null;let e=t.slice(vn.length),n=e.indexOf(":");if(n<1)return null;let r=e.slice(0,n),o=e.slice(n+1);return!Ym.has(r)||!Xm.test(o)?null:{action:r,taskSlug:o}}var tg=eg(Qm),Nr=new Map;async function La(t,e={}){let n=e.log??(()=>{}),r=ng(t),o=Da(r);if(!o){await q(t,"Unknown action",n);return}if(t.chat?.id===void 0){await q(t,"No chat context",n);return}let s=e.loadFarm??Je,i;try{i=await s(o.taskSlug)}catch(c){n("[farm-callback] loadFarm failed:",c),await q(t,"Farm load failed",n);return}if(!i){await q(t,"Farm not found (already GC\u2019d?)",n);return}try{await rg(o.action,t,i,e,n)}catch(c){n("[farm-callback] dispatch error:",c),await q(t,"Internal error",n)}}function ng(t){return t.callbackQuery?.data}async function q(t,e,n){try{await t.answerCbQuery(e)}catch(r){n("[farm-callback] answerCbQuery failed:",r)}}async function rg(t,e,n,r,o){switch(t){case"x":return ag(e,n,r,o);case"d":return cg(e,n,r,o);case"p":return Fa(`p:${n.taskSlug}`,e,n,r,o,og);case"r":return Fa(`r:${n.taskSlug}`,e,n,r,o,ig)}}async function Fa(t,e,n,r,o,s){let i=Nr.get(t);if(i){o(`[farm-callback] ${t} \u2014 second tap, awaiting in-flight lock`);try{await i}catch{}let a=r.loadFarm??Je,l;try{l=await a(n.taskSlug)}catch{await q(e,"Farm load failed",o);return}if(!l){await q(e,"Farm not found",o);return}return s(e,l,r,o)}let c=s(e,n,r,o);return Nr.set(t,c),c.finally(()=>{Nr.delete(t)}),c}async function og(t,e,n,r){if(e.prUrl){await q(t,`PR already open: ${e.prUrl}`,r);return}await q(t,"Opening PR\u2026",r);let o=n.checkGhReady??Ra,s;try{s=await o()}catch(m){r("[farm-callback] checkGhReady threw:",m);try{await t.reply("gh readiness check failed \u2014 see daemon logs")}catch{}return}if(!s.ok){try{await t.reply(s.hint)}catch{}return}let i=n.resolveWinnerBranch??Sn,c;try{c=await i(e)}catch(m){r("[farm-callback] resolveWinnerBranch failed:",m);try{await t.reply("Winner lookup failed")}catch{}return}let a=c.branch.branch,l=e.baseBranch??"main",d=`Auto PR: ${e.taskName}`,u=`Auto-generated by afk farm ${e.taskSlug} | winner: ${a} | created: ${new Date().toISOString()}`,p=n.createPr??Pa,y;try{y=await p({base:l,head:a,title:d,body:u})}catch(m){if(m instanceof bt){let b={"not-found":"gh CLI not found \u2014 install with: brew install gh","already-exists":"PR already exists for this branch",unauthed:"gh is not authenticated \u2014 run: gh auth login",network:"Network error \u2014 check gh connectivity",timeout:"gh timed out \u2014 check connectivity",unknown:"gh pr create failed \u2014 see daemon logs"};try{await t.reply(b[m.kind])}catch{}return}r("[farm-callback] createPr failed:",m);try{await t.reply("gh pr create failed \u2014 see daemon logs")}catch{}return}let f=n.recordPrCreated??Ta;try{await f(e.taskSlug,y)}catch(m){r("[farm-callback] recordPrCreated failed:",m)}try{await t.reply(`PR opened \u2713
@@ -1643,13 +1643,13 @@ No reset time available. Wait for the limit to reset, then send again \u2014 or
1643
1643
 
1644
1644
  Resets at ${c.toLocaleTimeString(void 0,{hour:"numeric",minute:"2-digit",hour12:!0})} (in ~${R} min).
1645
1645
 
1646
- I'll auto-resume when the limit resets \u2014 no need to retype.`;l=!0,u(x,!0).finally(()=>{l=!1})}},pg));continue}if(g.type==="resumed"){a!==null&&(clearInterval(a),a=null),c=null;let S=g.hotSwapped&&g.accountId?`\u25B6 **Resumed on ${g.accountId}**`:"\u25B6 **Resumed**";await u(S,!0);continue}if(g.type==="done"){a!==null&&(clearInterval(a),a=null),o.trim()&&await u(o,!0);break}if(g.type==="error")throw a!==null&&(clearInterval(a),a=null),g.error}})}finally{await Promise.resolve(y.return?.(void 0)).catch(()=>{})}if(o&&s){let h=Ot(Dt(o));if(h.length>1)for(let g=1;g<h.length;g++){let S=h[g];S&&await t.reply(S,{parse_mode:"HTML"})}}await Promise.resolve(y.return?.(void 0)).catch(h=>{r?.("iter.return error (session cleanup):",h)})}catch(p){throw r?.("Streaming error:",p),p}}async function $r(t,e,n,r,o){if(!r.has(e))try{await Promise.race([n.waitForInitialization(),new Promise((c,a)=>setTimeout(()=>a(new Error("timeout")),5e3))]);let s=n.getSessionMetadata(),i=[{command:"start",description:"Show welcome and command list"},{command:"help",description:"Show this command list"},{command:"clear",description:"Clear conversation history"},{command:"compact",description:"Compact conversation history"},{command:"model",description:"Switch Claude model (opus/sonnet/haiku)"},{command:"cd",description:"Show or change session working directory"}];if(s.slashCommands?.length)for(let c of s.slashCommands){let a=c.replace(/^\//,"");i.push({command:a,description:`SDK command: ${a}`})}if(s.skills?.length)for(let c of s.skills)i.push({command:c,description:`Run ${c} skill`});await t.telegram.setMyCommands(i,{scope:{type:"chat",chat_id:e}}),r.add(e),o(`Registered ${i.length} commands for chat ${e}`)}catch(s){o(`Could not register dynamic commands for chat ${e}:`,s)}}function mg(t){return t.length<3?null:t.length>=4&&t[0]===137&&t[1]===80&&t[2]===78&&t[3]===71?"image/png":t[0]===71&&t[1]===73&&t[2]===70?"image/gif":t.length>=12&&t[0]===82&&t[1]===73&&t[2]===70&&t[3]===70&&t[8]===87&&t[9]===69&&t[10]===66&&t[11]===80?"image/webp":t[0]===255&&t[1]===216&&t[2]===255?"image/jpeg":null}async function gg(t,e){let n=t.headers.get("content-length");if(n!=null){let c=Number(n);if(Number.isFinite(c)&&c>e)return{status:"too-large",bytesRead:c}}let r=t.body;if(!r)return{status:"missing-body"};let o=r.getReader(),s=[],i=0;try{for(;;){let{done:c,value:a}=await o.read();if(c)break;if(i+=a.byteLength,i>e)return await o.cancel().catch(()=>{}),{status:"too-large",bytesRead:i};s.push(Buffer.from(a))}}finally{o.releaseLock()}return{status:"ok",bytes:Buffer.concat(s,i)}}var En=class t{static MAX_QUEUE_DEPTH=5;sessionManager;messageQueues=new Map;registeredCommandChats;log;bot;constructor(e,n,r,o){this.bot=e,this.sessionManager=n,this.registeredCommandChats=r,this.log=o}async handlePhoto(e){let n=e.chat?.id,r=e.message,o=r?.photo;if(!n||!o?.length){this.log(`Photo handling: missing chatId or photo array for chat ${n??"(unknown)"}`);return}this.log(`\u{1F4F7} Photo from chat ID: ${n}`);let s=o[o.length-1];if(!s){this.log(`Photo handling: empty photo array for chat ${n}`);return}let i=5*1024*1024;if(s.file_size!=null&&s.file_size>i){this.log(`Photo handling: oversized file (${s.file_size} bytes) rejected for chat ${n}`),await e.reply("\u274C Image is too large (max 5 MB). Please send a smaller photo.");return}let c=r?.caption;try{let a=await this.sessionManager.getSession(n);if($r(this.bot,n,a,this.registeredCommandChats,this.log).catch(g=>this.log("Failed to register chat commands:",g)),a.state!=="idle"&&(this.messageQueues.get(n)?.length??0)>=t.MAX_QUEUE_DEPTH){await e.reply("\u23F3 Queue full. Please wait for your messages to be processed.");return}let l=await e.telegram.getFileLink(s.file_id),d=l instanceof URL?l:new URL(String(l));if(d.protocol!=="https:"||d.hostname!=="api.telegram.org"||d.port!==""&&d.port!=="443"){this.log(`Photo handling: unexpected file URL (protocol=${d.protocol} hostname=${d.hostname}) rejected for chat ${n}`),await e.reply("\u274C Couldn't download the image. Please try resending.");return}let u=await globalThis.fetch(d.href,{signal:AbortSignal.timeout(15e3),redirect:"error"});if(!u.ok){this.log(`Photo handling: fetch failed with status ${u.status} for chat ${n}`),await e.reply("\u274C Couldn't download the image. Please try resending.");return}let p=await gg(u,i);if(p.status==="too-large"){this.log(`Photo handling: downloaded file (${p.bytesRead} bytes) exceeds limit for chat ${n}`),await e.reply("\u274C Image is too large (max 5 MB). Please send a smaller photo.");return}if(p.status==="missing-body"){this.log(`Photo handling: fetch response had no body for chat ${n}`),await e.reply("\u274C Couldn't download the image. Please try resending.");return}let y=p.bytes,f=y.toString("base64"),m=["image/jpeg","image/png","image/gif","image/webp"],b=u.headers.get("content-type")??"",w=(b.split(";")[0]?.trim()??"").toLowerCase(),k;if(m.includes(w))k=w;else{let g=mg(y);if(g!==null)this.log(`Photo: sniffed ${g} (Content-Type was "${b}") for chat ${n}`),k=g;else{this.log(`Photo: unrecognised image format for chat ${n} (Content-Type: "${b}")`),await e.reply("\u274C Unsupported image format. Please send a JPEG, PNG, GIF, or WebP.");return}}let h=[];if(c!=null&&h.push({type:"text",text:`[User caption]: ${[...c].slice(0,1024).join("")}`}),h.push({type:"image",source:{type:"base64",media_type:k,data:f}}),a.state!=="idle"){this.enqueuePhoto(n,e,h)&&await e.reply("Message queued.");return}await this.processOne(n,e,h)}catch(a){let d=(a instanceof Error?a.message:String(a)).replace(/\/bot[^/]+\//g,"/bot[REDACTED]/");this.log("Photo handling error:",d),wt(a)?await e.reply("\u23F3 Rate limit reached. Please wait a moment and try again."):kt(a)?await e.reply("\u274C Couldn't download the image. Please try resending."):await e.reply(Ft())}}async handle(e){let n=e.chat?.id,r=e.message.text;if(!(!n||!r)&&(this.log(`\u{1F4EC} Message from chat ID: ${n}`),!r.startsWith("/")))try{let o=await this.sessionManager.getSession(n);if($r(this.bot,n,o,this.registeredCommandChats,this.log).catch(s=>this.log("Failed to register chat commands:",s)),o.state!=="idle"){this.enqueueMessage(n,e,r)&&await e.reply("Message queued.");return}await this.processOne(n,e,r)}catch(o){this.log("Message handling error:",o),wt(o)?await e.reply("\u23F3 Rate limit reached. Please wait a moment and try again."):kt(o)?await e.reply("\u{1F310} Network error. Please check your connection and try again."):await e.reply(Ft())}}async processClearDirect(e,n){try{await this.sessionManager.resetSession(e),this.registeredCommandChats.delete(e),await n.reply(Lt())}catch(r){this.log("Clear error:",r),await n.reply(j(r))}}enqueueMessage(e,n,r){let o=this.messageQueues.get(e);return o||(o=[],this.messageQueues.set(e,o)),o.length>=t.MAX_QUEUE_DEPTH?(n.reply("\u23F3 Queue full. Please wait for your messages to be processed.").catch(()=>{}),!1):(o.push({type:"message",ctx:n,text:r}),!0)}enqueuePhoto(e,n,r){let o=this.messageQueues.get(e);return o||(o=[],this.messageQueues.set(e,o)),o.length>=t.MAX_QUEUE_DEPTH?(n.reply("\u23F3 Queue full. Please wait for your messages to be processed.").catch(()=>{}),!1):(o.push({type:"photo",ctx:n,content:r}),!0)}enqueueClear(e,n){let r=this.messageQueues.get(e);r||(r=[],this.messageQueues.set(e,r)),r.push({type:"clear",ctx:n})}async processOne(e,n,r){let o=!1;try{let s=await this.sessionManager.getSession(e);await n.sendChatAction("typing").catch(()=>{}),await Ua(n,s,r,this.log)}catch(s){if(this.log("Message handling error:",s),(s?.message??"").includes("session is busy")){(typeof r=="string"?this.enqueueMessage(e,n,r):this.enqueuePhoto(e,n,r))&&await n.reply("Message queued."),o=!0;return}wt(s)?await n.reply("\u23F3 Rate limit reached. Please wait a moment and try again."):kt(s)?await n.reply("\u{1F310} Network error. Please check your connection and try again."):await n.reply(Ft())}finally{o||this.drainQueue(e).catch(s=>this.log("Drain error:",s))}}async drainQueue(e){let n=this.messageQueues.get(e);if(!n?.length)return;let r=n.shift();r.type==="message"?await this.processOne(e,r.ctx,r.text):r.type==="photo"?await this.processOne(e,r.ctx,r.content):await this.processClearDirect(e,r.ctx)}};var _n=class{bot;sessionManager;options;running=!1;registeredCommandChats=new Set;messageHandler;constructor(e){this.options=e,this.bot=new hg(e.botToken),this.sessionManager=new Mt(e),this.messageHandler=new En(this.bot,this.sessionManager,this.registeredCommandChats,this.log.bind(this)),this.setupHandlers()}setupHandlers(){this.bot.use(Ys(this.options.allowedChatIds,this.log.bind(this))),this.bot.command("start",n=>Nn(n)),this.bot.command("help",n=>$n(n,this.sessionManager)),this.bot.command("clear",async n=>{let r=n.chat?.id;if(!r){await n.reply(j("Could not identify chat"));return}(await this.sessionManager.getSession(r)).state!=="idle"?(this.messageHandler.enqueueClear(r,n),await n.reply("Clear queued.")):await Fr(n,this.sessionManager,this.registeredCommandChats,this.log.bind(this))}),this.bot.command("compact",n=>Sa(n,this.sessionManager,this.log.bind(this))),this.bot.command("model",n=>Lr(n,this.sessionManager,this.log.bind(this))),this.bot.command(["cd","cwd"],n=>va(n,this.sessionManager,this.log.bind(this))),this.bot.on("text",n=>this.messageHandler.handle(n)),this.bot.on("photo",n=>this.messageHandler.handlePhoto(n));let e=new RegExp(`^${yg(vn)}`);this.bot.action(e,n=>La(n,{log:this.log.bind(this)})),this.bot.catch((n,r)=>{this.log("Bot error:",n),r.reply(j("An unexpected error occurred. Please try again.")).catch(o=>this.log("Failed to send error message:",o))})}async start(){if(this.running)throw new Error("Bot is already running");this.log("Loading sessions..."),await this.sessionManager.loadSessions(),this.log("Starting bot..."),await this.bot.launch(),this.log("Registering bot commands..."),await this.bot.telegram.setMyCommands([{command:"start",description:"Show welcome and command list"},{command:"help",description:"Show this command list"},{command:"clear",description:"Clear conversation history"},{command:"compact",description:"Compact conversation history"},{command:"model",description:"Switch Claude model (opus/sonnet/haiku)"}]),this.running=!0,this.log("Bot started successfully");let e=async n=>{this.log(`Received ${n}, shutting down...`),await this.stop(),process.exit(0)};process.once("SIGINT",()=>e("SIGINT")),process.once("SIGTERM",()=>e("SIGTERM"))}async stop(){if(this.running){this.log("Stopping bot..."),this.running=!1,this.log("Closing sessions..."),await this.sessionManager.closeAll(),this.log("Stopping bot polling...");try{this.bot.stop()}catch(e){this.log("Error stopping bot (may not have been started):",e)}this.log("Bot stopped")}}getStats(){return{running:this.running,activeSessions:this.sessionManager.getSessionCount(),totalChats:this.sessionManager.getChatCount()}}async handleStart(e){return Nn(e)}async handleHelp(e){return $n(e,this.sessionManager)}async handleClear(e){let n=e.chat?.id;if(!n){await e.reply(j("Could not identify chat"));return}if((await this.sessionManager.getSession(n)).state!=="idle")this.messageHandler.enqueueClear(n,e),await e.reply("Clear queued.");else return Fr(e,this.sessionManager,this.registeredCommandChats,this.log.bind(this))}async handleMessage(e){return this.messageHandler.handle(e)}async handlePhoto(e){return this.messageHandler.handlePhoto(e)}async handleModelSwitch(e){return Lr(e,this.sessionManager,this.log.bind(this))}log(...e){this.options.verbose&&console.log("[TelegramBot]",...e)}};function yg(t){return t.replace(/[.*+?^${}()|[\]\\]/g,"\\$&")}import Tx from"chalk";var bg="https://api.telegram.org";async function ja(t){try{let e=await fetch(`${bg}/bot${t}/getMe`);if(!e.ok)return null;let n=await e.json();return!n.ok||!n.result?.id||!n.result?.first_name?null:{id:n.result.id,...n.result.username!==void 0?{username:n.result.username}:{},firstName:n.result.first_name}}catch{return null}}function Ha(){return Vo()}var wg=["shadow-verify","shadow_verify","resolve","diagnose","appmap","qualify","mint"],kg=[/\bverdict(s)?\b/i,/\brecommend(ation)?s?\b/i,/\bshould\s+(delete|remove|rewrite|refactor|rename|reject|merge|revert|disable)\b/i,/\b(USELESS|KEEP|REJECT|APPROVE|SALVAGE|BLOCK|FAIL)\b/,/\b(redundant|duplicated|superseded|obsolete)\b/i,/\bvulnerab\w*\b/i,/\bunused\b/i,/\bbroken\b/i,/\bregress\w*\b/i,/\|\s*(status|verdict|decision|severity|risk|finding|priority|holds\??)\s*\|/i,/\bfound\s+\d+\s*(issue|problem|bug|error|finding|vulnerabilit)/i,/\b(critical|high|medium|low)\s+(severity|priority|risk)\b/i,/\bclaim(s)?\b[^\n]{0,80}\b(holds?|refuted|verified|partial|confirmed|disputed)\b/i,/\b(root\s*cause|incident)\b/i,/\brecommend\s+(removing|deleting|rewriting|refactoring|merging|reverting)\b/i,/\bI\s+(applied|committed|pushed|edited|wrote|fixed|patched|reset|restored|staged)\b/i,/\b(applied|committed|pushed|fixed|patched)\s+(the|these|those)\s+(change|commit|fix|patch|edit)/i],Sg=[/\bverifier_verdict\b/i,/"\s*claim\s*"\s*:/i,/\bre-derived\b[^.\n]{0,80}\bindependent/i,/\bindependently\s+(re-derived|re-verified|verified|checked)\b/i,/\bverifier\s+(agrees|disagrees|confirms|refutes)\b/i],vg=`shadow-verify nudge:
1646
+ I'll auto-resume when the limit resets \u2014 no need to retype.`;l=!0,u(x,!0).finally(()=>{l=!1})}},pg));continue}if(g.type==="resumed"){a!==null&&(clearInterval(a),a=null),c=null;let S=g.hotSwapped&&g.accountId?`\u25B6 **Resumed on ${g.accountId}**`:"\u25B6 **Resumed**";await u(S,!0);continue}if(g.type==="done"){a!==null&&(clearInterval(a),a=null),o.trim()&&await u(o,!0);break}if(g.type==="error")throw a!==null&&(clearInterval(a),a=null),g.error}})}finally{await Promise.resolve(y.return?.(void 0)).catch(()=>{})}if(o&&s){let h=Ot(Dt(o));if(h.length>1)for(let g=1;g<h.length;g++){let S=h[g];S&&await t.reply(S,{parse_mode:"HTML"})}}await Promise.resolve(y.return?.(void 0)).catch(h=>{r?.("iter.return error (session cleanup):",h)})}catch(p){throw r?.("Streaming error:",p),p}}async function $r(t,e,n,r,o){if(!r.has(e))try{await Promise.race([n.waitForInitialization(),new Promise((c,a)=>setTimeout(()=>a(new Error("timeout")),5e3))]);let s=n.getSessionMetadata(),i=[{command:"start",description:"Show welcome and command list"},{command:"help",description:"Show this command list"},{command:"clear",description:"Clear conversation history"},{command:"compact",description:"Compact conversation history"},{command:"model",description:"Switch Claude model (opus/sonnet/haiku)"},{command:"cd",description:"Show or change session working directory"}];if(s.slashCommands?.length)for(let c of s.slashCommands){let a=c.replace(/^\//,"");i.push({command:a,description:`SDK command: ${a}`})}if(s.skills?.length)for(let c of s.skills)i.push({command:c,description:`Run ${c} skill`});await t.telegram.setMyCommands(i,{scope:{type:"chat",chat_id:e}}),r.add(e),o(`Registered ${i.length} commands for chat ${e}`)}catch(s){o(`Could not register dynamic commands for chat ${e}:`,s)}}function mg(t){return t.length<3?null:t.length>=4&&t[0]===137&&t[1]===80&&t[2]===78&&t[3]===71?"image/png":t[0]===71&&t[1]===73&&t[2]===70?"image/gif":t.length>=12&&t[0]===82&&t[1]===73&&t[2]===70&&t[3]===70&&t[8]===87&&t[9]===69&&t[10]===66&&t[11]===80?"image/webp":t[0]===255&&t[1]===216&&t[2]===255?"image/jpeg":null}async function gg(t,e){let n=t.headers.get("content-length");if(n!=null){let c=Number(n);if(Number.isFinite(c)&&c>e)return{status:"too-large",bytesRead:c}}let r=t.body;if(!r)return{status:"missing-body"};let o=r.getReader(),s=[],i=0;try{for(;;){let{done:c,value:a}=await o.read();if(c)break;if(i+=a.byteLength,i>e)return await o.cancel().catch(()=>{}),{status:"too-large",bytesRead:i};s.push(Buffer.from(a))}}finally{o.releaseLock()}return{status:"ok",bytes:Buffer.concat(s,i)}}var En=class t{static MAX_QUEUE_DEPTH=5;sessionManager;messageQueues=new Map;registeredCommandChats;log;bot;constructor(e,n,r,o){this.bot=e,this.sessionManager=n,this.registeredCommandChats=r,this.log=o}async handlePhoto(e){let n=e.chat?.id,r=e.message,o=r?.photo;if(!n||!o?.length){this.log(`Photo handling: missing chatId or photo array for chat ${n??"(unknown)"}`);return}this.log(`\u{1F4F7} Photo from chat ID: ${n}`);let s=o[o.length-1];if(!s){this.log(`Photo handling: empty photo array for chat ${n}`);return}let i=5*1024*1024;if(s.file_size!=null&&s.file_size>i){this.log(`Photo handling: oversized file (${s.file_size} bytes) rejected for chat ${n}`),await e.reply("\u274C Image is too large (max 5 MB). Please send a smaller photo.");return}let c=r?.caption;try{let a=await this.sessionManager.getSession(n);if($r(this.bot,n,a,this.registeredCommandChats,this.log).catch(g=>this.log("Failed to register chat commands:",g)),a.state!=="idle"&&(this.messageQueues.get(n)?.length??0)>=t.MAX_QUEUE_DEPTH){await e.reply("\u23F3 Queue full. Please wait for your messages to be processed.");return}let l=await e.telegram.getFileLink(s.file_id),d=l instanceof URL?l:new URL(String(l));if(d.protocol!=="https:"||d.hostname!=="api.telegram.org"||d.port!==""&&d.port!=="443"){this.log(`Photo handling: unexpected file URL (protocol=${d.protocol} hostname=${d.hostname}) rejected for chat ${n}`),await e.reply("\u274C Couldn't download the image. Please try resending.");return}let u=await globalThis.fetch(d.href,{signal:AbortSignal.timeout(15e3),redirect:"error"});if(!u.ok){this.log(`Photo handling: fetch failed with status ${u.status} for chat ${n}`),await e.reply("\u274C Couldn't download the image. Please try resending.");return}let p=await gg(u,i);if(p.status==="too-large"){this.log(`Photo handling: downloaded file (${p.bytesRead} bytes) exceeds limit for chat ${n}`),await e.reply("\u274C Image is too large (max 5 MB). Please send a smaller photo.");return}if(p.status==="missing-body"){this.log(`Photo handling: fetch response had no body for chat ${n}`),await e.reply("\u274C Couldn't download the image. Please try resending.");return}let y=p.bytes,f=y.toString("base64"),m=["image/jpeg","image/png","image/gif","image/webp"],b=u.headers.get("content-type")??"",w=(b.split(";")[0]?.trim()??"").toLowerCase(),k;if(m.includes(w))k=w;else{let g=mg(y);if(g!==null)this.log(`Photo: sniffed ${g} (Content-Type was "${b}") for chat ${n}`),k=g;else{this.log(`Photo: unrecognised image format for chat ${n} (Content-Type: "${b}")`),await e.reply("\u274C Unsupported image format. Please send a JPEG, PNG, GIF, or WebP.");return}}let h=[];if(c!=null&&h.push({type:"text",text:`[User caption]: ${[...c].slice(0,1024).join("")}`}),h.push({type:"image",source:{type:"base64",media_type:k,data:f}}),a.state!=="idle"){this.enqueuePhoto(n,e,h)&&await e.reply("Message queued.");return}await this.processOne(n,e,h)}catch(a){let d=(a instanceof Error?a.message:String(a)).replace(/\/bot[^/]+\//g,"/bot[REDACTED]/");this.log("Photo handling error:",d),wt(a)?await e.reply("\u23F3 Rate limit reached. Please wait a moment and try again."):kt(a)?await e.reply("\u274C Couldn't download the image. Please try resending."):await e.reply(Ft())}}async handle(e){let n=e.chat?.id,r=e.message.text;if(!(!n||!r)&&(this.log(`\u{1F4EC} Message from chat ID: ${n}`),!r.startsWith("/")))try{let o=await this.sessionManager.getSession(n);if($r(this.bot,n,o,this.registeredCommandChats,this.log).catch(s=>this.log("Failed to register chat commands:",s)),o.state!=="idle"){this.enqueueMessage(n,e,r)&&await e.reply("Message queued.");return}await this.processOne(n,e,r)}catch(o){this.log("Message handling error:",o),wt(o)?await e.reply("\u23F3 Rate limit reached. Please wait a moment and try again."):kt(o)?await e.reply("\u{1F310} Network error. Please check your connection and try again."):await e.reply(Ft())}}async processClearDirect(e,n){try{await this.sessionManager.resetSession(e),this.registeredCommandChats.delete(e),await n.reply(Lt())}catch(r){this.log("Clear error:",r),await n.reply(j(r))}}enqueueMessage(e,n,r){let o=this.messageQueues.get(e);return o||(o=[],this.messageQueues.set(e,o)),o.length>=t.MAX_QUEUE_DEPTH?(n.reply("\u23F3 Queue full. Please wait for your messages to be processed.").catch(()=>{}),!1):(o.push({type:"message",ctx:n,text:r}),!0)}enqueuePhoto(e,n,r){let o=this.messageQueues.get(e);return o||(o=[],this.messageQueues.set(e,o)),o.length>=t.MAX_QUEUE_DEPTH?(n.reply("\u23F3 Queue full. Please wait for your messages to be processed.").catch(()=>{}),!1):(o.push({type:"photo",ctx:n,content:r}),!0)}enqueueClear(e,n){let r=this.messageQueues.get(e);r||(r=[],this.messageQueues.set(e,r)),r.push({type:"clear",ctx:n})}async processOne(e,n,r){let o=!1;try{let s=await this.sessionManager.getSession(e);await n.sendChatAction("typing").catch(()=>{}),await Ua(n,s,r,this.log)}catch(s){if(this.log("Message handling error:",s),(s?.message??"").includes("session is busy")){(typeof r=="string"?this.enqueueMessage(e,n,r):this.enqueuePhoto(e,n,r))&&await n.reply("Message queued."),o=!0;return}wt(s)?await n.reply("\u23F3 Rate limit reached. Please wait a moment and try again."):kt(s)?await n.reply("\u{1F310} Network error. Please check your connection and try again."):await n.reply(Ft())}finally{o||this.drainQueue(e).catch(s=>this.log("Drain error:",s))}}async drainQueue(e){let n=this.messageQueues.get(e);if(!n?.length)return;let r=n.shift();r.type==="message"?await this.processOne(e,r.ctx,r.text):r.type==="photo"?await this.processOne(e,r.ctx,r.content):await this.processClearDirect(e,r.ctx)}};var _n=class{bot;sessionManager;options;running=!1;registeredCommandChats=new Set;messageHandler;constructor(e){this.options=e,this.bot=new hg(e.botToken),this.sessionManager=new Mt(e),this.messageHandler=new En(this.bot,this.sessionManager,this.registeredCommandChats,this.log.bind(this)),this.setupHandlers()}setupHandlers(){this.bot.use(Ys(this.options.allowedChatIds,this.log.bind(this))),this.bot.command("start",n=>Nn(n)),this.bot.command("help",n=>$n(n,this.sessionManager)),this.bot.command("clear",async n=>{let r=n.chat?.id;if(!r){await n.reply(j("Could not identify chat"));return}(await this.sessionManager.getSession(r)).state!=="idle"?(this.messageHandler.enqueueClear(r,n),await n.reply("Clear queued.")):await Fr(n,this.sessionManager,this.registeredCommandChats,this.log.bind(this))}),this.bot.command("compact",n=>Sa(n,this.sessionManager,this.log.bind(this))),this.bot.command("model",n=>Lr(n,this.sessionManager,this.log.bind(this))),this.bot.command(["cd","cwd"],n=>va(n,this.sessionManager,this.log.bind(this))),this.bot.on("text",n=>this.messageHandler.handle(n)),this.bot.on("photo",n=>this.messageHandler.handlePhoto(n));let e=new RegExp(`^${yg(vn)}`);this.bot.action(e,n=>La(n,{log:this.log.bind(this)})),this.bot.catch((n,r)=>{this.log("Bot error:",n),r.reply(j("An unexpected error occurred. Please try again.")).catch(o=>this.log("Failed to send error message:",o))})}async start(){if(this.running)throw new Error("Bot is already running");this.log("Loading sessions..."),await this.sessionManager.loadSessions(),this.log("Starting bot..."),await this.bot.launch(),this.log("Registering bot commands..."),await this.bot.telegram.setMyCommands([{command:"start",description:"Show welcome and command list"},{command:"help",description:"Show this command list"},{command:"clear",description:"Clear conversation history"},{command:"compact",description:"Compact conversation history"},{command:"model",description:"Switch Claude model (opus/sonnet/haiku)"}]),this.running=!0,this.log("Bot started successfully");let e=async n=>{this.log(`Received ${n}, shutting down...`),await this.stop(),process.exit(0)};process.once("SIGINT",()=>e("SIGINT")),process.once("SIGTERM",()=>e("SIGTERM"))}async stop(){if(this.running){this.log("Stopping bot..."),this.running=!1,this.log("Closing sessions..."),await this.sessionManager.closeAll(),this.log("Stopping bot polling...");try{this.bot.stop()}catch(e){this.log("Error stopping bot (may not have been started):",e)}this.log("Bot stopped")}}getStats(){return{running:this.running,activeSessions:this.sessionManager.getSessionCount(),totalChats:this.sessionManager.getChatCount()}}async handleStart(e){return Nn(e)}async handleHelp(e){return $n(e,this.sessionManager)}async handleClear(e){let n=e.chat?.id;if(!n){await e.reply(j("Could not identify chat"));return}if((await this.sessionManager.getSession(n)).state!=="idle")this.messageHandler.enqueueClear(n,e),await e.reply("Clear queued.");else return Fr(e,this.sessionManager,this.registeredCommandChats,this.log.bind(this))}async handleMessage(e){return this.messageHandler.handle(e)}async handlePhoto(e){return this.messageHandler.handlePhoto(e)}async handleModelSwitch(e){return Lr(e,this.sessionManager,this.log.bind(this))}log(...e){this.options.verbose&&console.log("[TelegramBot]",...e)}};function yg(t){return t.replace(/[.*+?^${}()|[\]\\]/g,"\\$&")}import Rx from"chalk";var bg="https://api.telegram.org";async function ja(t){try{let e=await fetch(`${bg}/bot${t}/getMe`);if(!e.ok)return null;let n=await e.json();return!n.ok||!n.result?.id||!n.result?.first_name?null:{id:n.result.id,...n.result.username!==void 0?{username:n.result.username}:{},firstName:n.result.first_name}}catch{return null}}function Ha(){return Vo()}var wg=["shadow-verify","shadow_verify","resolve","diagnose","appmap","qualify","mint"],kg=[/\bverdict(s)?\b/i,/\brecommend(ation)?s?\b/i,/\bshould\s+(delete|remove|rewrite|refactor|rename|reject|merge|revert|disable)\b/i,/\b(USELESS|KEEP|REJECT|APPROVE|SALVAGE|BLOCK|FAIL)\b/,/\b(redundant|duplicated|superseded|obsolete)\b/i,/\bvulnerab\w*\b/i,/\bunused\b/i,/\bbroken\b/i,/\bregress\w*\b/i,/\|\s*(status|verdict|decision|severity|risk|finding|priority|holds\??)\s*\|/i,/\bfound\s+\d+\s*(issue|problem|bug|error|finding|vulnerabilit)/i,/\b(critical|high|medium|low)\s+(severity|priority|risk)\b/i,/\bclaim(s)?\b[^\n]{0,80}\b(holds?|refuted|verified|partial|confirmed|disputed)\b/i,/\b(root\s*cause|incident)\b/i,/\brecommend\s+(removing|deleting|rewriting|refactoring|merging|reverting)\b/i,/\bI\s+(applied|committed|pushed|edited|wrote|fixed|patched|reset|restored|staged)\b/i,/\b(applied|committed|pushed|fixed|patched)\s+(the|these|those)\s+(change|commit|fix|patch|edit)/i],Sg=[/\bverifier_verdict\b/i,/"\s*claim\s*"\s*:/i,/\bre-derived\b[^.\n]{0,80}\bindependent/i,/\bindependently\s+(re-derived|re-verified|verified|checked)\b/i,/\bverifier\s+(agrees|disagrees|confirms|refutes)\b/i],vg=`shadow-verify nudge:
1647
1647
 
1648
1648
  The sub-agent that just finished returned output that reads like **decision-driving findings** (verdicts, recommendations, audit conclusions, or claim-style results that could drive file edits, deletions, commits, or external side-effects).
1649
1649
 
1650
1650
  Single-pass sub-agent reports are prone to confident hallucination \u2014 polished output that falls apart on re-derivation. Before acting on these conclusions, consider dispatching \`/shadow-verify\`. Independent verifiers will re-derive the 2\u20133 most load-bearing claims from scratch (without seeing the original reasoning) and flag any that don't hold up.
1651
1651
 
1652
- Skip when: the findings are purely exploratory, the sub-agent ran inside an already-verifying orchestrator, the user is about to dismiss the report, or the stakes are low (read-only Q&A).`;function Eg(t){if(!t)return!1;let e=t.toLowerCase();return wg.some(n=>e.includes(n))}function _g(t){return Sg.some(e=>e.test(t))}function Ag(t){let e=0;for(let n of kg)n.test(t)&&e++;return e}function Ba(t){if(t.event!=="SubagentStop")return{};let e=t.lastMessage??"";return e.length<600?{}:Eg(t.agentType)?{}:_g(e)?{}:Ag(e)<2?{}:{injectContext:vg}}var Tg=new Set(["Read","Glob","Grep","NotebookRead","LS","read_file","glob","grep","list_directory","memory_search"]),xg=new Set(["Write","Edit","NotebookEdit","MultiEdit","write_file","edit_file","memory_update","procedure_write","terminal_font_size"]),Rg=new Set(["Bash","BashOutput","KillBash","bash"]),Wa=new Set(["Agent","Task","agent"]),Ka=new Set(["Skill","skill"]),Ga=new Set(["Compose","compose"]),Dx=new Set([...Wa,...Ga,...Ka]),Pg=new Set(["WebFetch","WebSearch","send_telegram","web_scrape"]),Ig=new Set(["TaskCreate","TaskUpdate","TaskList","TaskGet","TaskOutput","TaskStop","EnterPlanMode","ExitPlanMode","ToolSearch"]),Cg=new Set(["create_schedule","list_schedules","get_schedule_history","cancel_schedule"]);function Te(t,e){if(t.has(e))return!0;let n=e.charAt(0).toUpperCase()+e.slice(1);return n!==e&&t.has(n)}function qa(t){return t.startsWith("mcp__")||t.startsWith("MCP__")?"mcp":Te(Tg,t)?"read":Te(xg,t)?"write":Te(Rg,t)?"shell":Te(Wa,t)?"subagent":Te(Ka,t)?"skill":Te(Ga,t)?"dag":Te(Pg,t)?"web":Te(Ig,t)?"planning":Cg.has(t)?"schedule":"other"}var Mg=["git commit","git push","git reset","rm ","mv ","mkdir","touch","chmod","chown","cp ","tee "," > "," >> ","npm install","pnpm install","pip install","apt ","apt-get ","brew install"," && "];function za(t){return function(n){if(n.event!=="PreToolUse")return{};if(t()!=="plan")return{};let{toolName:r}=n;if(qa(r)==="write")return{decision:"block",reason:`plan mode: ${r} is refused. Use /plan off to exit plan mode.`};if(r==="bash"){let o=typeof n.input=="object"&&n.input!==null?String(n.input.command??""):"";if(Mg.some(i=>o.includes(i)))return{decision:"block",reason:"plan mode: write-intent bash is refused. Use /plan off or rephrase as a read-only command."}}return{}}}import{mkdirSync as Fg,rmSync as Lg,writeFileSync as Ng}from"fs";import{join as Ur}from"path";function Og(t){let e=new Set;for(let c of t.nodes){if(e.has(c.id))throw new Error(`Duplicate node ID: ${c.id}`);e.add(c.id)}let n=new Set;for(let c of t.edges){if(!e.has(c.from))throw new Error(`Edge references non-existent node: ${c.from}`);if(!e.has(c.to))throw new Error(`Edge references non-existent node: ${c.to}`);let a=`${c.from}->${c.to}`;if(n.has(a))throw new Error(`Duplicate edge: ${c.from} -> ${c.to}`);n.add(a)}let r=Ja(t),o=new Map(r.inDegree),s=[];for(let[c,a]of o)a===0&&s.push(c);let i=0;for(;s.length>0;){let c=s.shift();i+=1;for(let a of r.downstream.get(c)??[]){let l=o.get(a)-1;o.set(a,l),l===0&&s.push(a)}}if(i!==e.size)throw new Error("Cycle detected in DAG")}function Ja(t){let e=new Map,n=new Map,r=new Map;for(let o of t.nodes)e.set(o.id,new Set),n.set(o.id,new Set),r.set(o.id,0);for(let o of t.edges)e.get(o.from).add(o.to),n.get(o.to).add(o.from),r.set(o.to,r.get(o.to)+1);return{downstream:e,upstream:n,inDegree:r}}function Dg(t,e,n){let r=[t];for(;r.length>0;){let o=r.shift();for(let s of e.get(o)??[])n.has(s)||(n.add(s),r.push(s))}}async function Va(t,e,n={}){if(t.nodes.length===0)return{outputs:{},failed:[],skipped:[]};Og(t);let{failFast:r=!0,nodeTimeoutMs:o}=n,s=o!==void 0&&Number.isFinite(o)&&o>0,i=Ja(t),c=new Map(t.nodes.map(m=>[m.id,m])),a={},l=[],d=new Set,u=new Set,p=new Map(i.inDegree),y=new AbortController,f=()=>{y.signal.aborted||y.abort(e.reason)};e.aborted?y.abort(e.reason):e.addEventListener("abort",f,{once:!0});try{for(;!y.signal.aborted;){let m=[];for(let[w,k]of p)k===0&&!u.has(w)&&!d.has(w)&&m.push(w);if(m.length===0)break;let b=await Promise.allSettled(m.map(async w=>{let k=c.get(w),h=new AbortController,g=()=>{h.signal.aborted||h.abort(y.signal.reason)};y.signal.aborted?h.abort(y.signal.reason):y.signal.addEventListener("abort",g,{once:!0});let S;s&&!h.signal.aborted&&(S=setTimeout(()=>{h.signal.aborted||h.abort(new de(`DAG node "${w}" exceeded nodeTimeoutMs of ${o}ms`,o))},o));let E={};for(let _ of i.upstream.get(w)??[])E[_]=a[_];try{let _=await k.run(E,h.signal);return{id:w,result:_}}finally{S!==void 0&&clearTimeout(S),y.signal.removeEventListener("abort",g)}}));for(let w=0;w<b.length;w++){let k=b[w];if(k.status==="fulfilled"){let{id:h,result:g}=k.value;a[h]=g,u.add(h),p.delete(h);for(let S of i.downstream.get(h)??[])p.set(S,p.get(S)-1)}else{let h=k.reason instanceof Error?k.reason:new Error(String(k.reason)),g=m[w];l.push({id:g,error:h}),u.add(g),p.delete(g),Dg(g,i.downstream,d),r&&y.abort("fail-fast")}}}}finally{e.removeEventListener("abort",f)}return{outputs:a,failed:l,skipped:Array.from(d)}}async function Ya(t){let{manager:e,parentSession:n,nodes:r,edges:o,failFast:s,nodeTimeoutMs:i}=t,c=n.abortSignal??new AbortController().signal,a=r.map(l=>({id:l.id,async run(d,u){let p=await e.forkSubagent({parent:{sessionId:n.sessionId},config:{model:l.model??"sonnet",systemPrompt:l.systemPrompt,...l.canUseTool!==void 0?{canUseTool:l.canUseTool}:{},...l.cwd!==void 0?{cwd:l.cwd}:{},...l.readRoots!==void 0?{readRoots:l.readRoots}:{},...l.writeRoots!==void 0?{writeRoots:l.writeRoots}:{}},idPrefix:l.idPrefix??`dag-${l.id}`,...l.outputSchema!==void 0?{outputSchema:l.outputSchema}:{},...l.agentType!==void 0?{agentType:l.agentType}:{},...l.parentId!==void 0?{parentId:l.parentId}:{}}),y=()=>{p.cancel().catch(()=>{})};u.aborted?p.cancel().catch(()=>{}):u.addEventListener("abort",y,{once:!0});try{if(u.aborted)throw new DOMException("Aborted","AbortError");let f=l.promptBuilder(d),m=await p.runToResult(f);if(m.status!=="succeeded"){let b,w=u.reason;throw w instanceof de?b=new Error(`Subagent ${l.id} aborted: ${w.message}`,m.error?{cause:m.error}:{}):b=m.error??new Error(`Subagent ${l.id} ${m.status}`),ss(b,{partialOutput:m.partialOutput,subagentId:m.id})}return m.output??m.message?.content}finally{u.removeEventListener("abort",y),await p.teardown().catch(()=>{})}}}));return Va({nodes:a,edges:o},c,{failFast:s,nodeTimeoutMs:i})}var Xa=1e3,An=36e5,Qa=1,Za=1e3;function $g(t){if(typeof t!="object"||t===null)throw new Error("Compose tool input must be an object");let e=t,n=e.nodes;if(!Array.isArray(n)||n.length===0)throw new Error('Compose tool requires a non-empty "nodes" array');let r=20;if(n.length>r)throw new Error(`Compose tool supports at most ${r} nodes (got ${n.length}). Split into multiple compose calls for larger workloads.`);let o=[],s=new Set;for(let u of n){if(typeof u!="object"||u===null)throw new Error("Each node must be an object");let p=u,y=p.id;if(typeof y!="string"||y.trim().length===0)throw new Error('Each node must have a non-empty "id" string');if(!/^[A-Za-z0-9_-]+$/.test(y)){let b=y.replace(/[\x00-\x1f\x7f]/g,"?").slice(0,32);throw new Error(`Node id "${b}" must match /^[A-Za-z0-9_-]+$/ (alphanumeric, underscore, hyphen)`)}if(s.has(y))throw new Error(`Duplicate node ID: ${y}`);s.add(y);let f=p.prompt;if(typeof f!="string"||f.trim().length===0)throw new Error(`Node "${y}" must have a non-empty "prompt" string`);let m;if(p.model!==void 0){if(typeof p.model!="string")throw new Error(`Node "${y}" model must be a string`);m=p.model}o.push({id:y,prompt:f,model:m})}let i;if(e.edges!==void 0){if(!Array.isArray(e.edges))throw new Error('"edges" must be an array');i=[];for(let u of e.edges){if(typeof u!="object"||u===null)throw new Error("Each edge must be an object");let p=u;if(typeof p.from!="string"||typeof p.to!="string")throw new Error('Each edge must have "from" and "to" strings');if(!s.has(p.from))throw new Error(`Edge references non-existent node: ${p.from}`);if(!s.has(p.to))throw new Error(`Edge references non-existent node: ${p.to}`);i.push({from:p.from,to:p.to})}}let c;if(e.fail_fast!==void 0){if(typeof e.fail_fast!="boolean")throw new Error('"fail_fast" must be a boolean');c=e.fail_fast}let a=[],l;if(e.node_timeout_ms!==void 0){let u=e.node_timeout_ms;if(typeof u!="number"||!Number.isFinite(u)||u<=0)throw new Error('"node_timeout_ms" must be a positive finite number (milliseconds)');if(u<Xa)throw new Error(`"node_timeout_ms" must be at least ${Xa}ms (got ${u}). Sub-second timeouts are almost always a unit mistake.`);l=Math.min(An,u),u>An&&a.push(`node_timeout_ms clamped: requested ${u}ms exceeds the maximum ${An}ms; using ${An}ms.`)}let d;if(e.max_tool_calls_per_node!==void 0){let u=e.max_tool_calls_per_node;if(typeof u!="number"||!Number.isFinite(u)||u<=0)throw new Error('"max_tool_calls_per_node" must be a positive finite number');if(!Number.isInteger(u))throw new Error(`"max_tool_calls_per_node" must be an integer (got ${u}). Tool calls are discrete events; fractional budgets are not meaningful.`);if(u<Qa)throw new Error(`"max_tool_calls_per_node" must be at least ${Qa}`);if(u>Za)throw new Error(`"max_tool_calls_per_node" must be at most ${Za} (got ${u}). A larger budget no longer constrains useful work.`);d=u}return{parsed:{nodes:o,edges:i,fail_fast:c,node_timeout_ms:l,max_tool_calls_per_node:d},warnings:a}}var St=8e3,ec=500,tc=4e3;function Ug(t){if(t==null)return;let e=typeof t=="string"?t:JSON.stringify(t);if(e.length!==0)return e.length>tc?e.slice(0,tc)+`
1652
+ Skip when: the findings are purely exploratory, the sub-agent ran inside an already-verifying orchestrator, the user is about to dismiss the report, or the stakes are low (read-only Q&A).`;function Eg(t){if(!t)return!1;let e=t.toLowerCase();return wg.some(n=>e.includes(n))}function _g(t){return Sg.some(e=>e.test(t))}function Ag(t){let e=0;for(let n of kg)n.test(t)&&e++;return e}function Ba(t){if(t.event!=="SubagentStop")return{};let e=t.lastMessage??"";return e.length<600?{}:Eg(t.agentType)?{}:_g(e)?{}:Ag(e)<2?{}:{injectContext:vg}}var Tg=new Set(["Read","Glob","Grep","NotebookRead","LS","read_file","glob","grep","list_directory","memory_search"]),xg=new Set(["Write","Edit","NotebookEdit","MultiEdit","write_file","edit_file","memory_update","procedure_write","terminal_font_size"]),Rg=new Set(["Bash","BashOutput","KillBash","bash"]),Wa=new Set(["Agent","Task","agent"]),Ka=new Set(["Skill","skill"]),Ga=new Set(["Compose","compose"]),Lx=new Set([...Wa,...Ga,...Ka]),Pg=new Set(["WebFetch","WebSearch","send_telegram","web_scrape"]),Ig=new Set(["TaskCreate","TaskUpdate","TaskList","TaskGet","TaskOutput","TaskStop","EnterPlanMode","ExitPlanMode","ToolSearch"]),Cg=new Set(["create_schedule","list_schedules","get_schedule_history","cancel_schedule"]);function Te(t,e){if(t.has(e))return!0;let n=e.charAt(0).toUpperCase()+e.slice(1);return n!==e&&t.has(n)}function qa(t){return t.startsWith("mcp__")||t.startsWith("MCP__")?"mcp":Te(Tg,t)?"read":Te(xg,t)?"write":Te(Rg,t)?"shell":Te(Wa,t)?"subagent":Te(Ka,t)?"skill":Te(Ga,t)?"dag":Te(Pg,t)?"web":Te(Ig,t)?"planning":Cg.has(t)?"schedule":"other"}var Mg=["git commit","git push","git reset","rm ","mv ","mkdir","touch","chmod","chown","cp ","tee "," > "," >> ","npm install","pnpm install","pip install","apt ","apt-get ","brew install"," && "];function za(t){return function(n){if(n.event!=="PreToolUse")return{};if(t()!=="plan")return{};let{toolName:r}=n;if(qa(r)==="write")return{decision:"block",reason:`plan mode: ${r} is refused. Use /plan off to exit plan mode.`};if(r==="bash"){let o=typeof n.input=="object"&&n.input!==null?String(n.input.command??""):"";if(Mg.some(i=>o.includes(i)))return{decision:"block",reason:"plan mode: write-intent bash is refused. Use /plan off or rephrase as a read-only command."}}return{}}}import{mkdirSync as Fg,rmSync as Lg,writeFileSync as Ng}from"fs";import{join as Ur}from"path";function Og(t){let e=new Set;for(let c of t.nodes){if(e.has(c.id))throw new Error(`Duplicate node ID: ${c.id}`);e.add(c.id)}let n=new Set;for(let c of t.edges){if(!e.has(c.from))throw new Error(`Edge references non-existent node: ${c.from}`);if(!e.has(c.to))throw new Error(`Edge references non-existent node: ${c.to}`);let a=`${c.from}->${c.to}`;if(n.has(a))throw new Error(`Duplicate edge: ${c.from} -> ${c.to}`);n.add(a)}let r=Ja(t),o=new Map(r.inDegree),s=[];for(let[c,a]of o)a===0&&s.push(c);let i=0;for(;s.length>0;){let c=s.shift();i+=1;for(let a of r.downstream.get(c)??[]){let l=o.get(a)-1;o.set(a,l),l===0&&s.push(a)}}if(i!==e.size)throw new Error("Cycle detected in DAG")}function Ja(t){let e=new Map,n=new Map,r=new Map;for(let o of t.nodes)e.set(o.id,new Set),n.set(o.id,new Set),r.set(o.id,0);for(let o of t.edges)e.get(o.from).add(o.to),n.get(o.to).add(o.from),r.set(o.to,r.get(o.to)+1);return{downstream:e,upstream:n,inDegree:r}}function Dg(t,e,n){let r=[t];for(;r.length>0;){let o=r.shift();for(let s of e.get(o)??[])n.has(s)||(n.add(s),r.push(s))}}async function Va(t,e,n={}){if(t.nodes.length===0)return{outputs:{},failed:[],skipped:[]};Og(t);let{failFast:r=!0,nodeTimeoutMs:o}=n,s=o!==void 0&&Number.isFinite(o)&&o>0,i=Ja(t),c=new Map(t.nodes.map(m=>[m.id,m])),a={},l=[],d=new Set,u=new Set,p=new Map(i.inDegree),y=new AbortController,f=()=>{y.signal.aborted||y.abort(e.reason)};e.aborted?y.abort(e.reason):e.addEventListener("abort",f,{once:!0});try{for(;!y.signal.aborted;){let m=[];for(let[w,k]of p)k===0&&!u.has(w)&&!d.has(w)&&m.push(w);if(m.length===0)break;let b=await Promise.allSettled(m.map(async w=>{let k=c.get(w),h=new AbortController,g=()=>{h.signal.aborted||h.abort(y.signal.reason)};y.signal.aborted?h.abort(y.signal.reason):y.signal.addEventListener("abort",g,{once:!0});let S;s&&!h.signal.aborted&&(S=setTimeout(()=>{h.signal.aborted||h.abort(new de(`DAG node "${w}" exceeded nodeTimeoutMs of ${o}ms`,o))},o));let E={};for(let _ of i.upstream.get(w)??[])E[_]=a[_];try{let _=await k.run(E,h.signal);return{id:w,result:_}}finally{S!==void 0&&clearTimeout(S),y.signal.removeEventListener("abort",g)}}));for(let w=0;w<b.length;w++){let k=b[w];if(k.status==="fulfilled"){let{id:h,result:g}=k.value;a[h]=g,u.add(h),p.delete(h);for(let S of i.downstream.get(h)??[])p.set(S,p.get(S)-1)}else{let h=k.reason instanceof Error?k.reason:new Error(String(k.reason)),g=m[w];l.push({id:g,error:h}),u.add(g),p.delete(g),Dg(g,i.downstream,d),r&&y.abort("fail-fast")}}}}finally{e.removeEventListener("abort",f)}return{outputs:a,failed:l,skipped:Array.from(d)}}async function Ya(t){let{manager:e,parentSession:n,nodes:r,edges:o,failFast:s,nodeTimeoutMs:i}=t,c=n.abortSignal??new AbortController().signal,a=r.map(l=>({id:l.id,async run(d,u){let p=await e.forkSubagent({parent:{sessionId:n.sessionId},config:{model:l.model??"sonnet",systemPrompt:l.systemPrompt,...l.canUseTool!==void 0?{canUseTool:l.canUseTool}:{},...l.cwd!==void 0?{cwd:l.cwd}:{},...l.readRoots!==void 0?{readRoots:l.readRoots}:{},...l.writeRoots!==void 0?{writeRoots:l.writeRoots}:{}},idPrefix:l.idPrefix??`dag-${l.id}`,...l.outputSchema!==void 0?{outputSchema:l.outputSchema}:{},...l.agentType!==void 0?{agentType:l.agentType}:{},...l.parentId!==void 0?{parentId:l.parentId}:{}}),y=()=>{p.cancel().catch(()=>{})};u.aborted?p.cancel().catch(()=>{}):u.addEventListener("abort",y,{once:!0});try{if(u.aborted)throw new DOMException("Aborted","AbortError");let f=l.promptBuilder(d),m=await p.runToResult(f);if(m.status!=="succeeded"){let b,w=u.reason;throw w instanceof de?b=new Error(`Subagent ${l.id} aborted: ${w.message}`,m.error?{cause:m.error}:{}):b=m.error??new Error(`Subagent ${l.id} ${m.status}`),ss(b,{partialOutput:m.partialOutput,subagentId:m.id})}return m.output??m.message?.content}finally{u.removeEventListener("abort",y),await p.teardown().catch(()=>{})}}}));return Va({nodes:a,edges:o},c,{failFast:s,nodeTimeoutMs:i})}var Xa=1e3,An=36e5,Qa=1,Za=1e3;function $g(t){if(typeof t!="object"||t===null)throw new Error("Compose tool input must be an object");let e=t,n=e.nodes;if(!Array.isArray(n)||n.length===0)throw new Error('Compose tool requires a non-empty "nodes" array');let r=20;if(n.length>r)throw new Error(`Compose tool supports at most ${r} nodes (got ${n.length}). Split into multiple compose calls for larger workloads.`);let o=[],s=new Set;for(let u of n){if(typeof u!="object"||u===null)throw new Error("Each node must be an object");let p=u,y=p.id;if(typeof y!="string"||y.trim().length===0)throw new Error('Each node must have a non-empty "id" string');if(!/^[A-Za-z0-9_-]+$/.test(y)){let b=y.replace(/[\x00-\x1f\x7f]/g,"?").slice(0,32);throw new Error(`Node id "${b}" must match /^[A-Za-z0-9_-]+$/ (alphanumeric, underscore, hyphen)`)}if(s.has(y))throw new Error(`Duplicate node ID: ${y}`);s.add(y);let f=p.prompt;if(typeof f!="string"||f.trim().length===0)throw new Error(`Node "${y}" must have a non-empty "prompt" string`);let m;if(p.model!==void 0){if(typeof p.model!="string")throw new Error(`Node "${y}" model must be a string`);m=p.model}o.push({id:y,prompt:f,model:m})}let i;if(e.edges!==void 0){if(!Array.isArray(e.edges))throw new Error('"edges" must be an array');i=[];for(let u of e.edges){if(typeof u!="object"||u===null)throw new Error("Each edge must be an object");let p=u;if(typeof p.from!="string"||typeof p.to!="string")throw new Error('Each edge must have "from" and "to" strings');if(!s.has(p.from))throw new Error(`Edge references non-existent node: ${p.from}`);if(!s.has(p.to))throw new Error(`Edge references non-existent node: ${p.to}`);i.push({from:p.from,to:p.to})}}let c;if(e.fail_fast!==void 0){if(typeof e.fail_fast!="boolean")throw new Error('"fail_fast" must be a boolean');c=e.fail_fast}let a=[],l;if(e.node_timeout_ms!==void 0){let u=e.node_timeout_ms;if(typeof u!="number"||!Number.isFinite(u)||u<=0)throw new Error('"node_timeout_ms" must be a positive finite number (milliseconds)');if(u<Xa)throw new Error(`"node_timeout_ms" must be at least ${Xa}ms (got ${u}). Sub-second timeouts are almost always a unit mistake.`);l=Math.min(An,u),u>An&&a.push(`node_timeout_ms clamped: requested ${u}ms exceeds the maximum ${An}ms; using ${An}ms.`)}let d;if(e.max_tool_calls_per_node!==void 0){let u=e.max_tool_calls_per_node;if(typeof u!="number"||!Number.isFinite(u)||u<=0)throw new Error('"max_tool_calls_per_node" must be a positive finite number');if(!Number.isInteger(u))throw new Error(`"max_tool_calls_per_node" must be an integer (got ${u}). Tool calls are discrete events; fractional budgets are not meaningful.`);if(u<Qa)throw new Error(`"max_tool_calls_per_node" must be at least ${Qa}`);if(u>Za)throw new Error(`"max_tool_calls_per_node" must be at most ${Za} (got ${u}). A larger budget no longer constrains useful work.`);d=u}return{parsed:{nodes:o,edges:i,fail_fast:c,node_timeout_ms:l,max_tool_calls_per_node:d},warnings:a}}var St=8e3,ec=500,tc=4e3;function Ug(t){if(t==null)return;let e=typeof t=="string"?t:JSON.stringify(t);if(e.length!==0)return e.length>tc?e.slice(0,tc)+`
1653
1653
  \u2026 (truncated)`:e}function jg(t,e,n,r){try{let o=Ur(et(),t,"compose",e);Fg(o,{recursive:!0});let s=Ur(o,`${n}.txt`);return Ng(s,r,"utf8"),s}catch{return}}function Hg(t,e){let n=[],r=[];for(let[o,s]of Object.entries(t.outputs)){let i=typeof s=="string"?s:s!=null?JSON.stringify(s):"(no output)",c;if(i.length>St){let a=jg(e.sessionId,e.callId,o,i);r.push({nodeId:o,emittedChars:St,totalChars:i.length,...a!==void 0?{spillPath:a}:{}});let l=a!==void 0?`
1654
1654
  \u2026 (truncated at ${St} / ${i.length} chars \u2014 full output at ${a})`:`
1655
1655
  \u2026 (truncated at ${St} / ${i.length} chars)`;c=i.slice(0,St)+l}else c=i;n.push(`## ${o}
@@ -1676,7 +1676,7 @@ ${h.map(_=>`> - ${_}`).join(`
1676
1676
 
1677
1677
  `:"")+b,E=f.failed.length>0;return{content:S,isError:E}}catch(u){let p=u instanceof Error?u.message:String(u);return V({event:"compose.failed",parent_session_id:this.ctx.parentSession.sessionId,error_message:p.slice(0,240),duration_ms:Date.now()-d}).catch(()=>{}),{content:`Compose execution error: ${p}`,isError:!0}}finally{await a.teardownAll()}}};function jr(t,e,n,r){let o=Ha();o.register("SubagentStop",Ba);let s=n??new X;return r!==void 0&&o.register("PreToolUse",za(r)),o.register("SessionEnd",Ln(s,e)),o.register("SessionEnd",i=>i.event!=="SessionEnd"?{}:(i.sessionId&&nc(i.sessionId),{})),t&&o.register("SubagentStop",i=>i.event!=="SubagentStop"?{}:i.status==="idle"||i.status==="running"?{}:(t({subagentId:i.subagentId,status:i.status,durationMs:i.durationMs,agentType:i.agentType}),{})),{registry:o,memoryStore:s}}var Wg="[skill-routing: active]\n\nRoute recurring work through registered skills instead of rolling ad-hoc solutions:\n\n- Bugs, failing tests, or regressions \u2192 `/diagnose`\n- High-stakes sub-agent output that will drive edits or commits \u2192 `/shadow-verify` before acting\n- Refactor needing parallel waves \u2192 `/parallelize`\n- Parallel or dependent multi-task work \u2192 `compose` tool (DAG of subagent nodes)\n- Greenfield feature where a written spec would genuinely help (novel scope, multi-day work, or external stakeholders involved) \u2192 `/mint`\n\nDo NOT reach for `/mint` for: bug fixes (use `/diagnose`), refactors with known shape, single-feature edits, work already spec'd in chat, or anything where the spec/approve pause would feel like ceremony. Implement directly in those cases.\n\nCommon composed sequences \u2014 reach for these when the task shape matches:\n\n- Bug with failing test and non-trivial fix \u2192 `/diagnose` \u2192 `/shadow-verify` on the proposed fix\n- Refactor needing parallel waves \u2192 plan \u2192 `/parallelize` \u2192 build waves\n- Diagnose + fix in parallel \u2192 `compose` with two independent nodes\n- Research \u2192 implement \u2192 verify pipeline \u2192 `compose` with edges: research\u2192implement\u2192verify\n- Multiple independent investigations \u2192 `compose` with N nodes, no edges\n\nReach for context-isolated investigators when the task is exploratory:\n\n- Map an unfamiliar module before editing \u2192 `/gather` or `/research`\n- Re-derive a load-bearing claim independently \u2192 `/shadow-verify`\n- Audit a diff before merge \u2192 `/review`\n- Survey git + infra + memory before non-trivial work \u2192 `/ground-state`\n- Generate alternatives before committing to a plan \u2192 `/devils-advocate`\n\nOr dispatch a raw `agent` call when no skill matches but the work is parallelizable, verification-heavy, or would otherwise consume substantial inline context.\n\nSkip orchestration for: single-line edits, trivial Q&A, and direct tool calls the user explicitly requested. The goal is leverage, not ceremony. If a skill would add overhead without adding value, don't invoke it.";function Hr(t,e){return!t||!e?t:`${t}
1678
1678
 
1679
- ${Wg}`}var gR=300*1e3;var xn=class extends Error{constructor(e,n){super(`Background job cap reached (${e}/${n} running). Wait for existing jobs to finish or cancel them before spawning more.`),this.name="BackgroundJobCapError"}};var qg=new Set;function rc(t){return qg.has(t)}var zg=new Set,Jg=new Set;function oc(t){for(let e of zg)e(t)}function sc(t){for(let e of Jg)e(t)}var Vg=240;function Yg(t,e=Vg){return t.length<=e?t:t.slice(0,e)+"\u2026"}function Xg(t){if(typeof t!="object"||t===null)return;let e=t.name;if(typeof e!="string")return;let n=e.trim();return n.length>0?n:void 0}function Qg(t){if(typeof t!="object"||t===null)throw new Error("Skill tool input must be an object");let e=t,n=e.name;if(typeof n!="string"||n.trim().length===0)throw new Error('Skill tool input must have a non-empty "name" field');let r,o=e.arguments;if(o!==void 0){if(typeof o!="string")throw new Error('Skill tool "arguments" must be a string');r=o}return{name:n.trim(),arguments:r}}var Ve=class{constructor(e){this.ctx=e}ctx;pluginBodies=null;async execute(e){if(e.signal.aborted)return{content:"Skill tool call aborted",isError:!0};let n=this.ctx.depth??0,r=this.ctx.maxDepth??Ye;if(n>=r){let a=Xg(e.input);return V({event:"delegation.skipped",parent_session_id:this.ctx.parentSession.sessionId,reason:"max_depth",depth:n,requested_name:a}).catch(()=>{}),{content:`Skill tool not available at nesting depth ${n} (max ${r})`,isError:!0}}let o;try{o=Qg(e.input)}catch(a){return{content:`Skill tool input validation failed: ${a instanceof Error?a.message:String(a)}`,isError:!0}}try{let a=pe(o.name);return await this.executeRegistrySkill(a,o.arguments,e)}catch{}let s=this.getPluginSkillBody(o.name);if(s)return await this.executePluginSkill(o.name,s.body,s.pluginPath,o.arguments,e);let c=Le(this.ctx.pluginConfigs).map(a=>a.name).join(", ");return{content:`Skill "${o.name}" not found. Available skills: ${c||"(none)"}`,isError:!0}}async executeRegistrySkill(e,n,r){if(r.signal.aborted)return{content:"Skill call aborted",isError:!0};if(e.context==="fork")return this.executeForkedRegistrySkill(e,n,r);let o=rc(e.name);o&&sc(e.name);let s=this.ctx.depth??0;V({event:"skill.dispatched",requested_name:e.name,parent_session_id:this.ctx.parentSession.sessionId,depth:s,...e.model!==void 0?{model:e.model}:{}}).catch(()=>{});let i=Date.now(),c,a;try{a=await e.handler(n&&n.length>0?n:void 0,this.ctx.parentSession,{apiKey:this.ctx.apiKey,defaultModel:this.ctx.defaultModel,defaultSubagentModel:this.ctx.defaultSubagentModel,callId:r.id,dispatchSkill:this.createDispatchSkillCallback(r)})}catch(d){c=d}finally{let d=Date.now()-i;o&&oc({skillName:e.name,durationMs:d,...c!==void 0?{isError:!0}:{}});let u=c!==void 0?c instanceof Error?c.message:String(c):void 0,p=c===void 0?typeof a=="string"?a.length:a!=null?JSON.stringify(a).length:0:void 0;V({event:"skill.completed",requested_name:e.name,parent_session_id:this.ctx.parentSession.sessionId,status:c!==void 0?"failed":"succeeded",duration_ms:d,depth:s,...p!==void 0?{content_chars:p}:{},...u!==void 0?{error_message:Yg(u)}:{},...e.model!==void 0?{model:e.model}:{}}).catch(()=>{})}return c!==void 0?{content:`Skill execution error: ${c instanceof Error?c.message:String(c)}`,isError:!0}:{content:typeof a=="string"?a:a!=null?JSON.stringify(a):"Skill completed successfully."}}buildForkedChildConfig(e,n){let r=this.ctx.depth??0,o=this.ctx.maxDepth??Ye,s={...e};if(!this.ctx.childProviderFactory||r>=o)return{childConfig:s,childManager:void 0};let i=new I({parentAbortSignal:n,...this.ctx.traceWriter!==void 0?{traceWriter:this.ctx.traceWriter}:{}}),c=new Xe({subagentManager:i,parentSession:vt(n),defaultConfig:{model:s.model,apiKey:this.ctx.apiKey,...this.ctx.baseUrl!==void 0?{baseUrl:this.ctx.baseUrl}:{}},defaultSubagentModel:this.ctx.defaultSubagentModel,childProviderFactory:this.ctx.childProviderFactory,childSkillExecutorFactory:this.ctx.childSkillExecutorFactory,depth:r+1,maxDepth:o}),a=this.ctx.childSkillExecutorFactory?this.ctx.childSkillExecutorFactory(r+1,o,n):void 0;return s.provider=this.ctx.childProviderFactory({childExecutor:c,...a!==void 0?{childSkillExecutor:a}:{},...s.model!==void 0?{model:s.model}:{}}),{childConfig:s,childManager:i}}async executeForkedRegistrySkill(e,n,r){if(r.signal.aborted)return{content:"Skill call aborted",isError:!0};let o;try{if(o=$(e.name)["system.md"],!o)return{content:`Skill "${e.name}" has context: "fork" but no prompts/system.md found`,isError:!0}}catch(a){return{content:`Failed to load skill prompts: ${a instanceof Error?a.message:String(a)}`,isError:!0}}let s=new I({parentAbortSignal:r.signal,apiKey:this.ctx.apiKey,...this.ctx.baseUrl!==void 0?{baseUrl:this.ctx.baseUrl}:{},...this.ctx.traceWriter!==void 0?{traceWriter:this.ctx.traceWriter}:{},progressSink:re()}),{childConfig:i,childManager:c}=this.buildForkedChildConfig({model:e.model??this.ctx.defaultSubagentModel??this.ctx.defaultModel??"sonnet",systemPrompt:o,...this.ctx.traceWriter!==void 0?{traceWriter:this.ctx.traceWriter}:{}},r.signal);try{let a=await s.forkSubagent({parent:this.ctx.parentSession,config:i,idPrefix:`skill-fork-${e.name}`,parentId:r.id,agentType:e.name}),l=n&&n.length>0?n:"Run the skill.",d=await a.runToResult(l);return d.status==="succeeded"&&d.message?{content:d.message.content}:d.status==="cancelled"&&typeof d.partialOutput=="string"&&d.partialOutput.length>0?{content:`[skill cancelled mid-flight \u2014 partial output preserved below]
1679
+ ${Wg}`}var yR=300*1e3;var xn=class extends Error{constructor(e,n){super(`Background job cap reached (${e}/${n} running). Wait for existing jobs to finish or cancel them before spawning more.`),this.name="BackgroundJobCapError"}};var qg=new Set;function rc(t){return qg.has(t)}var zg=new Set,Jg=new Set;function oc(t){for(let e of zg)e(t)}function sc(t){for(let e of Jg)e(t)}var Vg=240;function Yg(t,e=Vg){return t.length<=e?t:t.slice(0,e)+"\u2026"}function Xg(t){if(typeof t!="object"||t===null)return;let e=t.name;if(typeof e!="string")return;let n=e.trim();return n.length>0?n:void 0}function Qg(t){if(typeof t!="object"||t===null)throw new Error("Skill tool input must be an object");let e=t,n=e.name;if(typeof n!="string"||n.trim().length===0)throw new Error('Skill tool input must have a non-empty "name" field');let r,o=e.arguments;if(o!==void 0){if(typeof o!="string")throw new Error('Skill tool "arguments" must be a string');r=o}return{name:n.trim(),arguments:r}}var Ve=class{constructor(e){this.ctx=e}ctx;pluginBodies=null;async execute(e){if(e.signal.aborted)return{content:"Skill tool call aborted",isError:!0};let n=this.ctx.depth??0,r=this.ctx.maxDepth??Ye;if(n>=r){let a=Xg(e.input);return V({event:"delegation.skipped",parent_session_id:this.ctx.parentSession.sessionId,reason:"max_depth",depth:n,requested_name:a}).catch(()=>{}),{content:`Skill tool not available at nesting depth ${n} (max ${r})`,isError:!0}}let o;try{o=Qg(e.input)}catch(a){return{content:`Skill tool input validation failed: ${a instanceof Error?a.message:String(a)}`,isError:!0}}try{let a=pe(o.name);return await this.executeRegistrySkill(a,o.arguments,e)}catch{}let s=this.getPluginSkillBody(o.name);if(s)return await this.executePluginSkill(o.name,s.body,s.pluginPath,o.arguments,e);let c=Le(this.ctx.pluginConfigs).map(a=>a.name).join(", ");return{content:`Skill "${o.name}" not found. Available skills: ${c||"(none)"}`,isError:!0}}async executeRegistrySkill(e,n,r){if(r.signal.aborted)return{content:"Skill call aborted",isError:!0};if(e.context==="fork")return this.executeForkedRegistrySkill(e,n,r);let o=rc(e.name);o&&sc(e.name);let s=this.ctx.depth??0;V({event:"skill.dispatched",requested_name:e.name,parent_session_id:this.ctx.parentSession.sessionId,depth:s,...e.model!==void 0?{model:e.model}:{}}).catch(()=>{});let i=Date.now(),c,a;try{a=await e.handler(n&&n.length>0?n:void 0,this.ctx.parentSession,{apiKey:this.ctx.apiKey,defaultModel:this.ctx.defaultModel,defaultSubagentModel:this.ctx.defaultSubagentModel,callId:r.id,dispatchSkill:this.createDispatchSkillCallback(r)})}catch(d){c=d}finally{let d=Date.now()-i;o&&oc({skillName:e.name,durationMs:d,...c!==void 0?{isError:!0}:{}});let u=c!==void 0?c instanceof Error?c.message:String(c):void 0,p=c===void 0?typeof a=="string"?a.length:a!=null?JSON.stringify(a).length:0:void 0;V({event:"skill.completed",requested_name:e.name,parent_session_id:this.ctx.parentSession.sessionId,status:c!==void 0?"failed":"succeeded",duration_ms:d,depth:s,...p!==void 0?{content_chars:p}:{},...u!==void 0?{error_message:Yg(u)}:{},...e.model!==void 0?{model:e.model}:{}}).catch(()=>{})}return c!==void 0?{content:`Skill execution error: ${c instanceof Error?c.message:String(c)}`,isError:!0}:{content:typeof a=="string"?a:a!=null?JSON.stringify(a):"Skill completed successfully."}}buildForkedChildConfig(e,n){let r=this.ctx.depth??0,o=this.ctx.maxDepth??Ye,s={...e};if(!this.ctx.childProviderFactory||r>=o)return{childConfig:s,childManager:void 0};let i=new I({parentAbortSignal:n,...this.ctx.traceWriter!==void 0?{traceWriter:this.ctx.traceWriter}:{}}),c=new Xe({subagentManager:i,parentSession:vt(n),defaultConfig:{model:s.model,apiKey:this.ctx.apiKey,...this.ctx.baseUrl!==void 0?{baseUrl:this.ctx.baseUrl}:{}},defaultSubagentModel:this.ctx.defaultSubagentModel,childProviderFactory:this.ctx.childProviderFactory,childSkillExecutorFactory:this.ctx.childSkillExecutorFactory,depth:r+1,maxDepth:o}),a=this.ctx.childSkillExecutorFactory?this.ctx.childSkillExecutorFactory(r+1,o,n):void 0;return s.provider=this.ctx.childProviderFactory({childExecutor:c,...a!==void 0?{childSkillExecutor:a}:{},...s.model!==void 0?{model:s.model}:{}}),{childConfig:s,childManager:i}}async executeForkedRegistrySkill(e,n,r){if(r.signal.aborted)return{content:"Skill call aborted",isError:!0};let o;try{if(o=$(e.name)["system.md"],!o)return{content:`Skill "${e.name}" has context: "fork" but no prompts/system.md found`,isError:!0}}catch(a){return{content:`Failed to load skill prompts: ${a instanceof Error?a.message:String(a)}`,isError:!0}}let s=new I({parentAbortSignal:r.signal,apiKey:this.ctx.apiKey,...this.ctx.baseUrl!==void 0?{baseUrl:this.ctx.baseUrl}:{},...this.ctx.traceWriter!==void 0?{traceWriter:this.ctx.traceWriter}:{},progressSink:re()}),{childConfig:i,childManager:c}=this.buildForkedChildConfig({model:e.model??this.ctx.defaultSubagentModel??this.ctx.defaultModel??"sonnet",systemPrompt:o,...this.ctx.traceWriter!==void 0?{traceWriter:this.ctx.traceWriter}:{}},r.signal);try{let a=await s.forkSubagent({parent:this.ctx.parentSession,config:i,idPrefix:`skill-fork-${e.name}`,parentId:r.id,agentType:e.name}),l=n&&n.length>0?n:"Run the skill.",d=await a.runToResult(l);return d.status==="succeeded"&&d.message?{content:d.message.content}:d.status==="cancelled"&&typeof d.partialOutput=="string"&&d.partialOutput.length>0?{content:`[skill cancelled mid-flight \u2014 partial output preserved below]
1680
1680
 
1681
1681
  ${d.partialOutput}`}:{content:d.error?.message??"Forked skill failed with no output",isError:!0}}catch(a){return{content:`Forked skill execution error: ${a instanceof Error?a.message:String(a)}`,isError:!0}}finally{await c?.teardownAll(),await s.teardownAll()}}async executePluginSkill(e,n,r,o,s){if(s.signal.aborted)return{content:"Skill call aborted",isError:!0};let i=new I({parentAbortSignal:s.signal,apiKey:this.ctx.apiKey,...this.ctx.baseUrl!==void 0?{baseUrl:this.ctx.baseUrl}:{},...this.ctx.traceWriter!==void 0?{traceWriter:this.ctx.traceWriter}:{},progressSink:re()}),{childConfig:c,childManager:a}=this.buildForkedChildConfig({model:this.ctx.defaultSubagentModel??this.ctx.defaultModel??"sonnet",systemPrompt:n,env:{PLUGIN_ROOT:r},...this.ctx.traceWriter!==void 0?{traceWriter:this.ctx.traceWriter}:{}},s.signal);try{let l=await i.forkSubagent({parent:this.ctx.parentSession,config:c,idPrefix:`skill-${e}`,parentId:s.id,agentType:e}),d=o&&o.length>0?o:"Run the skill.",u=await l.runToResult(d);return u.status==="succeeded"&&u.message?{content:u.message.content}:u.status==="cancelled"&&typeof u.partialOutput=="string"&&u.partialOutput.length>0?{content:`[skill cancelled mid-flight \u2014 partial output preserved below]
1682
1682
 
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "agent-afk",
3
- "version": "3.23.1",
3
+ "version": "3.24.0",
4
4
  "description": "CLI tool for interacting with AI agents via multiple interfaces",
5
5
  "main": "dist/index.mjs",
6
6
  "type": "module",