agent-afk 3.31.0 → 3.32.1

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
@@ -1718,11 +1718,11 @@ Every turn must end in one externally identifiable terminal state. AFK users nee
1718
1718
 
1719
1719
  Never end a turn mid-loop without one of these. The terminal-state heading must be the last block of the response, with no trailing prose after it.`,ah=new Set(["repl","telegram"]);function Vr(t,e,n="one-shot"){if(!t)return t;let r=[t];return e&&r.push(sh),ah.has(n)&&r.push(ih),r.join(`
1720
1720
 
1721
- `)}var XR=300*1e3;var On=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 dh=new Set;function gc(t){return dh.has(t)}var uh=new Set,ph=new Set;function hc(t){for(let e of uh)e(t)}function yc(t){for(let e of ph)e(t)}var fh=240;function mh(t,e=fh){return t.length<=e?t:t.slice(0,e)+"\u2026"}function gh(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 hh(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 Xe=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??Qe;if(n>=r){let a=gh(e.input);return q({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=hh(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=gc(e.name);o&&yc(e.name);let s=this.ctx.depth??0;q({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&&hc({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;q({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:mh(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??Qe,s={...e};if(!this.ctx.childProviderFactory||r>=o)return{childConfig:s,childManager:void 0};let i=new C({parentAbortSignal:n,...this.ctx.traceWriter!==void 0?{traceWriter:this.ctx.traceWriter}:{}}),c=new Ze({subagentManager:i,parentSession:Tt(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,...this.ctx.backgroundRegistry!==void 0?{backgroundRegistry:this.ctx.backgroundRegistry}:{}}),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(l){return{content:`Failed to load skill prompts: ${l instanceof Error?l.message:String(l)}`,isError:!0}}let s=new C({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),a;try{a=await s.forkSubagent({parent:this.ctx.parentSession,config:i,idPrefix:`skill-fork-${e.name}`,parentId:r.id,agentType:e.name});let 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]
1721
+ `)}var XR=300*1e3;var On=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 dh=new Set;function gc(t){return dh.has(t)}var uh=new Set,ph=new Set;function hc(t){for(let e of uh)e(t)}function yc(t){for(let e of ph)e(t)}var fh=240;function mh(t,e=fh){return t.length<=e?t:t.slice(0,e)+"\u2026"}function gh(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 hh(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 Xe=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??Qe;if(n>=r){let a=gh(e.input);return q({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=hh(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=gc(e.name);o&&yc(e.name);let s=this.ctx.depth??0;q({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&&hc({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;q({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:mh(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??Qe,s={...e};if(!this.ctx.childProviderFactory||r>=o)return{childConfig:s,childManager:void 0};let i=new C({parentAbortSignal:n,...this.ctx.traceWriter!==void 0?{traceWriter:this.ctx.traceWriter}:{},...this.ctx.cwd!==void 0?{cwd:this.ctx.cwd}:{}}),c=new Ze({subagentManager:i,parentSession:Tt(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,...this.ctx.cwd!==void 0?{cwd:this.ctx.cwd}:{},...this.ctx.backgroundRegistry!==void 0?{backgroundRegistry:this.ctx.backgroundRegistry}:{}}),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(l){return{content:`Failed to load skill prompts: ${l instanceof Error?l.message:String(l)}`,isError:!0}}let s=new C({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(),...this.ctx.cwd!==void 0?{cwd:this.ctx.cwd}:{}}),{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),a;try{a=await s.forkSubagent({parent:this.ctx.parentSession,config:i,idPrefix:`skill-fork-${e.name}`,parentId:r.id,agentType:e.name});let 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]
1722
1722
 
1723
- ${d.partialOutput}`}:{content:d.error?.message??"Forked skill failed with no output",isError:!0}}catch(l){return{content:`Forked skill execution error: ${l instanceof Error?l.message:String(l)}`,isError:!0}}finally{a&&await a.teardown().catch(O),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 C({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),l;try{l=await i.forkSubagent({parent:this.ctx.parentSession,config:c,idPrefix:`skill-${e}`,parentId:s.id,agentType:e});let 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]
1723
+ ${d.partialOutput}`}:{content:d.error?.message??"Forked skill failed with no output",isError:!0}}catch(l){return{content:`Forked skill execution error: ${l instanceof Error?l.message:String(l)}`,isError:!0}}finally{a&&await a.teardown().catch(O),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 C({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(),...this.ctx.cwd!==void 0?{cwd:this.ctx.cwd}:{}}),{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),l;try{l=await i.forkSubagent({parent:this.ctx.parentSession,config:c,idPrefix:`skill-${e}`,parentId:s.id,agentType:e});let 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]
1724
1724
 
1725
- ${u.partialOutput}`}:{content:u.error?.message??"Plugin skill failed with no output",isError:!0}}catch(d){return{content:`Plugin skill execution error: ${d instanceof Error?d.message:String(d)}`,isError:!0}}finally{l&&await l.teardown().catch(O),await a?.teardownAll(),await i.teardownAll()}}getPluginSkillBody(e){return this.pluginBodies||(this.pluginBodies=yn(this.ctx.pluginConfigs)),this.pluginBodies.get(e)}createDispatchSkillCallback(e){return async(n,r)=>{let o={id:`${e.id}-dispatch-${n}`,name:"skill",input:{name:n,...r!==void 0?{arguments:r}:{}},signal:e.signal},s=await this.execute(o);if(s.isError)throw new Error(s.content);return s.content}}};var Qe=3;function Tt(t){return{sessionId:void 0,getInputStreamRef:()=>({pushUserMessage:()=>{}}),abortSignal:t}}var yh=[...ft,"agent","skill"];function bc(t={}){return({childExecutor:e,childSkillExecutor:n,model:r})=>{let o={permissions:{allowedTools:yh},subagentExecutor:e,...n!==void 0?{skillExecutor:n}:{}};return ee(typeof r=="string"?r:void 0)==="openai-compatible"?new ye({...o,...t.openaiBaseUrl!==void 0?{baseURL:t.openaiBaseUrl}:{}}):new ce(o)}}function wc(t,e,n,r,o,s){let i=(c,a,l)=>new Xe({parentSession:Tt(l),defaultModel:t,apiKey:e,...r!==void 0?{baseUrl:r}:{},depth:c,maxDepth:a,childProviderFactory:n,childSkillExecutorFactory:i,...o!==void 0?{traceWriter:o}:{},...s!==void 0?{backgroundRegistry:s}:{}});return i}function kc(t){return t.replace(/\x1b\[[0-9;]*[a-zA-Z]/g,"")}function bh(t){if(typeof t!="object"||t===null)throw new Error("Agent tool input must be an object");let e=t,n=e.prompt;if(typeof n!="string")throw new Error('Agent tool input must have a "prompt" field of type string');if(n.trim().length===0)throw new Error("Agent tool prompt cannot be empty");let r,o=e.model;if(o!==void 0){if(typeof o!="string")throw new Error("Agent tool model must be a string");r=o}let s=10,i=e.max_turns;if(i!==void 0){if(typeof i!="number")throw new Error("Agent tool max_turns must be a number");s=Math.max(1,Math.min(50,Math.floor(i)))}let c="agent-tool",a=e.id_prefix;if(a!==void 0){if(typeof a!="string")throw new Error("Agent tool id_prefix must be a string");c=a}let l="foreground",d=e.mode;if(d!==void 0){if(d!=="foreground"&&d!=="background")throw new Error(`Agent tool mode must be "foreground" or "background", got: ${JSON.stringify(d)}`);l=d}return{prompt:n,model:r,max_turns:s,id_prefix:c,mode:l}}function Dn(t){try{return q(t).catch(()=>{})}catch{return Promise.resolve()}}function et(t,e=240){return t.length<=e?t:t.slice(0,e)+"\u2026"}function vc(t){if(t!=null){if(typeof t=="string")return t.length;try{return JSON.stringify(t).length}catch{return}}}var wh=4096,Sc=1024;function kh(t){if(t==null)return;let e=vc(t);return e!==void 0&&e>wh?{truncated:!0,chars:e}:t}function Sh(t){let e={status:t.status,error:et(t.errorMessage,Sc),subagent_id:t.subagentId};t.schemaErrorMessage&&(e.schemaError=et(t.schemaErrorMessage,Sc));let n=kh(t.partialOutput);return n!==void 0&&(e.partialOutput=n),e}var Ze=class t{constructor(e){this.ctx=e}ctx;async execute(e){if(e.signal.aborted)return{content:"Agent tool call aborted",isError:!0};let n;try{n=bh(e.input)}catch(f){return{content:`Agent tool input validation failed: ${f instanceof Error?f.message:String(f)}`,isError:!0}}let r=this.ctx.depth??0,o=this.ctx.maxDepth??Qe,s,i=n.model??this.ctx.defaultSubagentModel??"sonnet",c=ee(i)==="openai-compatible",a={model:i,apiKey:c?void 0:this.ctx.defaultConfig.apiKey,systemPrompt:this.ctx.defaultConfig.systemPrompt,baseUrl:c?void 0:this.ctx.defaultConfig.baseUrl,maxTurns:n.max_turns},l;if(this.ctx.childProviderFactory&&r<o){s=new C({parentAbortSignal:e.signal}),l=Tt(e.signal);let f=new t({subagentManager:s,parentSession:l,defaultConfig:this.ctx.defaultConfig,defaultSubagentModel:this.ctx.defaultSubagentModel,childProviderFactory:this.ctx.childProviderFactory,childSkillExecutorFactory:this.ctx.childSkillExecutorFactory,depth:r+1,maxDepth:o}),m=this.ctx.childSkillExecutorFactory?this.ctx.childSkillExecutorFactory(r+1,o,e.signal):void 0;a.provider=this.ctx.childProviderFactory({childExecutor:f,...m!==void 0?{childSkillExecutor:m}:{},...a.model!==void 0?{model:a.model}:{}})}let d;try{d=await this.ctx.subagentManager.forkSubagent({parent:this.ctx.parentSession,parentId:e.id,config:a,idPrefix:n.id_prefix,agentType:n.id_prefix&&n.id_prefix!=="agent-tool"?kc(n.id_prefix).replace(/[\r\n]+/g," ").trim()||"agent":kc(n.prompt).replace(/[\r\n]+/g," ").slice(0,40).trim()||"agent",denyElicitations:n.mode==="background"}),l!==void 0&&(l.sessionId=d.id)}catch(f){let m=f instanceof Error?f.message:String(f);return Dn({event:"subagent.failed",subagent_id:"unknown",id_prefix:n.id_prefix,parent_session_id:this.ctx.parentSession.sessionId,status:"failed",error_message:et(m),depth:r}),{content:`Failed to fork subagent: ${m}`,isError:!0}}if(n.mode==="background"){let f=this.ctx.backgroundRegistry;if(!f)return await d.teardown().catch(b=>O("subagent-executor: handle teardown failed: "+(b instanceof Error?b.message:String(b)))),{content:'Background mode is not available in this session \u2014 no BackgroundAgentRegistry is wired. Re-issue the call with mode="foreground" or run inside `afk interactive`.',isError:!0};let m;try{m=f.register({handle:d,prompt:n.prompt,model:a.model??"sonnet",parentSessionId:this.ctx.parentSession.sessionId})}catch(b){if(b instanceof On)return await d.teardown().catch(S=>O("subagent-executor: handle teardown failed after cap error: "+(S instanceof Error?S.message:String(S)))),{content:b.message,isError:!0};throw b}let w={status:"running",jobId:m.jobId,subagentId:m.subagentId,label:m.label,message:`Background subagent started (jobId=${m.jobId}). It is running detached and its result will NOT auto-inject into this context. Retrieve it later via /bgsub:join ${m.jobId} or ask the user to join.`};return{content:JSON.stringify(w)}}let u=()=>{d.cancel()};e.signal.addEventListener("abort",u,{once:!0});let p=Date.now(),g=this.ctx.parentSession.sessionId;try{let f=await d.runToResult(n.prompt);if(f.status==="succeeded"&&f.message){let S=f.message.content,y=typeof S=="string"?S:JSON.stringify(S),h=f.trace;return Dn({event:"subagent.completed",subagent_id:d.id,parent_session_id:g,status:f.status,duration_ms:Date.now()-p,content_chars:y.length,depth:r,tool_call_count:h?.toolCalls.length,thinking_present:h?.thinkingPresent,tool_names:h?.toolCalls.length?JSON.stringify([...new Set(h.toolCalls.map(k=>k.name))]):void 0}),{content:y}}let m=f.error?.message??"Subagent failed with no output",w=f.trace;Dn({event:"subagent.failed",subagent_id:d.id,id_prefix:n.id_prefix,parent_session_id:g,status:f.status,duration_ms:Date.now()-p,error_message:et(m),schema_error:f.schemaError?et(f.schemaError.message):void 0,partial_output_chars:vc(f.partialOutput),depth:r,tool_call_count:w?.toolCalls.length,thinking_present:w?.thinkingPresent,tool_names:w?.toolCalls.length?JSON.stringify([...new Set(w.toolCalls.map(S=>S.name))]):void 0});let b=Sh({status:f.status,errorMessage:m,schemaErrorMessage:f.schemaError?.message,partialOutput:f.partialOutput,subagentId:d.id});return{content:JSON.stringify(b),isError:!0}}catch(f){let m=f instanceof Error?f.message:String(f);throw Dn({event:"subagent.failed",subagent_id:d.id,id_prefix:n.id_prefix,parent_session_id:g,status:"failed",duration_ms:Date.now()-p,error_message:et(m),depth:r}),f}finally{e.signal.removeEventListener("abort",u),await s?.teardownAll(),await d.teardown()}}};var Tc="unknown";try{let t=Ac(_c(Ec(import.meta.url)),"..","package.json"),e=JSON.parse(Jr(t,"utf8"));e.version&&(Tc=e.version)}catch{console.warn("\u26A0\uFE0F [daemon] Could not read package.json at startup \u2014 version drift check disabled.")}async function Eh(){let t;try{t=Pr()}catch(l){console.error("\u274C Configuration error:",l.message),process.exit(1)}let e=ee(t.model);if(e==="openai-compatible"||e==="openai-codex"){let l=v.OPENAI_API_KEY||v.CODEX_API_KEY;console.log(l?"\u{1F4DD} Using OPENAI_API_KEY / CODEX_API_KEY for OpenAI auth":"\u{1F4DD} Will attempt API key from ~/.codex/auth.json (run `afk provider auth diagnose` for details)")}else{let l=kt();(!l||l.length===0)&&(console.error("\u274C Claude models require ANTHROPIC_API_KEY or CLAUDE_CODE_OAUTH_TOKEN."),console.error(" Set one in your environment, run `afk login`, or sign in to Claude Code."),process.exit(1)),Ht(l)==="oauth"?(process.env.CLAUDE_CODE_OAUTH_TOKEN=l,console.log("\u{1F4DD} Using CLAUDE_CODE_OAUTH_TOKEN for Anthropic auth (OAuth, auto-refresh on 401)")):(process.env.ANTHROPIC_API_KEY=l,console.log("\u{1F4DD} Using ANTHROPIC_API_KEY for Anthropic auth")),t.apiKey=l}Th(rt());let n=v.TELEGRAM_BOT_TOKEN;n||(console.error("\u274C Error: TELEGRAM_BOT_TOKEN environment variable is required"),console.error(`
1725
+ ${u.partialOutput}`}:{content:u.error?.message??"Plugin skill failed with no output",isError:!0}}catch(d){return{content:`Plugin skill execution error: ${d instanceof Error?d.message:String(d)}`,isError:!0}}finally{l&&await l.teardown().catch(O),await a?.teardownAll(),await i.teardownAll()}}getPluginSkillBody(e){return this.pluginBodies||(this.pluginBodies=yn(this.ctx.pluginConfigs)),this.pluginBodies.get(e)}createDispatchSkillCallback(e){return async(n,r)=>{let o={id:`${e.id}-dispatch-${n}`,name:"skill",input:{name:n,...r!==void 0?{arguments:r}:{}},signal:e.signal},s=await this.execute(o);if(s.isError)throw new Error(s.content);return s.content}}};var Qe=3;function Tt(t){return{sessionId:void 0,getInputStreamRef:()=>({pushUserMessage:()=>{}}),abortSignal:t}}var yh=[...ft,"agent","skill"];function bc(t={}){return({childExecutor:e,childSkillExecutor:n,model:r})=>{let o={permissions:{allowedTools:yh},subagentExecutor:e,...n!==void 0?{skillExecutor:n}:{}};return ee(typeof r=="string"?r:void 0)==="openai-compatible"?new ye({...o,...t.openaiBaseUrl!==void 0?{baseURL:t.openaiBaseUrl}:{}}):new ce(o)}}function wc(t,e,n,r,o,s,i){let c=(a,l,d)=>new Xe({parentSession:Tt(d),defaultModel:t,apiKey:e,...r!==void 0?{baseUrl:r}:{},depth:a,maxDepth:l,childProviderFactory:n,childSkillExecutorFactory:c,...o!==void 0?{traceWriter:o}:{},...s!==void 0?{backgroundRegistry:s}:{},...i!==void 0?{cwd:i}:{}});return c}function kc(t){return t.replace(/\x1b\[[0-9;]*[a-zA-Z]/g,"")}function bh(t){if(typeof t!="object"||t===null)throw new Error("Agent tool input must be an object");let e=t,n=e.prompt;if(typeof n!="string")throw new Error('Agent tool input must have a "prompt" field of type string');if(n.trim().length===0)throw new Error("Agent tool prompt cannot be empty");let r,o=e.model;if(o!==void 0){if(typeof o!="string")throw new Error("Agent tool model must be a string");r=o}let s=10,i=e.max_turns;if(i!==void 0){if(typeof i!="number")throw new Error("Agent tool max_turns must be a number");s=Math.max(1,Math.min(50,Math.floor(i)))}let c="agent-tool",a=e.id_prefix;if(a!==void 0){if(typeof a!="string")throw new Error("Agent tool id_prefix must be a string");c=a}let l="foreground",d=e.mode;if(d!==void 0){if(d!=="foreground"&&d!=="background")throw new Error(`Agent tool mode must be "foreground" or "background", got: ${JSON.stringify(d)}`);l=d}return{prompt:n,model:r,max_turns:s,id_prefix:c,mode:l}}function Dn(t){try{return q(t).catch(()=>{})}catch{return Promise.resolve()}}function et(t,e=240){return t.length<=e?t:t.slice(0,e)+"\u2026"}function vc(t){if(t!=null){if(typeof t=="string")return t.length;try{return JSON.stringify(t).length}catch{return}}}var wh=4096,Sc=1024;function kh(t){if(t==null)return;let e=vc(t);return e!==void 0&&e>wh?{truncated:!0,chars:e}:t}function Sh(t){let e={status:t.status,error:et(t.errorMessage,Sc),subagent_id:t.subagentId};t.schemaErrorMessage&&(e.schemaError=et(t.schemaErrorMessage,Sc));let n=kh(t.partialOutput);return n!==void 0&&(e.partialOutput=n),e}var Ze=class t{constructor(e){this.ctx=e}ctx;async execute(e){if(e.signal.aborted)return{content:"Agent tool call aborted",isError:!0};let n;try{n=bh(e.input)}catch(f){return{content:`Agent tool input validation failed: ${f instanceof Error?f.message:String(f)}`,isError:!0}}let r=this.ctx.depth??0,o=this.ctx.maxDepth??Qe,s,i=n.model??this.ctx.defaultSubagentModel??"sonnet",c=ee(i)==="openai-compatible",a={model:i,apiKey:c?void 0:this.ctx.defaultConfig.apiKey,systemPrompt:this.ctx.defaultConfig.systemPrompt,baseUrl:c?void 0:this.ctx.defaultConfig.baseUrl,maxTurns:n.max_turns},l;if(this.ctx.childProviderFactory&&r<o){s=new C({parentAbortSignal:e.signal,...this.ctx.cwd!==void 0?{cwd:this.ctx.cwd}:{}}),l=Tt(e.signal);let f=new t({subagentManager:s,parentSession:l,defaultConfig:this.ctx.defaultConfig,defaultSubagentModel:this.ctx.defaultSubagentModel,childProviderFactory:this.ctx.childProviderFactory,childSkillExecutorFactory:this.ctx.childSkillExecutorFactory,depth:r+1,maxDepth:o,...this.ctx.cwd!==void 0?{cwd:this.ctx.cwd}:{}}),m=this.ctx.childSkillExecutorFactory?this.ctx.childSkillExecutorFactory(r+1,o,e.signal):void 0;a.provider=this.ctx.childProviderFactory({childExecutor:f,...m!==void 0?{childSkillExecutor:m}:{},...a.model!==void 0?{model:a.model}:{}})}let d;try{d=await this.ctx.subagentManager.forkSubagent({parent:this.ctx.parentSession,parentId:e.id,config:a,idPrefix:n.id_prefix,agentType:n.id_prefix&&n.id_prefix!=="agent-tool"?kc(n.id_prefix).replace(/[\r\n]+/g," ").trim()||"agent":kc(n.prompt).replace(/[\r\n]+/g," ").slice(0,40).trim()||"agent",denyElicitations:n.mode==="background"}),l!==void 0&&(l.sessionId=d.id)}catch(f){let m=f instanceof Error?f.message:String(f);return Dn({event:"subagent.failed",subagent_id:"unknown",id_prefix:n.id_prefix,parent_session_id:this.ctx.parentSession.sessionId,status:"failed",error_message:et(m),depth:r}),{content:`Failed to fork subagent: ${m}`,isError:!0}}if(n.mode==="background"){let f=this.ctx.backgroundRegistry;if(!f)return await d.teardown().catch(b=>O("subagent-executor: handle teardown failed: "+(b instanceof Error?b.message:String(b)))),{content:'Background mode is not available in this session \u2014 no BackgroundAgentRegistry is wired. Re-issue the call with mode="foreground" or run inside `afk interactive`.',isError:!0};let m;try{m=f.register({handle:d,prompt:n.prompt,model:a.model??"sonnet",parentSessionId:this.ctx.parentSession.sessionId})}catch(b){if(b instanceof On)return await d.teardown().catch(S=>O("subagent-executor: handle teardown failed after cap error: "+(S instanceof Error?S.message:String(S)))),{content:b.message,isError:!0};throw b}let w={status:"running",jobId:m.jobId,subagentId:m.subagentId,label:m.label,message:`Background subagent started (jobId=${m.jobId}). It is running detached and its result will NOT auto-inject into this context. Retrieve it later via /bgsub:join ${m.jobId} or ask the user to join.`};return{content:JSON.stringify(w)}}let u=()=>{d.cancel()};e.signal.addEventListener("abort",u,{once:!0});let p=Date.now(),g=this.ctx.parentSession.sessionId;try{let f=await d.runToResult(n.prompt);if(f.status==="succeeded"&&f.message){let S=f.message.content,y=typeof S=="string"?S:JSON.stringify(S),h=f.trace;return Dn({event:"subagent.completed",subagent_id:d.id,parent_session_id:g,status:f.status,duration_ms:Date.now()-p,content_chars:y.length,depth:r,tool_call_count:h?.toolCalls.length,thinking_present:h?.thinkingPresent,tool_names:h?.toolCalls.length?JSON.stringify([...new Set(h.toolCalls.map(k=>k.name))]):void 0}),{content:y}}let m=f.error?.message??"Subagent failed with no output",w=f.trace;Dn({event:"subagent.failed",subagent_id:d.id,id_prefix:n.id_prefix,parent_session_id:g,status:f.status,duration_ms:Date.now()-p,error_message:et(m),schema_error:f.schemaError?et(f.schemaError.message):void 0,partial_output_chars:vc(f.partialOutput),depth:r,tool_call_count:w?.toolCalls.length,thinking_present:w?.thinkingPresent,tool_names:w?.toolCalls.length?JSON.stringify([...new Set(w.toolCalls.map(S=>S.name))]):void 0});let b=Sh({status:f.status,errorMessage:m,schemaErrorMessage:f.schemaError?.message,partialOutput:f.partialOutput,subagentId:d.id});return{content:JSON.stringify(b),isError:!0}}catch(f){let m=f instanceof Error?f.message:String(f);throw Dn({event:"subagent.failed",subagent_id:d.id,id_prefix:n.id_prefix,parent_session_id:g,status:"failed",duration_ms:Date.now()-p,error_message:et(m),depth:r}),f}finally{e.signal.removeEventListener("abort",u),await s?.teardownAll(),await d.teardown()}}};var Tc="unknown";try{let t=Ac(_c(Ec(import.meta.url)),"..","package.json"),e=JSON.parse(Jr(t,"utf8"));e.version&&(Tc=e.version)}catch{console.warn("\u26A0\uFE0F [daemon] Could not read package.json at startup \u2014 version drift check disabled.")}async function Eh(){let t;try{t=Pr()}catch(l){console.error("\u274C Configuration error:",l.message),process.exit(1)}let e=ee(t.model);if(e==="openai-compatible"||e==="openai-codex"){let l=v.OPENAI_API_KEY||v.CODEX_API_KEY;console.log(l?"\u{1F4DD} Using OPENAI_API_KEY / CODEX_API_KEY for OpenAI auth":"\u{1F4DD} Will attempt API key from ~/.codex/auth.json (run `afk provider auth diagnose` for details)")}else{let l=kt();(!l||l.length===0)&&(console.error("\u274C Claude models require ANTHROPIC_API_KEY or CLAUDE_CODE_OAUTH_TOKEN."),console.error(" Set one in your environment, run `afk login`, or sign in to Claude Code."),process.exit(1)),Ht(l)==="oauth"?(process.env.CLAUDE_CODE_OAUTH_TOKEN=l,console.log("\u{1F4DD} Using CLAUDE_CODE_OAUTH_TOKEN for Anthropic auth (OAuth, auto-refresh on 401)")):(process.env.ANTHROPIC_API_KEY=l,console.log("\u{1F4DD} Using ANTHROPIC_API_KEY for Anthropic auth")),t.apiKey=l}Th(rt());let n=v.TELEGRAM_BOT_TOKEN;n||(console.error("\u274C Error: TELEGRAM_BOT_TOKEN environment variable is required"),console.error(`
1726
1726
  How to get a bot token:`),console.error(" 1. Open Telegram and search for @BotFather"),console.error(" 2. Send /newbot and follow the instructions"),console.error(" 3. Run: afk telegram setup"),process.exit(1));let r=mt(v.AFK_TELEGRAM_ALLOWED_CHAT_IDS,console.warn);r.size===0&&(console.error("\u274C Error: AFK_TELEGRAM_ALLOWED_CHAT_IDS must list at least one chat ID"),console.error(`
1727
1727
  This is an allowlist that gates who can message the bot.`),console.error("Run `afk telegram setup` to set it interactively, or set it manually:"),console.error(" AFK_TELEGRAM_ALLOWED_CHAT_IDS=123456789,-100987654321"),process.exit(1)),console.log("\u{1F50E} Validating bot token...");let o=await Qa(n);o||(console.error("\u274C Error: TELEGRAM_BOT_TOKEN was rejected by Telegram (getMe failed)"),console.error(" The token may be revoked, malformed, or your network may be unreachable."),console.error(" Re-run `afk telegram setup` to refresh it."),process.exit(1));let s=o.username?`@${o.username}`:o.firstName;console.log(""),console.log(`\u{1F916} Starting Agent AFK Telegram Bot as ${s} (id ${o.id})`),console.log(`\u{1F4E1} Model: ${t.model} \xB7 Provider: ${e}`),console.log(`\u{1F512} Allowlist: ${r.size} chat ID(s)`);let i=new X,c=v.AFK_TELEGRAM_CWD,a=new In({botToken:n,apiKey:t.apiKey??"",dataDir:v.TELEGRAM_DATA_DIR||"./data/telegram-sessions",defaultModel:t.model,verbose:v.TELEGRAM_VERBOSE==="true",allowedChatIds:r,settingSources:["user","project"],...c!==void 0&&c.length>0?{botCwd:c}:{},createSession:async l=>{let d=Ie(l.model)??l.model;console.log(`Creating session with model: ${l.model} -> ${d}`);let u=ee(d),p=u==="openai-compatible"||u==="openai-codex",g=p?void 0:Wi(),f;if(!p){let h,k=l.apiKey??t.apiKey??"",E=t.baseUrl,_=l.cwd??c,T=new C({apiKey:k,...E!==void 0?{baseUrl:E}:{},..._!==void 0&&_.length>0?{cwd:_}:{}}),I={get sessionId(){return h?.sessionId},getInputStreamRef(){return h?.getInputStreamRef?.()??{pushUserMessage:()=>{}}},get abortSignal(){return h?.abortSignal??new AbortController().signal}},A=bc(),M=wc(l.model,k,A,E),x=new Ze({subagentManager:T,parentSession:I,defaultConfig:{apiKey:k,systemPrompt:l.systemPrompt??t.systemPrompt,...E!==void 0?{baseUrl:E}:{}},defaultSubagentModel:hn(l.model),childProviderFactory:A,childSkillExecutorFactory:M}),R=new Xe({parentSession:I,defaultModel:l.model,defaultSubagentModel:hn(l.model),apiKey:k,childProviderFactory:A,childSkillExecutorFactory:M,...E!==void 0?{baseUrl:E}:{}}),F=l.systemPrompt??t.systemPrompt,D=new Mn({parentSession:I,defaultModel:l.model,defaultSubagentModel:hn(l.model),apiKey:k,...E!==void 0?{baseUrl:E}:{},systemPrompt:typeof F=="string"?F:""}),B=[...ft,...Ft,"agent","skill","compose"];f=new ce({permissions:{allowedTools:B},subagentExecutor:x,skillExecutor:R,composeExecutor:D});let W=l.systemPrompt??t.systemPrompt,V=t.autoRouting?.telegram??!1,le=typeof W=="string"?Vr(W,V,"telegram"):W,K=new Se({...l.apiKey!==void 0?{apiKey:l.apiKey}:{},model:l.model,...le!==void 0?{systemPrompt:le}:{},maxTurns:100,...g!==void 0?{maxOutputTokens:g}:{},...E!==void 0?{baseUrl:E}:{},..._!==void 0&&_.length>0?{cwd:_}:{},provider:f,hookRegistry:zr(void 0,"telegram",i).registry});return h=K,K}let m=l.systemPrompt??t.systemPrompt,w=t.autoRouting?.telegram??!1,b=typeof m=="string"?Vr(m,w,"telegram"):m,S=l.cwd??c;return new Se({...l.apiKey!==void 0?{apiKey:l.apiKey}:{},model:l.model,...b!==void 0?{systemPrompt:b}:{},maxTurns:100,...g!==void 0?{maxOutputTokens:g}:{},...S!==void 0&&S.length>0?{cwd:S}:{},hookRegistry:zr(void 0,"telegram",i).registry})}});try{a.start(),console.log("\u2705 Bot started successfully!"),console.log(`
1728
1728
  \u{1F4DD} Slash commands (Agent SDK):`),console.log(" /start - Welcome and command list"),console.log(" /help - Show command list"),console.log(" /clear - Clear conversation history"),console.log(" /compact - Compact history (summarize older messages)"),console.log(" /model - Switch model (opus/sonnet/haiku/gpt-5.4/...)"),console.log(`
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "agent-afk",
3
- "version": "3.31.0",
3
+ "version": "3.32.1",
4
4
  "description": "CLI tool for interacting with AI agents via multiple interfaces",
5
5
  "main": "dist/index.mjs",
6
6
  "type": "module",