grix-connector 1.1.0 → 1.1.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (40) hide show
  1. package/dist/adapter/agy/agy-adapter.js +2 -2
  2. package/dist/adapter/claude/claude-adapter.js +17 -17
  3. package/dist/adapter/deepseek/deepseek-adapter.js +3 -3
  4. package/dist/bridge/acp-toolbar-persist.js +1 -1
  5. package/dist/bridge/bridge.js +9 -9
  6. package/dist/core/installer/env-detect.js +3 -2
  7. package/dist/core/installer/preflight.js +3 -2
  8. package/dist/grix.js +0 -0
  9. package/package.json +2 -2
  10. package/scripts/install-guardian.sh +0 -0
  11. package/scripts/upgrade-guardian.sh +0 -0
  12. package/dist/adapter/claude/claude-bridge-server.js +0 -1
  13. package/dist/adapter/claude/claude-tools.js +0 -1
  14. package/dist/adapter/claude/claude-worker-client.js +0 -1
  15. package/dist/adapter/claude/mcp-http-launcher.js +0 -2
  16. package/dist/adapter/claude/result-timeout.js +0 -1
  17. package/dist/adapter/qwen/index.js +0 -1
  18. package/dist/adapter/qwen/qwen-adapter.js +0 -4
  19. package/dist/aibot/client.js +0 -1
  20. package/dist/aibot/index.js +0 -1
  21. package/dist/aibot/types.js +0 -0
  22. package/dist/core/file-ops/handler.js +0 -1
  23. package/dist/core/file-ops/list-files.js +0 -1
  24. package/dist/core/file-ops/types.js +0 -0
  25. package/dist/log.js +0 -3
  26. package/dist/main.js +0 -31
  27. package/dist/mcp/stream-http/config.js +0 -1
  28. package/dist/mcp/stream-http/connection-binding.js +0 -1
  29. package/dist/mcp/stream-http/event-tool-executor.js +0 -1
  30. package/dist/mcp/stream-http/gateway.js +0 -1
  31. package/dist/mcp/stream-http/index.js +0 -1
  32. package/dist/mcp/stream-http/security.js +0 -1
  33. package/dist/mcp/stream-http/session-manager.js +0 -1
  34. package/dist/mcp/stream-http/tool-executor.js +0 -1
  35. package/dist/mcp/stream-http/tool-registry.js +0 -1
  36. package/dist/mcp/stream-http/tool-schemas.js +0 -1
  37. package/dist/session/index.js +0 -1
  38. package/dist/session/manager.js +0 -1
  39. package/dist/transport/index.js +0 -1
  40. package/dist/transport/json-rpc.js +0 -3
@@ -1,16 +1,16 @@
1
- import M from"node:path";import{realpath as G,stat as H}from"node:fs/promises";import{tmpdir as V}from"node:os";import{randomUUID as U}from"node:crypto";import{ConnectionManager as J}from"../core/aibot/index.js";import{ClaudeAdapter as y}from"../adapter/claude/index.js";import{CodexAdapter as X}from"../adapter/codex/index.js";import{PiAdapter as Y}from"../adapter/pi/index.js";import{AcpAdapter as A}from"../adapter/acp/index.js";import{OpenHumanAdapter as Z}from"../adapter/openhuman/index.js";import{CursorAdapter as L}from"../adapter/cursor/index.js";import{CodeWhaleAdapter as q}from"../adapter/codewhale/index.js";import{OpenCodeAdapter as ee}from"../adapter/opencode/index.js";import{AgyAdapter as O}from"../adapter/agy/index.js";import{getCachedAgyModels as te}from"../adapter/agy/model-list.js";import{getCachedAgyQuotaInfo as ie}from"../adapter/agy/quota.js";import{LOCAL_ACTION_ERROR_CODES as x,LOCAL_ACTION_TYPES as S,SESSION_CONTROL_ERROR_CODES as h,SESSION_CONTROL_VERBS as _,SESSION_MODE_IDS as C}from"../adapter/claude/protocol-contract.js";import{parseClaudeSessionUsage as ne}from"../adapter/claude/usage-parser.js";import{fetchAvailableModels as I,getCachedModels as P,readSettingsEnv as se}from"../adapter/claude/model-list.js";import{parseAcpSessionUsage as oe}from"../adapter/acp/usage-parser.js";import{parseCodexSessionUsage as re}from"../adapter/codex/usage-parser.js";import{readCodexProviderSettings as de}from"../adapter/codex/codex-trust.js";import{scanCodexSessions as ae}from"../adapter/codex/session-scanner.js";import{scanClaudeSessions as ce}from"../adapter/claude/session-scanner.js";import{scanAcpSessions as le}from"../adapter/acp/session-scanner.js";import{parsePiSessionUsage as ue}from"../adapter/pi/usage-parser.js";import{SessionScanCache as D,resolveCodexLeafDirs as he,resolveClaudeLeafDirs as ge,resolveAcpLeafDirs as me}from"./session-scan-cache.js";import{log as u,ConversationLog as _e,AgentApiPacketLog as fe,BridgeEventLog as pe}from"../core/log/index.js";import{RevokeHandler as ve}from"./revoke-handler.js";import{AdapterPool as be,PoolFullError as Se}from"./adapter-pool.js";import{parseSessionControlCommand as we,handleSessionControlCommand as F,handleSessionControlLocalAction as Ce}from"./session-controller.js";import{handleAcpSetModel as W,handleAcpSetMode as N,resolveAcpInitialDefaults as Ae}from"./acp-toolbar-persist.js";import{SessionBindingStore as ke}from"../core/persistence/session-binding-store.js";import{handleFileListAction as Ee,handleCreateFolderAction as Re,serveLocalFile as $e,realHomeDir as K}from"../core/files/index.js";import{AllowlistGate as xe}from"../core/access/allowlist-gate.js";import{ActiveEventStore as Me}from"../core/persistence/active-event-store.js";import{DEFAULT_CONNECTOR_RUNTIME_CONFIG as ye,applyConnectorRuntimeConfigPatch as Te,extractConnectorRuntimeConfigPatch as He}from"./runtime-config.js";import{SendController as Le}from"./send-controller.js";import{queryProviderQuota as z}from"../core/provider-quota/index.js";import{queryKiroQuota as B}from"../core/provider-quota/kiro.js";import{buildToolUseCard as R,buildToolResultCard as $,buildLocalGrixCardLink as Ie}from"./tool-card-utils.js";import{DeferredEventManager as Pe}from"./deferred-events.js";import{buildAgentProbeResult as De,PROBE_CACHE_TTL_STATIC_MS as Be,PROBE_CACHE_TTL_FULL_MS as Qe}from"./probe-helper.js";const Ue=600*1e3,qe=60*1e3,j=30*1e3,T=new Set(["claude","acp","agy","cursor","codex"]),Q=3;class Tt{config;name;aibotHandle;aibotConfig;pool;stopped=!1;revokeHandler=new ve;sessionBindings=new Map;deferredMgr;sendCtrl=new Le(ye);bindingStore;globalConfigStore;upgradeTrigger=null;allowlistGate;activeEventStore;cachedRateLimits=null;cachedRateLimitsSampledAtMs=null;cachedCodexContextWindow=null;cachedCodexTokenUsage=null;cachedCodexUsageSampledAtMs=null;cachedAcpContextWindow=null;cachedAcpContextWindowSampledAtMs=null;cachedClaudeRateLimitState=null;cachedProviderQuota=null;cachedProviderQuotaSampledAtMs=null;claudeWorkerStatus=new Map;conversationLog=null;packetLog=null;kiroQuotaTimer=null;eventSessionIndex=new Map;inflightEvents=new Map;restartCount=new Map;probeCache=new Map;sessionScanCache;isRateLimitsCacheFresh(e){if(!Number.isFinite(e))return!1;const i=Number(e);return i>0&&Date.now()-i<=qe}async maybeQueryProviderQuota(){if(this.config.aibot.clientType==="kiro"){if(this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)&&this.cachedProviderQuota)return this.cachedProviderQuota;try{const t=await B();return this.cachedProviderQuota=t,this.cachedProviderQuotaSampledAtMs=Date.now(),u.info(this.name,`[provider-quota] kiro queried: success=${t.success}`+(t.balance?` balance=${t.balance.remaining} ${t.balance.unit}`:"")+(t.planName?` plan=${t.planName}`:"")+(t.error?` error=${t.error}`:"")),t}catch(t){return u.warn(this.name,`[provider-quota] kiro query failed: ${t instanceof Error?t.message:String(t)}`),null}}let e=this.config.providerBaseUrl,i=this.config.providerApiKey;if((!e||!i)&&(this.config.adapterType??"acp")==="claude"){const t=se();e||(e=(t.ANTHROPIC_BASE_URL??"").trim()||void 0),i||(i=(t.ANTHROPIC_API_KEY??"").trim()||(t.ANTHROPIC_AUTH_TOKEN??"").trim()||void 0)}if((!e||!i)&&(this.config.adapterType??"acp")==="codex"){const t=de();!e&&t.baseUrl&&(e=t.baseUrl),!i&&t.apiKey&&(i=t.apiKey)}if(!e||!i)return null;if(this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)&&this.cachedProviderQuota)return this.cachedProviderQuota;try{const t=await z(e,i);return this.cachedProviderQuota=t,this.cachedProviderQuotaSampledAtMs=Date.now(),u.info(this.name,`[provider-quota] queried: provider=${t.provider} success=${t.success}`+(t.tiers.length>0?` tiers=${t.tiers.map(s=>`${s.name}=${s.usedPercent}%`).join(",")}`:"")+(t.balance?` balance=${t.balance.remaining} ${t.balance.unit}`:"")+(t.error?` error=${t.error}`:"")),t}catch(t){return u.warn(this.name,`[provider-quota] query failed: ${t instanceof Error?t.message:String(t)}`),null}}startKiroQuotaTimer(){this.kiroQuotaTimer||(this.kiroQuotaTimer=setInterval(()=>{if(this.stopped){this.stopKiroQuotaTimer();return}this.refreshAndPushKiroQuota().catch(()=>{})},j),u.info(this.name,`[kiro-quota-timer] started (interval=${j}ms)`))}stopKiroQuotaTimer(){this.kiroQuotaTimer&&(clearInterval(this.kiroQuotaTimer),this.kiroQuotaTimer=null,u.info(this.name,"[kiro-quota-timer] stopped"))}async refreshAndPushKiroQuota(){try{const e=await B();this.cachedProviderQuota=e,this.cachedProviderQuotaSampledAtMs=Date.now(),e.success&&this.pushKiroQuotaToBindings(e)}catch{}}pushKiroQuotaToBindings(e){const i=this.providerQuotaToRateLimits(e);for(const[t,s]of this.sessionBindings.entries()){if(!s)continue;const n={provider_quota:e};i&&(n.rate_limits=i);const o=this.cachedAcpContextWindow;o&&("usedPercentage"in o?n.context_window={usedPercentage:o.usedPercentage,remainingPercentage:100-o.usedPercentage}:n.context_window=o),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:s,meta:n})}}getFreshClaudeRateLimitState(){const e=this.cachedClaudeRateLimitState;return e&&this.isRateLimitsCacheFresh(e.sampledAt)?e:null}getFreshCodexGlobalRateLimitCache(){const e=this.cachedRateLimits,i=this.cachedCodexContextWindow,t=this.cachedCodexTokenUsage;return{sampledAt:Math.max(this.cachedRateLimitsSampledAtMs??0,this.cachedCodexUsageSampledAtMs??0)||null,rateLimits:e,contextWindow:i,tokenUsage:t,hasData:!!(e||i||t)}}getStatus(){const e=this.pool?.getStatus()??{total:0,ready:0,busy:0};return{name:this.name,alive:!this.stopped,busy:e.busy>0,exhausted:this.pool?[...this.pool.getAllSlots()].some(i=>i.respawn.exhausted):!1,adapterType:this.config.adapterType??"acp",clientType:this.config.aibot.clientType,pool:e}}async probe(e={}){const i=e.conversation?"full":"static",t=e.conversation?Qe:Be;if(!e.fresh){const r=this.probeCache.get(i);if(r&&Date.now()-r.sampledAt<t)return{...r.result,cached:!0}}const s=this.config.adapterType??"acp",n=this.createAdapter(s,"__probe__"),o=e.conversation&&s==="acp"?()=>this.runAcpConversationProbe(n,e.timeoutMs??1e4):void 0;let d;try{d=await De({adapter:n,agentName:this.name,clientType:this.config.aibot.clientType,adapterType:s,providerBaseUrl:this.config.providerBaseUrl??null,opts:e,launchConversationProbe:o})}finally{n.stop().catch(()=>{})}return this.probeCache.set(i,{result:d,sampledAt:d.probed_at}),d}async runAcpConversationProbe(e,i){const t=Date.now(),s="__probe__",n=`probe-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,o=this.config.agent.cwd||V(),d=()=>Date.now()-t;try{await e.start()}catch(g){return{attempted:!0,ok:!1,latency_ms:d(),error:{code:"conversation_failed",message:`start failed: ${g instanceof Error?g.message:String(g)}`}}}if(!e.isAlive())return{attempted:!0,ok:!1,latency_ms:d(),error:{code:"process_not_started",message:"agent process not alive"}};if(e instanceof A)try{await e.bindSession(s,o)}catch{}let r=null;const a=new Promise(g=>{r=m=>{m===n&&g()},e.on("eventDone",r)});e.deliverInboundEvent({event_id:n,session_id:s,content:"ping",msg_id:n}),e.deliverStopEvent(n,s);let c=null;const l=await Promise.race([a.then(()=>!1),new Promise(g=>{c=setTimeout(()=>g(!0),i)})]);return c&&clearTimeout(c),r&&e.removeListener("eventDone",r),l?{attempted:!0,ok:!1,latency_ms:d(),error:{code:"conversation_timeout",message:`no eventDone within ${i}ms`}}:{attempted:!0,ok:!0,latency_ms:d()}}constructor(e,i){this.config=e,this.name=e.name;const t=e.adapterType??"acp";this.aibotConfig={...e.aibot,...t==="claude"?{localActions:e.aibot.localActions??["session_control","claude_interaction_reply","get_session_usage","get_rate_limits","set_model","set_mode","thread_compact","get_agent_global_config"]}:{}},e.eventQueue&&(this.aibotConfig.concurrency={max_concurrent:e.eventQueue.maxConcurrent,max_queued:e.eventQueue.maxQueued,queue_timeout_ms:e.eventQueue.queueTimeoutMs,cancelable_queued:e.eventQueue.cancelableQueued,cancelable_running:e.eventQueue.cancelableRunning}),this.conversationLog=e.logDir?new _e(e.logDir):null,this.packetLog=e.logDir?new fe(e.logDir):null,this.bindingStore=new ke(e.bindingsPath),this.bindingStore.load(),this.globalConfigStore=i??null,this.deferredMgr=new Pe(this.name),this.allowlistGate=e.allowlistPath?new xe(e.allowlistPath):null,this.activeEventStore=e.activeEventStorePath?new Me(e.activeEventStorePath):null,t==="codex"?this.sessionScanCache=new D(ae,he):t==="claude"?this.sessionScanCache=new D(ce,ge):this.sessionScanCache=new D(le,me)}async start(){(this.config.adapterType??"acp")==="claude"&&(await I().catch(()=>{}),this.maybeQueryProviderQuota().catch(()=>{})),this.config.aibot.clientType==="kiro"&&(this.maybeQueryProviderQuota().catch(()=>{}),this.startKiroQuotaTimer()),(this.config.adapterType??"acp")==="codex"&&this.maybeQueryProviderQuota().catch(()=>{}),await this.connectAibot(),this.sendCtrl.bind(this.aibotHandle);const e=this.config.adapterType??"acp";if(e==="agy")for(const[i,t]of this.bindingStore.entries())t.cwd&&this.aibotHandle.sendUpdateBindingCard({session_id:i,worker_status:"ready",cwd:t.cwd,meta:this.buildAgyToolbarMeta(i)});this.pool=new be({maxPoolSize:this.config.poolMaxSize??20,idleTimeoutMs:this.config.poolIdleTimeoutMs??3e5,eventQueue:this.config.eventQueue},i=>{const t=this.createAdapter(e,i);return t instanceof A&&t.on("acpSessionReady",s=>{this.bindingStore.setAcpSessionId(i,s),this.sessionScanCache.invalidate()}),t},(i,t)=>{this.aibotHandle.sendEventAck({event_id:i,session_id:t,received_at:Date.now()})}),this.pool.setEventStateHandler((i,t,s,n)=>{u.info(this.name,`[queue-debug] send event_state session=${t} event=${i} state=${s} queue_pos=${n?.queue_position??""} queue_total=${n?.queue_total??""}`),this.aibotHandle.sendEventState({event_id:i,session_id:t,state:s,content_preview:n?.content_preview,queue_position:n?.queue_position,queue_total:n?.queue_total,actions:n?.actions,reason:n?.reason,updated_at:Date.now()}),this.pushQueueSnapshotForSession(t),(s==="canceled"||s==="failed")&&this.aibotHandle.sendEventResult({event_id:i,status:s==="canceled"?"canceled":"failed",msg:n?.reason,updated_at:Date.now()})}),this.pool.setQueueComposingHandler((i,t,s)=>{this.aibotHandle.sendSessionActivitySet({session_id:i,kind:"composing",active:t,...t?{ttl_ms:3e4,ref_event_id:s}:{}})}),this.pool.setInternalErrorHandler(i=>{this.handleSessionInternalError(i).catch(t=>{u.error(this.name,`[recovery] handleSessionInternalError failed event=${i.eventId} session=${i.sessionId}: ${t instanceof Error?t.message:String(t)}`)})}),this.pool.setEventStartedHandler((i,t)=>{if(this.config.adapterType!=="claude")return;this.claudeWorkerStatus.set(t,"busy");const s=this.bindingStore.get(t);s?.cwd&&this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"busy",cwd:s.cwd,meta:this.buildClaudeToolbarMeta(t)})}),this.pool.setEventDoneHandler((i,t)=>{if(this.config.adapterType!=="claude")return;this.claudeWorkerStatus.set(t,"ready");const s=this.bindingStore.get(t);s?.cwd&&this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:s.cwd,meta:this.buildClaudeToolbarMeta(t)})}),this.pool.startIdleSweep(),u.info(this.name,`Ready (adapter: ${e}, poolMax: ${this.config.poolMaxSize??20})`)}async stop(){this.stopped=!0,this.pool?.stopIdleSweep(),this.stopKiroQuotaTimer();const e=this.pool?.collectActiveEventIds()??[];e.length>0&&this.activeEventStore&&await this.activeEventStore.save(e);for(const t of e)u.info(this.name,`Canceling active event on shutdown: ${t}`),this.sendEventResultWithCleanup(t,"canceled","process shutting down");e.length>0&&await new Promise(t=>setTimeout(t,100)),this.pool?.clearActiveEventsForShutdown();const i=this.deferredMgr.getAllDeferredEvents();for(const t of i)u.info(this.name,`Failing deferred event on shutdown: ${t.event_id}`),this.sendEventResultWithCleanup(t.event_id,"failed","process shutting down");this.deferredMgr.clearAll(),await this.pool?.stop(),this.aibotHandle?.disconnect(),e.length>0&&this.activeEventStore&&await this.activeEventStore.save([]),this.eventSessionIndex.clear(),this.inflightEvents.clear(),this.restartCount.clear()}createAdapter(e,i){switch(e){case"claude":return this.createClaudeAdapter(i);case"codex":return this.createCodexAdapter(i);case"pi":return this.createPiAdapter(i);case"openhuman":return this.createOpenHumanAdapter(i);case"codewhale":return this.createCodeWhaleAdapter(i);case"cursor":return this.createCursorAdapter(i);case"opencode":return this.createOpenCodeAdapter(i);case"agy":return this.createAgyAdapter(i);default:return this.createAcpAdapter(i)}}createCursorAdapter(e){const i={...this.config.adapterOptions??{}},t=this.bindingStore.get(e),s=t?.modelId??this.globalConfigStore?.get(this.name)?.modelId;s&&(i.model=s),t?.modeId&&(i.mode=t.modeId);const n={sendStreamChunk:(o,d,r,a,c)=>{this.sendStreamChunkByRuntimeConfig(o,d,r,a,c)},sendEventResult:(o,d,r)=>{this.sendEventResultWithCleanup(o,d,r)},sendEventAck:(o,d)=>{this.aibotHandle.sendEventAck({event_id:o,session_id:d,received_at:Date.now()})},sendRawEventEnvelope:(o,d,r)=>{this.aibotHandle.sendMsg({event_id:o,session_id:d,msg_type:1,content:"[cursor] raw_event",extra:{channel_data:{cursor:{raw_event:r}},agent_api_origin:!0}})},agentInvoke:async(o,d)=>this.platformInvoke(o,d),sendLocalActionResult:(o,d,r,a,c)=>{this.aibotHandle.sendLocalActionResult({action_id:o,status:d,...r!==void 0?{result:r}:{},...a?{error_code:a}:{},...c?{error_msg:c}:{}})}};return new L({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:i},n)}createClaudeAdapter(e){const i={sendReply:(n,o,d,r,a)=>{this.sendReplyByRuntimeConfig(n,o,d,r,a)},sendStreamChunk:(n,o,d,r,a,c,l)=>{this.sendStreamChunkByRuntimeConfig(n,o,d,r,a,c,l)},sendMedia:(n,o,d,r,a,c,l)=>{this.aibotHandle.sendMedia({event_id:n,session_id:o,content:d,msg_type:2,quoted_message_id:a||void 0,client_msg_id:c||void 0,extra:l?{media_caption:r,...l}:{media_caption:r}})},sendEventResult:(n,o,d,r)=>{this.sendEventResultWithCleanup(n,o,d,r)},sendEventAck:(n,o)=>{this.aibotHandle.sendEventAck({event_id:n,session_id:o,received_at:Date.now()})},agentInvoke:async(n,o,d)=>this.platformInvoke(n,o,d),sendLocalActionResult:(n,o,d,r,a)=>{this.aibotHandle.sendLocalActionResult({action_id:n,status:o,...d!==void 0?{result:d}:{},...r?{error_code:r}:{},...a?{error_msg:a}:{}})},sendToolUse:(n,o,d,r)=>{this.sendToolExecutionCard(n,o,R(d,r))},sendToolResult:(n,o,d,r)=>{this.sendToolExecutionCard(n,o,$(d,r))},getWsUrl:()=>this.config.aibot.url,getAgentId:()=>this.config.aibot.agentId,getApiKey:()=>this.config.aibot.apiKey,getActiveEventCount:()=>0,getPendingPermissionCount:()=>0,getPendingElicitationCount:()=>0,sendAgentQuestionCard:(n,o,d)=>{const r=d.questions.map(c=>c.header).join(", "),a=Ie(`[Agent Question] ${d.request_id}`,"agent_question",d);this.aibotHandle.sendText({event_id:n,session_id:o,content:a,msg_type:1,extra:{card_type:"agent_question",summary_text:r}})},sendPermissionCard:n=>{this.aibotHandle.sendMsg({event_id:n.eventId,session_id:n.sessionId,client_msg_id:`perm_${U()}`,msg_type:1,content:n.toolTitle?`Permission required: ${n.toolTitle}`:"Permission request",extra:{channel_data:{execApproval:{approvalId:n.approvalId,approvalSlug:n.toolName},grix:{execApproval:{approval_command_id:n.approvalId,command:n.toolTitle||n.toolName,host:"claude"}}},agent_api_origin:!0}})},sendDirectMessage:n=>{this.aibotHandle.sendMsg({session_id:n.sessionId,msg_type:1,content:n.content,...n.clientMsgId?{client_msg_id:n.clientMsgId}:{},...n.quotedMessageId?{quoted_message_id:n.quotedMessageId}:{}})},onStatusLineUpdated:n=>{(n.rateLimits?.fiveHour||n.rateLimits?.sevenDay)&&(this.cachedClaudeRateLimitState=n);const o=this.bindingStore.get(e);o?.cwd&&this.aibotHandle.sendUpdateBindingCard({session_id:e,worker_status:this.claudeWorkerStatus.get(e)??"ready",cwd:o.cwd,meta:this.buildClaudeToolbarMeta(e)})}},t=this.config.adapterOptions??{},s={...t,sessionRuntimeResolver:()=>{const n=this.bindingStore.get(e);return{cwd:n?.cwd,modeId:n?.modeId??C.fullAuto,modelId:n?.modelId??this.globalConfigStore?.get(this.name)?.modelId,pluginDir:t.pluginDir,claudeSessionId:n?.claudeSessionId,onSessionIdAssigned:o=>{this.bindingStore.setClaudeSessionId(e,o),this.sessionScanCache.invalidate()}}}};return new y({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:s},i)}createCodexAdapter(e){let i=null;const t={sendEventResult:(o,d,r)=>{this.sendEventResultWithCleanup(o,d,r)},sendEventAck:(o,d)=>this.aibotHandle.sendEventAck({event_id:o,session_id:d,received_at:Date.now()}),sendCodexEvent:o=>{this.shouldDropCodexDisplayEvent(o.event_id,o.codex_method)||(this.aibotHandle.sendCodexEvent(o),this.logCodexEventToConversation(o))},sendCodexEventReliable:async o=>{if(!this.shouldDropCodexDisplayEvent(o.event_id,o.codex_method)){try{await this.aibotHandle.sendCodexEventReliable(o)}catch(d){u.warn("bridge",`[codex] sendCodexEventReliable ACK failed event=${o.event_id}: ${d}`)}u.info("bridge",`[codex] sendCodexEventReliable done event=${o.event_id} method=${o.codex_method}`),this.logCodexEventToConversation(o)}},sendRunError:(o,d,r)=>{this.sendStreamChunkByRuntimeConfig(o,d,`
1
+ import M from"node:path";import{realpath as G,stat as H}from"node:fs/promises";import{tmpdir as V}from"node:os";import{randomUUID as U}from"node:crypto";import{ConnectionManager as J}from"../core/aibot/index.js";import{ClaudeAdapter as y}from"../adapter/claude/index.js";import{CodexAdapter as X}from"../adapter/codex/index.js";import{PiAdapter as Y}from"../adapter/pi/index.js";import{AcpAdapter as A}from"../adapter/acp/index.js";import{OpenHumanAdapter as Z}from"../adapter/openhuman/index.js";import{CursorAdapter as L}from"../adapter/cursor/index.js";import{CodeWhaleAdapter as q}from"../adapter/codewhale/index.js";import{OpenCodeAdapter as ee}from"../adapter/opencode/index.js";import{AgyAdapter as O}from"../adapter/agy/index.js";import{getCachedAgyModels as te}from"../adapter/agy/model-list.js";import{getCachedAgyQuotaInfo as ie}from"../adapter/agy/quota.js";import{LOCAL_ACTION_ERROR_CODES as x,LOCAL_ACTION_TYPES as w,SESSION_CONTROL_ERROR_CODES as h,SESSION_CONTROL_VERBS as _,SESSION_MODE_IDS as C}from"../adapter/claude/protocol-contract.js";import{parseClaudeSessionUsage as ne}from"../adapter/claude/usage-parser.js";import{fetchAvailableModels as P,getCachedModels as D,readSettingsEnv as se}from"../adapter/claude/model-list.js";import{parseAcpSessionUsage as oe}from"../adapter/acp/usage-parser.js";import{parseCodexSessionUsage as re}from"../adapter/codex/usage-parser.js";import{readCodexProviderSettings as de}from"../adapter/codex/codex-trust.js";import{scanCodexSessions as ae}from"../adapter/codex/session-scanner.js";import{scanClaudeSessions as ce}from"../adapter/claude/session-scanner.js";import{scanAcpSessions as le}from"../adapter/acp/session-scanner.js";import{parsePiSessionUsage as ue}from"../adapter/pi/usage-parser.js";import{SessionScanCache as I,resolveCodexLeafDirs as he,resolveClaudeLeafDirs as ge,resolveAcpLeafDirs as me}from"./session-scan-cache.js";import{log as u,ConversationLog as _e,AgentApiPacketLog as fe,BridgeEventLog as pe}from"../core/log/index.js";import{RevokeHandler as ve}from"./revoke-handler.js";import{AdapterPool as be,PoolFullError as we}from"./adapter-pool.js";import{parseSessionControlCommand as Se,handleSessionControlCommand as F,handleSessionControlLocalAction as Ce}from"./session-controller.js";import{handleAcpSetModel as W,handleAcpSetMode as N,resolveAcpInitialDefaults as Ae}from"./acp-toolbar-persist.js";import{SessionBindingStore as ke}from"../core/persistence/session-binding-store.js";import{handleFileListAction as Ee,handleCreateFolderAction as Re,serveLocalFile as $e,realHomeDir as K}from"../core/files/index.js";import{AllowlistGate as xe}from"../core/access/allowlist-gate.js";import{ActiveEventStore as Me}from"../core/persistence/active-event-store.js";import{DEFAULT_CONNECTOR_RUNTIME_CONFIG as ye,applyConnectorRuntimeConfigPatch as Te,extractConnectorRuntimeConfigPatch as He}from"./runtime-config.js";import{SendController as Le}from"./send-controller.js";import{queryProviderQuota as z}from"../core/provider-quota/index.js";import{queryKiroQuota as B}from"../core/provider-quota/kiro.js";import{buildToolUseCard as R,buildToolResultCard as $,buildLocalGrixCardLink as Pe}from"./tool-card-utils.js";import{DeferredEventManager as De}from"./deferred-events.js";import{buildAgentProbeResult as Ie,PROBE_CACHE_TTL_STATIC_MS as Be,PROBE_CACHE_TTL_FULL_MS as Qe}from"./probe-helper.js";const Ue=600*1e3,qe=60*1e3,j=30*1e3,T=new Set(["claude","acp","agy","cursor","codex"]),Q=3;class Tt{config;name;aibotHandle;aibotConfig;pool;stopped=!1;revokeHandler=new ve;sessionBindings=new Map;deferredMgr;sendCtrl=new Le(ye);bindingStore;globalConfigStore;upgradeTrigger=null;allowlistGate;activeEventStore;cachedRateLimits=null;cachedRateLimitsSampledAtMs=null;cachedCodexContextWindow=null;cachedCodexTokenUsage=null;cachedCodexUsageSampledAtMs=null;cachedAcpContextWindow=null;cachedAcpContextWindowSampledAtMs=null;cachedClaudeRateLimitState=null;cachedProviderQuota=null;cachedProviderQuotaSampledAtMs=null;claudeWorkerStatus=new Map;conversationLog=null;packetLog=null;kiroQuotaTimer=null;eventSessionIndex=new Map;inflightEvents=new Map;restartCount=new Map;probeCache=new Map;sessionScanCache;isRateLimitsCacheFresh(e){if(!Number.isFinite(e))return!1;const n=Number(e);return n>0&&Date.now()-n<=qe}async maybeQueryProviderQuota(){if(this.config.aibot.clientType==="kiro"){if(this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)&&this.cachedProviderQuota)return this.cachedProviderQuota;try{const t=await B();return this.cachedProviderQuota=t,this.cachedProviderQuotaSampledAtMs=Date.now(),u.info(this.name,`[provider-quota] kiro queried: success=${t.success}`+(t.balance?` balance=${t.balance.remaining} ${t.balance.unit}`:"")+(t.planName?` plan=${t.planName}`:"")+(t.error?` error=${t.error}`:"")),t}catch(t){return u.warn(this.name,`[provider-quota] kiro query failed: ${t instanceof Error?t.message:String(t)}`),null}}let e=this.config.providerBaseUrl,n=this.config.providerApiKey;if((!e||!n)&&(this.config.adapterType??"acp")==="claude"){const t=se();e||(e=(t.ANTHROPIC_BASE_URL??"").trim()||void 0),n||(n=(t.ANTHROPIC_API_KEY??"").trim()||(t.ANTHROPIC_AUTH_TOKEN??"").trim()||void 0)}if((!e||!n)&&(this.config.adapterType??"acp")==="codex"){const t=de();!e&&t.baseUrl&&(e=t.baseUrl),!n&&t.apiKey&&(n=t.apiKey)}if(!e||!n)return null;if(this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)&&this.cachedProviderQuota)return this.cachedProviderQuota;try{const t=await z(e,n);return this.cachedProviderQuota=t,this.cachedProviderQuotaSampledAtMs=Date.now(),u.info(this.name,`[provider-quota] queried: provider=${t.provider} success=${t.success}`+(t.tiers.length>0?` tiers=${t.tiers.map(s=>`${s.name}=${s.usedPercent}%`).join(",")}`:"")+(t.balance?` balance=${t.balance.remaining} ${t.balance.unit}`:"")+(t.error?` error=${t.error}`:"")),t}catch(t){return u.warn(this.name,`[provider-quota] query failed: ${t instanceof Error?t.message:String(t)}`),null}}startKiroQuotaTimer(){this.kiroQuotaTimer||(this.kiroQuotaTimer=setInterval(()=>{if(this.stopped){this.stopKiroQuotaTimer();return}this.refreshAndPushKiroQuota().catch(()=>{})},j),u.info(this.name,`[kiro-quota-timer] started (interval=${j}ms)`))}stopKiroQuotaTimer(){this.kiroQuotaTimer&&(clearInterval(this.kiroQuotaTimer),this.kiroQuotaTimer=null,u.info(this.name,"[kiro-quota-timer] stopped"))}async refreshAndPushKiroQuota(){try{const e=await B();this.cachedProviderQuota=e,this.cachedProviderQuotaSampledAtMs=Date.now(),e.success&&this.pushKiroQuotaToBindings(e)}catch{}}pushKiroQuotaToBindings(e){const n=this.providerQuotaToRateLimits(e);for(const[t,s]of this.sessionBindings.entries()){if(!s)continue;const i={provider_quota:e};n&&(i.rate_limits=n);const o=this.cachedAcpContextWindow;o&&("usedPercentage"in o?i.context_window={usedPercentage:o.usedPercentage,remainingPercentage:100-o.usedPercentage}:i.context_window=o),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:s,meta:i})}}getFreshClaudeRateLimitState(){const e=this.cachedClaudeRateLimitState;return e&&this.isRateLimitsCacheFresh(e.sampledAt)?e:null}getFreshCodexGlobalRateLimitCache(){const e=this.cachedRateLimits,n=this.cachedCodexContextWindow,t=this.cachedCodexTokenUsage;return{sampledAt:Math.max(this.cachedRateLimitsSampledAtMs??0,this.cachedCodexUsageSampledAtMs??0)||null,rateLimits:e,contextWindow:n,tokenUsage:t,hasData:!!(e||n||t)}}getStatus(){const e=this.pool?.getStatus()??{total:0,ready:0,busy:0};return{name:this.name,alive:!this.stopped,busy:e.busy>0,exhausted:this.pool?[...this.pool.getAllSlots()].some(n=>n.respawn.exhausted):!1,adapterType:this.config.adapterType??"acp",clientType:this.config.aibot.clientType,pool:e}}async probe(e={}){const n=e.conversation?"full":"static",t=e.conversation?Qe:Be;if(!e.fresh){const r=this.probeCache.get(n);if(r&&Date.now()-r.sampledAt<t)return{...r.result,cached:!0}}const s=this.config.adapterType??"acp",i=this.createAdapter(s,"__probe__"),o=e.conversation&&s==="acp"?()=>this.runAcpConversationProbe(i,e.timeoutMs??1e4):void 0;let d;try{d=await Ie({adapter:i,agentName:this.name,clientType:this.config.aibot.clientType,adapterType:s,providerBaseUrl:this.config.providerBaseUrl??null,opts:e,launchConversationProbe:o})}finally{i.stop().catch(()=>{})}return this.probeCache.set(n,{result:d,sampledAt:d.probed_at}),d}async runAcpConversationProbe(e,n){const t=Date.now(),s="__probe__",i=`probe-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,o=this.config.agent.cwd||V(),d=()=>Date.now()-t;try{await e.start()}catch(g){return{attempted:!0,ok:!1,latency_ms:d(),error:{code:"conversation_failed",message:`start failed: ${g instanceof Error?g.message:String(g)}`}}}if(!e.isAlive())return{attempted:!0,ok:!1,latency_ms:d(),error:{code:"process_not_started",message:"agent process not alive"}};if(e instanceof A)try{await e.bindSession(s,o)}catch{}let r=null;const a=new Promise(g=>{r=m=>{m===i&&g()},e.on("eventDone",r)});e.deliverInboundEvent({event_id:i,session_id:s,content:"ping",msg_id:i}),e.deliverStopEvent(i,s);let c=null;const l=await Promise.race([a.then(()=>!1),new Promise(g=>{c=setTimeout(()=>g(!0),n)})]);return c&&clearTimeout(c),r&&e.removeListener("eventDone",r),l?{attempted:!0,ok:!1,latency_ms:d(),error:{code:"conversation_timeout",message:`no eventDone within ${n}ms`}}:{attempted:!0,ok:!0,latency_ms:d()}}constructor(e,n){this.config=e,this.name=e.name;const t=e.adapterType??"acp";this.aibotConfig={...e.aibot,...t==="claude"?{localActions:e.aibot.localActions??["session_control","claude_interaction_reply","get_session_usage","get_rate_limits","set_model","set_mode","thread_compact","get_agent_global_config"]}:{}},e.eventQueue&&(this.aibotConfig.concurrency={max_concurrent:e.eventQueue.maxConcurrent,max_queued:e.eventQueue.maxQueued,queue_timeout_ms:e.eventQueue.queueTimeoutMs,cancelable_queued:e.eventQueue.cancelableQueued,cancelable_running:e.eventQueue.cancelableRunning}),this.conversationLog=e.logDir?new _e(e.logDir):null,this.packetLog=e.logDir?new fe(e.logDir):null,this.bindingStore=new ke(e.bindingsPath),this.bindingStore.load(),this.globalConfigStore=n??null,this.deferredMgr=new De(this.name),this.allowlistGate=e.allowlistPath?new xe(e.allowlistPath):null,this.activeEventStore=e.activeEventStorePath?new Me(e.activeEventStorePath):null,t==="codex"?this.sessionScanCache=new I(ae,he):t==="claude"?this.sessionScanCache=new I(ce,ge):this.sessionScanCache=new I(le,me)}async start(){(this.config.adapterType??"acp")==="claude"&&(await P().catch(()=>{}),this.maybeQueryProviderQuota().catch(()=>{})),this.config.aibot.clientType==="kiro"&&(this.maybeQueryProviderQuota().catch(()=>{}),this.startKiroQuotaTimer()),(this.config.adapterType??"acp")==="codex"&&this.maybeQueryProviderQuota().catch(()=>{}),await this.connectAibot(),this.sendCtrl.bind(this.aibotHandle);const e=this.config.adapterType??"acp";if(e==="agy")for(const[n,t]of this.bindingStore.entries())t.cwd&&this.aibotHandle.sendUpdateBindingCard({session_id:n,worker_status:"ready",cwd:t.cwd,meta:this.buildAgyToolbarMeta(n)});this.pool=new be({maxPoolSize:this.config.poolMaxSize??20,idleTimeoutMs:this.config.poolIdleTimeoutMs??3e5,eventQueue:this.config.eventQueue},n=>{const t=this.createAdapter(e,n);return t instanceof A&&t.on("acpSessionReady",s=>{this.bindingStore.setAcpSessionId(n,s),this.sessionScanCache.invalidate()}),t},(n,t)=>{this.aibotHandle.sendEventAck({event_id:n,session_id:t,received_at:Date.now()})}),this.pool.setEventStateHandler((n,t,s,i)=>{u.info(this.name,`[queue-debug] send event_state session=${t} event=${n} state=${s} queue_pos=${i?.queue_position??""} queue_total=${i?.queue_total??""}`),this.aibotHandle.sendEventState({event_id:n,session_id:t,state:s,content_preview:i?.content_preview,queue_position:i?.queue_position,queue_total:i?.queue_total,actions:i?.actions,reason:i?.reason,updated_at:Date.now()}),this.pushQueueSnapshotForSession(t),(s==="canceled"||s==="failed")&&this.aibotHandle.sendEventResult({event_id:n,status:s==="canceled"?"canceled":"failed",msg:i?.reason,updated_at:Date.now()})}),this.pool.setQueueComposingHandler((n,t,s)=>{this.aibotHandle.sendSessionActivitySet({session_id:n,kind:"composing",active:t,...t?{ttl_ms:3e4,ref_event_id:s}:{}})}),this.pool.setInternalErrorHandler(n=>{this.handleSessionInternalError(n).catch(t=>{u.error(this.name,`[recovery] handleSessionInternalError failed event=${n.eventId} session=${n.sessionId}: ${t instanceof Error?t.message:String(t)}`)})}),this.pool.setEventStartedHandler((n,t)=>{if(this.config.adapterType!=="claude")return;this.claudeWorkerStatus.set(t,"busy");const s=this.bindingStore.get(t);s?.cwd&&this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"busy",cwd:s.cwd,meta:this.buildClaudeToolbarMeta(t)})}),this.pool.setEventDoneHandler((n,t)=>{if(this.config.adapterType!=="claude")return;this.claudeWorkerStatus.set(t,"ready");const s=this.bindingStore.get(t);s?.cwd&&this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:s.cwd,meta:this.buildClaudeToolbarMeta(t)})}),this.pool.startIdleSweep(),u.info(this.name,`Ready (adapter: ${e}, poolMax: ${this.config.poolMaxSize??20})`)}async stop(){this.stopped=!0,this.pool?.stopIdleSweep(),this.stopKiroQuotaTimer();const e=this.pool?.collectActiveEventIds()??[];e.length>0&&this.activeEventStore&&await this.activeEventStore.save(e);for(const t of e)u.info(this.name,`Canceling active event on shutdown: ${t}`),this.sendEventResultWithCleanup(t,"canceled","process shutting down");e.length>0&&await new Promise(t=>setTimeout(t,100)),this.pool?.clearActiveEventsForShutdown();const n=this.deferredMgr.getAllDeferredEvents();for(const t of n)u.info(this.name,`Failing deferred event on shutdown: ${t.event_id}`),this.sendEventResultWithCleanup(t.event_id,"failed","process shutting down");this.deferredMgr.clearAll(),await this.pool?.stop(),this.aibotHandle?.disconnect(),e.length>0&&this.activeEventStore&&await this.activeEventStore.save([]),this.eventSessionIndex.clear(),this.inflightEvents.clear(),this.restartCount.clear()}createAdapter(e,n){switch(e){case"claude":return this.createClaudeAdapter(n);case"codex":return this.createCodexAdapter(n);case"pi":return this.createPiAdapter(n);case"openhuman":return this.createOpenHumanAdapter(n);case"codewhale":return this.createCodeWhaleAdapter(n);case"cursor":return this.createCursorAdapter(n);case"opencode":return this.createOpenCodeAdapter(n);case"agy":return this.createAgyAdapter(n);default:return this.createAcpAdapter(n)}}createCursorAdapter(e){const n={...this.config.adapterOptions??{}},t=this.bindingStore.get(e);t?.modelId&&(n.model=t.modelId),t?.modeId&&(n.mode=t.modeId);const s={sendStreamChunk:(i,o,d,r,a)=>{this.sendStreamChunkByRuntimeConfig(i,o,d,r,a)},sendEventResult:(i,o,d)=>{this.sendEventResultWithCleanup(i,o,d)},sendEventAck:(i,o)=>{this.aibotHandle.sendEventAck({event_id:i,session_id:o,received_at:Date.now()})},sendRawEventEnvelope:(i,o,d)=>{this.aibotHandle.sendMsg({event_id:i,session_id:o,msg_type:1,content:"[cursor] raw_event",extra:{channel_data:{cursor:{raw_event:d}},agent_api_origin:!0}})},agentInvoke:async(i,o)=>this.platformInvoke(i,o),sendLocalActionResult:(i,o,d,r,a)=>{this.aibotHandle.sendLocalActionResult({action_id:i,status:o,...d!==void 0?{result:d}:{},...r?{error_code:r}:{},...a?{error_msg:a}:{}})}};return new L({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:n},s)}createClaudeAdapter(e){const n={sendReply:(i,o,d,r,a)=>{this.sendReplyByRuntimeConfig(i,o,d,r,a)},sendStreamChunk:(i,o,d,r,a,c,l)=>{this.sendStreamChunkByRuntimeConfig(i,o,d,r,a,c,l)},sendMedia:(i,o,d,r,a,c,l)=>{this.aibotHandle.sendMedia({event_id:i,session_id:o,content:d,msg_type:2,quoted_message_id:a||void 0,client_msg_id:c||void 0,extra:l?{media_caption:r,...l}:{media_caption:r}})},sendEventResult:(i,o,d,r)=>{this.sendEventResultWithCleanup(i,o,d,r)},sendEventAck:(i,o)=>{this.aibotHandle.sendEventAck({event_id:i,session_id:o,received_at:Date.now()})},agentInvoke:async(i,o,d)=>this.platformInvoke(i,o,d),sendLocalActionResult:(i,o,d,r,a)=>{this.aibotHandle.sendLocalActionResult({action_id:i,status:o,...d!==void 0?{result:d}:{},...r?{error_code:r}:{},...a?{error_msg:a}:{}})},sendToolUse:(i,o,d,r)=>{this.sendToolExecutionCard(i,o,R(d,r))},sendToolResult:(i,o,d,r)=>{this.sendToolExecutionCard(i,o,$(d,r))},getWsUrl:()=>this.config.aibot.url,getAgentId:()=>this.config.aibot.agentId,getApiKey:()=>this.config.aibot.apiKey,getActiveEventCount:()=>0,getPendingPermissionCount:()=>0,getPendingElicitationCount:()=>0,sendAgentQuestionCard:(i,o,d)=>{const r=d.questions.map(c=>c.header).join(", "),a=Pe(`[Agent Question] ${d.request_id}`,"agent_question",d);this.aibotHandle.sendText({event_id:i,session_id:o,content:a,msg_type:1,extra:{card_type:"agent_question",summary_text:r}})},sendPermissionCard:i=>{this.aibotHandle.sendMsg({event_id:i.eventId,session_id:i.sessionId,client_msg_id:`perm_${U()}`,msg_type:1,content:i.toolTitle?`Permission required: ${i.toolTitle}`:"Permission request",extra:{channel_data:{execApproval:{approvalId:i.approvalId,approvalSlug:i.toolName},grix:{execApproval:{approval_command_id:i.approvalId,command:i.toolTitle||i.toolName,host:"claude"}}},agent_api_origin:!0}})},sendDirectMessage:i=>{this.aibotHandle.sendMsg({session_id:i.sessionId,msg_type:1,content:i.content,...i.clientMsgId?{client_msg_id:i.clientMsgId}:{},...i.quotedMessageId?{quoted_message_id:i.quotedMessageId}:{}})},onStatusLineUpdated:i=>{(i.rateLimits?.fiveHour||i.rateLimits?.sevenDay)&&(this.cachedClaudeRateLimitState=i);const o=this.bindingStore.get(e);o?.cwd&&this.aibotHandle.sendUpdateBindingCard({session_id:e,worker_status:this.claudeWorkerStatus.get(e)??"ready",cwd:o.cwd,meta:this.buildClaudeToolbarMeta(e)})}},t=this.config.adapterOptions??{},s={...t,sessionRuntimeResolver:()=>{const i=this.bindingStore.get(e);return{cwd:i?.cwd,modeId:i?.modeId??C.fullAuto,modelId:i?.modelId,pluginDir:t.pluginDir,claudeSessionId:i?.claudeSessionId,onSessionIdAssigned:o=>{this.bindingStore.setClaudeSessionId(e,o),this.sessionScanCache.invalidate()}}}};return new y({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:s},n)}createCodexAdapter(e){let n=null;const t={sendEventResult:(o,d,r)=>{this.sendEventResultWithCleanup(o,d,r)},sendEventAck:(o,d)=>this.aibotHandle.sendEventAck({event_id:o,session_id:d,received_at:Date.now()}),sendCodexEvent:o=>{this.shouldDropCodexDisplayEvent(o.event_id,o.codex_method)||(this.aibotHandle.sendCodexEvent(o),this.logCodexEventToConversation(o))},sendCodexEventReliable:async o=>{if(!this.shouldDropCodexDisplayEvent(o.event_id,o.codex_method)){try{await this.aibotHandle.sendCodexEventReliable(o)}catch(d){u.warn("bridge",`[codex] sendCodexEventReliable ACK failed event=${o.event_id}: ${d}`)}u.info("bridge",`[codex] sendCodexEventReliable done event=${o.event_id} method=${o.codex_method}`),this.logCodexEventToConversation(o)}},sendRunError:(o,d,r)=>{this.sendStreamChunkByRuntimeConfig(o,d,`
2
2
 
3
- Error: ${r}`,1,!1)},sendUpdateBindingCard:(o,d,r,a)=>{const c={...a??{}};if(!c.rate_limits&&this.cachedProviderQuota?.success){this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)||this.maybeQueryProviderQuota().catch(()=>{});const l=this.providerQuotaToCodexRateLimits(this.cachedProviderQuota);l&&(c.rate_limits=l.rateLimits,c.rate_limit_primary_percent=l.primaryPercent,c.rate_limit_secondary_percent=l.secondaryPercent,c.rate_limit_primary_window_min=l.primaryWindowMin,c.rate_limit_secondary_window_min=l.secondaryWindowMin)}!c.provider_quota&&this.cachedProviderQuota?.success&&(c.provider_quota=this.cachedProviderQuota),this.aibotHandle.sendUpdateBindingCard({session_id:o,worker_status:d,cwd:r,...Object.keys(c).length>0?{meta:c}:{}})},agentInvoke:async(o,d)=>this.platformInvoke(o,d),sendLocalActionResult:(o,d,r,a,c)=>this.aibotHandle.sendLocalActionResult({action_id:o,status:d,...r!==void 0?{result:r}:{},...a?{error_code:a}:{},...c?{error_msg:c}:{}}),sendSessionActivitySet:(o,d,r,a)=>{this.aibotHandle.sendSessionActivitySet({session_id:o,kind:d,active:r,...a??{}})},getConversationLog:()=>this.conversationLog,onRateLimitsUpdated:o=>{this.cachedRateLimits=o,this.cachedRateLimitsSampledAtMs=Date.now(),this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)||this.maybeQueryProviderQuota().catch(()=>{});const d=this.bindingStore.get(e);if(d?.cwd){const r=this.cachedRateLimitsSampledAtMs,a={rate_limits:{primary:o.primary,secondary:o.secondary,sampledAt:r},rate_limit_primary_percent:o.primary.usedPercent,rate_limit_secondary_percent:o.secondary.usedPercent,rate_limit_primary_window_min:o.primary.windowMinutes,rate_limit_secondary_window_min:o.secondary.windowMinutes,...i?.getEffortMeta()};this.cachedProviderQuota?.success&&(a.provider_quota=this.cachedProviderQuota),this.aibotHandle.sendUpdateBindingCard({session_id:e,worker_status:"ready",cwd:d.cwd,meta:a})}},onContextWindowUpdated:o=>{if(!o)return;this.cachedCodexContextWindow=o,this.cachedCodexUsageSampledAtMs=Date.now();const d=this.bindingStore.get(e);d?.cwd&&this.aibotHandle.sendUpdateBindingCard({session_id:e,worker_status:"ready",cwd:d.cwd,meta:{context_window:o,...i?.getEffortMeta()}})},onTokenUsageUpdated:o=>{o&&(this.cachedCodexTokenUsage=o,this.cachedCodexUsageSampledAtMs=Date.now())}},s=this.config.adapterOptions??{},n=this.globalConfigStore?.get(this.name);return i=new X({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:{...s,model:this.bindingStore.getCodexModelId(e)??s.model,collaborationMode:this.bindingStore.getCodexModeId(e)??s.collaborationMode,reasoningEffort:n?.codexReasoningEffort??s.reasoningEffort,sandboxMode:n?.codexSandboxMode??s.sandboxMode,aibotSessionId:e,bindingStore:this.bindingStore}},t),i}createCodeWhaleAdapter(e){const i={sendEventResult:(s,n,o)=>{this.sendEventResultWithCleanup(s,n,o)},sendEventAck:(s,n)=>this.aibotHandle.sendEventAck({event_id:s,session_id:n,received_at:Date.now()}),sendStreamChunk:(s,n,o,d,r,a)=>{this.sendStreamChunkByRuntimeConfig(s,n,o,d,r,a)},sendUpdateBindingCard:(s,n,o,d)=>this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:n,cwd:o,...d?{meta:d}:{}}),sendLocalActionResult:(s,n,o,d,r)=>this.aibotHandle.sendLocalActionResult({action_id:s,status:n,...o!==void 0?{result:o}:{},...d?{error_code:d}:{},...r?{error_msg:r}:{}}),sendSessionActivitySet:(s,n,o,d)=>{this.aibotHandle.sendSessionActivitySet({session_id:s,kind:n,active:o,...d??{}})},sendToolUse:(s,n,o,d)=>{this.sendToolExecutionCard(s,n,R(o,d))},sendToolResult:(s,n,o,d)=>{this.sendToolExecutionCard(s,n,$(o,d))},agentInvoke:async(s,n)=>this.platformInvoke(s,n),getConversationLog:()=>this.conversationLog},t=this.config.adapterOptions??{};return new q({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:{...t,aibotSessionId:e,bindingStore:this.bindingStore}},i)}createPiAdapter(e){const i={sendEventResult:(s,n,o)=>{this.sendEventResultWithCleanup(s,n,o),u.info("bridge",`[pi] sendEventResult event=${s} status=${n}`)},sendEventAck:(s,n)=>this.aibotHandle.sendEventAck({event_id:s,session_id:n,received_at:Date.now()}),sendUpdateBindingCard:(s,n,o)=>this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:n,cwd:o}),agentInvoke:async(s,n)=>this.platformInvoke(s,n),sendLocalActionResult:(s,n,o,d,r)=>{this.aibotHandle.sendLocalActionResult({action_id:s,status:n,...o!==void 0?{result:o}:{},...d?{error_code:d}:{},...r?{error_msg:r}:{}})},sendSessionActivitySet:(s,n,o,d)=>{this.aibotHandle.sendSessionActivitySet({session_id:s,kind:n,active:o,...d??{}})},sendToolUse:(s,n,o,d)=>{this.sendToolExecutionCard(s,n,R(o,d))},sendToolResult:(s,n,o,d)=>{this.sendToolExecutionCard(s,n,$(o,d))},sendStreamChunk:(s,n,o,d,r,a)=>{this.sendStreamChunkByRuntimeConfig(s,n,o,d,r,a),r&&u.info("bridge",`[pi] sendFinalStreamChunk event=${s} seq=${d}`)},sendFinalStreamChunkReliable:async(s,n,o,d)=>{if(this.finalizeThinking(s,n),this.resolveEventRuntimeConfig(s).responseDelivery==="single_message"&&s){this.bufferStreamChunk(s,n,"",!0);return}try{await this.aibotHandle.sendStreamChunkRequest({event_id:s,session_id:n,delta_content:"",chunk_seq:o,is_finish:!0,...d?{client_msg_id:d}:{}})}catch(a){u.warn("bridge",`[pi] sendFinalStreamChunkReliable ACK failed event=${s}: ${a}`)}u.info("bridge",`[pi] sendFinalStreamChunkReliable done event=${s} seq=${o}`)},sendThinking:(s,n,o)=>{this.sendThinkingByRuntimeConfig(s,n,o)},sendRunError:(s,n,o,d,r)=>{this.sendStreamChunkByRuntimeConfig(s,n,`
3
+ Error: ${r}`,1,!1)},sendUpdateBindingCard:(o,d,r,a)=>{const c={...a??{}};if(!c.rate_limits&&this.cachedProviderQuota?.success){this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)||this.maybeQueryProviderQuota().catch(()=>{});const l=this.providerQuotaToCodexRateLimits(this.cachedProviderQuota);l&&(c.rate_limits=l.rateLimits,c.rate_limit_primary_percent=l.primaryPercent,c.rate_limit_secondary_percent=l.secondaryPercent,c.rate_limit_primary_window_min=l.primaryWindowMin,c.rate_limit_secondary_window_min=l.secondaryWindowMin)}!c.provider_quota&&this.cachedProviderQuota?.success&&(c.provider_quota=this.cachedProviderQuota),this.aibotHandle.sendUpdateBindingCard({session_id:o,worker_status:d,cwd:r,...Object.keys(c).length>0?{meta:c}:{}})},agentInvoke:async(o,d)=>this.platformInvoke(o,d),sendLocalActionResult:(o,d,r,a,c)=>this.aibotHandle.sendLocalActionResult({action_id:o,status:d,...r!==void 0?{result:r}:{},...a?{error_code:a}:{},...c?{error_msg:c}:{}}),sendSessionActivitySet:(o,d,r,a)=>{this.aibotHandle.sendSessionActivitySet({session_id:o,kind:d,active:r,...a??{}})},getConversationLog:()=>this.conversationLog,onRateLimitsUpdated:o=>{this.cachedRateLimits=o,this.cachedRateLimitsSampledAtMs=Date.now(),this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)||this.maybeQueryProviderQuota().catch(()=>{});const d=this.bindingStore.get(e);if(d?.cwd){const r=this.cachedRateLimitsSampledAtMs,a={rate_limits:{primary:o.primary,secondary:o.secondary,sampledAt:r},rate_limit_primary_percent:o.primary.usedPercent,rate_limit_secondary_percent:o.secondary.usedPercent,rate_limit_primary_window_min:o.primary.windowMinutes,rate_limit_secondary_window_min:o.secondary.windowMinutes,...n?.getEffortMeta()};this.cachedProviderQuota?.success&&(a.provider_quota=this.cachedProviderQuota),this.aibotHandle.sendUpdateBindingCard({session_id:e,worker_status:"ready",cwd:d.cwd,meta:a})}},onContextWindowUpdated:o=>{if(!o)return;this.cachedCodexContextWindow=o,this.cachedCodexUsageSampledAtMs=Date.now();const d=this.bindingStore.get(e);d?.cwd&&this.aibotHandle.sendUpdateBindingCard({session_id:e,worker_status:"ready",cwd:d.cwd,meta:{context_window:o,...n?.getEffortMeta()}})},onTokenUsageUpdated:o=>{o&&(this.cachedCodexTokenUsage=o,this.cachedCodexUsageSampledAtMs=Date.now())}},s=this.config.adapterOptions??{},i=this.globalConfigStore?.get(this.name);return n=new X({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:{...s,model:this.bindingStore.getCodexModelId(e)??s.model,collaborationMode:this.bindingStore.getCodexModeId(e)??s.collaborationMode,reasoningEffort:i?.codexReasoningEffort??s.reasoningEffort,sandboxMode:i?.codexSandboxMode??s.sandboxMode,aibotSessionId:e,bindingStore:this.bindingStore}},t),n}createCodeWhaleAdapter(e){const n={sendEventResult:(s,i,o)=>{this.sendEventResultWithCleanup(s,i,o)},sendEventAck:(s,i)=>this.aibotHandle.sendEventAck({event_id:s,session_id:i,received_at:Date.now()}),sendStreamChunk:(s,i,o,d,r,a)=>{this.sendStreamChunkByRuntimeConfig(s,i,o,d,r,a)},sendUpdateBindingCard:(s,i,o,d)=>this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:i,cwd:o,...d?{meta:d}:{}}),sendLocalActionResult:(s,i,o,d,r)=>this.aibotHandle.sendLocalActionResult({action_id:s,status:i,...o!==void 0?{result:o}:{},...d?{error_code:d}:{},...r?{error_msg:r}:{}}),sendSessionActivitySet:(s,i,o,d)=>{this.aibotHandle.sendSessionActivitySet({session_id:s,kind:i,active:o,...d??{}})},sendToolUse:(s,i,o,d)=>{this.sendToolExecutionCard(s,i,R(o,d))},sendToolResult:(s,i,o,d)=>{this.sendToolExecutionCard(s,i,$(o,d))},agentInvoke:async(s,i)=>this.platformInvoke(s,i),getConversationLog:()=>this.conversationLog},t=this.config.adapterOptions??{};return new q({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:{...t,aibotSessionId:e,bindingStore:this.bindingStore}},n)}createPiAdapter(e){const n={sendEventResult:(s,i,o)=>{this.sendEventResultWithCleanup(s,i,o),u.info("bridge",`[pi] sendEventResult event=${s} status=${i}`)},sendEventAck:(s,i)=>this.aibotHandle.sendEventAck({event_id:s,session_id:i,received_at:Date.now()}),sendUpdateBindingCard:(s,i,o)=>this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:i,cwd:o}),agentInvoke:async(s,i)=>this.platformInvoke(s,i),sendLocalActionResult:(s,i,o,d,r)=>{this.aibotHandle.sendLocalActionResult({action_id:s,status:i,...o!==void 0?{result:o}:{},...d?{error_code:d}:{},...r?{error_msg:r}:{}})},sendSessionActivitySet:(s,i,o,d)=>{this.aibotHandle.sendSessionActivitySet({session_id:s,kind:i,active:o,...d??{}})},sendToolUse:(s,i,o,d)=>{this.sendToolExecutionCard(s,i,R(o,d))},sendToolResult:(s,i,o,d)=>{this.sendToolExecutionCard(s,i,$(o,d))},sendStreamChunk:(s,i,o,d,r,a)=>{this.sendStreamChunkByRuntimeConfig(s,i,o,d,r,a),r&&u.info("bridge",`[pi] sendFinalStreamChunk event=${s} seq=${d}`)},sendFinalStreamChunkReliable:async(s,i,o,d)=>{if(this.finalizeThinking(s,i),this.resolveEventRuntimeConfig(s).responseDelivery==="single_message"&&s){this.bufferStreamChunk(s,i,"",!0);return}try{await this.aibotHandle.sendStreamChunkRequest({event_id:s,session_id:i,delta_content:"",chunk_seq:o,is_finish:!0,...d?{client_msg_id:d}:{}})}catch(a){u.warn("bridge",`[pi] sendFinalStreamChunkReliable ACK failed event=${s}: ${a}`)}u.info("bridge",`[pi] sendFinalStreamChunkReliable done event=${s} seq=${o}`)},sendThinking:(s,i,o)=>{this.sendThinkingByRuntimeConfig(s,i,o)},sendRunError:(s,i,o,d,r)=>{this.sendStreamChunkByRuntimeConfig(s,i,`
4
4
 
5
- Error: ${o}`,d,!1,r)}},t=this.config.adapterOptions??{};return new Y({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:{...t,aibotSessionId:e,bindingStore:this.bindingStore}},i)}createOpenHumanAdapter(e){const i={sendStreamChunk:(s,n,o,d,r,a)=>{this.finalizeThinking(s,n),this.sendStreamChunkByRuntimeConfig(s,n,o,d,r,a)},sendFinalStreamChunkReliable:async(s,n,o,d)=>{if(this.finalizeThinking(s,n),this.resolveEventRuntimeConfig(s).responseDelivery==="single_message"&&s){this.bufferStreamChunk(s,n,"",!0);return}try{await this.aibotHandle.sendStreamChunkRequest({event_id:s,session_id:n,delta_content:"",chunk_seq:o,is_finish:!0,...d?{client_msg_id:d}:{}})}catch(a){u.warn("bridge",`[openhuman] sendFinalStreamChunkReliable ACK failed event=${s}: ${a}`)}},sendEventResult:(s,n,o)=>{this.sendEventResultWithCleanup(s,n,o),u.info("bridge",`[openhuman] sendEventResult event=${s} status=${n}`)},sendEventAck:(s,n)=>this.aibotHandle.sendEventAck({event_id:s,session_id:n,received_at:Date.now()}),sendToolUse:(s,n,o,d)=>{this.sendToolExecutionCard(s,n,R(o,d))},sendToolResult:(s,n,o,d)=>{this.sendToolExecutionCard(s,n,$(o,d))},sendThinking:(s,n,o)=>{this.sendThinkingByRuntimeConfig(s,n,o)},sendRunError:(s,n,o)=>{this.sendStreamChunkByRuntimeConfig(s,n,`
5
+ Error: ${o}`,d,!1,r)}},t=this.config.adapterOptions??{};return new Y({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:{...t,aibotSessionId:e,bindingStore:this.bindingStore}},n)}createOpenHumanAdapter(e){const n={sendStreamChunk:(s,i,o,d,r,a)=>{this.finalizeThinking(s,i),this.sendStreamChunkByRuntimeConfig(s,i,o,d,r,a)},sendFinalStreamChunkReliable:async(s,i,o,d)=>{if(this.finalizeThinking(s,i),this.resolveEventRuntimeConfig(s).responseDelivery==="single_message"&&s){this.bufferStreamChunk(s,i,"",!0);return}try{await this.aibotHandle.sendStreamChunkRequest({event_id:s,session_id:i,delta_content:"",chunk_seq:o,is_finish:!0,...d?{client_msg_id:d}:{}})}catch(a){u.warn("bridge",`[openhuman] sendFinalStreamChunkReliable ACK failed event=${s}: ${a}`)}},sendEventResult:(s,i,o)=>{this.sendEventResultWithCleanup(s,i,o),u.info("bridge",`[openhuman] sendEventResult event=${s} status=${i}`)},sendEventAck:(s,i)=>this.aibotHandle.sendEventAck({event_id:s,session_id:i,received_at:Date.now()}),sendToolUse:(s,i,o,d)=>{this.sendToolExecutionCard(s,i,R(o,d))},sendToolResult:(s,i,o,d)=>{this.sendToolExecutionCard(s,i,$(o,d))},sendThinking:(s,i,o)=>{this.sendThinkingByRuntimeConfig(s,i,o)},sendRunError:(s,i,o)=>{this.sendStreamChunkByRuntimeConfig(s,i,`
6
6
 
7
- Error: ${o}`,0,!1)},sendUpdateBindingCard:(s,n,o)=>this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:n,cwd:o}),agentInvoke:async(s,n)=>this.platformInvoke(s,n),sendLocalActionResult:(s,n,o,d,r)=>{this.aibotHandle.sendLocalActionResult({action_id:s,status:n,...o!==void 0?{result:o}:{},...d?{error_code:d}:{},...r?{error_msg:r}:{}})}},t=this.config.adapterOptions??{};return new Z({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:t},i,{port:t.port,host:t.host,workspaceDir:t.workspace_dir,sessionToken:t.session_token,enableSessionBinding:!0,aibotSessionId:e})}createOpenCodeAdapter(e){const i={sendStreamChunk:(s,n,o,d,r,a)=>{this.sendStreamChunkByRuntimeConfig(s,n,o,d,r,a)},sendFinalStreamChunkReliable:async(s,n,o,d)=>{if(this.finalizeThinking(s,n),this.resolveEventRuntimeConfig(s).responseDelivery==="single_message"&&s){this.bufferStreamChunk(s,n,"",!0);return}try{await this.aibotHandle.sendStreamChunkRequest({event_id:s,session_id:n,delta_content:"",chunk_seq:o,is_finish:!0,...d?{client_msg_id:d}:{}})}catch(a){u.warn("bridge",`[opencode] sendFinalStreamChunkReliable ACK failed event=${s}: ${a}`)}},sendEventResult:(s,n,o)=>{this.sendEventResultWithCleanup(s,n,o),u.info("bridge",`[opencode] sendEventResult event=${s} status=${n}`)},sendEventAck:(s,n)=>this.aibotHandle.sendEventAck({event_id:s,session_id:n,received_at:Date.now()}),sendToolUse:(s,n,o,d)=>{this.sendToolExecutionCard(s,n,R(o,d))},sendToolResult:(s,n,o,d)=>{this.sendToolExecutionCard(s,n,$(o,d))},sendThinking:(s,n,o)=>{this.sendThinkingByRuntimeConfig(s,n,o)},sendRunError:(s,n,o)=>{this.sendStreamChunkByRuntimeConfig(s,n,`
7
+ Error: ${o}`,0,!1)},sendUpdateBindingCard:(s,i,o)=>this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:i,cwd:o}),agentInvoke:async(s,i)=>this.platformInvoke(s,i),sendLocalActionResult:(s,i,o,d,r)=>{this.aibotHandle.sendLocalActionResult({action_id:s,status:i,...o!==void 0?{result:o}:{},...d?{error_code:d}:{},...r?{error_msg:r}:{}})}},t=this.config.adapterOptions??{};return new Z({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:t},n,{port:t.port,host:t.host,workspaceDir:t.workspace_dir,sessionToken:t.session_token,enableSessionBinding:!0,aibotSessionId:e})}createOpenCodeAdapter(e){const n={sendStreamChunk:(s,i,o,d,r,a)=>{this.sendStreamChunkByRuntimeConfig(s,i,o,d,r,a)},sendFinalStreamChunkReliable:async(s,i,o,d)=>{if(this.finalizeThinking(s,i),this.resolveEventRuntimeConfig(s).responseDelivery==="single_message"&&s){this.bufferStreamChunk(s,i,"",!0);return}try{await this.aibotHandle.sendStreamChunkRequest({event_id:s,session_id:i,delta_content:"",chunk_seq:o,is_finish:!0,...d?{client_msg_id:d}:{}})}catch(a){u.warn("bridge",`[opencode] sendFinalStreamChunkReliable ACK failed event=${s}: ${a}`)}},sendEventResult:(s,i,o)=>{this.sendEventResultWithCleanup(s,i,o),u.info("bridge",`[opencode] sendEventResult event=${s} status=${i}`)},sendEventAck:(s,i)=>this.aibotHandle.sendEventAck({event_id:s,session_id:i,received_at:Date.now()}),sendToolUse:(s,i,o,d)=>{this.sendToolExecutionCard(s,i,R(o,d))},sendToolResult:(s,i,o,d)=>{this.sendToolExecutionCard(s,i,$(o,d))},sendThinking:(s,i,o)=>{this.sendThinkingByRuntimeConfig(s,i,o)},sendRunError:(s,i,o)=>{this.sendStreamChunkByRuntimeConfig(s,i,`
8
8
 
9
- Error: ${o}`,0,!1)},sendUpdateBindingCard:(s,n,o)=>this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:n,cwd:o}),agentInvoke:async(s,n)=>this.platformInvoke(s,n),sendLocalActionResult:(s,n,o,d,r)=>{this.aibotHandle.sendLocalActionResult({action_id:s,status:n,...o!==void 0?{result:o}:{},...d?{error_code:d}:{},...r?{error_msg:r}:{}})}},t=this.config.adapterOptions??{};return new ee({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:t},i,{port:t.port,hostname:t.hostname,model:t.model,agent:t.agent,permissionPolicy:t.permission_policy,enableSessionBinding:!0,aibotSessionId:e})}createAgyAdapter(e){const i={sendStreamChunk:(s,n,o,d,r)=>{this.sendStreamChunkByRuntimeConfig(s,n,o,d,r)},sendEventResult:(s,n,o)=>{this.sendEventResultWithCleanup(s,n,o)},sendEventAck:(s,n)=>{this.aibotHandle.sendEventAck({event_id:s,session_id:n,received_at:Date.now()})},agentInvoke:async(s,n,o)=>this.platformInvoke(s,n,o),forceCompleteInternalEvent:(s,n)=>{this.pool.eventComplete(s,n),this.pushQueueSnapshotForSession(n)}},t=s=>{const n=this.bindingStore.get(s);return{cwd:n?.cwd,modelId:n?.modelId??this.globalConfigStore?.get(this.name)?.modelId}};return new O({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:this.config.adapterOptions??{}},i,t)}createAcpAdapter(e){const i=this.isAcpRawTransportEnabled(),t={sendStreamChunk:(r,a,c,l,g,m)=>{this.finalizeThinking(r,a),this.sendStreamChunkByRuntimeConfig(r,a,c,l,g,m)},sendFinalStreamChunkReliable:async(r,a,c,l)=>{if(this.finalizeThinking(r,a),this.resolveEventRuntimeConfig(r).responseDelivery==="single_message"&&r){this.bufferStreamChunk(r,a,"",!0);return}try{await this.aibotHandle.sendStreamChunkRequest({event_id:r,session_id:a,delta_content:"",chunk_seq:c,is_finish:!0,...l?{client_msg_id:l}:{}})}catch(m){u.warn("bridge",`[acp] sendFinalStreamChunkReliable ACK failed event=${r}: ${m}`)}u.info("bridge",`[acp] sendFinalStreamChunkReliable done event=${r} seq=${c}`)},sendEventResult:(r,a,c)=>{this.sendEventResultWithCleanup(r,a,c)},sendEventAck:(r,a)=>{this.aibotHandle.sendEventAck({event_id:r,session_id:a,received_at:Date.now()})},agentInvoke:async(r,a)=>this.platformInvoke(r,a),sendLocalActionResult:(r,a,c,l,g)=>{this.aibotHandle.sendLocalActionResult({action_id:r,status:a,...c!==void 0?{result:c}:{},...l?{error_code:l}:{},...g?{error_msg:g}:{}})},sendRawEventEnvelope:(r,a,c)=>{this.sendAcpRawEventEnvelope(r,a,c)},sendToolUse:(r,a,c,l)=>{if(i){this.sendAcpRawEventEnvelope(r,a,{type:"tool_use",payload:{tool_name:c,tool_input:l??""}});return}this.sendToolExecutionCard(r,a,R(c,l))},sendToolResult:(r,a,c,l)=>{this.sendToolExecutionCard(r,a,$(c,l))},sendThinking:(r,a,c)=>{this.sendThinkingByRuntimeConfig(r,a,c)},sendRunError:(r,a,c)=>{this.sendStreamChunkByRuntimeConfig(r,a,`
9
+ Error: ${o}`,0,!1)},sendUpdateBindingCard:(s,i,o)=>this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:i,cwd:o}),agentInvoke:async(s,i)=>this.platformInvoke(s,i),sendLocalActionResult:(s,i,o,d,r)=>{this.aibotHandle.sendLocalActionResult({action_id:s,status:i,...o!==void 0?{result:o}:{},...d?{error_code:d}:{},...r?{error_msg:r}:{}})}},t=this.config.adapterOptions??{};return new ee({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:t},n,{port:t.port,hostname:t.hostname,model:t.model,agent:t.agent,permissionPolicy:t.permission_policy,enableSessionBinding:!0,aibotSessionId:e})}createAgyAdapter(e){const n={sendStreamChunk:(s,i,o,d,r)=>{this.sendStreamChunkByRuntimeConfig(s,i,o,d,r)},sendEventResult:(s,i,o)=>{this.sendEventResultWithCleanup(s,i,o)},sendEventAck:(s,i)=>{this.aibotHandle.sendEventAck({event_id:s,session_id:i,received_at:Date.now()})},agentInvoke:async(s,i,o)=>this.platformInvoke(s,i,o),forceCompleteInternalEvent:(s,i)=>{this.pool.eventComplete(s,i),this.pushQueueSnapshotForSession(i)}},t=s=>{const i=this.bindingStore.get(s);return{cwd:i?.cwd,modelId:i?.modelId}};return new O({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env,options:this.config.adapterOptions??{}},n,t)}createAcpAdapter(e){const n=this.isAcpRawTransportEnabled(),t={sendStreamChunk:(r,a,c,l,g,m)=>{this.finalizeThinking(r,a),this.sendStreamChunkByRuntimeConfig(r,a,c,l,g,m)},sendFinalStreamChunkReliable:async(r,a,c,l)=>{if(this.finalizeThinking(r,a),this.resolveEventRuntimeConfig(r).responseDelivery==="single_message"&&r){this.bufferStreamChunk(r,a,"",!0);return}try{await this.aibotHandle.sendStreamChunkRequest({event_id:r,session_id:a,delta_content:"",chunk_seq:c,is_finish:!0,...l?{client_msg_id:l}:{}})}catch(m){u.warn("bridge",`[acp] sendFinalStreamChunkReliable ACK failed event=${r}: ${m}`)}u.info("bridge",`[acp] sendFinalStreamChunkReliable done event=${r} seq=${c}`)},sendEventResult:(r,a,c)=>{this.sendEventResultWithCleanup(r,a,c)},sendEventAck:(r,a)=>{this.aibotHandle.sendEventAck({event_id:r,session_id:a,received_at:Date.now()})},agentInvoke:async(r,a)=>this.platformInvoke(r,a),sendLocalActionResult:(r,a,c,l,g)=>{this.aibotHandle.sendLocalActionResult({action_id:r,status:a,...c!==void 0?{result:c}:{},...l?{error_code:l}:{},...g?{error_msg:g}:{}})},sendRawEventEnvelope:(r,a,c)=>{this.sendAcpRawEventEnvelope(r,a,c)},sendToolUse:(r,a,c,l)=>{if(n){this.sendAcpRawEventEnvelope(r,a,{type:"tool_use",payload:{tool_name:c,tool_input:l??""}});return}this.sendToolExecutionCard(r,a,R(c,l))},sendToolResult:(r,a,c,l)=>{this.sendToolExecutionCard(r,a,$(c,l))},sendThinking:(r,a,c)=>{this.sendThinkingByRuntimeConfig(r,a,c)},sendRunError:(r,a,c)=>{this.sendStreamChunkByRuntimeConfig(r,a,`
10
10
 
11
- Error: ${c}`,1,!1)},sendPermissionCard:r=>{if(i){this.sendAcpRawEventEnvelope(r.eventId,r.sessionId,{type:"permission_request",payload:{tool_call_id:r.toolCallId,tool_name:r.toolName,tool_title:r.toolTitle,options:r.options}});return}this.aibotHandle.sendMsg({event_id:r.eventId,session_id:r.sessionId,client_msg_id:`perm_${U()}`,msg_type:1,content:r.toolTitle?`Permission required: ${r.toolTitle}`:"Permission request",extra:{channel_data:{execApproval:{approvalId:r.toolCallId,approvalSlug:r.toolName},grix:{execApproval:{approval_command_id:r.toolCallId,command:r.toolTitle||r.toolName,host:"acp"}}},agent_api_origin:!0}})},sendAuthNotification:(r,a)=>{r&&this.aibotHandle.sendMsg({session_id:r,msg_type:1,content:a,extra:{biz_card:{version:1,type:"agent_error",payload:{error:{name:"AuthRequired",message:a}}}}})},sendUpdateBindingCard:(r,a,c,l)=>{const g={...l??{}};if(!g.rate_limits&&this.cachedProviderQuota?.success){this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)||this.maybeQueryProviderQuota().catch(()=>{});const m=this.providerQuotaToRateLimits(this.cachedProviderQuota);m&&(g.rate_limits=m)}!g.provider_quota&&this.cachedProviderQuota?.success&&(g.provider_quota=this.cachedProviderQuota),this.aibotHandle.sendUpdateBindingCard({session_id:r,worker_status:a,cwd:c,...Object.keys(g).length>0?{meta:g}:{}})},onSkillsUpdate:r=>{try{this.aibotHandle.sendSkillsUpdate({skills:r})}catch(a){}},onContextWindowUpdated:r=>{this.cachedAcpContextWindow=r,this.cachedAcpContextWindowSampledAtMs=Date.now();const a="usedPercentage"in r?r.usedPercentage.toFixed(1):(r.used/r.size*100).toFixed(1);u.info(this.name,`[acp] context_window updated: ${a}%`);const c=this.bindingStore.get(e);if(c?.cwd){const l={context_window:"usedPercentage"in r?{usedPercentage:r.usedPercentage,remainingPercentage:100-r.usedPercentage}:r};if(this.config.aibot.clientType==="kiro"&&this.cachedProviderQuota?.success){this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)||this.maybeQueryProviderQuota().catch(()=>{}),l.provider_quota=this.cachedProviderQuota;const g=this.providerQuotaToRateLimits(this.cachedProviderQuota);g&&(l.rate_limits=g)}this.aibotHandle.sendUpdateBindingCard({session_id:e,worker_status:"ready",cwd:c.cwd,meta:l})}},sendMcpFrame:r=>{this.aibotHandle.sendMcpFrame(e,r)}},s=e?this.bindingStore.get(e):void 0,n=this.globalConfigStore?.get(this.name),{initialModel:o,initialMode:d}=Ae({sessionBinding:s,globalDefaults:n,configInitialMode:this.config.acpInitialMode});return(o||d)&&u.info(this.name,`[toolbar] hydrate from binding: session=${e} model=${o??"<none>"} mode=${d??"<none>"}`),new A({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env},t,{acpAuthMethod:this.config.acpAuthMethod,acpInitialMode:d,acpInitialModel:o,acpMcpTools:this.config.acpMcpTools,rawTransport:i,eventResultsPath:this.config.eventResultsPath,approvalMode:this.config.approvalMode,bindingStore:this.config.enableSessionBinding?this.bindingStore:void 0,aibotSessionId:e,autoInjectArgs:this.config.autoInjectArgs,bridgeLog:this.config.logDir?new pe(this.config.logDir,`${this.name}-${e}`):null})}async connectAibot(){const e=new J;if(this.aibotHandle=await e.connect(this.aibotConfig,{aborted:()=>this.stopped,label:this.name,packetLog:this.packetLog}),this.aibotHandle.onEvent(i=>{this.handleAibotEvent(i).catch(t=>{u.error(this.name,`handleAibotEvent failed: ${t}`),this.aibotHandle.sendEventAck({event_id:i.event_id,session_id:i.session_id,received_at:Date.now()});const s=t instanceof Error?t.message:String(t);if(/CWD must be|Bound directory does not exist|Bound path is not a directory/i.test(s)&&i.session_id){this.bindingStore.delete(i.session_id),this.sessionBindings.delete(i.session_id);const o=this.pool.getSlot(i.session_id);o?.adapter instanceof A&&o.adapter.getSessionBindings().delete(i.session_id);const d=this.config.adapterType??"acp",r=this.resolveBindingChannelKey(d);this.aibotHandle.sendMsg({event_id:i.event_id,session_id:i.session_id,msg_type:1,content:s,extra:{channel_data:{[r]:{sessionBinding:{status:"missing",reason:"binding_stale",error_code:h.invalidCwd}}}},quoted_message_id:i.msg_id});return}this.aibotHandle.sendEventResult({event_id:i.event_id,status:"failed",msg:s,updated_at:Date.now()})})}),this.aibotHandle.onStop(i=>{try{this.handleAibotStop(i)}catch(t){u.error(this.name,`handleAibotStop failed: ${t}`)}}),this.aibotHandle.onRevoke(i=>{try{this.handleAibotRevoke(i)}catch(t){u.error(this.name,`handleAibotRevoke failed: ${t}`)}}),this.aibotHandle.onLocalAction(i=>{this.handleAibotLocalAction(i).catch(t=>{u.error(this.name,`handleAibotLocalAction failed: ${t}`)})}),this.aibotHandle.onEventCancel(i=>{u.info(this.name,`recv event_cancel event_id=${i.event_id} session_id=${i.session_id}`),this.handleEventCancel(i).catch(t=>{u.error(this.name,`handleEventCancel failed: ${t}`)})}),this.aibotHandle.onMcpFrame((i,t)=>{const s=this.pool.getSlot(i)?.adapter;s?.deliverMcpFrameToAgent?s.deliverMcpFrameToAgent(t):u.warn(this.name,`mcp_frame: no adapter for session=${i}`)}),this.aibotHandle.onQueueClear(i=>{const t=this.pool.clearQueue(i.session_id);this.aibotHandle.sendQueueClearResult({session_id:i.session_id,canceled_event_ids:t}),this.pushQueueSnapshotForSession(i.session_id)}),this.aibotHandle.onQueueSnapshotQuery(i=>{this.replyQueueSnapshotForSession(i.session_id)}),u.info(this.name,"Connected to aibot"),this.activeEventStore){const i=await this.activeEventStore.drain();if(i.length>0){u.warn(this.name,`Recovering ${i.length} stale event(s) from previous run`);for(const t of i)u.info(this.name,`Failing stale event on startup: ${t}`),this.aibotHandle.sendEventResult({event_id:t,status:"failed",msg:"process restarted, event lost",updated_at:Date.now()})}}this.pushQueueSnapshots(),this.aibotHandle.onReconnected(()=>{this.pushQueueSnapshots()}),this.aibotHandle.onStreamRejected((i,t)=>{this.sendCtrl.markEventRejected(i)})}pushQueueSnapshots(){if(!(!this.config.eventQueue||!this.pool))for(const e of this.pool.getAllSlots())this.pushQueueSnapshotForSession(e.sessionId)}pushQueueSnapshotForSession(e){if(!this.config.eventQueue||!this.pool)return;const i=this.pool.getQueueSnapshot(e);if(!i){u.info(this.name,`[queue-debug] push snapshot session=${e} slot=evicted running=0 queued=0`),this.aibotHandle.sendQueueSnapshot({session_id:e,running:[],running_items:[],queued:[]});return}u.info(this.name,`[queue-debug] push snapshot session=${e} running=${i.running.length} queued=${i.queued.length} running_ids=[${i.running.join(",")}]`),this.aibotHandle.sendQueueSnapshot({session_id:e,running:i.running,running_items:i.running_items.map(t=>({event_id:t.event_id,...t.content_preview?{content_preview:t.content_preview}:{},...t.title?{title:t.title}:{},...t.summary?{summary:t.summary}:{},actions:[{type:"stop"}]})),queued:i.queued.map(t=>({event_id:t.event_id,position:t.position,...t.content_preview?{content_preview:t.content_preview}:{},...t.title?{title:t.title}:{},...t.summary?{summary:t.summary}:{},actions:[{type:"cancel"}]}))})}replyQueueSnapshotForSession(e){if(!this.config.eventQueue||!this.pool)return;const i=this.pool.getQueueSnapshot(e);if(i){this.aibotHandle.sendQueueSnapshot({session_id:e,running:i.running,running_items:i.running_items.map(t=>({event_id:t.event_id,...t.content_preview?{content_preview:t.content_preview}:{},...t.title?{title:t.title}:{},...t.summary?{summary:t.summary}:{},actions:[{type:"stop"}]})),queued:i.queued.map(t=>({event_id:t.event_id,position:t.position,...t.content_preview?{content_preview:t.content_preview}:{},...t.title?{title:t.title}:{},...t.summary?{summary:t.summary}:{},actions:[{type:"cancel"}]}))});return}this.aibotHandle.sendQueueSnapshot({session_id:e,running:[],running_items:[],queued:[]})}async platformInvoke(e,i,t){return e==="file_link"?$e(i):this.aibotHandle.agentInvoke(e,i,t)}sendReplyByRuntimeConfig(e,i,t,s,n){this.indexEventSession(e,i),this.sendCtrl.sendReply(e,i,t,s,n),t&&this.conversationLog?.logOutbound?.(i,e,"reply",t)}sendStreamChunkByRuntimeConfig(e,i,t,s,n,o,d){this.indexEventSession(e,i),this.sendCtrl.sendStreamChunk(e,i,t,s,n,o,d),(t||n)&&this.conversationLog?.logOutbound?.(i,e,n?"stream_chunk_finish":"stream_chunk",t)}sendRunErrorAsChunk(e,i,t){this.sendStreamChunkByRuntimeConfig(e,i,`
11
+ Error: ${c}`,1,!1)},sendPermissionCard:r=>{if(n){this.sendAcpRawEventEnvelope(r.eventId,r.sessionId,{type:"permission_request",payload:{tool_call_id:r.toolCallId,tool_name:r.toolName,tool_title:r.toolTitle,options:r.options}});return}this.aibotHandle.sendMsg({event_id:r.eventId,session_id:r.sessionId,client_msg_id:`perm_${U()}`,msg_type:1,content:r.toolTitle?`Permission required: ${r.toolTitle}`:"Permission request",extra:{channel_data:{execApproval:{approvalId:r.toolCallId,approvalSlug:r.toolName},grix:{execApproval:{approval_command_id:r.toolCallId,command:r.toolTitle||r.toolName,host:"acp"}}},agent_api_origin:!0}})},sendAuthNotification:(r,a)=>{r&&this.aibotHandle.sendMsg({session_id:r,msg_type:1,content:a,extra:{biz_card:{version:1,type:"agent_error",payload:{error:{name:"AuthRequired",message:a}}}}})},sendUpdateBindingCard:(r,a,c,l)=>{const g={...l??{}};if(!g.rate_limits&&this.cachedProviderQuota?.success){this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)||this.maybeQueryProviderQuota().catch(()=>{});const m=this.providerQuotaToRateLimits(this.cachedProviderQuota);m&&(g.rate_limits=m)}!g.provider_quota&&this.cachedProviderQuota?.success&&(g.provider_quota=this.cachedProviderQuota),this.aibotHandle.sendUpdateBindingCard({session_id:r,worker_status:a,cwd:c,...Object.keys(g).length>0?{meta:g}:{}})},onSkillsUpdate:r=>{try{this.aibotHandle.sendSkillsUpdate({skills:r})}catch(a){}},onContextWindowUpdated:r=>{this.cachedAcpContextWindow=r,this.cachedAcpContextWindowSampledAtMs=Date.now();const a="usedPercentage"in r?r.usedPercentage.toFixed(1):(r.used/r.size*100).toFixed(1);u.info(this.name,`[acp] context_window updated: ${a}%`);const c=this.bindingStore.get(e);if(c?.cwd){const l={context_window:"usedPercentage"in r?{usedPercentage:r.usedPercentage,remainingPercentage:100-r.usedPercentage}:r};if(this.config.aibot.clientType==="kiro"&&this.cachedProviderQuota?.success){this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)||this.maybeQueryProviderQuota().catch(()=>{}),l.provider_quota=this.cachedProviderQuota;const g=this.providerQuotaToRateLimits(this.cachedProviderQuota);g&&(l.rate_limits=g)}this.aibotHandle.sendUpdateBindingCard({session_id:e,worker_status:"ready",cwd:c.cwd,meta:l})}},sendMcpFrame:r=>{this.aibotHandle.sendMcpFrame(e,r)}},s=e?this.bindingStore.get(e):void 0,i=this.globalConfigStore?.get(this.name),{initialModel:o,initialMode:d}=Ae({sessionBinding:s,globalDefaults:i,configInitialMode:this.config.acpInitialMode});return(o||d)&&u.info(this.name,`[toolbar] hydrate from binding: session=${e} model=${o??"<none>"} mode=${d??"<none>"}`),new A({command:this.config.agent.command,args:this.config.agent.args,env:this.config.agent.env},t,{acpAuthMethod:this.config.acpAuthMethod,acpInitialMode:d,acpInitialModel:o,acpMcpTools:this.config.acpMcpTools,rawTransport:n,eventResultsPath:this.config.eventResultsPath,approvalMode:this.config.approvalMode,bindingStore:this.config.enableSessionBinding?this.bindingStore:void 0,aibotSessionId:e,autoInjectArgs:this.config.autoInjectArgs,bridgeLog:this.config.logDir?new pe(this.config.logDir,`${this.name}-${e}`):null})}async connectAibot(){const e=new J;if(this.aibotHandle=await e.connect(this.aibotConfig,{aborted:()=>this.stopped,label:this.name,packetLog:this.packetLog}),this.aibotHandle.onEvent(n=>{this.handleAibotEvent(n).catch(t=>{u.error(this.name,`handleAibotEvent failed: ${t}`),this.aibotHandle.sendEventAck({event_id:n.event_id,session_id:n.session_id,received_at:Date.now()});const s=t instanceof Error?t.message:String(t);if(/CWD must be|Bound directory does not exist|Bound path is not a directory/i.test(s)&&n.session_id){this.bindingStore.delete(n.session_id),this.sessionBindings.delete(n.session_id);const o=this.pool.getSlot(n.session_id);o?.adapter instanceof A&&o.adapter.getSessionBindings().delete(n.session_id);const d=this.config.adapterType??"acp",r=this.resolveBindingChannelKey(d);this.aibotHandle.sendMsg({event_id:n.event_id,session_id:n.session_id,msg_type:1,content:s,extra:{channel_data:{[r]:{sessionBinding:{status:"missing",reason:"binding_stale",error_code:h.invalidCwd}}}},quoted_message_id:n.msg_id});return}this.aibotHandle.sendEventResult({event_id:n.event_id,status:"failed",msg:s,updated_at:Date.now()})})}),this.aibotHandle.onStop(n=>{try{this.handleAibotStop(n)}catch(t){u.error(this.name,`handleAibotStop failed: ${t}`)}}),this.aibotHandle.onRevoke(n=>{try{this.handleAibotRevoke(n)}catch(t){u.error(this.name,`handleAibotRevoke failed: ${t}`)}}),this.aibotHandle.onLocalAction(n=>{this.handleAibotLocalAction(n).catch(t=>{u.error(this.name,`handleAibotLocalAction failed: ${t}`)})}),this.aibotHandle.onEventCancel(n=>{u.info(this.name,`recv event_cancel event_id=${n.event_id} session_id=${n.session_id}`),this.handleEventCancel(n).catch(t=>{u.error(this.name,`handleEventCancel failed: ${t}`)})}),this.aibotHandle.onMcpFrame((n,t)=>{const s=this.pool.getSlot(n)?.adapter;s?.deliverMcpFrameToAgent?s.deliverMcpFrameToAgent(t):u.warn(this.name,`mcp_frame: no adapter for session=${n}`)}),this.aibotHandle.onQueueClear(n=>{const t=this.pool.clearQueue(n.session_id);this.aibotHandle.sendQueueClearResult({session_id:n.session_id,canceled_event_ids:t}),this.pushQueueSnapshotForSession(n.session_id)}),this.aibotHandle.onQueueSnapshotQuery(n=>{this.replyQueueSnapshotForSession(n.session_id)}),u.info(this.name,"Connected to aibot"),this.activeEventStore){const n=await this.activeEventStore.drain();if(n.length>0){u.warn(this.name,`Recovering ${n.length} stale event(s) from previous run`);for(const t of n)u.info(this.name,`Failing stale event on startup: ${t}`),this.aibotHandle.sendEventResult({event_id:t,status:"failed",msg:"process restarted, event lost",updated_at:Date.now()})}}this.pushQueueSnapshots(),this.aibotHandle.onReconnected(()=>{this.pushQueueSnapshots()}),this.aibotHandle.onStreamRejected((n,t)=>{this.sendCtrl.markEventRejected(n)})}pushQueueSnapshots(){if(!(!this.config.eventQueue||!this.pool))for(const e of this.pool.getAllSlots())this.pushQueueSnapshotForSession(e.sessionId)}pushQueueSnapshotForSession(e){if(!this.config.eventQueue||!this.pool)return;const n=this.pool.getQueueSnapshot(e);if(!n){u.info(this.name,`[queue-debug] push snapshot session=${e} slot=evicted running=0 queued=0`),this.aibotHandle.sendQueueSnapshot({session_id:e,running:[],running_items:[],queued:[]});return}u.info(this.name,`[queue-debug] push snapshot session=${e} running=${n.running.length} queued=${n.queued.length} running_ids=[${n.running.join(",")}]`),this.aibotHandle.sendQueueSnapshot({session_id:e,running:n.running,running_items:n.running_items.map(t=>({event_id:t.event_id,...t.content_preview?{content_preview:t.content_preview}:{},...t.title?{title:t.title}:{},...t.summary?{summary:t.summary}:{},actions:[{type:"stop"}]})),queued:n.queued.map(t=>({event_id:t.event_id,position:t.position,...t.content_preview?{content_preview:t.content_preview}:{},...t.title?{title:t.title}:{},...t.summary?{summary:t.summary}:{},actions:[{type:"cancel"}]}))})}replyQueueSnapshotForSession(e){if(!this.config.eventQueue||!this.pool)return;const n=this.pool.getQueueSnapshot(e);if(n){this.aibotHandle.sendQueueSnapshot({session_id:e,running:n.running,running_items:n.running_items.map(t=>({event_id:t.event_id,...t.content_preview?{content_preview:t.content_preview}:{},...t.title?{title:t.title}:{},...t.summary?{summary:t.summary}:{},actions:[{type:"stop"}]})),queued:n.queued.map(t=>({event_id:t.event_id,position:t.position,...t.content_preview?{content_preview:t.content_preview}:{},...t.title?{title:t.title}:{},...t.summary?{summary:t.summary}:{},actions:[{type:"cancel"}]}))});return}this.aibotHandle.sendQueueSnapshot({session_id:e,running:[],running_items:[],queued:[]})}async platformInvoke(e,n,t){return e==="file_link"?$e(n):this.aibotHandle.agentInvoke(e,n,t)}sendReplyByRuntimeConfig(e,n,t,s,i){this.indexEventSession(e,n),this.sendCtrl.sendReply(e,n,t,s,i),t&&this.conversationLog?.logOutbound?.(n,e,"reply",t)}sendStreamChunkByRuntimeConfig(e,n,t,s,i,o,d){this.indexEventSession(e,n),this.sendCtrl.sendStreamChunk(e,n,t,s,i,o,d),(t||i)&&this.conversationLog?.logOutbound?.(n,e,i?"stream_chunk_finish":"stream_chunk",t)}sendRunErrorAsChunk(e,n,t){this.sendStreamChunkByRuntimeConfig(e,n,`
12
12
 
13
- Error: ${t}`,1,!1)}sendEventResultWithCleanup(e,i,t,s){this.sendCtrl.sendEventResult(e,i,t,s);const n=this.eventSessionIndex.get(e);n&&(this.pool.eventComplete(e,n),this.pushQueueSnapshotForSession(n),this.conversationLog?.logResult?.(n,e,i,t),this.eventSessionIndex.delete(e)),this.inflightEvents.delete(e),this.restartCount.delete(e),i==="responded"&&(this.cachedProviderQuotaSampledAtMs=null,this.maybeQueryProviderQuota().catch(()=>{}))}async handleSessionInternalError(e){const{eventId:i,sessionId:t,errorMsg:s}=e;if(this.stopped)return;const n=this.inflightEvents.get(i);if(!n){u.warn(this.name,`[recovery] no inflight event for internalError event=${i} session=${t}; surface failure directly`),this.sendRunErrorAsChunk(i,t,s),this.sendEventResultWithCleanup(i,"failed",s,"agent_stop_failure");return}const o=(this.restartCount.get(i)??0)+1;this.restartCount.set(i,o);const d=this.config.adapterType??"acp";if(o>Q){u.error(this.name,`[recovery] adapter=${d} session=${t} event=${i} restart=${o}/${Q} outcome=give-up err=${s}`),this.sendRunErrorAsChunk(i,t,s),this.sendEventResultWithCleanup(i,"failed",s,"agent_stop_failure");return}u.info(this.name,`[recovery] adapter=${d} session=${t} event=${i} restart=${o}/${Q} outcome=restarting err=${s}`);const r=this.pool.drainQueuedForSession(t);r.length>0&&u.info(this.name,`[recovery] session=${t} preserved ${r.length} queued sibling event(s) across restart`);try{await this.pool.removeSlot(t)}catch(l){u.warn(this.name,`[recovery] removeSlot failed session=${t}: ${l instanceof Error?l.message:String(l)}`)}if(this.stopped)return;const a=this.resolveRecoveryPrompt(d,n),c={...n,content:a};try{await this.pool.deliverInboundEvent(c)}catch(l){u.error(this.name,`[recovery] redeliver failed event=${i} session=${t}: ${l instanceof Error?l.message:String(l)}`),this.sendEventResultWithCleanup(i,"failed",l instanceof Error?l.message:String(l));return}for(const l of r){if(this.stopped)break;try{await this.pool.deliverInboundEvent(l)}catch(g){u.error(this.name,`[recovery] sibling redeliver failed event=${l.event_id} session=${t}: ${g instanceof Error?g.message:String(g)}`),this.sendEventResultWithCleanup(l.event_id,"failed",g instanceof Error?g.message:String(g))}}}resolveRecoveryPrompt(e,i){return e==="acp"?"continue":i.content}sendThinkingByRuntimeConfig(e,i,t){this.sendCtrl.sendThinking(e,i,t)}bufferStreamChunk(e,i,t,s,n){this.sendCtrl.bufferOnly(e,i,t,s,n)}flushBufferedStreamText(e){}resolveEventRuntimeConfig(e){return this.sendCtrl.resolveEventRuntimeConfig(e)}captureEventRuntimeConfig(e){this.sendCtrl.captureEventRuntimeConfig(e),this.indexEventSession(e.event_id,e.session_id),e.event_id&&this.inflightEvents.set(e.event_id,e)}indexEventSession(e,i){!e||!i||this.eventSessionIndex.set(e,i)}shouldDropToolDisplayEvent(e){return this.sendCtrl.shouldDropToolDisplayEvent(e)}shouldDropThinkingDisplayEvent(e){return this.sendCtrl.shouldDropThinkingDisplayEvent(e)}shouldDropCodexDisplayEvent(e,i){return this.sendCtrl.shouldDropCodexDisplayEvent(e,i)}logCodexEventToConversation(e){if(!this.conversationLog||e.codex_method!=="item/agentMessage/delta")return;const t=e.codex_payload?.params?.delta;if(!t)return;const s=this.eventSessionIndex.get(e.event_id)??e.session_id;this.conversationLog.append(s,{ts:Date.now(),dir:"outbound",event_id:e.event_id,kind:"codex_delta",text_len:t.length,content:t})}isAcpRawTransportEnabled(){return(this.config.adapterOptions??{}).raw_transport===!0}shouldDropAcpRawDisplayEvent(e,i){return this.sendCtrl.shouldDropAcpRawDisplayEvent(e,i)}sendAcpRawEventEnvelope(e,i,t){this.shouldDropAcpRawDisplayEvent(e,t.type)||this.aibotHandle.sendMsg({event_id:e,session_id:i,msg_type:1,content:this.buildAcpRawEventFallbackText(t),extra:{channel_data:{acp:{raw_event:t}},agent_api_origin:!0}})}buildAcpRawEventFallbackText(e){const i=String(e.type??"").trim();if(!i)return"[acp] event";switch(i){case"permission_request":return`Permission required: ${String(e.payload?.tool_title??e.payload?.tool_name??"permission request")}`;case"tool_use":return`[tool] ${String(e.payload?.tool_name??"tool")}`;case"tool_result":return"[tool result]";case"thinking":return"[thinking]";case"error":return`[error] ${String(e.payload?.message??"agent error")}`;case"result":return"[result]";default:return`[acp] ${i}`}}sendToolExecutionCard(e,i,t,s){this.sendCtrl.sendToolExecutionCard(e,i,t,s)}async handleAibotEvent(e){if(this.stopped){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",msg:"agent shutting down",updated_at:Date.now()});return}this.logInboundConversation(e);const i=this.config.adapterType??"acp";if(this.allowlistGate&&e.sender_id&&!await this.allowlistGate.checkAccess(e.sender_id)){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"responded",code:"access_denied",msg:`sender ${e.sender_id} is not authorized`,updated_at:Date.now()});return}const t=He(e.extra);if(t.error){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",code:"connector_config_invalid",msg:t.error,updated_at:Date.now()});return}const s=t.patch,n=we(e);if(n){if(n.verb===_.exec){await this.handleSessionControlCommand(n,e);return}if(n.verb===_.listSessions){await this.handleListSessionsTextCommand(e);return}if(i==="claude"){await this.handleSessionControlCommand(n,e);return}if(i==="codex"&&n.verb===_.open){await this.handleCodexSessionControlOpen(n,e);return}if(i==="pi"&&n.verb===_.open){await this.handlePiSessionControlOpen(n,e);return}if(i==="pi"&&n.verb===_.restart){await this.handlePiSessionControlRestart(e);return}if((i==="openhuman"||i==="opencode")&&n.verb===_.open){await this.handleOpenHumanSessionControlOpen(n,e);return}if(i==="codewhale"&&n.verb===_.open){await this.handleCodeWhaleSessionControlOpen(n,e);return}if(this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),n.verb===_.open){const r=n.args.trim();if(!r){this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",code:h.cwdRequired,msg:"cwd is required",updated_at:Date.now()});return}try{const a=M.resolve(r);if(!(await H(a)).isDirectory()){this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",code:h.invalidCwd,msg:`Path is not a directory: ${a}`,updated_at:Date.now()});return}}catch(a){const c=String(a?.code??""),l=c==="ENOENT"?`Directory does not exist: ${M.resolve(r)}`:c==="EACCES"||c==="EPERM"?"Directory is not accessible":`Invalid path: ${r}`;this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",code:h.invalidCwd,msg:l,updated_at:Date.now()});return}}if(n.verb===_.open){const r=this.bindingStore.get(e.session_id);if(r?.cwd)try{await H(r.cwd)}catch{u.info("bridge",`Stale binding detected for session ${e.session_id}: ${r.cwd} no longer exists, clearing`),this.bindingStore.delete(e.session_id),this.sessionBindings.delete(e.session_id);const a=this.pool.getSlot(e.session_id);a?.adapter instanceof A&&a.adapter.getSessionBindings().delete(e.session_id)}}if(await this.handleSessionControlForPool(n,e),i==="acp"&&n.verb===_.stop){const a=this.bindingStore.get(e.session_id)?.cwd??"";await this.pool.removeSlot(e.session_id).catch(()=>{}),this.sessionBindings.delete(e.session_id),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"responded",msg:`Session worker stopped for ${a}`,updated_at:Date.now()});return}if(F(n,e,this.sessionControlCtx(e.session_id),{...this.sessionControlSenders(),sendEventAck:()=>{}}),n.verb===_.open)if(i==="pi")await this.deferredMgr.replayDeferredPiEvents(e.session_id,this.deferredCallbacks());else if(i==="openhuman")await this.deferredMgr.replayDeferredOpenHumanEvents(e.session_id,this.deferredCallbacks());else if(i==="opencode")await this.deferredMgr.replayDeferredOpenCodeEvents(e.session_id,this.deferredCallbacks());else if(i==="agy"){await this.deferredMgr.replayDeferredAgyEvents(e.session_id,this.deferredCallbacks());const r=this.bindingStore.get(e.session_id)?.cwd??"";r&&this.aibotHandle.sendUpdateBindingCard({session_id:e.session_id,worker_status:"ready",cwd:r,meta:this.buildAgyToolbarMeta(e.session_id)})}else await this.deferredMgr.replayDeferredAcpEvents(e.session_id,this.deferredCallbacks());return}if(e.mirror_mode==="record_only"){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"responded",updated_at:Date.now()});return}if(i==="claude"){if(this.isStaleClaudeEvent(e)){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",code:"event_stale",msg:"event is stale and will not be processed",updated_at:Date.now()});return}if(!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferClaudeEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(i);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}}if(i==="codex"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferCodexEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(i);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(i==="cursor"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferCursorEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(i);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(i==="codewhale"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferCodeWhaleEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(i);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(i==="opencode"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferOpenCodeEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(i);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(i==="pi"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferPiEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(i);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(i==="openhuman"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferOpenHumanEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(i);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(i==="agy"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferAgyEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(i);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(i==="acp"&&!this.bindingStore.get(e.session_id)?.cwd){u.info(this.name,`[acp] binding missing session_id=${e.session_id} event_id=${e.event_id}`),this.deferredMgr.deferAcpEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(i);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if((this.config.adapterType??"acp")==="acp"){const a=String(e.content??"").trim().match(/^\/(\S+)\s*(.*)/);if(a){const[,c,l]=a,m=this.pool.getSlot(e.session_id)?.adapter;if(m?.execCommand&&(m.getSupportedCommands?.()??[]).some(f=>f.name===c||f.name===`/${c}`)){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()});try{const f=await m.execCommand(c,l.trim(),e.session_id);f.status==="options"&&f.data&&this.handleExecCommandOptions(e.session_id,c,f.data),this.aibotHandle.sendEventResult({event_id:e.event_id,status:f.status==="failed"?"failed":"responded",msg:f.message,updated_at:Date.now()})}catch(f){this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",msg:f instanceof Error?f.message:String(f),updated_at:Date.now()})}return}}}const d=this.buildInboundEvent(e,s);try{this.captureEventRuntimeConfig(d),await this.pool.deliverInboundEvent(d)}catch(r){if(this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.inflightEvents.delete(e.event_id),this.restartCount.delete(e.event_id),this.eventSessionIndex.delete(e.event_id),r instanceof Se)this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",msg:r.message,updated_at:Date.now()});else throw r}}async handleCodexSessionControlOpen(e,i){const t=i.session_id,s=e.args.trim(),n=()=>this.aibotHandle.sendEventAck({event_id:i.event_id,session_id:t,received_at:Date.now()}),o=(d,r)=>this.aibotHandle.sendEventResult({event_id:i.event_id,status:d,...r,updated_at:Date.now()});if(!s){n(),o("failed",{msg:"Usage: /grix open <working-directory>",code:h.cwdRequired});return}try{const d=await this.resolveCwdForBinding(s),r=this.bindingStore.get(t);if(r?.cwd){let a=!1;try{if(await this.resolveCwdForBinding(r.cwd)===d){n(),o("responded",{msg:`Session already bound to ${r.cwd}`});return}}catch{a=!0,u.info("bridge",`Stale codex binding for session ${t}: ${r.cwd} no longer exists, allowing rebind`)}if(!a){n(),o("failed",{msg:`Session already bound to ${r.cwd}. Rebinding is not allowed.`,code:h.rebindForbidden});return}}this.bindingStore.set(t,d),this.sessionBindings.set(t,d),this.deferredMgr.sendCodexDeferredReplayComposing(t,this.deferredCallbacks()),await this.bindSessionForPool(t,d),await this.deferredMgr.replayDeferredCodexEvents(t,this.deferredCallbacks(),{announceComposing:!1}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d}),n(),o("responded",{msg:`Session bound to ${d}`})}catch(d){n(),o("failed",{code:h.invalidCwd,msg:d instanceof Error?d.message:String(d)})}}async handleSessionControlForPool(e,i){e.verb===_.open&&await this.bindSessionForPool(i.session_id,e.args.trim())}async replayDeferredEventsForSession(e){const i=this.config.adapterType??"acp";i==="pi"?await this.deferredMgr.replayDeferredPiEvents(e,this.deferredCallbacks()):i==="openhuman"?await this.deferredMgr.replayDeferredOpenHumanEvents(e,this.deferredCallbacks()):i==="opencode"?await this.deferredMgr.replayDeferredOpenCodeEvents(e,this.deferredCallbacks()):i==="agy"?await this.deferredMgr.replayDeferredAgyEvents(e,this.deferredCallbacks()):i==="claude"||i==="codex"||i==="cursor"||i==="codewhale"?await this.deferredMgr.replayDeferredClaudeEvents(e,this.deferredCallbacks()):await this.deferredMgr.replayDeferredAcpEvents(e,this.deferredCallbacks())}async handleSessionControlLocalActionForPool(e){if(String(e.action_type??"")!==S.sessionControl)return;const i=e.params??{};if(String(i.verb??"").trim().toLowerCase()!==_.open)return;const s=String(i.session_id??"").trim(),n=String(i.cwd??"").trim();!s||!n||await this.bindSessionForPool(s,n)}async handleCodexSessionControlLocalActionOpen(e){const i=e.params??{},t=String(i.session_id??"").trim(),s=String(i.cwd??"").trim(),n=String(i.agent_session_id??"").trim(),o=(d,r,a,c)=>{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:d,...r!==void 0?{result:r}:{},...a?{error_code:a}:{},...c?{error_msg:c}:{}})};if(!t||!s){o("failed",void 0,h.cwdRequired,"session cwd is required");return}try{const d=await this.resolveCwdForBinding(s);this.ensureImportedAgentSession(n,d);const r=this.bindingStore.get(t);if(r?.cwd){let a=!1;try{const c=await this.resolveCwdForBinding(r.cwd);if(c===d){this.setResolvedAgentSessionId(t,n),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,c)});return}}catch{a=!0,u.info("bridge",`Stale codex binding for session ${t}: ${r.cwd} no longer exists, allowing rebind`)}if(!a){o("failed",void 0,h.rebindForbidden,`Session already bound to ${r.cwd}`);return}}this.bindingStore.set(t,d),this.setResolvedAgentSessionId(t,n),this.sessionBindings.set(t,d),this.deferredMgr.sendCodexDeferredReplayComposing(t,this.deferredCallbacks()),await this.bindSessionForPool(t,d),await this.deferredMgr.replayDeferredCodexEvents(t,this.deferredCallbacks(),{announceComposing:!1}),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,d)})}catch(d){o("failed",void 0,d?.sessionControlErrorCode??h.invalidCwd,d instanceof Error?d.message:String(d))}}async handleCursorSessionControlLocalActionOpen(e){const i=e.params??{},t=String(i.session_id??"").trim(),s=String(i.cwd??"").trim(),n=String(i.agent_session_id??"").trim(),o=(d,r,a,c)=>{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:d,...r!==void 0?{result:r}:{},...a?{error_code:a}:{},...c?{error_msg:c}:{}})};if(!t||!s){o("failed",void 0,h.cwdRequired,"session cwd is required");return}try{const d=await this.resolveCwdForBinding(s);this.ensureImportedAgentSession(n,d);const r=this.bindingStore.get(t);if(r?.cwd){let a=!1;try{const c=await this.resolveCwdForBinding(r.cwd);if(c===d){this.setResolvedAgentSessionId(t,n),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,c)});return}}catch{a=!0,u.info("bridge",`Stale cursor binding for session ${t}: ${r.cwd} no longer exists, allowing rebind`)}if(!a){o("failed",void 0,h.rebindForbidden,`Session already bound to ${r.cwd}`);return}}this.bindingStore.set(t,d),this.setResolvedAgentSessionId(t,n),this.sessionBindings.set(t,d),this.deferredMgr.sendCursorDeferredReplayComposing(t,this.deferredCallbacks()),await this.bindSessionForPool(t,d),await this.deferredMgr.replayDeferredCursorEvents(t,this.deferredCallbacks(),{announceComposing:!1}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d}),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,d)})}catch(d){o("failed",void 0,d?.sessionControlErrorCode??h.invalidCwd,d instanceof Error?d.message:String(d))}}async handlePiSessionControlOpen(e,i){const t=i.session_id,s=e.args.trim(),n=()=>this.aibotHandle.sendEventAck({event_id:i.event_id,session_id:t,received_at:Date.now()}),o=(d,r)=>this.aibotHandle.sendEventResult({event_id:i.event_id,status:d,...r,updated_at:Date.now()});if(!s){n(),o("failed",{msg:"Usage: /grix open <working-directory>",code:h.cwdRequired});return}try{const d=await this.resolveCwdForBinding(s),r=this.bindingStore.get(t);if(r?.cwd){let a=!1;try{if(await this.resolveCwdForBinding(r.cwd)===d){n(),o("responded",{msg:`Session already bound to ${r.cwd}`}),this.aibotHandle.sendMsg({event_id:i.event_id,session_id:t,msg_type:1,content:`\u2705 Session already bound to \`${r.cwd}\``,quoted_message_id:i.msg_id});return}}catch{a=!0,u.info("bridge",`Stale pi binding for session ${t}: ${r.cwd} no longer exists, allowing rebind`)}if(!a){n(),o("failed",{msg:`Session already bound to ${r.cwd}. Rebinding is not allowed.`,code:h.rebindForbidden});return}}this.bindingStore.set(t,d),this.sessionBindings.set(t,d),await this.deferredMgr.replayDeferredPiEvents(t,this.deferredCallbacks()),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d}),n(),o("responded",{msg:`Session bound to ${d}`}),this.aibotHandle.sendMsg({event_id:i.event_id,session_id:t,msg_type:1,content:`\u2705 Working directory bound: \`${d}\``,quoted_message_id:i.msg_id})}catch(d){n(),o("failed",{code:h.invalidCwd,msg:d instanceof Error?d.message:String(d)})}}async handlePiSessionControlRestart(e){const i=e.session_id,t=()=>this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:i,received_at:Date.now()}),s=(d,r)=>this.aibotHandle.sendEventResult({event_id:e.event_id,status:d,...r,updated_at:Date.now()}),o=this.bindingStore.get(i)?.cwd??"";if(!o){t(),s("failed",{msg:"session binding was not found",code:h.bindingMissing});return}t(),await this.pool.removeSlot(i).catch(()=>{}),this.aibotHandle.sendUpdateBindingCard({session_id:i,worker_status:"ready",cwd:o}),s("responded",{msg:`Session worker restarted for ${o}`})}async handlePiSessionControlRestartLocalAction(e){const i=e.params??{},t=String(i.session_id??"").trim(),n=(t?this.bindingStore.get(t):void 0)?.cwd??"",o=(d,r,a,c)=>{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:d,...r!==void 0?{result:r}:{},...a?{error_code:a}:{},...c?{error_msg:c}:{}})};if(!t){o("failed",void 0,"session_id_required","session_id is required for restart");return}if(!n){o("failed",void 0,h.bindingMissing,"Session binding missing. Open a workspace first.");return}await this.pool.removeSlot(t).catch(()=>{}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:n}),o("ok",{outcome:"restarted",binding:{aibotSessionId:t,cwd:n,workerStatus:"ready"}})}async bindSessionForPool(e,i){const t=String(i??"").trim();if(!e||!t)return;const s=this.pool.getOrCreateSlot(e);if(!s||(s.startPromise&&await s.startPromise,!s.adapter))return;const n=s.adapter instanceof A?s.adapter:null;if(!n?.hasSessionBinding)return;const o=await this.resolveCwdForBinding(t);n.announceDeferredComposing(e),await n.bindSession(e,o),this.sessionBindings.set(e,o),this.sessionScanCache.invalidate(),n.replayDeferredEvents(e)}deferredCallbacks(){return{captureEventRuntimeConfig:e=>this.captureEventRuntimeConfig(e),deliverInboundEvent:e=>this.pool.deliverInboundEvent(e),sendEventResult:(e,i,t)=>this.sendEventResultWithCleanup(e,i,t),sendSessionComposing:(e,i,t)=>{const s={};i&&(s.ttl_ms=t?.ttlMs??3e4,t?.activity&&(s.activity=t.activity)),this.aibotHandle.sendSessionActivitySet({session_id:e,kind:"composing",active:i,...s})}}}async handleOpenHumanSessionControlOpen(e,i){const t=()=>this.aibotHandle.sendEventAck({event_id:i.event_id,session_id:i.session_id,received_at:Date.now()});try{await this.resolveCwdForBinding(e.args.trim()),await this.handleSessionControlForPool(e,i),F(e,i,this.sessionControlCtx(i.session_id),this.sessionControlSenders()),this.deferredMgr.hasDeferred("openhuman",i.session_id)&&await this.deferredMgr.replayDeferredOpenHumanEvents(i.session_id,this.deferredCallbacks()),this.deferredMgr.hasDeferred("opencode",i.session_id)&&await this.deferredMgr.replayDeferredOpenCodeEvents(i.session_id,this.deferredCallbacks())}catch(s){t(),this.aibotHandle.sendEventResult({event_id:i.event_id,status:"failed",code:s?.cwdErrorCode,msg:s instanceof Error?s.message:String(s),updated_at:Date.now()})}}async handleCodeWhaleSessionControlOpen(e,i){const t=i.session_id,s=e.args.trim(),n=()=>this.aibotHandle.sendEventAck({event_id:i.event_id,session_id:t,received_at:Date.now()}),o=(d,r)=>this.aibotHandle.sendEventResult({event_id:i.event_id,status:d,...r,updated_at:Date.now()});if(!s){n(),o("failed",{msg:"Usage: /grix open <working-directory>",code:h.cwdRequired});return}try{const d=await this.resolveCwdForBinding(s),r=this.bindingStore.get(t);if(r?.cwd){if(await this.resolveCwdForBinding(r.cwd)!==d){n(),o("failed",{msg:`Session already bound to ${r.cwd}. Rebinding is not allowed.`,code:h.rebindForbidden});return}n(),o("responded",{msg:`Session already bound to ${r.cwd}`});return}this.bindingStore.set(t,d),this.sessionBindings.set(t,d),await this.bindSessionForPool(t,d),await this.deferredMgr.replayDeferredCodeWhaleEvents(t,this.deferredCallbacks(),{announceComposing:!1}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d}),n(),o("responded",{msg:`Session bound to ${d}`})}catch(d){n(),o("failed",{code:h.invalidCwd,msg:d instanceof Error?d.message:String(d)})}}async handleCodeWhaleSessionControlLocalActionOpen(e){const i=e.params??{},t=String(i.session_id??"").trim(),s=String(i.cwd??"").trim(),n=String(i.agent_session_id??"").trim(),o=(d,r,a,c)=>{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:d,...r!==void 0?{result:r}:{},...a?{error_code:a}:{},...c?{error_msg:c}:{}})};if(!t||!s){o("failed",void 0,h.cwdRequired,"session cwd is required");return}try{const d=await this.resolveCwdForBinding(s);this.ensureImportedAgentSession(n,d);const r=this.bindingStore.get(t);if(r?.cwd){const a=await this.resolveCwdForBinding(r.cwd);if(a!==d){o("failed",void 0,h.rebindForbidden,`Session already bound to ${r.cwd}`);return}this.setResolvedAgentSessionId(t,n),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,a)});return}this.bindingStore.set(t,d),this.setResolvedAgentSessionId(t,n),this.sessionBindings.set(t,d),await this.bindSessionForPool(t,d),await this.deferredMgr.replayDeferredCodeWhaleEvents(t,this.deferredCallbacks(),{announceComposing:!1}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d}),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,d)})}catch(d){o("failed",void 0,d?.sessionControlErrorCode??h.invalidCwd,d instanceof Error?d.message:String(d))}}normalizeClaudeModeId(e){return String(e??"").trim().toLowerCase()===C.approval?C.approval:C.fullAuto}handleExecCommandOptions(e,i,t){const s=Array.isArray(t?.options)?t.options:[];if(s.length===0)return;const o=this.bindingStore.get(e)?.cwd??"",d={};if(i==="model"){d.available_models=s.map(a=>({id:a.id,displayName:a.label}));const r=s.find(a=>a.current);r&&(d.model_id=r.id)}else if(i==="mode"){d.available_modes=s.map(a=>({id:a.id,displayName:a.label}));const r=s.find(a=>a.current);r&&(d.mode_id=r.id)}else d[`${i}_options`]=s;this.aibotHandle.sendUpdateBindingCard({session_id:e,worker_status:"ready",cwd:o,meta:d})}buildAgyToolbarMeta(e){const i=this.bindingStore.get(e);if(!i)return;const t=te(this.config.agent.command),s=t.length>0?t[0].id:"";let n=i.modelId||this.globalConfigStore?.get(this.name)?.modelId||s;t.length>0&&!t.some(d=>d.id===n)&&(n=s);const o=ie();return{model_id:n,currentModelId:n,available_models:t.map(d=>({id:d.id,displayName:d.displayName})),...o.plan!==void 0&&{plan:o.plan},...o.quota_exhausted!==void 0&&{quota_exhausted:o.quota_exhausted},...o.quota_reset_at!==void 0&&{quota_reset_at:o.quota_reset_at},...o.available_credits!==void 0&&{available_credits:o.available_credits}}}async handleAgySetModel(e,i){const t=e.params??{},s=String(t.model_id??t.modelId??"").trim();if(!i){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"session_id_required",error_msg:"session_id is required for set_model"});return}if(!s){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"set_model_invalid",error_msg:"model_id is required"});return}const n=this.bindingStore.get(i);if(!n?.cwd){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.bindingMissing,error_msg:"session binding was not found"});return}n.modelId!==s&&(this.bindingStore.setModelId(i,s),this.globalConfigStore?.set(this.name,{modelId:s})),this.aibotHandle.sendUpdateBindingCard({session_id:i,worker_status:"ready",cwd:n.cwd,meta:this.buildAgyToolbarMeta(i)}),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"model_set",model_id:s,binding:{cwd:n.cwd,model_id:s}}})}buildClaudeToolbarMeta(e){const i=this.bindingStore.get(e);if(!i)return;const t=P(),s=t.length>0?t[0].id:"";let n=i.modelId||this.globalConfigStore?.get(this.name)?.modelId||s;t.length>0&&!t.some(l=>l.id===n)&&(n=s);const o=this.normalizeClaudeModeId(i.modeId),d={model_id:n,mode_id:o,currentModelId:n,currentModeId:o,available_models:t.map(l=>({id:l.id,displayName:l.displayName}))},r=this.pool.getSlot(e),a=r?.adapter instanceof y?r.adapter.getSessionState():null;(a?.rateLimits?.fiveHour||a?.rateLimits?.sevenDay)&&(this.cachedClaudeRateLimitState=a);const c=a??this.getFreshClaudeRateLimitState();if(c){const l=c.rateLimits;l&&(l.fiveHour||l.sevenDay)&&(d.rate_limits={...l.fiveHour?{fiveHour:l.fiveHour}:{},...l.sevenDay?{sevenDay:l.sevenDay}:{},sampledAt:c.sampledAt||Date.now()})}if(a){const l=a.contextWindow;l.usedPercentage!=null&&(d.context_window={usedPercentage:l.usedPercentage,remainingPercentage:l.remainingPercentage})}if(!d.rate_limits&&this.cachedProviderQuota?.success){this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)||this.maybeQueryProviderQuota().catch(()=>{});const l=this.providerQuotaToRateLimits(this.cachedProviderQuota);l&&(d.rate_limits=l),u.info(this.name,`[toolbar-meta] provider quota fallback: hasCached=${!!this.cachedProviderQuota} fromProvider=${JSON.stringify(l)}`)}return d.rate_limits&&u.info(this.name,`[toolbar-meta] rate_limits included: ${JSON.stringify(d.rate_limits)}`),d}providerQuotaToCodexRateLimits(e){const i=e.tiers.find(n=>n.name==="five_hour"),t=e.tiers.find(n=>n.name==="weekly_limit"),s=n=>n||null;if(i||t){const n={credits:{hasCredits:!0,unlimited:!1,balance:null},sampledAt:this.cachedProviderQuotaSampledAtMs??Date.now()};let o=0,d=0,r=0,a=0;return i&&(n.primary={usedPercent:i.usedPercent,windowMinutes:300,resetsAt:s(i.resetsAt)},o=i.usedPercent,d=300),t&&(n.secondary={usedPercent:t.usedPercent,windowMinutes:10080,resetsAt:s(t.resetsAt)},r=t.usedPercent,a=10080),{rateLimits:n,primaryPercent:o,secondaryPercent:r,primaryWindowMin:d,secondaryWindowMin:a}}if(e.balance){const n=e.balance,o=n.total&&n.total>0?Math.min(100,(n.used??n.total-n.remaining)/n.total*100):0;return{rateLimits:{primary:{usedPercent:o,windowMinutes:0,resetsAt:null},secondary:{usedPercent:0,windowMinutes:0,resetsAt:null},credits:{hasCredits:!0,unlimited:!1,balance:n.remaining},sampledAt:this.cachedProviderQuotaSampledAtMs??Date.now()},primaryPercent:o,secondaryPercent:0,primaryWindowMin:0,secondaryWindowMin:0}}return null}providerQuotaToRateLimits(e){const i=n=>{if(!n)return 0;const o=Date.parse(n);return Number.isFinite(o)?Math.floor(o/1e3):0},t=e.tiers.find(n=>n.name==="five_hour"),s=e.tiers.find(n=>n.name==="weekly_limit");if(t||s)return{...t?{fiveHour:{usedPercentage:t.usedPercent,resetsAt:i(t.resetsAt)}}:{},...s?{sevenDay:{usedPercentage:s.usedPercent,resetsAt:i(s.resetsAt)}}:{},sampledAt:this.cachedProviderQuotaSampledAtMs??Date.now()};if(e.balance){const n=e.balance;return{credit:{remaining:n.remaining,total:n.total,used:n.used,unit:n.unit,resetsAt:n.resetsAt?i(n.resetsAt):0},planName:e.planName,sampledAt:this.cachedProviderQuotaSampledAtMs??Date.now()}}return null}async resolveCwdForBinding(e){const i=String(e??"").trim();if(process.platform!=="win32"&&(/^[a-zA-Z]:[\\/]/.test(i)||/^\\\\/.test(i))){const n=new Error(`Specified path is not valid on this host: ${i}`);throw n.cwdErrorCode=h.invalidCwd,n}const t=M.resolve(i);let s;try{s=await H(t)}catch(n){const o=String(n?.code??"");if(o==="ENOENT"){const d=new Error(`Specified path does not exist: ${t}`);throw d.cwdErrorCode=h.invalidCwd,d}if(o==="EACCES"||o==="EPERM"){const d=new Error("Specified path is not accessible.");throw d.cwdErrorCode=h.invalidCwd,d}throw n}if(!s.isDirectory()){const n=new Error("Specified path is not a directory.");throw n.cwdErrorCode=h.invalidCwd,n}try{return await G(t)}catch{return t}}getClaudeWorkerStatus(e){const i=this.pool.getSlot(e);return i?i.state==="starting"?"starting":i.state==="stopped"?"stopped":i.adapter.getStatus().busy?"busy":"ready":"stopped"}async ensureSlotStarted(e,i=6e4){const t=this.pool.getOrCreateSlot(e);if(!t)throw new Error("Failed to allocate session slot");t.startPromise&&(u.info(this.name,`ensureSlotStarted: awaiting startPromise for session=${e}`),await Promise.race([t.startPromise,new Promise((s,n)=>setTimeout(()=>n(new Error(`ensureSlotStarted timeout (${i}ms) session=${e}`)),i))]),u.info(this.name,`ensureSlotStarted: startPromise resolved for session=${e}`))}resolveAgentSessionId(e){switch(this.config.adapterType??"acp"){case"claude":return e.claudeSessionId;case"codex":return e.codexThreadId;case"pi":return e.piSessionPath;case"codewhale":return e.codewhaleThreadId;default:return e.acpSessionId}}providerKeyForAdapter(){const e=this.config.adapterType??"acp";switch(e){case"claude":case"codex":case"pi":case"codewhale":return e;default:return"acp"}}setResolvedAgentSessionId(e,i){const t=String(e??"").trim(),s=String(i??"").trim();if(!t||!s)return;switch(this.config.adapterType??"acp"){case"claude":this.bindingStore.setClaudeSessionId(t,s);break;case"codex":this.bindingStore.setCodexThreadId(t,s);break;case"pi":this.bindingStore.setPiSessionPath(t,s);break;case"codewhale":this.bindingStore.setCodeWhaleThreadId(t,s);break;default:this.bindingStore.setAcpSessionId(t,s);break}this.sessionScanCache.invalidate()}normalizePathForCompare(e){const i=String(e??"").trim();if(!i)return"";const t=M.resolve(i);return process.platform==="win32"?t.toLowerCase():t}ensureImportedAgentSession(e,i){const t=String(e??"").trim();if(!t)return;const s=this.normalizePathForCompare(i);let n="";const o=this.config.adapterType??"acp";if(o==="codex"?n=this.sessionScanCache.get().find(a=>a.threadId===t)?.cwd??"":o==="claude"?n=this.sessionScanCache.get().find(a=>a.sessionId===t)?.cwd??"":o==="acp"&&(n=this.sessionScanCache.get().find(a=>a.sessionId===t)?.cwd??""),!n){for(const[,r]of this.bindingStore.entries())if(this.resolveAgentSessionId(r)===t){n=r.cwd??"";break}}if(!n){const r=new Error(`agent session not found: ${t}`);throw r.sessionControlErrorCode=h.invalidAgentSession,r}const d=this.normalizePathForCompare(n);if(d&&s&&d!==s){const r=new Error(`agent session cwd mismatch: expected ${i}, got ${n}`);throw r.sessionControlErrorCode=h.invalidAgentSession,r}}buildOpenedBindingResult(e,i,t="ready"){const s=this.bindingStore.get(e),n=s?String(this.resolveAgentSessionId(s)??"").trim():"",o={aibotSessionId:e,providerKey:this.providerKeyForAdapter(),cwd:i,workerStatus:t};return n&&(o.bindingId=n,o.agentSessionId=n),o}hasDiskScanner(){const e=this.config.adapterType??"acp";return e==="codex"||e==="claude"||e==="acp"}async handleListSessionsTextCommand(e){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()});const i=this.config.adapterType??"acp",t=new Map;for(const r of this.pool.getAllSlots())t.set(r.sessionId,r);const s=Array.from(this.bindingStore.entries()),n=new Map;for(const[r,a]of s){const c=this.resolveAgentSessionId(a);if(c){const l=t.get(r),g=l?l.adapter.getStatus().busy?"busy":"ready":"inactive";n.set(c,{aibotSessionId:r,workerStatus:g})}}const o=[],d=new Set;if(i==="codex"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.threadId);const c=n.get(a.threadId);a.title&&o.push(` Title: ${a.title}`),o.push(` Agent: ${a.threadId}`),c&&o.push(` AIBot: ${c.aibotSessionId}`),o.push(` CWD: ${a.cwd||"-"}`),o.push(` State: ${c?.workerStatus??(a.archived?"archived":"inactive")}`),o.push(` Created: ${new Date(a.createdAt).toISOString()}`),o.push(` Updated: ${new Date(a.updatedAt).toISOString()}`),o.push("---")}}else if(i==="claude"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.sessionId);const c=n.get(a.sessionId);a.title&&o.push(` Title: ${a.title}`),o.push(` Agent: ${a.sessionId}`),c&&o.push(` AIBot: ${c.aibotSessionId}`),o.push(` CWD: ${a.cwd||"-"}`),o.push(` State: ${c?.workerStatus??"inactive"}`),o.push(` Updated: ${new Date(a.updatedAt).toISOString()}`),o.push("---")}}else if(i==="acp"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.sessionId);const c=n.get(a.sessionId);a.title&&o.push(` Title: ${a.title}`),o.push(` Agent: ${a.sessionId}`),c&&o.push(` AIBot: ${c.aibotSessionId}`),o.push(` CWD: ${a.cwd||"-"}`),o.push(` State: ${c?.workerStatus??"inactive"}`),o.push(` Updated: ${new Date(a.updatedAt).toISOString()}`),o.push("---")}}for(const[r,a]of s){const c=this.resolveAgentSessionId(a);if(c&&d.has(c))continue;const l=t.get(r),g=l?l.adapter.getStatus().busy?"busy":"ready":"closed";o.push(` Title: ${c?c.slice(0,8)+"\u2026":r.slice(0,8)+"\u2026"}`),o.push(` AIBot: ${r}`),c&&o.push(` Agent: ${c}`),o.push(` CWD: ${a.cwd??"-"}`),o.push(` State: ${g}`),o.push(` Updated: ${new Date(a.updatedAt).toISOString()}`),o.push("---")}if(o.length===0){this.aibotHandle.sendEventResult({event_id:e.event_id,status:"responded",msg:"No sessions found.",updated_at:Date.now()});return}this.aibotHandle.sendEventResult({event_id:e.event_id,status:"responded",msg:`Sessions (${o.filter(r=>r==="---").length}):
13
+ Error: ${t}`,1,!1)}sendEventResultWithCleanup(e,n,t,s){this.sendCtrl.sendEventResult(e,n,t,s);const i=this.eventSessionIndex.get(e);i&&(this.pool.eventComplete(e,i),this.pushQueueSnapshotForSession(i),this.conversationLog?.logResult?.(i,e,n,t),this.eventSessionIndex.delete(e)),this.inflightEvents.delete(e),this.restartCount.delete(e),n==="responded"&&(this.cachedProviderQuotaSampledAtMs=null,this.maybeQueryProviderQuota().catch(()=>{}))}async handleSessionInternalError(e){const{eventId:n,sessionId:t,errorMsg:s}=e;if(this.stopped)return;const i=this.inflightEvents.get(n);if(!i){u.warn(this.name,`[recovery] no inflight event for internalError event=${n} session=${t}; surface failure directly`),this.sendRunErrorAsChunk(n,t,s),this.sendEventResultWithCleanup(n,"failed",s,"agent_stop_failure");return}const o=(this.restartCount.get(n)??0)+1;this.restartCount.set(n,o);const d=this.config.adapterType??"acp";if(o>Q){u.error(this.name,`[recovery] adapter=${d} session=${t} event=${n} restart=${o}/${Q} outcome=give-up err=${s}`),this.sendRunErrorAsChunk(n,t,s),this.sendEventResultWithCleanup(n,"failed",s,"agent_stop_failure");return}u.info(this.name,`[recovery] adapter=${d} session=${t} event=${n} restart=${o}/${Q} outcome=restarting err=${s}`);const r=this.pool.drainQueuedForSession(t);r.length>0&&u.info(this.name,`[recovery] session=${t} preserved ${r.length} queued sibling event(s) across restart`);try{await this.pool.removeSlot(t)}catch(l){u.warn(this.name,`[recovery] removeSlot failed session=${t}: ${l instanceof Error?l.message:String(l)}`)}if(this.stopped)return;const a=this.resolveRecoveryPrompt(d,i),c={...i,content:a};try{await this.pool.deliverInboundEvent(c)}catch(l){u.error(this.name,`[recovery] redeliver failed event=${n} session=${t}: ${l instanceof Error?l.message:String(l)}`),this.sendEventResultWithCleanup(n,"failed",l instanceof Error?l.message:String(l));return}for(const l of r){if(this.stopped)break;try{await this.pool.deliverInboundEvent(l)}catch(g){u.error(this.name,`[recovery] sibling redeliver failed event=${l.event_id} session=${t}: ${g instanceof Error?g.message:String(g)}`),this.sendEventResultWithCleanup(l.event_id,"failed",g instanceof Error?g.message:String(g))}}}resolveRecoveryPrompt(e,n){return e==="acp"?"continue":n.content}sendThinkingByRuntimeConfig(e,n,t){this.sendCtrl.sendThinking(e,n,t)}bufferStreamChunk(e,n,t,s,i){this.sendCtrl.bufferOnly(e,n,t,s,i)}flushBufferedStreamText(e){}resolveEventRuntimeConfig(e){return this.sendCtrl.resolveEventRuntimeConfig(e)}captureEventRuntimeConfig(e){this.sendCtrl.captureEventRuntimeConfig(e),this.indexEventSession(e.event_id,e.session_id),e.event_id&&this.inflightEvents.set(e.event_id,e)}indexEventSession(e,n){!e||!n||this.eventSessionIndex.set(e,n)}shouldDropToolDisplayEvent(e){return this.sendCtrl.shouldDropToolDisplayEvent(e)}shouldDropThinkingDisplayEvent(e){return this.sendCtrl.shouldDropThinkingDisplayEvent(e)}shouldDropCodexDisplayEvent(e,n){return this.sendCtrl.shouldDropCodexDisplayEvent(e,n)}logCodexEventToConversation(e){if(!this.conversationLog||e.codex_method!=="item/agentMessage/delta")return;const t=e.codex_payload?.params?.delta;if(!t)return;const s=this.eventSessionIndex.get(e.event_id)??e.session_id;this.conversationLog.append(s,{ts:Date.now(),dir:"outbound",event_id:e.event_id,kind:"codex_delta",text_len:t.length,content:t})}isAcpRawTransportEnabled(){return(this.config.adapterOptions??{}).raw_transport===!0}shouldDropAcpRawDisplayEvent(e,n){return this.sendCtrl.shouldDropAcpRawDisplayEvent(e,n)}sendAcpRawEventEnvelope(e,n,t){this.shouldDropAcpRawDisplayEvent(e,t.type)||this.aibotHandle.sendMsg({event_id:e,session_id:n,msg_type:1,content:this.buildAcpRawEventFallbackText(t),extra:{channel_data:{acp:{raw_event:t}},agent_api_origin:!0}})}buildAcpRawEventFallbackText(e){const n=String(e.type??"").trim();if(!n)return"[acp] event";switch(n){case"permission_request":return`Permission required: ${String(e.payload?.tool_title??e.payload?.tool_name??"permission request")}`;case"tool_use":return`[tool] ${String(e.payload?.tool_name??"tool")}`;case"tool_result":return"[tool result]";case"thinking":return"[thinking]";case"error":return`[error] ${String(e.payload?.message??"agent error")}`;case"result":return"[result]";default:return`[acp] ${n}`}}sendToolExecutionCard(e,n,t,s){this.sendCtrl.sendToolExecutionCard(e,n,t,s)}async handleAibotEvent(e){if(this.stopped){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",msg:"agent shutting down",updated_at:Date.now()});return}this.logInboundConversation(e);const n=this.config.adapterType??"acp";if(this.allowlistGate&&e.sender_id&&!await this.allowlistGate.checkAccess(e.sender_id)){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"responded",code:"access_denied",msg:`sender ${e.sender_id} is not authorized`,updated_at:Date.now()});return}const t=He(e.extra);if(t.error){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",code:"connector_config_invalid",msg:t.error,updated_at:Date.now()});return}const s=t.patch,i=Se(e);if(i){if(i.verb===_.exec){await this.handleSessionControlCommand(i,e);return}if(i.verb===_.listSessions){await this.handleListSessionsTextCommand(e);return}if(n==="claude"){await this.handleSessionControlCommand(i,e);return}if(n==="codex"&&i.verb===_.open){await this.handleCodexSessionControlOpen(i,e);return}if(n==="pi"&&i.verb===_.open){await this.handlePiSessionControlOpen(i,e);return}if(n==="pi"&&i.verb===_.restart){await this.handlePiSessionControlRestart(e);return}if((n==="openhuman"||n==="opencode")&&i.verb===_.open){await this.handleOpenHumanSessionControlOpen(i,e);return}if(n==="codewhale"&&i.verb===_.open){await this.handleCodeWhaleSessionControlOpen(i,e);return}if(this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),i.verb===_.open){const r=i.args.trim();if(!r){this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",code:h.cwdRequired,msg:"cwd is required",updated_at:Date.now()});return}try{const a=M.resolve(r);if(!(await H(a)).isDirectory()){this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",code:h.invalidCwd,msg:`Path is not a directory: ${a}`,updated_at:Date.now()});return}}catch(a){const c=String(a?.code??""),l=c==="ENOENT"?`Directory does not exist: ${M.resolve(r)}`:c==="EACCES"||c==="EPERM"?"Directory is not accessible":`Invalid path: ${r}`;this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",code:h.invalidCwd,msg:l,updated_at:Date.now()});return}}if(i.verb===_.open){const r=this.bindingStore.get(e.session_id);if(r?.cwd)try{await H(r.cwd)}catch{u.info("bridge",`Stale binding detected for session ${e.session_id}: ${r.cwd} no longer exists, clearing`),this.bindingStore.delete(e.session_id),this.sessionBindings.delete(e.session_id);const a=this.pool.getSlot(e.session_id);a?.adapter instanceof A&&a.adapter.getSessionBindings().delete(e.session_id)}}if(await this.handleSessionControlForPool(i,e),n==="acp"&&i.verb===_.stop){const a=this.bindingStore.get(e.session_id)?.cwd??"";await this.pool.removeSlot(e.session_id).catch(()=>{}),this.sessionBindings.delete(e.session_id),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"responded",msg:`Session worker stopped for ${a}`,updated_at:Date.now()});return}if(F(i,e,this.sessionControlCtx(e.session_id),{...this.sessionControlSenders(),sendEventAck:()=>{}}),i.verb===_.open)if(n==="pi")await this.deferredMgr.replayDeferredPiEvents(e.session_id,this.deferredCallbacks());else if(n==="openhuman")await this.deferredMgr.replayDeferredOpenHumanEvents(e.session_id,this.deferredCallbacks());else if(n==="opencode")await this.deferredMgr.replayDeferredOpenCodeEvents(e.session_id,this.deferredCallbacks());else if(n==="agy"){await this.deferredMgr.replayDeferredAgyEvents(e.session_id,this.deferredCallbacks());const r=this.bindingStore.get(e.session_id)?.cwd??"";r&&this.aibotHandle.sendUpdateBindingCard({session_id:e.session_id,worker_status:"ready",cwd:r,meta:this.buildAgyToolbarMeta(e.session_id)})}else await this.deferredMgr.replayDeferredAcpEvents(e.session_id,this.deferredCallbacks());return}if(e.mirror_mode==="record_only"){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"responded",updated_at:Date.now()});return}if(n==="claude"){if(this.isStaleClaudeEvent(e)){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",code:"event_stale",msg:"event is stale and will not be processed",updated_at:Date.now()});return}if(!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferClaudeEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(n);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}}if(n==="codex"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferCodexEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(n);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(n==="cursor"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferCursorEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(n);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(n==="codewhale"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferCodeWhaleEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(n);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(n==="opencode"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferOpenCodeEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(n);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(n==="pi"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferPiEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(n);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(n==="openhuman"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferOpenHumanEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(n);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(n==="agy"&&!this.bindingStore.get(e.session_id)?.cwd){this.deferredMgr.deferAgyEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(n);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if(n==="acp"&&!this.bindingStore.get(e.session_id)?.cwd){u.info(this.name,`[acp] binding missing session_id=${e.session_id} event_id=${e.event_id}`),this.deferredMgr.deferAcpEvent(String(e.session_id??"").trim(),this.buildInboundEvent(e,s));const a=this.resolveBindingChannelKey(n);this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.aibotHandle.sendMsg({event_id:e.event_id,session_id:e.session_id,msg_type:1,content:"Session binding missing.",extra:{channel_data:{[a]:{sessionBinding:{status:"missing",reason:"binding_missing",error_code:h.bindingMissing}}}},quoted_message_id:e.msg_id});return}if((this.config.adapterType??"acp")==="acp"){const a=String(e.content??"").trim().match(/^\/(\S+)\s*(.*)/);if(a){const[,c,l]=a,m=this.pool.getSlot(e.session_id)?.adapter;if(m?.execCommand&&(m.getSupportedCommands?.()??[]).some(f=>f.name===c||f.name===`/${c}`)){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()});try{const f=await m.execCommand(c,l.trim(),e.session_id);f.status==="options"&&f.data&&this.handleExecCommandOptions(e.session_id,c,f.data),this.aibotHandle.sendEventResult({event_id:e.event_id,status:f.status==="failed"?"failed":"responded",msg:f.message,updated_at:Date.now()})}catch(f){this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",msg:f instanceof Error?f.message:String(f),updated_at:Date.now()})}return}}}const d=this.buildInboundEvent(e,s);try{this.captureEventRuntimeConfig(d),await this.pool.deliverInboundEvent(d)}catch(r){if(this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()}),this.inflightEvents.delete(e.event_id),this.restartCount.delete(e.event_id),this.eventSessionIndex.delete(e.event_id),r instanceof we)this.aibotHandle.sendEventResult({event_id:e.event_id,status:"failed",msg:r.message,updated_at:Date.now()});else throw r}}async handleCodexSessionControlOpen(e,n){const t=n.session_id,s=e.args.trim(),i=()=>this.aibotHandle.sendEventAck({event_id:n.event_id,session_id:t,received_at:Date.now()}),o=(d,r)=>this.aibotHandle.sendEventResult({event_id:n.event_id,status:d,...r,updated_at:Date.now()});if(!s){i(),o("failed",{msg:"Usage: /grix open <working-directory>",code:h.cwdRequired});return}try{const d=await this.resolveCwdForBinding(s),r=this.bindingStore.get(t);if(r?.cwd){let a=!1;try{if(await this.resolveCwdForBinding(r.cwd)===d){i(),o("responded",{msg:`Session already bound to ${r.cwd}`});return}}catch{a=!0,u.info("bridge",`Stale codex binding for session ${t}: ${r.cwd} no longer exists, allowing rebind`)}if(!a){i(),o("failed",{msg:`Session already bound to ${r.cwd}. Rebinding is not allowed.`,code:h.rebindForbidden});return}}this.bindingStore.set(t,d),this.sessionBindings.set(t,d),this.deferredMgr.sendCodexDeferredReplayComposing(t,this.deferredCallbacks()),await this.bindSessionForPool(t,d),await this.deferredMgr.replayDeferredCodexEvents(t,this.deferredCallbacks(),{announceComposing:!1}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d}),i(),o("responded",{msg:`Session bound to ${d}`})}catch(d){i(),o("failed",{code:h.invalidCwd,msg:d instanceof Error?d.message:String(d)})}}async handleSessionControlForPool(e,n){e.verb===_.open&&await this.bindSessionForPool(n.session_id,e.args.trim())}async replayDeferredEventsForSession(e){const n=this.config.adapterType??"acp";n==="pi"?await this.deferredMgr.replayDeferredPiEvents(e,this.deferredCallbacks()):n==="openhuman"?await this.deferredMgr.replayDeferredOpenHumanEvents(e,this.deferredCallbacks()):n==="opencode"?await this.deferredMgr.replayDeferredOpenCodeEvents(e,this.deferredCallbacks()):n==="agy"?await this.deferredMgr.replayDeferredAgyEvents(e,this.deferredCallbacks()):n==="claude"||n==="codex"||n==="cursor"||n==="codewhale"?await this.deferredMgr.replayDeferredClaudeEvents(e,this.deferredCallbacks()):await this.deferredMgr.replayDeferredAcpEvents(e,this.deferredCallbacks())}async handleSessionControlLocalActionForPool(e){if(String(e.action_type??"")!==w.sessionControl)return;const n=e.params??{};if(String(n.verb??"").trim().toLowerCase()!==_.open)return;const s=String(n.session_id??"").trim(),i=String(n.cwd??"").trim();!s||!i||await this.bindSessionForPool(s,i)}async handleCodexSessionControlLocalActionOpen(e){const n=e.params??{},t=String(n.session_id??"").trim(),s=String(n.cwd??"").trim(),i=String(n.agent_session_id??"").trim(),o=(d,r,a,c)=>{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:d,...r!==void 0?{result:r}:{},...a?{error_code:a}:{},...c?{error_msg:c}:{}})};if(!t||!s){o("failed",void 0,h.cwdRequired,"session cwd is required");return}try{const d=await this.resolveCwdForBinding(s);this.ensureImportedAgentSession(i,d);const r=this.bindingStore.get(t);if(r?.cwd){let a=!1;try{const c=await this.resolveCwdForBinding(r.cwd);if(c===d){this.setResolvedAgentSessionId(t,i),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,c)});return}}catch{a=!0,u.info("bridge",`Stale codex binding for session ${t}: ${r.cwd} no longer exists, allowing rebind`)}if(!a){o("failed",void 0,h.rebindForbidden,`Session already bound to ${r.cwd}`);return}}this.bindingStore.set(t,d),this.setResolvedAgentSessionId(t,i),this.sessionBindings.set(t,d),this.deferredMgr.sendCodexDeferredReplayComposing(t,this.deferredCallbacks()),await this.bindSessionForPool(t,d),await this.deferredMgr.replayDeferredCodexEvents(t,this.deferredCallbacks(),{announceComposing:!1}),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,d)})}catch(d){o("failed",void 0,d?.sessionControlErrorCode??h.invalidCwd,d instanceof Error?d.message:String(d))}}async handleCursorSessionControlLocalActionOpen(e){const n=e.params??{},t=String(n.session_id??"").trim(),s=String(n.cwd??"").trim(),i=String(n.agent_session_id??"").trim(),o=(d,r,a,c)=>{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:d,...r!==void 0?{result:r}:{},...a?{error_code:a}:{},...c?{error_msg:c}:{}})};if(!t||!s){o("failed",void 0,h.cwdRequired,"session cwd is required");return}try{const d=await this.resolveCwdForBinding(s);this.ensureImportedAgentSession(i,d);const r=this.bindingStore.get(t);if(r?.cwd){let a=!1;try{const c=await this.resolveCwdForBinding(r.cwd);if(c===d){this.setResolvedAgentSessionId(t,i),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,c)});return}}catch{a=!0,u.info("bridge",`Stale cursor binding for session ${t}: ${r.cwd} no longer exists, allowing rebind`)}if(!a){o("failed",void 0,h.rebindForbidden,`Session already bound to ${r.cwd}`);return}}this.bindingStore.set(t,d),this.setResolvedAgentSessionId(t,i),this.sessionBindings.set(t,d),this.deferredMgr.sendCursorDeferredReplayComposing(t,this.deferredCallbacks()),await this.bindSessionForPool(t,d),await this.deferredMgr.replayDeferredCursorEvents(t,this.deferredCallbacks(),{announceComposing:!1}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d}),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,d)})}catch(d){o("failed",void 0,d?.sessionControlErrorCode??h.invalidCwd,d instanceof Error?d.message:String(d))}}async handlePiSessionControlOpen(e,n){const t=n.session_id,s=e.args.trim(),i=()=>this.aibotHandle.sendEventAck({event_id:n.event_id,session_id:t,received_at:Date.now()}),o=(d,r)=>this.aibotHandle.sendEventResult({event_id:n.event_id,status:d,...r,updated_at:Date.now()});if(!s){i(),o("failed",{msg:"Usage: /grix open <working-directory>",code:h.cwdRequired});return}try{const d=await this.resolveCwdForBinding(s),r=this.bindingStore.get(t);if(r?.cwd){let a=!1;try{if(await this.resolveCwdForBinding(r.cwd)===d){i(),o("responded",{msg:`Session already bound to ${r.cwd}`}),this.aibotHandle.sendMsg({event_id:n.event_id,session_id:t,msg_type:1,content:`\u2705 Session already bound to \`${r.cwd}\``,quoted_message_id:n.msg_id});return}}catch{a=!0,u.info("bridge",`Stale pi binding for session ${t}: ${r.cwd} no longer exists, allowing rebind`)}if(!a){i(),o("failed",{msg:`Session already bound to ${r.cwd}. Rebinding is not allowed.`,code:h.rebindForbidden});return}}this.bindingStore.set(t,d),this.sessionBindings.set(t,d),await this.deferredMgr.replayDeferredPiEvents(t,this.deferredCallbacks()),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d}),i(),o("responded",{msg:`Session bound to ${d}`}),this.aibotHandle.sendMsg({event_id:n.event_id,session_id:t,msg_type:1,content:`\u2705 Working directory bound: \`${d}\``,quoted_message_id:n.msg_id})}catch(d){i(),o("failed",{code:h.invalidCwd,msg:d instanceof Error?d.message:String(d)})}}async handlePiSessionControlRestart(e){const n=e.session_id,t=()=>this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:n,received_at:Date.now()}),s=(d,r)=>this.aibotHandle.sendEventResult({event_id:e.event_id,status:d,...r,updated_at:Date.now()}),o=this.bindingStore.get(n)?.cwd??"";if(!o){t(),s("failed",{msg:"session binding was not found",code:h.bindingMissing});return}t(),await this.pool.removeSlot(n).catch(()=>{}),this.aibotHandle.sendUpdateBindingCard({session_id:n,worker_status:"ready",cwd:o}),s("responded",{msg:`Session worker restarted for ${o}`})}async handlePiSessionControlRestartLocalAction(e){const n=e.params??{},t=String(n.session_id??"").trim(),i=(t?this.bindingStore.get(t):void 0)?.cwd??"",o=(d,r,a,c)=>{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:d,...r!==void 0?{result:r}:{},...a?{error_code:a}:{},...c?{error_msg:c}:{}})};if(!t){o("failed",void 0,"session_id_required","session_id is required for restart");return}if(!i){o("failed",void 0,h.bindingMissing,"Session binding missing. Open a workspace first.");return}await this.pool.removeSlot(t).catch(()=>{}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:i}),o("ok",{outcome:"restarted",binding:{aibotSessionId:t,cwd:i,workerStatus:"ready"}})}async bindSessionForPool(e,n){const t=String(n??"").trim();if(!e||!t)return;const s=this.pool.getOrCreateSlot(e);if(!s||(s.startPromise&&await s.startPromise,!s.adapter))return;const i=s.adapter instanceof A?s.adapter:null;if(!i?.hasSessionBinding)return;const o=await this.resolveCwdForBinding(t);i.announceDeferredComposing(e),await i.bindSession(e,o),this.sessionBindings.set(e,o),this.sessionScanCache.invalidate(),i.replayDeferredEvents(e)}deferredCallbacks(){return{captureEventRuntimeConfig:e=>this.captureEventRuntimeConfig(e),deliverInboundEvent:e=>this.pool.deliverInboundEvent(e),sendEventResult:(e,n,t)=>this.sendEventResultWithCleanup(e,n,t),sendSessionComposing:(e,n,t)=>{const s={};n&&(s.ttl_ms=t?.ttlMs??3e4,t?.activity&&(s.activity=t.activity)),this.aibotHandle.sendSessionActivitySet({session_id:e,kind:"composing",active:n,...s})}}}async handleOpenHumanSessionControlOpen(e,n){const t=()=>this.aibotHandle.sendEventAck({event_id:n.event_id,session_id:n.session_id,received_at:Date.now()});try{await this.resolveCwdForBinding(e.args.trim()),await this.handleSessionControlForPool(e,n),F(e,n,this.sessionControlCtx(n.session_id),this.sessionControlSenders()),this.deferredMgr.hasDeferred("openhuman",n.session_id)&&await this.deferredMgr.replayDeferredOpenHumanEvents(n.session_id,this.deferredCallbacks()),this.deferredMgr.hasDeferred("opencode",n.session_id)&&await this.deferredMgr.replayDeferredOpenCodeEvents(n.session_id,this.deferredCallbacks())}catch(s){t(),this.aibotHandle.sendEventResult({event_id:n.event_id,status:"failed",code:s?.cwdErrorCode,msg:s instanceof Error?s.message:String(s),updated_at:Date.now()})}}async handleCodeWhaleSessionControlOpen(e,n){const t=n.session_id,s=e.args.trim(),i=()=>this.aibotHandle.sendEventAck({event_id:n.event_id,session_id:t,received_at:Date.now()}),o=(d,r)=>this.aibotHandle.sendEventResult({event_id:n.event_id,status:d,...r,updated_at:Date.now()});if(!s){i(),o("failed",{msg:"Usage: /grix open <working-directory>",code:h.cwdRequired});return}try{const d=await this.resolveCwdForBinding(s),r=this.bindingStore.get(t);if(r?.cwd){if(await this.resolveCwdForBinding(r.cwd)!==d){i(),o("failed",{msg:`Session already bound to ${r.cwd}. Rebinding is not allowed.`,code:h.rebindForbidden});return}i(),o("responded",{msg:`Session already bound to ${r.cwd}`});return}this.bindingStore.set(t,d),this.sessionBindings.set(t,d),await this.bindSessionForPool(t,d),await this.deferredMgr.replayDeferredCodeWhaleEvents(t,this.deferredCallbacks(),{announceComposing:!1}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d}),i(),o("responded",{msg:`Session bound to ${d}`})}catch(d){i(),o("failed",{code:h.invalidCwd,msg:d instanceof Error?d.message:String(d)})}}async handleCodeWhaleSessionControlLocalActionOpen(e){const n=e.params??{},t=String(n.session_id??"").trim(),s=String(n.cwd??"").trim(),i=String(n.agent_session_id??"").trim(),o=(d,r,a,c)=>{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:d,...r!==void 0?{result:r}:{},...a?{error_code:a}:{},...c?{error_msg:c}:{}})};if(!t||!s){o("failed",void 0,h.cwdRequired,"session cwd is required");return}try{const d=await this.resolveCwdForBinding(s);this.ensureImportedAgentSession(i,d);const r=this.bindingStore.get(t);if(r?.cwd){const a=await this.resolveCwdForBinding(r.cwd);if(a!==d){o("failed",void 0,h.rebindForbidden,`Session already bound to ${r.cwd}`);return}this.setResolvedAgentSessionId(t,i),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,a)});return}this.bindingStore.set(t,d),this.setResolvedAgentSessionId(t,i),this.sessionBindings.set(t,d),await this.bindSessionForPool(t,d),await this.deferredMgr.replayDeferredCodeWhaleEvents(t,this.deferredCallbacks(),{announceComposing:!1}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d}),o("ok",{outcome:"opened",binding:this.buildOpenedBindingResult(t,d)})}catch(d){o("failed",void 0,d?.sessionControlErrorCode??h.invalidCwd,d instanceof Error?d.message:String(d))}}normalizeClaudeModeId(e){return String(e??"").trim().toLowerCase()===C.approval?C.approval:C.fullAuto}handleExecCommandOptions(e,n,t){const s=Array.isArray(t?.options)?t.options:[];if(s.length===0)return;const o=this.bindingStore.get(e)?.cwd??"",d={};if(n==="model"){d.available_models=s.map(a=>({id:a.id,displayName:a.label}));const r=s.find(a=>a.current);r&&(d.model_id=r.id)}else if(n==="mode"){d.available_modes=s.map(a=>({id:a.id,displayName:a.label}));const r=s.find(a=>a.current);r&&(d.mode_id=r.id)}else d[`${n}_options`]=s;this.aibotHandle.sendUpdateBindingCard({session_id:e,worker_status:"ready",cwd:o,meta:d})}buildAgyToolbarMeta(e){const n=this.bindingStore.get(e);if(!n)return;const t=te(this.config.agent.command),s=t.length>0?t[0].id:"";let i=n.modelId||s;t.length>0&&!t.some(d=>d.id===i)&&(i=s);const o=ie();return{model_id:i,currentModelId:i,available_models:t.map(d=>({id:d.id,displayName:d.displayName})),...o.plan!==void 0&&{plan:o.plan},...o.quota_exhausted!==void 0&&{quota_exhausted:o.quota_exhausted},...o.quota_reset_at!==void 0&&{quota_reset_at:o.quota_reset_at},...o.available_credits!==void 0&&{available_credits:o.available_credits}}}async handleAgySetModel(e,n){const t=e.params??{},s=String(t.model_id??t.modelId??"").trim();if(!n){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"session_id_required",error_msg:"session_id is required for set_model"});return}if(!s){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"set_model_invalid",error_msg:"model_id is required"});return}const i=this.bindingStore.get(n);if(!i?.cwd){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.bindingMissing,error_msg:"session binding was not found"});return}i.modelId!==s&&this.bindingStore.setModelId(n,s),this.aibotHandle.sendUpdateBindingCard({session_id:n,worker_status:"ready",cwd:i.cwd,meta:this.buildAgyToolbarMeta(n)}),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"model_set",model_id:s,binding:{cwd:i.cwd,model_id:s}}})}buildClaudeToolbarMeta(e){const n=this.bindingStore.get(e);if(!n)return;const t=D(),s=t.length>0?t[0].id:"";let i=n.modelId||s;t.length>0&&!t.some(l=>l.id===i)&&(i=s);const o=this.normalizeClaudeModeId(n.modeId),d={model_id:i,mode_id:o,currentModelId:i,currentModeId:o,available_models:t.map(l=>({id:l.id,displayName:l.displayName}))},r=this.pool.getSlot(e),a=r?.adapter instanceof y?r.adapter.getSessionState():null;(a?.rateLimits?.fiveHour||a?.rateLimits?.sevenDay)&&(this.cachedClaudeRateLimitState=a);const c=a??this.getFreshClaudeRateLimitState();if(c){const l=c.rateLimits;l&&(l.fiveHour||l.sevenDay)&&(d.rate_limits={...l.fiveHour?{fiveHour:l.fiveHour}:{},...l.sevenDay?{sevenDay:l.sevenDay}:{},sampledAt:c.sampledAt||Date.now()})}if(a){const l=a.contextWindow;l.usedPercentage!=null&&(d.context_window={usedPercentage:l.usedPercentage,remainingPercentage:l.remainingPercentage})}if(!d.rate_limits&&this.cachedProviderQuota?.success){this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)||this.maybeQueryProviderQuota().catch(()=>{});const l=this.providerQuotaToRateLimits(this.cachedProviderQuota);l&&(d.rate_limits=l),u.info(this.name,`[toolbar-meta] provider quota fallback: hasCached=${!!this.cachedProviderQuota} fromProvider=${JSON.stringify(l)}`)}return d.rate_limits&&u.info(this.name,`[toolbar-meta] rate_limits included: ${JSON.stringify(d.rate_limits)}`),d}providerQuotaToCodexRateLimits(e){const n=e.tiers.find(i=>i.name==="five_hour"),t=e.tiers.find(i=>i.name==="weekly_limit"),s=i=>i||null;if(n||t){const i={credits:{hasCredits:!0,unlimited:!1,balance:null},sampledAt:this.cachedProviderQuotaSampledAtMs??Date.now()};let o=0,d=0,r=0,a=0;return n&&(i.primary={usedPercent:n.usedPercent,windowMinutes:300,resetsAt:s(n.resetsAt)},o=n.usedPercent,d=300),t&&(i.secondary={usedPercent:t.usedPercent,windowMinutes:10080,resetsAt:s(t.resetsAt)},r=t.usedPercent,a=10080),{rateLimits:i,primaryPercent:o,secondaryPercent:r,primaryWindowMin:d,secondaryWindowMin:a}}if(e.balance){const i=e.balance,o=i.total&&i.total>0?Math.min(100,(i.used??i.total-i.remaining)/i.total*100):0;return{rateLimits:{primary:{usedPercent:o,windowMinutes:0,resetsAt:null},secondary:{usedPercent:0,windowMinutes:0,resetsAt:null},credits:{hasCredits:!0,unlimited:!1,balance:i.remaining},sampledAt:this.cachedProviderQuotaSampledAtMs??Date.now()},primaryPercent:o,secondaryPercent:0,primaryWindowMin:0,secondaryWindowMin:0}}return null}providerQuotaToRateLimits(e){const n=i=>{if(!i)return 0;const o=Date.parse(i);return Number.isFinite(o)?Math.floor(o/1e3):0},t=e.tiers.find(i=>i.name==="five_hour"),s=e.tiers.find(i=>i.name==="weekly_limit");if(t||s)return{...t?{fiveHour:{usedPercentage:t.usedPercent,resetsAt:n(t.resetsAt)}}:{},...s?{sevenDay:{usedPercentage:s.usedPercent,resetsAt:n(s.resetsAt)}}:{},sampledAt:this.cachedProviderQuotaSampledAtMs??Date.now()};if(e.balance){const i=e.balance;return{credit:{remaining:i.remaining,total:i.total,used:i.used,unit:i.unit,resetsAt:i.resetsAt?n(i.resetsAt):0},planName:e.planName,sampledAt:this.cachedProviderQuotaSampledAtMs??Date.now()}}return null}async resolveCwdForBinding(e){const n=String(e??"").trim();if(process.platform!=="win32"&&(/^[a-zA-Z]:[\\/]/.test(n)||/^\\\\/.test(n))){const i=new Error(`Specified path is not valid on this host: ${n}`);throw i.cwdErrorCode=h.invalidCwd,i}const t=M.resolve(n);let s;try{s=await H(t)}catch(i){const o=String(i?.code??"");if(o==="ENOENT"){const d=new Error(`Specified path does not exist: ${t}`);throw d.cwdErrorCode=h.invalidCwd,d}if(o==="EACCES"||o==="EPERM"){const d=new Error("Specified path is not accessible.");throw d.cwdErrorCode=h.invalidCwd,d}throw i}if(!s.isDirectory()){const i=new Error("Specified path is not a directory.");throw i.cwdErrorCode=h.invalidCwd,i}try{return await G(t)}catch{return t}}getClaudeWorkerStatus(e){const n=this.pool.getSlot(e);return n?n.state==="starting"?"starting":n.state==="stopped"?"stopped":n.adapter.getStatus().busy?"busy":"ready":"stopped"}async ensureSlotStarted(e,n=6e4){const t=this.pool.getOrCreateSlot(e);if(!t)throw new Error("Failed to allocate session slot");t.startPromise&&(u.info(this.name,`ensureSlotStarted: awaiting startPromise for session=${e}`),await Promise.race([t.startPromise,new Promise((s,i)=>setTimeout(()=>i(new Error(`ensureSlotStarted timeout (${n}ms) session=${e}`)),n))]),u.info(this.name,`ensureSlotStarted: startPromise resolved for session=${e}`))}resolveAgentSessionId(e){switch(this.config.adapterType??"acp"){case"claude":return e.claudeSessionId;case"codex":return e.codexThreadId;case"pi":return e.piSessionPath;case"codewhale":return e.codewhaleThreadId;default:return e.acpSessionId}}providerKeyForAdapter(){const e=this.config.adapterType??"acp";switch(e){case"claude":case"codex":case"pi":case"codewhale":return e;default:return"acp"}}setResolvedAgentSessionId(e,n){const t=String(e??"").trim(),s=String(n??"").trim();if(!t||!s)return;switch(this.config.adapterType??"acp"){case"claude":this.bindingStore.setClaudeSessionId(t,s);break;case"codex":this.bindingStore.setCodexThreadId(t,s);break;case"pi":this.bindingStore.setPiSessionPath(t,s);break;case"codewhale":this.bindingStore.setCodeWhaleThreadId(t,s);break;default:this.bindingStore.setAcpSessionId(t,s);break}this.sessionScanCache.invalidate()}normalizePathForCompare(e){const n=String(e??"").trim();if(!n)return"";const t=M.resolve(n);return process.platform==="win32"?t.toLowerCase():t}ensureImportedAgentSession(e,n){const t=String(e??"").trim();if(!t)return;const s=this.normalizePathForCompare(n);let i="";const o=this.config.adapterType??"acp";if(o==="codex"?i=this.sessionScanCache.get().find(a=>a.threadId===t)?.cwd??"":o==="claude"?i=this.sessionScanCache.get().find(a=>a.sessionId===t)?.cwd??"":o==="acp"&&(i=this.sessionScanCache.get().find(a=>a.sessionId===t)?.cwd??""),!i){for(const[,r]of this.bindingStore.entries())if(this.resolveAgentSessionId(r)===t){i=r.cwd??"";break}}if(!i){const r=new Error(`agent session not found: ${t}`);throw r.sessionControlErrorCode=h.invalidAgentSession,r}const d=this.normalizePathForCompare(i);if(d&&s&&d!==s){const r=new Error(`agent session cwd mismatch: expected ${n}, got ${i}`);throw r.sessionControlErrorCode=h.invalidAgentSession,r}}buildOpenedBindingResult(e,n,t="ready"){const s=this.bindingStore.get(e),i=s?String(this.resolveAgentSessionId(s)??"").trim():"",o={aibotSessionId:e,providerKey:this.providerKeyForAdapter(),cwd:n,workerStatus:t};return i&&(o.bindingId=i,o.agentSessionId=i),o}hasDiskScanner(){const e=this.config.adapterType??"acp";return e==="codex"||e==="claude"||e==="acp"}async handleListSessionsTextCommand(e){this.aibotHandle.sendEventAck({event_id:e.event_id,session_id:e.session_id,received_at:Date.now()});const n=this.config.adapterType??"acp",t=new Map;for(const r of this.pool.getAllSlots())t.set(r.sessionId,r);const s=Array.from(this.bindingStore.entries()),i=new Map;for(const[r,a]of s){const c=this.resolveAgentSessionId(a);if(c){const l=t.get(r),g=l?l.adapter.getStatus().busy?"busy":"ready":"inactive";i.set(c,{aibotSessionId:r,workerStatus:g})}}const o=[],d=new Set;if(n==="codex"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.threadId);const c=i.get(a.threadId);a.title&&o.push(` Title: ${a.title}`),o.push(` Agent: ${a.threadId}`),c&&o.push(` AIBot: ${c.aibotSessionId}`),o.push(` CWD: ${a.cwd||"-"}`),o.push(` State: ${c?.workerStatus??(a.archived?"archived":"inactive")}`),o.push(` Created: ${new Date(a.createdAt).toISOString()}`),o.push(` Updated: ${new Date(a.updatedAt).toISOString()}`),o.push("---")}}else if(n==="claude"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.sessionId);const c=i.get(a.sessionId);a.title&&o.push(` Title: ${a.title}`),o.push(` Agent: ${a.sessionId}`),c&&o.push(` AIBot: ${c.aibotSessionId}`),o.push(` CWD: ${a.cwd||"-"}`),o.push(` State: ${c?.workerStatus??"inactive"}`),o.push(` Updated: ${new Date(a.updatedAt).toISOString()}`),o.push("---")}}else if(n==="acp"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.sessionId);const c=i.get(a.sessionId);a.title&&o.push(` Title: ${a.title}`),o.push(` Agent: ${a.sessionId}`),c&&o.push(` AIBot: ${c.aibotSessionId}`),o.push(` CWD: ${a.cwd||"-"}`),o.push(` State: ${c?.workerStatus??"inactive"}`),o.push(` Updated: ${new Date(a.updatedAt).toISOString()}`),o.push("---")}}for(const[r,a]of s){const c=this.resolveAgentSessionId(a);if(c&&d.has(c))continue;const l=t.get(r),g=l?l.adapter.getStatus().busy?"busy":"ready":"closed";o.push(` Title: ${c?c.slice(0,8)+"\u2026":r.slice(0,8)+"\u2026"}`),o.push(` AIBot: ${r}`),c&&o.push(` Agent: ${c}`),o.push(` CWD: ${a.cwd??"-"}`),o.push(` State: ${g}`),o.push(` Updated: ${new Date(a.updatedAt).toISOString()}`),o.push("---")}if(o.length===0){this.aibotHandle.sendEventResult({event_id:e.event_id,status:"responded",msg:"No sessions found.",updated_at:Date.now()});return}this.aibotHandle.sendEventResult({event_id:e.event_id,status:"responded",msg:`Sessions (${o.filter(r=>r==="---").length}):
14
14
  ${o.join(`
15
- `)}`,updated_at:Date.now()})}async handleListSessionsLocalAction(e){const i=this.config.adapterType??"acp",t=new Map;for(const r of this.pool.getAllSlots())t.set(r.sessionId,r);const s=Array.from(this.bindingStore.entries()),n=new Map;for(const[r,a]of s){const c=this.resolveAgentSessionId(a);if(c){const l=t.get(r),g=l?l.adapter.getStatus().busy?"busy":"ready":"inactive";n.set(c,{aibotSessionId:r,workerStatus:g,bindingUpdatedAt:a.updatedAt??0})}}const o=[],d=new Set;if(i==="codex"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.threadId);const c=n.get(a.threadId),l={agentSessionId:a.threadId,cwd:a.cwd||null,workerStatus:c?.workerStatus??(a.archived?"archived":"inactive"),createdAt:a.createdAt,updatedAt:a.updatedAt,archived:a.archived};c&&(l.aibotSessionId=c.aibotSessionId),a.title&&(l.title=a.title),o.push(l)}}else if(i==="claude"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.sessionId);const c=n.get(a.sessionId),l={agentSessionId:a.sessionId,cwd:a.cwd||null,workerStatus:c?.workerStatus??"inactive",updatedAt:a.updatedAt};c&&(l.aibotSessionId=c.aibotSessionId),a.title&&(l.title=a.title),o.push(l)}}else if(i==="acp"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.sessionId);const c=n.get(a.sessionId),l=c&&c.bindingUpdatedAt>0?c.bindingUpdatedAt:a.updatedAt,g={agentSessionId:a.sessionId,cwd:a.cwd||null,agentType:a.agentType,workerStatus:c?.workerStatus??"inactive",updatedAt:l};c&&(g.aibotSessionId=c.aibotSessionId),a.title&&(g.title=a.title),a.createdAt&&(g.createdAt=a.createdAt),o.push(g)}}for(const[r,a]of s){const c=this.resolveAgentSessionId(a);if(c&&d.has(c))continue;const l=t.get(r),g=l?l.adapter.getStatus().busy?"busy":"ready":"closed",m={aibotSessionId:r,cwd:a.cwd??null,workerStatus:g,updatedAt:a.updatedAt,title:c?`${c.slice(0,8)}\u2026`:`${r.slice(0,8)}\u2026`};c&&(m.agentSessionId=c),o.push(m)}o.sort((r,a)=>a.updatedAt-r.updatedAt),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"sessions_listed",sessions:o,total:o.length}})}async handleSessionControlCommand(e,i){const t=i.session_id,s=(n,o,d)=>{this.aibotHandle.sendEventResult({event_id:i.event_id,status:n,...o?{msg:o}:{},...d?{code:d}:{},updated_at:Date.now()})};this.aibotHandle.sendEventAck({event_id:i.event_id,session_id:t,received_at:Date.now()});try{switch(e.verb){case _.open:{await I().catch(()=>{});const n=e.args.trim();if(!n){s("failed","Usage: /grix open <working-directory>",h.cwdRequired);return}let o="";try{o=await this.resolveCwdForBinding(n)}catch(r){s("failed",r instanceof Error?r.message:String(r),h.invalidCwd);return}const d=this.bindingStore.get(t);if(d?.cwd){const r=await this.resolveCwdForBinding(d.cwd);if(r!==o){s("failed","session binding cannot be changed to another working directory",h.rebindForbidden);return}this.bindingStore.ensureModeId(t,C.fullAuto),this.sessionBindings.set(t,r),await this.ensureSlotStarted(t),this.claudeWorkerStatus.set(t,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:r,meta:this.buildClaudeToolbarMeta(t)}),await this.deferredMgr.replayDeferredClaudeEvents(t,this.deferredCallbacks()),s("responded",`Working directory already bound: ${r}`);return}this.bindingStore.set(t,o,{modeId:C.fullAuto}),this.sessionBindings.set(t,o),await this.ensureSlotStarted(t),this.claudeWorkerStatus.set(t,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:o,meta:this.buildClaudeToolbarMeta(t)}),await this.deferredMgr.replayDeferredClaudeEvents(t,this.deferredCallbacks()),s("responded",`Session bound to ${o}`);return}case _.where:{const n=this.bindingStore.get(t);if(!n?.cwd){s("failed","session binding was not found",h.bindingMissing);return}s("responded",`Working directory: ${n.cwd}`);return}case _.status:{const n=this.bindingStore.get(t);if(!n?.cwd){s("failed","session binding was not found",h.bindingMissing);return}const o=this.normalizeClaudeModeId(n.modeId),d=this.getClaudeWorkerStatus(t);s("responded",`Status: worker=${d} mode=${o} cwd=${n.cwd}`);return}case _.stop:{const n=this.bindingStore.get(t);if(!n?.cwd){s("failed","session binding was not found",h.bindingMissing);return}await this.pool.removeSlot(t),this.claudeWorkerStatus.set(t,"stopped"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"stopped",cwd:n.cwd}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"stopped",cwd:n.cwd,meta:this.buildClaudeToolbarMeta(t)}),s("responded",`Session worker stopped for ${n.cwd}`);return}case _.restart:{const n=this.bindingStore.get(t);if(!n?.cwd){s("failed","session binding was not found",h.bindingMissing);return}await this.pool.removeSlot(t),await this.ensureSlotStarted(t),this.claudeWorkerStatus.set(t,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:n.cwd,meta:this.buildClaudeToolbarMeta(t)}),s("responded",`Session worker restarted for ${n.cwd}`);return}case _.setMode:{const n=e.args.trim();if(!n){s("failed","Usage: /grix set_mode <mode-id>",x.modeInvalid);return}const o=this.normalizeClaudeModeId(n);if(o!==n.toLowerCase()){s("failed","set mode_id is invalid",x.modeInvalid);return}const d=this.bindingStore.get(t);if(!d?.cwd){s("failed","session binding was not found",h.bindingMissing);return}if(this.normalizeClaudeModeId(d.modeId)===o){s("responded",`Mode unchanged: ${o}`);return}this.bindingStore.setModeId(t,o),await this.pool.removeSlot(t),await this.ensureSlotStarted(t),this.claudeWorkerStatus.set(t,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d.cwd,meta:this.buildClaudeToolbarMeta(t)}),s("responded",`Mode set to ${o}`);return}case _.setModel:{const n=e.args.trim();if(!n){s("failed","Usage: /grix set_model <model-id>");return}const o=this.bindingStore.get(t);if(!o?.cwd){s("failed","session binding was not found",h.bindingMissing);return}if((o.modelId??"")===n){s("responded",`Model unchanged: ${n}`);return}this.bindingStore.setModelId(t,n),this.globalConfigStore?.set(this.name,{modelId:n}),await this.pool.removeSlot(t),await this.ensureSlotStarted(t),this.claudeWorkerStatus.set(t,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:o.cwd,meta:this.buildClaudeToolbarMeta(t)}),s("responded",`Model set to ${n}`);return}case _.listOptions:{const n=P(),o=this.bindingStore.get(t),d=n.map(a=>a.id).join(", "),r=`${C.fullAuto}, ${C.approval}`;s("responded",`Modes (current: ${this.normalizeClaudeModeId(o?.modeId)}): ${r}
16
- Models (current: ${o?.modelId??"default"}): ${d}`);return}case _.exec:{const[n,...o]=e.args.trim().split(/\s+/);if(!n){s("failed","Usage: /grix exec <command> [args]",h.verbInvalid);return}const r=this.pool.getSlot(t)?.adapter;if(!r?.execCommand){s("failed","Agent does not support command execution",h.verbInvalid);return}const a=r.getSupportedCommands?.()??[];if(!a.some(c=>c.name===n)){s("failed",`Unknown command: ${n}. Supported: ${a.map(c=>c.name).join(", ")}`,h.verbInvalid);return}try{const c=await r.execCommand(n,o.join(" "),t);s(c.status==="ok"?"responded":"failed",c.message??`${n} ${c.status}`,c.status==="ok"?void 0:h.runtimeError)}catch(c){s("failed",`exec error: ${c instanceof Error?c.message:c}`,h.runtimeError)}return}default:s("failed",`Unsupported command for Claude: /grix ${e.verb}`,h.verbInvalid)}}catch(n){s("failed",n instanceof Error?n.message:String(n),h.runtimeError)}}async handleSessionControlLocalAction(e){const i=String(e.action_type??"").trim();if(i!==S.sessionControl&&i!==S.setMode&&i!=="set_mode"&&i!==S.setModel&&i!=="set_model")return!1;const t=e.params??{},s=String(t.session_id??"").trim(),n=i===S.setMode?_.setMode:i===S.setModel?_.setModel:String(t.verb??"").trim().toLowerCase();if(u.info(this.name,`handleSessionControlLocalAction verb=${n} action_id=${e.action_id} session_id=${s}`),!s)return this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:x.localActionRouteMissing,error_msg:"local action session_id is required"}),!0;const o=r=>{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:r})},d=(r,a)=>{u.warn(this.name,`session_control local_action failed action_id=${e.action_id} session_id=${s} verb=${n} code=${r} msg=${a}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:r,error_msg:a})};try{switch(n){case _.open:{await I().catch(()=>{});const r=String(t.cwd??"").trim(),a=String(t.agent_session_id??"").trim();if(!r)return d(h.cwdRequired,"session control cwd is required"),!0;u.info(this.name,`handleSessionControlLocalAction open cwd=${r} session_id=${s}`);const c=await this.resolveCwdForBinding(r);this.ensureImportedAgentSession(a,c);const l=this.bindingStore.get(s);if(l?.cwd){const g=await this.resolveCwdForBinding(l.cwd);return g!==c?(d(h.rebindForbidden,"session binding cannot be changed to another working directory"),!0):(this.bindingStore.ensureModeId(s,C.fullAuto),this.setResolvedAgentSessionId(s,a),this.sessionBindings.set(s,g),await this.ensureSlotStarted(s),await this.replayDeferredEventsForSession(s),this.claudeWorkerStatus.set(s,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:"ready",cwd:g,meta:this.buildClaudeToolbarMeta(s)}),o({outcome:"opened",binding:{...this.buildOpenedBindingResult(s,g),mode_id:this.normalizeClaudeModeId(this.bindingStore.get(s)?.modeId)}}),!0)}return this.bindingStore.set(s,c,{modeId:C.fullAuto}),this.setResolvedAgentSessionId(s,a),this.sessionBindings.set(s,c),await this.ensureSlotStarted(s),await this.replayDeferredEventsForSession(s),this.claudeWorkerStatus.set(s,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:"ready",cwd:c,meta:this.buildClaudeToolbarMeta(s)}),o({outcome:"opened",binding:{...this.buildOpenedBindingResult(s,c),mode_id:this.normalizeClaudeModeId(this.bindingStore.get(s)?.modeId)}}),!0}case _.status:case _.where:{const r=this.bindingStore.get(s);return r?.cwd?(o({outcome:n,binding:{cwd:r.cwd,mode_id:this.normalizeClaudeModeId(r.modeId),worker_status:this.getClaudeWorkerStatus(s)}}),!0):(d(h.bindingMissing,"session binding was not found"),!0)}case _.stop:{const r=this.bindingStore.get(s);return r?.cwd?(await this.pool.removeSlot(s),o({outcome:"stopped",binding:{cwd:r.cwd,mode_id:this.normalizeClaudeModeId(r.modeId),worker_status:"stopped"}}),!0):(d(h.bindingMissing,"session binding was not found"),!0)}case _.restart:{const r=this.bindingStore.get(s);return r?.cwd?(await this.pool.removeSlot(s),await this.ensureSlotStarted(s),this.claudeWorkerStatus.set(s,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:"ready",cwd:r.cwd,meta:this.buildClaudeToolbarMeta(s)}),o({outcome:"restarted",binding:{cwd:r.cwd,mode_id:this.normalizeClaudeModeId(r.modeId)}}),!0):(d(h.bindingMissing,"session binding was not found"),!0)}case _.setMode:{const r=String(t.mode_id??t.modeId??"").trim();if(!r)return d(x.modeInvalid,"set mode_id is invalid"),!0;const a=this.normalizeClaudeModeId(r);if(a!==r.toLowerCase())return d(x.modeInvalid,"set mode_id is invalid"),!0;const c=this.bindingStore.get(s);return c?.cwd?(this.normalizeClaudeModeId(c.modeId)!==a&&(this.bindingStore.setModeId(s,a),await this.pool.removeSlot(s),await this.ensureSlotStarted(s),this.claudeWorkerStatus.set(s,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:"ready",cwd:c.cwd,meta:this.buildClaudeToolbarMeta(s)})),o({outcome:"mode_set",mode_id:a,binding:{cwd:c.cwd,mode_id:a}}),!0):(d(h.bindingMissing,"session binding was not found"),!0)}case _.setModel:{const r=String(t.model_id??t.modelId??"").trim();if(!r)return d("set_model_invalid","model_id is required"),!0;const a=this.bindingStore.get(s);return a?.cwd?((a.modelId??"")!==r&&(this.bindingStore.setModelId(s,r),this.globalConfigStore?.set(this.name,{modelId:r}),await this.pool.removeSlot(s),await this.ensureSlotStarted(s),this.claudeWorkerStatus.set(s,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:"ready",cwd:a.cwd,meta:this.buildClaudeToolbarMeta(s)})),o({outcome:"model_set",model_id:r,binding:{cwd:a.cwd,model_id:r}}),!0):(d(h.bindingMissing,"session binding was not found"),!0)}case _.listOptions:{const r=P(),a=this.pool.getSlot(s);return o({modes:[C.fullAuto,C.approval],currentModeId:this.normalizeClaudeModeId(this.bindingStore.get(s)?.modeId),models:r.map(c=>({modelId:c.id,name:c.displayName})),currentModelId:this.bindingStore.get(s)?.modelId??"",available_models:r.map(c=>({id:c.id,displayName:c.displayName})),agent_commands:a?.adapter?.getSupportedCommands?.()??[]}),!0}case _.exec:{const[r,...a]=String(t.args??"").trim().split(/\s+/);if(!r)return d(h.verbInvalid,"Usage: exec <command> [args]"),!0;await this.ensureSlotStarted(s);const l=this.pool.getSlot(s)?.adapter;if(!l?.execCommand)return d(h.verbInvalid,"Agent does not support command execution"),!0;const g=l.getSupportedCommands?.()??[];if(!g.some(m=>m.name===r))return d(h.verbInvalid,`Unknown command: ${r}. Supported: ${g.map(m=>m.name).join(", ")}`),!0;try{const m=await l.execCommand(r,a.join(" "),s);m.status==="ok"?o({outcome:"exec",command:r,message:m.message,data:m.data}):d(h.runtimeError,m.message??`${r} failed`)}catch(m){d(h.runtimeError,`exec error: ${m instanceof Error?m.message:m}`)}return!0}default:return d(h.verbInvalid,`session control verb ${n} is not supported`),!0}}catch(r){const a=r instanceof Error&&r.cwdErrorCode?r.cwdErrorCode:r instanceof Error&&r.sessionControlErrorCode?r.sessionControlErrorCode:h.runtimeError;return u.error(this.name,`handleSessionControlLocalAction error verb=${n} session_id=${s}: ${r instanceof Error?r.message:r}`),d(a,r instanceof Error?r.message:String(r)),!0}}async handleEventCancel(e){const{event_id:i,session_id:t}=e;if(u.info(this.name,`handleEventCancel start event_id=${i} session_id=${t}`),this.pool.cancelEvent(i,t)){if(!(this.pool.getSlot(t)?.adapter?.getActiveEventIds().includes(i)??!1)){this.sendEventResultWithCleanup(i,"canceled","canceled"),this.aibotHandle.sendEventCancelResult({event_id:i,accepted:!0,final_state:"canceled"});return}await this.waitForEventDone(i,t,15e3),this.aibotHandle.sendEventCancelResult({event_id:i,accepted:!0,final_state:"canceled"}),this.pushQueueSnapshotForSession(t);return}this.aibotHandle.sendEventCancelResult({event_id:i,accepted:!1,reason:"event not found or not cancelable"})}waitForEventDone(e,i,t){return new Promise(s=>{const n=this.pool.getSlot(i);if(!n?.adapter){s();return}const o=setTimeout(()=>{n.adapter.removeListener("eventDone",d),s()},t),d=r=>{r===e&&(clearTimeout(o),n.adapter.removeListener("eventDone",d),s())};n.adapter.on("eventDone",d)})}handleAibotStop(e){u.info(this.name,`[stop-trace] handleAibotStop begin session=${e.session_id} event=${e.event_id} stop_id=${e.stop_id||"-"} adapterType=${this.config.adapterType??"acp"}`),this.aibotHandle.sendEventStopAck({stop_id:e.stop_id,event_id:e.event_id,accepted:!0,updated_at:Date.now()});const i=this.pool.getSlot(e.session_id),t=i?.adapter?.getStatus().busy??!1,s=(this.config.adapterType??"acp")==="acp",n=i?.adapter?.getActiveEventIds,o=typeof n=="function"?n.call(i.adapter):[],d=o.length>0?o.includes(e.event_id):t,r=(s||(this.config.adapterType??"acp")==="codex")&&d;if(u.info(this.name,`[stop-trace] handleAibotStop decision session=${e.session_id} event=${e.event_id} slotExists=${!!i?.adapter} busy=${t} activeIds=[${o.join(",")}] stoppingActiveEvent=${d} killOnStop=${r}`),d&&this.sendCtrl.markEventStopped(e.event_id),i?.adapter&&t){const a=r?this.pool.drainQueuedForSession(e.session_id):[],c=l=>{l===e.event_id&&(i.adapter.removeListener("eventDone",c),u.info(this.name,`[stop-trace] handleAibotStop eventDone -> stopResult(stopped) session=${e.session_id} event=${e.event_id} stop_id=${e.stop_id||"-"} killOnStop=${r}`),this.aibotHandle.sendEventStopResult({stop_id:e.stop_id,event_id:e.event_id,status:"stopped",updated_at:Date.now()}),r&&this.killAndResumeStopSlot(e.session_id,a))};i.adapter.on("eventDone",c),this.pool.deliverStopEvent(e.event_id,e.session_id)}else u.info(this.name,`[stop-trace] handleAibotStop slot-not-busy -> immediate stopResult(stopped) session=${e.session_id} event=${e.event_id} slotExists=${!!i?.adapter}`),this.pool.deliverStopEvent(e.event_id,e.session_id),this.aibotHandle.sendEventStopResult({stop_id:e.stop_id,event_id:e.event_id,status:"stopped",updated_at:Date.now()});(this.config.adapterType??"acp")==="pi"&&this.aibotHandle.sendText({event_id:e.event_id,session_id:e.session_id,content:"/stop",msg_type:0})}async killAndResumeStopSlot(e,i){if(!this.stopped){u.info(this.name,`[stop-trace] killAndResumeStopSlot begin session=${e} siblings=${i.length} -> removeSlot (kill process group)`);try{await this.pool.removeSlot(e),u.info(this.name,`[stop-trace] killAndResumeStopSlot removeSlot done session=${e} (process killed) -> redeliver ${i.length} sibling(s)`)}catch(t){u.warn(this.name,`[acp-stop] removeSlot failed session=${e}: ${t instanceof Error?t.message:String(t)}`)}if(!this.stopped)for(const t of i){if(this.stopped)break;try{await this.pool.deliverInboundEvent(t)}catch(s){u.error(this.name,`[acp-stop] sibling redeliver failed event=${t.event_id} session=${e}: ${s instanceof Error?s.message:String(s)}`),this.sendEventResultWithCleanup(t.event_id,"failed",s instanceof Error?s.message:String(s))}}}}handleAibotRevoke(e){if(!e.event_id||!this.revokeHandler.checkAndTrack(e.event_id))return;const i=e.event_id,t=e.session_id;if(t&&this.pool.cancelEvent(i,t)){(this.pool.getSlot(t)?.adapter?.getActiveEventIds().includes(i)??!1)||this.sendEventResultWithCleanup(i,"canceled","revoked");return}if(this.deferredMgr.removeEvent(i)){this.aibotHandle.sendEventResult({event_id:i,status:"canceled",msg:"revoked",updated_at:Date.now()});return}this.pool.deliverStopEvent(i,t||void 0)}async handleAibotLocalAction(e){const i=e.action_type??"",t=String((e.params??{}).session_id??""),s=String((e.params??{}).verb??"").trim().toLowerCase();u.debug(this.name,`local_action received action_type=${i} verb=${s||"-"} action_id=${e.action_id} session_id=${t}`);const n=(this.config.adapterType??"acp")==="claude";if(i===S.sessionControl&&s===_.exec&&await this.handleSessionControlLocalAction(e))return;if(i===S.sessionControl&&s===_.listSessions){await this.handleListSessionsLocalAction(e);return}if(n&&(i===S.interactionReply||i==="exec_approve"||i==="exec_reject")&&(await this.pool.deliverLocalAction(e)).handled||n&&await this.handleSessionControlLocalAction(e))return;if(i===S.sessionControl){const c=this.config.adapterType??"acp",l=c==="codex",g=c==="pi",m=String((e.params??{}).verb??"").trim().toLowerCase();if(l&&m===_.open){await this.handleCodexSessionControlLocalActionOpen(e);return}if(c==="cursor"&&m===_.open){await this.handleCursorSessionControlLocalActionOpen(e);return}if(l&&m==="restart"){const v=this.bindingStore.get(t)?.cwd??"";await this.pool.removeSlot(t).catch(()=>{}),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"restarted",binding:{aibotSessionId:t,cwd:v,workerStatus:"ready"}}});return}if(g&&m===_.open){try{const p=e.params??{},v=String(p.cwd??"").trim();if(!v){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.cwdRequired,error_msg:"session cwd is required"});return}const w=await this.resolveCwdForBinding(v),k=String(p.agent_session_id??"").trim();this.ensureImportedAgentSession(k,w),this.bindingStore.set(t,w),this.setResolvedAgentSessionId(t,k),this.sessionBindings.set(t,w),await this.deferredMgr.replayDeferredPiEvents(t,this.deferredCallbacks()),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:w}),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"opened",binding:this.buildOpenedBindingResult(t,w)}})}catch(p){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:p?.sessionControlErrorCode??p?.cwdErrorCode??h.invalidCwd,error_msg:p instanceof Error?p.message:String(p)})}return}if(g&&m===_.restart){await this.handlePiSessionControlRestartLocalAction(e);return}const b=c==="openhuman",f=c==="opencode";if((b||f)&&m===_.open){try{const p=e.params??{},v=String(p.cwd??"").trim();if(!v){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.cwdRequired,error_msg:"session cwd is required"});return}const w=await this.resolveCwdForBinding(v),k=String(p.agent_session_id??"").trim();this.ensureImportedAgentSession(k,w),this.bindingStore.set(t,w),this.setResolvedAgentSessionId(t,k),this.sessionBindings.set(t,w),b&&await this.deferredMgr.replayDeferredOpenHumanEvents(t,this.deferredCallbacks()),f&&await this.deferredMgr.replayDeferredOpenCodeEvents(t,this.deferredCallbacks()),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"opened",binding:this.buildOpenedBindingResult(t,w)}})}catch(p){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:p?.sessionControlErrorCode??p?.cwdErrorCode??h.invalidCwd,error_msg:p instanceof Error?p.message:String(p)})}return}if(c==="codewhale"&&m===_.open){await this.handleCodeWhaleSessionControlLocalActionOpen(e);return}if(c==="acp"&&m===_.stop){const v=this.bindingStore.get(t)?.cwd??"";await this.pool.removeSlot(t).catch(()=>{}),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"stopped",binding:{aibotSessionId:t,cwd:v,workerStatus:"stopped"}}});return}try{if(m===_.open){const p=e.params??{},v=String(p.cwd??"").trim();if(!v){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.cwdRequired,error_msg:"session cwd is required"});return}const w=String(p.agent_session_id??"").trim();if(w){const k=await this.resolveCwdForBinding(v);this.ensureImportedAgentSession(w,k)}}await this.handleSessionControlLocalActionForPool(e)}catch(p){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:p?.sessionControlErrorCode??p?.cwdErrorCode??h.runtimeError,error_msg:p instanceof Error?p.message:String(p)});return}if(m===_.open){const p=e.params??{},v=await this.resolveCwdForBinding(String(p.cwd??"").trim());this.setResolvedAgentSessionId(t,String(p.agent_session_id??"").trim()),c==="pi"?await this.deferredMgr.replayDeferredPiEvents(t,this.deferredCallbacks()):c==="openhuman"?await this.deferredMgr.replayDeferredOpenHumanEvents(t,this.deferredCallbacks()):c==="opencode"?await this.deferredMgr.replayDeferredOpenCodeEvents(t,this.deferredCallbacks()):c==="agy"?(this.bindingStore.set(t,v),this.sessionBindings.set(t,v),await this.deferredMgr.replayDeferredAgyEvents(t,this.deferredCallbacks())):await this.deferredMgr.replayDeferredAcpEvents(t,this.deferredCallbacks()),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"opened",binding:this.buildOpenedBindingResult(t,v)}}),(this.config.adapterType??"acp")==="agy"&&this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:v,meta:this.buildAgyToolbarMeta(t)})}else Ce(e,this.sessionControlCtx(t),this.sessionControlSenders());return}if(i==="file_list"){const c=Date.now(),l=t?this.bindingStore.get(t)?.cwd:void 0,g=e.params??{},m=String(g.parent_id??"").trim(),b=Array.isArray(g.allowed_extensions)?g.allowed_extensions.filter(v=>typeof v=="string").map(v=>v.trim()).filter(v=>v.length>0):[];u.info("file-list-diag",`plugin << recv action_id=${e.action_id} session_id=${t} parent_id=${m||"<root>"} show_hidden=${!!g.show_hidden} ext_count=${b.length} bound_cwd=${l??"<none>"}`);const f=await Ee({parent_id:m||null,session_id:t,show_hidden:!!g.show_hidden,allowed_extensions:b},{resolveCwd:()=>l??this.config.agent.cwd??process.cwd(),fallbackDir:K()}),E=Date.now()-c,p=f.result?.files?.length??0;u.info("file-list-diag",`plugin -> reply action_id=${e.action_id} status=${f.status} elapsed=${E}ms count=${p} current_path=${f.result?.current_path??""} error_code=${f.error_code??""} error_msg=${f.error_msg??""}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:f.status,...f.result?{result:f.result}:{},...f.error_code?{error_code:f.error_code}:{},...f.error_msg?{error_msg:f.error_msg}:{}});return}if(i==="create_folder"){const c=t?this.bindingStore.get(t)?.cwd:void 0,l=String((e.params??{}).parent_id??"").trim(),g=String((e.params??{}).name??"").trim(),m=await Re({parent_id:l||null,name:g,session_id:t},{resolveCwd:()=>c??this.config.agent.cwd??process.cwd(),fallbackDir:K()});this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:m.status,...m.result?{result:m.result}:{},...m.error_code?{error_code:m.error_code}:{},...m.error_msg?{error_msg:m.error_msg}:{}});return}if(i===S.setModel&&(this.config.adapterType??"acp")==="agy"){await this.handleAgySetModel(e,t);return}if(i===S.getSessionUsage){await this.handleGetSessionUsage(e,t);return}if(i===S.getRateLimits){await this.handleGetRateLimits(e,t);return}const o=(this.config.adapterType??"acp")==="acp";if((n||o)&&i===S.threadCompact){await this.handleThreadCompact(e,t);return}if(i==="connector_rollback"){await this.handleConnectorRollback(e);return}if(i==="connector_upgrade_push"){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{accepted:!0}}),this.upgradeTrigger?.();return}if(i===S.getAgentGlobalConfig){const c=this.globalConfigStore?.get(this.name);this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{agentName:this.name,...c?{config:c}:{config:null}}});return}const d=this.config.adapterType??"acp",r=(d==="codex"||d==="cursor"||d==="pi"||d==="openhuman"||d==="opencode"||d==="acp")&&!!t&&!!this.bindingStore.get(t)?.cwd,a=await this.pool.deliverLocalAction(e,{autoCreateSlot:r});if(a.handled){if(a.kind==="set_mode"){const c=String((e.params??{}).mode_id??"");c&&(d==="cursor"||d==="claude"?this.bindingStore.setModeId(t,c):T.has(d)||this.globalConfigStore?.set(this.name,{acpInitialMode:c}))}else if(a.kind==="set_model"){const c=String((e.params??{}).model_id??"");c&&(d==="cursor"?(this.bindingStore.setModelId(t,c),this.globalConfigStore?.set(this.name,{modelId:c})):T.has(d)||this.globalConfigStore?.set(this.name,{modelId:c}))}else if(a.kind==="set_reasoning_effort"){const c=String((e.params??{}).reasoning_effort??(e.params??{}).reasoning_eff??(e.params??{}).effort??"");c&&this.globalConfigStore?.set(this.name,{codexReasoningEffort:c})}else if(a.kind==="set_sandbox_mode"){const c=String((e.params??{}).sandbox_mode??(e.params??{}).sandboxMode??"");if(c){const l=c==="default"?void 0:c;this.globalConfigStore?.set(this.name,{codexSandboxMode:l})}}return}if((d==="codex"||d==="cursor"||d==="pi"||d==="openhuman"||d==="opencode")&&t&&!this.bindingStore.get(t)?.cwd){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.bindingMissing,error_msg:"Session binding missing. Open a workspace first."});return}if(d==="acp"&&(i==="set_mode"||i==="set_model")){const c=this.sessionControlSenders(),l=this.pool.getSlot(t)?.adapter,g={bindingStore:this.bindingStore,acpAdapter:l instanceof A?l:null,globalConfigStore:this.globalConfigStore,agentName:this.name,log:u};if(i==="set_mode"){const m=String((e.params??{}).mode_id??""),b=await N(g,t,m);if(b.status==="failed")c.sendLocalActionResult(e.action_id,"failed",void 0,b.errorCode,b.errorMsg);else{const f=l instanceof A?l.buildToolbarContext(b.result?.outcome==="mode_set"?"mode_set":"mode_set_failed"):null,E=f?{...f,...b.result}:b.result;c.sendLocalActionResult(e.action_id,"ok",E)}}else{const m=String((e.params??{}).model_id??""),b=await W(g,t,m);if(b.status==="failed")c.sendLocalActionResult(e.action_id,"failed",void 0,b.errorCode,b.errorMsg);else{const f=l instanceof A?l.buildToolbarContext(b.result?.outcome==="model_set"?"model_set":"model_set_failed"):null,E=f?{...f,...b.result}:b.result;c.sendLocalActionResult(e.action_id,"ok",E)}}return}this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"unsupported_local_action",error_msg:`action type ${i} is not supported`})}async handleGetSessionUsage(e,i){if(!i){u.warn(this.name,`[usage] get_session_usage rejected: no session_id in params, action_id=${e.action_id}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"session_id_required",error_msg:"session_id is required for get_session_usage"});return}const t=this.config.adapterType??"acp",s=this.bindingStore.get(i),n=s?.cwd??this.config.agent.cwd??process.cwd();u.info(this.name,`[usage] get_session_usage action_id=${e.action_id} session_id=${i} adapterType=${t} hasBinding=${!!s} cwd=${n}`);let o=null,d,r;switch(t){case"claude":{if(d=s?.claudeSessionId,!d){u.warn(this.name,`[usage] no claude binding for session_id=${i}, action_id=${e.action_id}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No Claude session binding found"});return}u.info(this.name,`[usage] parsing claude usage: claudeSessionId=${d} cwd=${n}`),o=await ne(d,n),r="claude";break}case"agy":{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{sessionId:i,adapterType:"agy",available:!1,reason:"agy print-mode adapter does not track token usage"}});return}case"acp":default:{if(d=s?.acpSessionId,!d){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No ACP session binding found"});return}o=await oe(d,n,this.config.aibot.clientType),r=t,(!r||r==="acp")&&(r="acp");break}case"codex":{if(d=s?.codexThreadId,!d){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No Codex thread binding found"});return}o=await re(d),r="codex";break}case"pi":{if(d=s?.piSessionPath,!d){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No Pi session path binding found"});return}o=await ue(d),r="pi";break}case"cursor":{const c=this.pool.getSlot(i)?.adapter;if(!(c instanceof L)){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No Cursor session found"});return}const l=c.getUsageSnapshot(i);if(!l){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"usage_not_found",error_msg:"No usage data found for this session"});return}this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{sessionId:i,adapterType:"cursor",models:[{modelId:this.bindingStore.get(i)?.modelId??"auto",turns:l.turns,input:l.total.input,output:l.total.output,cacheRead:l.total.cacheRead,cacheWrite:l.total.cacheWrite}],total:{input:l.total.input,output:l.total.output,cacheRead:l.total.cacheRead,cacheWrite:l.total.cacheWrite},turns:l.turns,sampledAt:l.sampledAt}});return}case"codewhale":{const c=this.pool.getSlot(i)?.adapter;if(!(c instanceof q)){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No CodeWhale session found"});return}const l=c.getUsageSnapshot();if(!l){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"usage_not_found",error_msg:"No usage data found for this session"});return}this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{sessionId:i,adapterType:"codewhale",models:[{modelId:this.bindingStore.get(i)?.modelId??"codewhale",turns:l.turns,input:l.total.input,output:l.total.output}],total:{input:l.total.input,output:l.total.output},turns:l.turns,sampledAt:l.sampledAt}});return}}if(!o){u.info(this.name,`[usage] no usage data found: session_id=${i} adapterSessionId=${d} adapterType=${r}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"usage_not_found",error_msg:"No usage data found for this session"});return}u.info(this.name,`[usage] result ok: session_id=${i} adapterSessionId=${d} turns=${o.turns} models=${o.models.length}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{sessionId:d,adapterType:r,models:o.models,total:o.total,turns:o.turns,sampledAt:new Date().toISOString()}})}async handleThreadCompact(e,i){if(!i){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"session_id_required",error_msg:"session_id is required for thread_compact"});return}if(!this.bindingStore.get(i)?.cwd){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.bindingMissing,error_msg:"session binding was not found"});return}try{await this.ensureSlotStarted(i);const n=this.pool.getSlot(i)?.adapter;if(!n?.execCommand){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.runtimeError,error_msg:"Agent does not support command execution"});return}u.info(this.name,`thread_compact session_id=${i} action_id=${e.action_id}`);const o=await n.execCommand("compact","",i);o.status==="ok"?this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"compacted",message:o.message,data:o.data}}):this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.runtimeError,error_msg:o.message??"compact failed"})}catch(s){u.warn(this.name,`thread_compact error session_id=${i}: ${s instanceof Error?s.message:s}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.runtimeError,error_msg:s instanceof Error?s.message:String(s)})}}async handleGetRateLimits(e,i){const t=this.config.adapterType??"acp";if(this.config.aibot.clientType==="kiro"){const s=this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs),n=this.cachedAcpContextWindow;if(s&&this.cachedProviderQuota){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"acp",available:!0,cached:!0,sampledAt:this.cachedProviderQuotaSampledAtMs,rateLimits:null,contextWindow:n,tokenUsage:null,providerQuota:this.cachedProviderQuota}});return}try{const o=await B();this.cachedProviderQuota=o,this.cachedProviderQuotaSampledAtMs=Date.now(),u.info(this.name,`[rate-limits] kiro quota queried: success=${o.success}`+(o.balance?` balance=${o.balance.remaining} ${o.balance.unit}`:"")+(o.error?` error=${o.error}`:"")),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"acp",available:!0,cached:!1,sampledAt:this.cachedProviderQuotaSampledAtMs,rateLimits:null,contextWindow:n,tokenUsage:null,providerQuota:o}})}catch(o){u.warn(this.name,`[rate-limits] kiro quota query failed: ${o instanceof Error?o.message:String(o)}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"acp",available:!1,cached:!1,sampledAt:null,rateLimits:null,contextWindow:n,tokenUsage:null}})}return}switch(t){case"codex":{const s=this.maybeQueryProviderQuota(),n=this.getFreshCodexGlobalRateLimitCache();if(n.hasData&&!this.isRateLimitsCacheFresh(this.cachedRateLimitsSampledAtMs)&&!this.isRateLimitsCacheFresh(this.cachedCodexUsageSampledAtMs)){const l=this.pool.getAllSlots().find(g=>g.state==="ready"&&g.adapter);if(l&&(u.info(this.name,`[rate-limits] codex cache stale, refreshing from slot: session=${l.sessionId}`),(await l.adapter.handleLocalAction?.(e))?.handled))return}if(n.hasData){n.rateLimits?u.info(this.name,`[rate-limits] codex cached: primary=${n.rateLimits.primary.usedPercent.toFixed(1)}% resetsAt=${n.rateLimits.primary.resetsAt} secondary=${n.rateLimits.secondary.usedPercent.toFixed(1)}% resetsAt=${n.rateLimits.secondary.resetsAt}`):u.info(this.name,`[rate-limits] codex cached context/token only: hasContext=${!!n.contextWindow} hasToken=${!!n.tokenUsage}`);const l=await s;this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"codex",available:n.hasData||!!l,cached:!0,sampledAt:n.sampledAt,rateLimits:n.rateLimits,contextWindow:n.contextWindow,tokenUsage:n.tokenUsage,providerQuota:l}});return}const d=this.pool.getAllSlots().find(l=>l.state==="ready"&&l.adapter);if(d&&(u.info(this.name,`[rate-limits] codex reuse existing slot: session=${d.sessionId}`),(await d.adapter.handleLocalAction?.(e))?.handled))return;const r=this.resolveRateLimitWakeSessionId(i,t);if(r){const l=await this.wakeRateLimitSlot(r,t);if(l?.adapter&&(await l.adapter.handleLocalAction?.(e))?.handled)return}const a=await s,c=!!a;u.info(this.name,`[rate-limits] codex no native data, providerQuota=${a?a.provider:"none"} available=${c}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"codex",available:c,cached:!1,sampledAt:null,rateLimits:null,contextWindow:null,tokenUsage:null,providerQuota:a}});return}case"claude":{const s=this.maybeQueryProviderQuota(),n=this.getFreshClaudeRateLimitState();if(n){const c=n.rateLimits,l=await s;u.info(this.name,`[rate-limits] claude global cached: sampledAt=${n.sampledAt} hasRateLimits=${!!c}`+(l?` providerQuota=${l.provider}`:"")),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"claude",available:!0,cached:!0,sampledAt:n.sampledAt,rateLimits:n.rateLimits??null,contextWindow:n.contextWindow,tokenUsage:null,providerQuota:l}});return}let o=this.pool.getAllSlots().find(c=>c.state==="ready"&&c.adapter instanceof y)??null;if(!o){const c=this.resolveRateLimitWakeSessionId(i,t);c&&(o=await this.wakeRateLimitSlot(c,t))}u.info(this.name,`[rate-limits] handleGetRateLimits: session_id=${i} adapterType=claude hasSlot=${!!o} hasAdapter=${!!o?.adapter}`);const d=o?.adapter,r=d instanceof y?d.getSessionState():null,a=await s;if(r){(r.rateLimits?.fiveHour||r.rateLimits?.sevenDay||r.contextWindow?.usedPercentage!=null)&&(this.cachedClaudeRateLimitState=r);const c=this.getFreshClaudeRateLimitState(),l=c?.rateLimits&&!r.rateLimits?{...r,rateLimits:c.rateLimits}:r,g=l.rateLimits;u.info(this.name,`[rate-limits] claude global state: sampledAt=${l.sampledAt} hasRateLimits=${!!g}`+(g?` fiveHour=${g.fiveHour?.usedPercentage??"n/a"}% resetsAt=${g.fiveHour?.resetsAt??"n/a"} sevenDay=${g.sevenDay?.usedPercentage??"n/a"}% resetsAt=${g.sevenDay?.resetsAt??"n/a"}`:"")+(c?.rateLimits&&!r.rateLimits?" source=live+cached-fallback":" source=live")+(a?` providerQuota=${a.provider}`:"")),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"claude",available:!0,cached:!1,sampledAt:l.sampledAt,rateLimits:l.rateLimits??null,contextWindow:l.contextWindow,tokenUsage:null,providerQuota:a}})}else u.info(this.name,`[rate-limits] claude no global state: hasAdapter=${!!d} adapterType=${d?.constructor?.name??"n/a"}`+(a?` providerQuota=${a.provider}`:"")),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"claude",available:!!a,cached:!1,sampledAt:null,rateLimits:null,contextWindow:null,tokenUsage:null,providerQuota:a}});return}case"cursor":{const n=(i?this.pool.getSlot(i):null)?.adapter,o=n instanceof L?n.getRateLimitsSnapshot():{adapterType:"cursor",available:!1,cached:!1,sampledAt:null,rateLimits:null,contextWindow:null,tokenUsage:null};this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:o});return}default:{const s=this.config.providerBaseUrl,n=this.config.providerApiKey;if(!s||!n){u.info(this.name,`[rate-limits] no provider config for adapterType=${t}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:t,available:!1,cached:!1,sampledAt:null,rateLimits:null,contextWindow:null,tokenUsage:null}});return}if(this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)&&this.cachedProviderQuota){u.info(this.name,`[rate-limits] provider quota cached: provider=${this.cachedProviderQuota.provider} success=${this.cachedProviderQuota.success}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:t,available:!0,cached:!0,sampledAt:this.cachedProviderQuotaSampledAtMs,rateLimits:null,contextWindow:null,tokenUsage:null,providerQuota:this.cachedProviderQuota}});return}try{const d=await z(s,n);this.cachedProviderQuota=d,this.cachedProviderQuotaSampledAtMs=Date.now(),u.info(this.name,`[rate-limits] provider quota queried: provider=${d.provider} success=${d.success}`+(d.tiers.length>0?` tiers=${d.tiers.map(r=>`${r.name}=${r.usedPercent}%`).join(",")}`:"")+(d.balance?` balance=${d.balance.remaining} ${d.balance.unit}`:"")+(d.error?` error=${d.error}`:"")),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:t,available:!0,cached:!1,sampledAt:this.cachedProviderQuotaSampledAtMs,rateLimits:null,contextWindow:null,tokenUsage:null,providerQuota:d}})}catch(d){u.warn(this.name,`[rate-limits] provider quota query failed: ${d instanceof Error?d.message:String(d)}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:t,available:!1,cached:!1,sampledAt:null,rateLimits:null,contextWindow:null,tokenUsage:null}})}return}}}resolveRateLimitWakeSessionId(e,i){const t=String(e??"").trim(),s=this.bindingStore.getMostRecentlyUpdatedSessionId({requireCwd:!0});return s?t&&t===s?t:i==="codex"||i==="claude"?(t&&t!==s&&u.info(this.name,`[rate-limits] ${i} remap wake session: requested=${t} use_recent=${s}`),s):t&&t!==s?(u.info(this.name,`[rate-limits] skip wake slot: session=${t} adapterType=${i} reason=not_recent_session recent=${s}`),null):t||null:(u.info(this.name,`[rate-limits] skip wake slot: adapterType=${i} reason=no_recent_binding`),null)}async wakeRateLimitSlot(e,i){const t=String(e??"").trim();if(!t)return null;if(!this.bindingStore.get(t)?.cwd)return u.info(this.name,`[rate-limits] skip wake slot: session=${t} adapterType=${i} reason=binding_cwd_missing`),null;try{const n=this.pool.getOrCreateSlot(t);if(!n)return u.info(this.name,`[rate-limits] skip wake slot: session=${t} adapterType=${i} reason=slot_unavailable`),null;if(n.startPromise){const o=i==="claude"?6e4:2e4;await Promise.race([n.startPromise,new Promise((d,r)=>setTimeout(()=>r(new Error(`wake rate-limits slot timeout (${o}ms)`)),o))])}return u.info(this.name,`[rate-limits] wake slot success: session=${t} adapterType=${i}`),this.pool.getSlot(t)??n}catch(n){return u.warn(this.name,`[rate-limits] wake slot failed: session=${t} adapterType=${i} err=${n instanceof Error?n.message:String(n)}`),null}}sessionControlCtx(e){const i=this.pool.getSlot(e),t=i?.adapter instanceof A?i.adapter:null,s=(this.config.adapterType??"acp")==="acp",n={bindingStore:this.bindingStore,acpAdapter:t,globalConfigStore:this.globalConfigStore,agentName:this.name,log:u};return{getCwd:()=>this.bindingStore.get(e)?.cwd??this.config.agent.cwd??process.cwd(),getSessionBindings:()=>{if(t)return t.getSessionBindings();const o=this.sessionBindings;if(e&&!o.has(e)){const d=this.bindingStore.get(e);d?.cwd&&o.set(e,d.cwd)}return o},getStatus:()=>this.getStatus(),isAcpAlive:!!t?.isAlive(),getAcpSessionOptions:()=>t?.acpSessionOptions??null,setMode:o=>t?t.setMode(o):Promise.resolve(!1),setModel:o=>t?t.setModel(o):Promise.resolve(!1),acpSetMode:s?(o,d)=>N(n,o,d):void 0,acpSetModel:s?(o,d)=>W(n,o,d):void 0,getPendingApproval:o=>{const d=t?.pendingApprovalEntries.get(o);return d?{requestId:d}:void 0},deletePendingApproval:o=>t?.pendingApprovalEntries.delete(o)??!1,respondPermission:(o,d)=>(t&&t.respondToPermission(o,d),Promise.resolve()),onSessionBound:(o,d)=>{this.bindingStore.set(o,d)},onSessionUnbound:o=>{const r=this.bindingStore.get(o)?.cwd??"";this.bindingStore.delete(o),this.sessionBindings.delete(o),this.claudeWorkerStatus.delete(o),this.deferredMgr.clearSession(o),r&&this.aibotHandle.sendUpdateBindingCard({session_id:o,worker_status:"stopped",cwd:r})},cancelActiveRun:()=>(this.config.adapterType??"acp")==="agy"&&i?.adapter instanceof O?(i.adapter.cancelCurrentRun(),Promise.resolve()):i?.adapter?.cancel("")??Promise.resolve(),onModeSet:o=>{const d=this.config.adapterType??"acp";T.has(d)||this.globalConfigStore?.set(this.name,{acpInitialMode:o})},onModelSet:o=>{const d=this.config.adapterType??"acp";T.has(d)||this.globalConfigStore?.set(this.name,{modelId:o})}}}sessionControlSenders(){return{sendEventAck:(e,i)=>this.aibotHandle.sendEventAck({event_id:e,session_id:i,received_at:Date.now()}),sendEventResult:(e,i,t)=>this.aibotHandle.sendEventResult({event_id:e,status:i,...t?.msg?{msg:t.msg}:{},...t?.code?{code:t.code}:{},updated_at:Date.now()}),sendLocalActionResult:(e,i,t,s,n)=>this.aibotHandle.sendLocalActionResult({action_id:e,status:i,...t?{result:t}:{},...s?{error_code:s}:{},...n?{error_msg:n}:{}})}}resolveBindingChannelKey(e){switch(e){case"claude":return"grix-claude";case"codex":return"codex";case"cursor":return"cursor";case"pi":return"pi";case"openhuman":return"openhuman";case"codewhale":return"codewhale";case"opencode":return"opencode";case"agy":return"acp";case"acp":return this.config.aibot.clientType==="qwen"?"qwen":"acp";default:return e}}finalizeThinking(e,i){this.sendCtrl.finalizeThinking(e,i)}logInboundConversation(e){const i=String(e.session_id??"").trim();i&&this.conversationLog?.logInbound(i,{event_id:e.event_id,msg_id:e.msg_id,sender_id:e.sender_id,msg_type:e.msg_type,content:e.content??""})}buildInboundEvent(e,i){const t=Te(this.sendCtrl.getGlobalRuntimeConfig(),i);return{event_id:e.event_id,session_id:e.session_id,thread_id:e.thread_id,sender_id:e.sender_id,msg_id:e.msg_id,msg_type:e.msg_type,content:e.content??"",quoted_message_id:e.quoted_message_id,context_messages_json:e.context_messages?JSON.stringify(e.context_messages):void 0,extra_json:e.extra?JSON.stringify(e.extra):void 0,connector_runtime_config:{response_delivery:t.responseDelivery,tool_events:t.toolEvents,thinking_events:t.thinkingEvents},session_type:e.session_type,created_at:e.created_at}}isStaleClaudeEvent(e){const i=Number(e.created_at);return!Number.isFinite(i)||i<=0?!1:Date.now()-i>Ue}async handleConnectorRollback(e){const i=String((e.params??{}).target_version??"").trim(),t=String((e.params??{}).reason??"server_initiated");if(!i){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"MISSING_TARGET_VERSION",error_msg:"target_version is required for connector_rollback"});return}u.info(this.name,`connector_rollback: target=${i} reason=${t}`);try{const{npmInstall:s,writePending:n,removePending:o,upgradeLog:d}=await import("../core/upgrade/npm-upgrader.js"),{resolveClientVersion:r}=await import("../core/util/client-version.js"),a=r();d(`server rollback: ${a} -> ${i} reason=${t}`),n(a,i),await s("grix-connector",i),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{rolled_back_to:i}}),process.kill(process.pid,"SIGTERM")}catch(s){try{const{removePending:o}=await import("../core/upgrade/npm-upgrader.js");o()}catch{}const n=s instanceof Error?s.message:String(s);u.error(this.name,`connector_rollback failed: ${n}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"ROLLBACK_FAILED",error_msg:n})}}setUpgradeTrigger(e){this.upgradeTrigger=e}}export{Tt as AgentInstance};
15
+ `)}`,updated_at:Date.now()})}async handleListSessionsLocalAction(e){const n=this.config.adapterType??"acp",t=new Map;for(const r of this.pool.getAllSlots())t.set(r.sessionId,r);const s=Array.from(this.bindingStore.entries()),i=new Map;for(const[r,a]of s){const c=this.resolveAgentSessionId(a);if(c){const l=t.get(r),g=l?l.adapter.getStatus().busy?"busy":"ready":"inactive";i.set(c,{aibotSessionId:r,workerStatus:g,bindingUpdatedAt:a.updatedAt??0})}}const o=[],d=new Set;if(n==="codex"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.threadId);const c=i.get(a.threadId),l={agentSessionId:a.threadId,cwd:a.cwd||null,workerStatus:c?.workerStatus??(a.archived?"archived":"inactive"),createdAt:a.createdAt,updatedAt:a.updatedAt,archived:a.archived};c&&(l.aibotSessionId=c.aibotSessionId),a.title&&(l.title=a.title),o.push(l)}}else if(n==="claude"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.sessionId);const c=i.get(a.sessionId),l={agentSessionId:a.sessionId,cwd:a.cwd||null,workerStatus:c?.workerStatus??"inactive",updatedAt:a.updatedAt};c&&(l.aibotSessionId=c.aibotSessionId),a.title&&(l.title=a.title),o.push(l)}}else if(n==="acp"){const r=this.sessionScanCache.get();for(const a of r){d.add(a.sessionId);const c=i.get(a.sessionId),l=c&&c.bindingUpdatedAt>0?c.bindingUpdatedAt:a.updatedAt,g={agentSessionId:a.sessionId,cwd:a.cwd||null,agentType:a.agentType,workerStatus:c?.workerStatus??"inactive",updatedAt:l};c&&(g.aibotSessionId=c.aibotSessionId),a.title&&(g.title=a.title),a.createdAt&&(g.createdAt=a.createdAt),o.push(g)}}for(const[r,a]of s){const c=this.resolveAgentSessionId(a);if(c&&d.has(c))continue;const l=t.get(r),g=l?l.adapter.getStatus().busy?"busy":"ready":"closed",m={aibotSessionId:r,cwd:a.cwd??null,workerStatus:g,updatedAt:a.updatedAt,title:c?`${c.slice(0,8)}\u2026`:`${r.slice(0,8)}\u2026`};c&&(m.agentSessionId=c),o.push(m)}o.sort((r,a)=>a.updatedAt-r.updatedAt),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"sessions_listed",sessions:o,total:o.length}})}async handleSessionControlCommand(e,n){const t=n.session_id,s=(i,o,d)=>{this.aibotHandle.sendEventResult({event_id:n.event_id,status:i,...o?{msg:o}:{},...d?{code:d}:{},updated_at:Date.now()})};this.aibotHandle.sendEventAck({event_id:n.event_id,session_id:t,received_at:Date.now()});try{switch(e.verb){case _.open:{await P().catch(()=>{});const i=e.args.trim();if(!i){s("failed","Usage: /grix open <working-directory>",h.cwdRequired);return}let o="";try{o=await this.resolveCwdForBinding(i)}catch(r){s("failed",r instanceof Error?r.message:String(r),h.invalidCwd);return}const d=this.bindingStore.get(t);if(d?.cwd){const r=await this.resolveCwdForBinding(d.cwd);if(r!==o){s("failed","session binding cannot be changed to another working directory",h.rebindForbidden);return}this.bindingStore.ensureModeId(t,C.fullAuto),this.sessionBindings.set(t,r),await this.ensureSlotStarted(t),this.claudeWorkerStatus.set(t,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:r,meta:this.buildClaudeToolbarMeta(t)}),await this.deferredMgr.replayDeferredClaudeEvents(t,this.deferredCallbacks()),s("responded",`Working directory already bound: ${r}`);return}this.bindingStore.set(t,o,{modeId:C.fullAuto}),this.sessionBindings.set(t,o),await this.ensureSlotStarted(t),this.claudeWorkerStatus.set(t,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:o,meta:this.buildClaudeToolbarMeta(t)}),await this.deferredMgr.replayDeferredClaudeEvents(t,this.deferredCallbacks()),s("responded",`Session bound to ${o}`);return}case _.where:{const i=this.bindingStore.get(t);if(!i?.cwd){s("failed","session binding was not found",h.bindingMissing);return}s("responded",`Working directory: ${i.cwd}`);return}case _.status:{const i=this.bindingStore.get(t);if(!i?.cwd){s("failed","session binding was not found",h.bindingMissing);return}const o=this.normalizeClaudeModeId(i.modeId),d=this.getClaudeWorkerStatus(t);s("responded",`Status: worker=${d} mode=${o} cwd=${i.cwd}`);return}case _.stop:{const i=this.bindingStore.get(t);if(!i?.cwd){s("failed","session binding was not found",h.bindingMissing);return}await this.pool.removeSlot(t),this.claudeWorkerStatus.set(t,"stopped"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"stopped",cwd:i.cwd}),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"stopped",cwd:i.cwd,meta:this.buildClaudeToolbarMeta(t)}),s("responded",`Session worker stopped for ${i.cwd}`);return}case _.restart:{const i=this.bindingStore.get(t);if(!i?.cwd){s("failed","session binding was not found",h.bindingMissing);return}await this.pool.removeSlot(t),await this.ensureSlotStarted(t),this.claudeWorkerStatus.set(t,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:i.cwd,meta:this.buildClaudeToolbarMeta(t)}),s("responded",`Session worker restarted for ${i.cwd}`);return}case _.setMode:{const i=e.args.trim();if(!i){s("failed","Usage: /grix set_mode <mode-id>",x.modeInvalid);return}const o=this.normalizeClaudeModeId(i);if(o!==i.toLowerCase()){s("failed","set mode_id is invalid",x.modeInvalid);return}const d=this.bindingStore.get(t);if(!d?.cwd){s("failed","session binding was not found",h.bindingMissing);return}if(this.normalizeClaudeModeId(d.modeId)===o){s("responded",`Mode unchanged: ${o}`);return}this.bindingStore.setModeId(t,o),await this.pool.removeSlot(t),await this.ensureSlotStarted(t),this.claudeWorkerStatus.set(t,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:d.cwd,meta:this.buildClaudeToolbarMeta(t)}),s("responded",`Mode set to ${o}`);return}case _.setModel:{const i=e.args.trim();if(!i){s("failed","Usage: /grix set_model <model-id>");return}const o=this.bindingStore.get(t);if(!o?.cwd){s("failed","session binding was not found",h.bindingMissing);return}if((o.modelId??"")===i){s("responded",`Model unchanged: ${i}`);return}this.bindingStore.setModelId(t,i),await this.pool.removeSlot(t),await this.ensureSlotStarted(t),this.claudeWorkerStatus.set(t,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:o.cwd,meta:this.buildClaudeToolbarMeta(t)}),s("responded",`Model set to ${i}`);return}case _.listOptions:{const i=D(),o=this.bindingStore.get(t),d=i.map(a=>a.id).join(", "),r=`${C.fullAuto}, ${C.approval}`;s("responded",`Modes (current: ${this.normalizeClaudeModeId(o?.modeId)}): ${r}
16
+ Models (current: ${o?.modelId??"default"}): ${d}`);return}case _.exec:{const[i,...o]=e.args.trim().split(/\s+/);if(!i){s("failed","Usage: /grix exec <command> [args]",h.verbInvalid);return}const r=this.pool.getSlot(t)?.adapter;if(!r?.execCommand){s("failed","Agent does not support command execution",h.verbInvalid);return}const a=r.getSupportedCommands?.()??[];if(!a.some(c=>c.name===i)){s("failed",`Unknown command: ${i}. Supported: ${a.map(c=>c.name).join(", ")}`,h.verbInvalid);return}try{const c=await r.execCommand(i,o.join(" "),t);s(c.status==="ok"?"responded":"failed",c.message??`${i} ${c.status}`,c.status==="ok"?void 0:h.runtimeError)}catch(c){s("failed",`exec error: ${c instanceof Error?c.message:c}`,h.runtimeError)}return}default:s("failed",`Unsupported command for Claude: /grix ${e.verb}`,h.verbInvalid)}}catch(i){s("failed",i instanceof Error?i.message:String(i),h.runtimeError)}}async handleSessionControlLocalAction(e){const n=String(e.action_type??"").trim();if(n!==w.sessionControl&&n!==w.setMode&&n!=="set_mode"&&n!==w.setModel&&n!=="set_model")return!1;const t=e.params??{},s=String(t.session_id??"").trim(),i=n===w.setMode?_.setMode:n===w.setModel?_.setModel:String(t.verb??"").trim().toLowerCase();if(u.info(this.name,`handleSessionControlLocalAction verb=${i} action_id=${e.action_id} session_id=${s}`),!s)return this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:x.localActionRouteMissing,error_msg:"local action session_id is required"}),!0;const o=r=>{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:r})},d=(r,a)=>{u.warn(this.name,`session_control local_action failed action_id=${e.action_id} session_id=${s} verb=${i} code=${r} msg=${a}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:r,error_msg:a})};try{switch(i){case _.open:{await P().catch(()=>{});const r=String(t.cwd??"").trim(),a=String(t.agent_session_id??"").trim();if(!r)return d(h.cwdRequired,"session control cwd is required"),!0;u.info(this.name,`handleSessionControlLocalAction open cwd=${r} session_id=${s}`);const c=await this.resolveCwdForBinding(r);this.ensureImportedAgentSession(a,c);const l=this.bindingStore.get(s);if(l?.cwd){const g=await this.resolveCwdForBinding(l.cwd);return g!==c?(d(h.rebindForbidden,"session binding cannot be changed to another working directory"),!0):(this.bindingStore.ensureModeId(s,C.fullAuto),this.setResolvedAgentSessionId(s,a),this.sessionBindings.set(s,g),await this.ensureSlotStarted(s),await this.replayDeferredEventsForSession(s),this.claudeWorkerStatus.set(s,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:"ready",cwd:g,meta:this.buildClaudeToolbarMeta(s)}),o({outcome:"opened",binding:{...this.buildOpenedBindingResult(s,g),mode_id:this.normalizeClaudeModeId(this.bindingStore.get(s)?.modeId)}}),!0)}return this.bindingStore.set(s,c,{modeId:C.fullAuto}),this.setResolvedAgentSessionId(s,a),this.sessionBindings.set(s,c),await this.ensureSlotStarted(s),await this.replayDeferredEventsForSession(s),this.claudeWorkerStatus.set(s,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:"ready",cwd:c,meta:this.buildClaudeToolbarMeta(s)}),o({outcome:"opened",binding:{...this.buildOpenedBindingResult(s,c),mode_id:this.normalizeClaudeModeId(this.bindingStore.get(s)?.modeId)}}),!0}case _.status:case _.where:{const r=this.bindingStore.get(s);return r?.cwd?(o({outcome:i,binding:{cwd:r.cwd,mode_id:this.normalizeClaudeModeId(r.modeId),worker_status:this.getClaudeWorkerStatus(s)}}),!0):(d(h.bindingMissing,"session binding was not found"),!0)}case _.stop:{const r=this.bindingStore.get(s);return r?.cwd?(await this.pool.removeSlot(s),o({outcome:"stopped",binding:{cwd:r.cwd,mode_id:this.normalizeClaudeModeId(r.modeId),worker_status:"stopped"}}),!0):(d(h.bindingMissing,"session binding was not found"),!0)}case _.restart:{const r=this.bindingStore.get(s);return r?.cwd?(await this.pool.removeSlot(s),await this.ensureSlotStarted(s),this.claudeWorkerStatus.set(s,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:"ready",cwd:r.cwd,meta:this.buildClaudeToolbarMeta(s)}),o({outcome:"restarted",binding:{cwd:r.cwd,mode_id:this.normalizeClaudeModeId(r.modeId)}}),!0):(d(h.bindingMissing,"session binding was not found"),!0)}case _.setMode:{const r=String(t.mode_id??t.modeId??"").trim();if(!r)return d(x.modeInvalid,"set mode_id is invalid"),!0;const a=this.normalizeClaudeModeId(r);if(a!==r.toLowerCase())return d(x.modeInvalid,"set mode_id is invalid"),!0;const c=this.bindingStore.get(s);return c?.cwd?(this.normalizeClaudeModeId(c.modeId)!==a&&(this.bindingStore.setModeId(s,a),await this.pool.removeSlot(s),await this.ensureSlotStarted(s),this.claudeWorkerStatus.set(s,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:"ready",cwd:c.cwd,meta:this.buildClaudeToolbarMeta(s)})),o({outcome:"mode_set",mode_id:a,binding:{cwd:c.cwd,mode_id:a}}),!0):(d(h.bindingMissing,"session binding was not found"),!0)}case _.setModel:{const r=String(t.model_id??t.modelId??"").trim();if(!r)return d("set_model_invalid","model_id is required"),!0;const a=this.bindingStore.get(s);return a?.cwd?((a.modelId??"")!==r&&(this.bindingStore.setModelId(s,r),await this.pool.removeSlot(s),await this.ensureSlotStarted(s),this.claudeWorkerStatus.set(s,"ready"),this.aibotHandle.sendUpdateBindingCard({session_id:s,worker_status:"ready",cwd:a.cwd,meta:this.buildClaudeToolbarMeta(s)})),o({outcome:"model_set",model_id:r,binding:{cwd:a.cwd,model_id:r}}),!0):(d(h.bindingMissing,"session binding was not found"),!0)}case _.listOptions:{const r=D(),a=this.pool.getSlot(s);return o({modes:[C.fullAuto,C.approval],currentModeId:this.normalizeClaudeModeId(this.bindingStore.get(s)?.modeId),models:r.map(c=>({modelId:c.id,name:c.displayName})),currentModelId:this.bindingStore.get(s)?.modelId??"",available_models:r.map(c=>({id:c.id,displayName:c.displayName})),agent_commands:a?.adapter?.getSupportedCommands?.()??[]}),!0}case _.exec:{const[r,...a]=String(t.args??"").trim().split(/\s+/);if(!r)return d(h.verbInvalid,"Usage: exec <command> [args]"),!0;await this.ensureSlotStarted(s);const l=this.pool.getSlot(s)?.adapter;if(!l?.execCommand)return d(h.verbInvalid,"Agent does not support command execution"),!0;const g=l.getSupportedCommands?.()??[];if(!g.some(m=>m.name===r))return d(h.verbInvalid,`Unknown command: ${r}. Supported: ${g.map(m=>m.name).join(", ")}`),!0;try{const m=await l.execCommand(r,a.join(" "),s);m.status==="ok"?o({outcome:"exec",command:r,message:m.message,data:m.data}):d(h.runtimeError,m.message??`${r} failed`)}catch(m){d(h.runtimeError,`exec error: ${m instanceof Error?m.message:m}`)}return!0}default:return d(h.verbInvalid,`session control verb ${i} is not supported`),!0}}catch(r){const a=r instanceof Error&&r.cwdErrorCode?r.cwdErrorCode:r instanceof Error&&r.sessionControlErrorCode?r.sessionControlErrorCode:h.runtimeError;return u.error(this.name,`handleSessionControlLocalAction error verb=${i} session_id=${s}: ${r instanceof Error?r.message:r}`),d(a,r instanceof Error?r.message:String(r)),!0}}async handleEventCancel(e){const{event_id:n,session_id:t}=e;if(u.info(this.name,`handleEventCancel start event_id=${n} session_id=${t}`),this.pool.cancelEvent(n,t)){if(!(this.pool.getSlot(t)?.adapter?.getActiveEventIds().includes(n)??!1)){this.sendEventResultWithCleanup(n,"canceled","canceled"),this.aibotHandle.sendEventCancelResult({event_id:n,accepted:!0,final_state:"canceled"});return}await this.waitForEventDone(n,t,15e3),this.aibotHandle.sendEventCancelResult({event_id:n,accepted:!0,final_state:"canceled"}),this.pushQueueSnapshotForSession(t);return}this.aibotHandle.sendEventCancelResult({event_id:n,accepted:!1,reason:"event not found or not cancelable"})}waitForEventDone(e,n,t){return new Promise(s=>{const i=this.pool.getSlot(n);if(!i?.adapter){s();return}const o=setTimeout(()=>{i.adapter.removeListener("eventDone",d),s()},t),d=r=>{r===e&&(clearTimeout(o),i.adapter.removeListener("eventDone",d),s())};i.adapter.on("eventDone",d)})}handleAibotStop(e){u.info(this.name,`[stop-trace] handleAibotStop begin session=${e.session_id} event=${e.event_id} stop_id=${e.stop_id||"-"} adapterType=${this.config.adapterType??"acp"}`),this.aibotHandle.sendEventStopAck({stop_id:e.stop_id,event_id:e.event_id,accepted:!0,updated_at:Date.now()});const n=this.pool.getSlot(e.session_id),t=n?.adapter?.getStatus().busy??!1,s=(this.config.adapterType??"acp")==="acp",i=n?.adapter?.getActiveEventIds,o=typeof i=="function"?i.call(n.adapter):[],d=o.length>0?o.includes(e.event_id):t,r=(s||(this.config.adapterType??"acp")==="codex")&&d;if(u.info(this.name,`[stop-trace] handleAibotStop decision session=${e.session_id} event=${e.event_id} slotExists=${!!n?.adapter} busy=${t} activeIds=[${o.join(",")}] stoppingActiveEvent=${d} killOnStop=${r}`),d&&this.sendCtrl.markEventStopped(e.event_id),n?.adapter&&t){const a=r?this.pool.drainQueuedForSession(e.session_id):[],c=l=>{l===e.event_id&&(n.adapter.removeListener("eventDone",c),u.info(this.name,`[stop-trace] handleAibotStop eventDone -> stopResult(stopped) session=${e.session_id} event=${e.event_id} stop_id=${e.stop_id||"-"} killOnStop=${r}`),this.aibotHandle.sendEventStopResult({stop_id:e.stop_id,event_id:e.event_id,status:"stopped",updated_at:Date.now()}),r&&this.killAndResumeStopSlot(e.session_id,a))};n.adapter.on("eventDone",c),this.pool.deliverStopEvent(e.event_id,e.session_id)}else u.info(this.name,`[stop-trace] handleAibotStop slot-not-busy -> immediate stopResult(stopped) session=${e.session_id} event=${e.event_id} slotExists=${!!n?.adapter}`),this.pool.deliverStopEvent(e.event_id,e.session_id),this.aibotHandle.sendEventStopResult({stop_id:e.stop_id,event_id:e.event_id,status:"stopped",updated_at:Date.now()});(this.config.adapterType??"acp")==="pi"&&this.aibotHandle.sendText({event_id:e.event_id,session_id:e.session_id,content:"/stop",msg_type:0})}async killAndResumeStopSlot(e,n){if(!this.stopped){u.info(this.name,`[stop-trace] killAndResumeStopSlot begin session=${e} siblings=${n.length} -> removeSlot (kill process group)`);try{await this.pool.removeSlot(e),u.info(this.name,`[stop-trace] killAndResumeStopSlot removeSlot done session=${e} (process killed) -> redeliver ${n.length} sibling(s)`)}catch(t){u.warn(this.name,`[acp-stop] removeSlot failed session=${e}: ${t instanceof Error?t.message:String(t)}`)}if(!this.stopped)for(const t of n){if(this.stopped)break;try{await this.pool.deliverInboundEvent(t)}catch(s){u.error(this.name,`[acp-stop] sibling redeliver failed event=${t.event_id} session=${e}: ${s instanceof Error?s.message:String(s)}`),this.sendEventResultWithCleanup(t.event_id,"failed",s instanceof Error?s.message:String(s))}}}}handleAibotRevoke(e){if(!e.event_id||!this.revokeHandler.checkAndTrack(e.event_id))return;const n=e.event_id,t=e.session_id;if(t&&this.pool.cancelEvent(n,t)){(this.pool.getSlot(t)?.adapter?.getActiveEventIds().includes(n)??!1)||this.sendEventResultWithCleanup(n,"canceled","revoked");return}if(this.deferredMgr.removeEvent(n)){this.aibotHandle.sendEventResult({event_id:n,status:"canceled",msg:"revoked",updated_at:Date.now()});return}this.pool.deliverStopEvent(n,t||void 0)}async handleAibotLocalAction(e){const n=e.action_type??"",t=String((e.params??{}).session_id??""),s=String((e.params??{}).verb??"").trim().toLowerCase();u.debug(this.name,`local_action received action_type=${n} verb=${s||"-"} action_id=${e.action_id} session_id=${t}`);const i=(this.config.adapterType??"acp")==="claude";if(n===w.sessionControl&&s===_.exec&&await this.handleSessionControlLocalAction(e))return;if(n===w.sessionControl&&s===_.listSessions){await this.handleListSessionsLocalAction(e);return}if(i&&(n===w.interactionReply||n==="exec_approve"||n==="exec_reject")&&(await this.pool.deliverLocalAction(e)).handled||i&&await this.handleSessionControlLocalAction(e))return;if(n===w.sessionControl){const c=this.config.adapterType??"acp",l=c==="codex",g=c==="pi",m=String((e.params??{}).verb??"").trim().toLowerCase();if(l&&m===_.open){await this.handleCodexSessionControlLocalActionOpen(e);return}if(c==="cursor"&&m===_.open){await this.handleCursorSessionControlLocalActionOpen(e);return}if(l&&m==="restart"){const v=this.bindingStore.get(t)?.cwd??"";await this.pool.removeSlot(t).catch(()=>{}),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"restarted",binding:{aibotSessionId:t,cwd:v,workerStatus:"ready"}}});return}if(g&&m===_.open){try{const p=e.params??{},v=String(p.cwd??"").trim();if(!v){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.cwdRequired,error_msg:"session cwd is required"});return}const S=await this.resolveCwdForBinding(v),k=String(p.agent_session_id??"").trim();this.ensureImportedAgentSession(k,S),this.bindingStore.set(t,S),this.setResolvedAgentSessionId(t,k),this.sessionBindings.set(t,S),await this.deferredMgr.replayDeferredPiEvents(t,this.deferredCallbacks()),this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:S}),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"opened",binding:this.buildOpenedBindingResult(t,S)}})}catch(p){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:p?.sessionControlErrorCode??p?.cwdErrorCode??h.invalidCwd,error_msg:p instanceof Error?p.message:String(p)})}return}if(g&&m===_.restart){await this.handlePiSessionControlRestartLocalAction(e);return}const b=c==="openhuman",f=c==="opencode";if((b||f)&&m===_.open){try{const p=e.params??{},v=String(p.cwd??"").trim();if(!v){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.cwdRequired,error_msg:"session cwd is required"});return}const S=await this.resolveCwdForBinding(v),k=String(p.agent_session_id??"").trim();this.ensureImportedAgentSession(k,S),this.bindingStore.set(t,S),this.setResolvedAgentSessionId(t,k),this.sessionBindings.set(t,S),b&&await this.deferredMgr.replayDeferredOpenHumanEvents(t,this.deferredCallbacks()),f&&await this.deferredMgr.replayDeferredOpenCodeEvents(t,this.deferredCallbacks()),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"opened",binding:this.buildOpenedBindingResult(t,S)}})}catch(p){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:p?.sessionControlErrorCode??p?.cwdErrorCode??h.invalidCwd,error_msg:p instanceof Error?p.message:String(p)})}return}if(c==="codewhale"&&m===_.open){await this.handleCodeWhaleSessionControlLocalActionOpen(e);return}if(c==="acp"&&m===_.stop){const v=this.bindingStore.get(t)?.cwd??"";await this.pool.removeSlot(t).catch(()=>{}),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"stopped",binding:{aibotSessionId:t,cwd:v,workerStatus:"stopped"}}});return}try{if(m===_.open){const p=e.params??{},v=String(p.cwd??"").trim();if(!v){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.cwdRequired,error_msg:"session cwd is required"});return}const S=String(p.agent_session_id??"").trim();if(S){const k=await this.resolveCwdForBinding(v);this.ensureImportedAgentSession(S,k)}}await this.handleSessionControlLocalActionForPool(e)}catch(p){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:p?.sessionControlErrorCode??p?.cwdErrorCode??h.runtimeError,error_msg:p instanceof Error?p.message:String(p)});return}if(m===_.open){const p=e.params??{},v=await this.resolveCwdForBinding(String(p.cwd??"").trim());this.setResolvedAgentSessionId(t,String(p.agent_session_id??"").trim()),c==="pi"?await this.deferredMgr.replayDeferredPiEvents(t,this.deferredCallbacks()):c==="openhuman"?await this.deferredMgr.replayDeferredOpenHumanEvents(t,this.deferredCallbacks()):c==="opencode"?await this.deferredMgr.replayDeferredOpenCodeEvents(t,this.deferredCallbacks()):c==="agy"?(this.bindingStore.set(t,v),this.sessionBindings.set(t,v),await this.deferredMgr.replayDeferredAgyEvents(t,this.deferredCallbacks())):await this.deferredMgr.replayDeferredAcpEvents(t,this.deferredCallbacks()),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"opened",binding:this.buildOpenedBindingResult(t,v)}}),(this.config.adapterType??"acp")==="agy"&&this.aibotHandle.sendUpdateBindingCard({session_id:t,worker_status:"ready",cwd:v,meta:this.buildAgyToolbarMeta(t)})}else Ce(e,this.sessionControlCtx(t),this.sessionControlSenders());return}if(n==="file_list"){const c=Date.now(),l=t?this.bindingStore.get(t)?.cwd:void 0,g=e.params??{},m=String(g.parent_id??"").trim(),b=Array.isArray(g.allowed_extensions)?g.allowed_extensions.filter(v=>typeof v=="string").map(v=>v.trim()).filter(v=>v.length>0):[];u.info("file-list-diag",`plugin << recv action_id=${e.action_id} session_id=${t} parent_id=${m||"<root>"} show_hidden=${!!g.show_hidden} ext_count=${b.length} bound_cwd=${l??"<none>"}`);const f=await Ee({parent_id:m||null,session_id:t,show_hidden:!!g.show_hidden,allowed_extensions:b},{resolveCwd:()=>l??this.config.agent.cwd??process.cwd(),fallbackDir:K()}),E=Date.now()-c,p=f.result?.files?.length??0;u.info("file-list-diag",`plugin -> reply action_id=${e.action_id} status=${f.status} elapsed=${E}ms count=${p} current_path=${f.result?.current_path??""} error_code=${f.error_code??""} error_msg=${f.error_msg??""}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:f.status,...f.result?{result:f.result}:{},...f.error_code?{error_code:f.error_code}:{},...f.error_msg?{error_msg:f.error_msg}:{}});return}if(n==="create_folder"){const c=t?this.bindingStore.get(t)?.cwd:void 0,l=String((e.params??{}).parent_id??"").trim(),g=String((e.params??{}).name??"").trim(),m=await Re({parent_id:l||null,name:g,session_id:t},{resolveCwd:()=>c??this.config.agent.cwd??process.cwd(),fallbackDir:K()});this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:m.status,...m.result?{result:m.result}:{},...m.error_code?{error_code:m.error_code}:{},...m.error_msg?{error_msg:m.error_msg}:{}});return}if(n===w.setModel&&(this.config.adapterType??"acp")==="agy"){await this.handleAgySetModel(e,t);return}if(n===w.getSessionUsage){await this.handleGetSessionUsage(e,t);return}if(n===w.getRateLimits){await this.handleGetRateLimits(e,t);return}const o=(this.config.adapterType??"acp")==="acp";if((i||o)&&n===w.threadCompact){await this.handleThreadCompact(e,t);return}if(n==="connector_rollback"){await this.handleConnectorRollback(e);return}if(n==="connector_upgrade_push"){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{accepted:!0}}),this.upgradeTrigger?.();return}if(n===w.getAgentGlobalConfig){const c=this.globalConfigStore?.get(this.name);this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{agentName:this.name,...c?{config:c}:{config:null}}});return}const d=this.config.adapterType??"acp",r=(d==="codex"||d==="cursor"||d==="pi"||d==="openhuman"||d==="opencode"||d==="acp")&&!!t&&!!this.bindingStore.get(t)?.cwd,a=await this.pool.deliverLocalAction(e,{autoCreateSlot:r});if(a.handled){if(a.kind==="set_mode"){const c=String((e.params??{}).mode_id??"");c&&(d==="cursor"||d==="claude"?this.bindingStore.setModeId(t,c):T.has(d)||this.globalConfigStore?.set(this.name,{acpInitialMode:c}))}else if(a.kind==="set_model"){const c=String((e.params??{}).model_id??"");c&&(d==="cursor"?this.bindingStore.setModelId(t,c):T.has(d)||this.globalConfigStore?.set(this.name,{modelId:c}))}else if(a.kind==="set_reasoning_effort"){const c=String((e.params??{}).reasoning_effort??(e.params??{}).reasoning_eff??(e.params??{}).effort??"");c&&this.globalConfigStore?.set(this.name,{codexReasoningEffort:c})}else if(a.kind==="set_sandbox_mode"){const c=String((e.params??{}).sandbox_mode??(e.params??{}).sandboxMode??"");if(c){const l=c==="default"?void 0:c;this.globalConfigStore?.set(this.name,{codexSandboxMode:l})}}return}if((d==="codex"||d==="cursor"||d==="pi"||d==="openhuman"||d==="opencode")&&t&&!this.bindingStore.get(t)?.cwd){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.bindingMissing,error_msg:"Session binding missing. Open a workspace first."});return}if(d==="acp"&&(n==="set_mode"||n==="set_model")){const c=this.sessionControlSenders(),l=this.pool.getSlot(t)?.adapter,g={bindingStore:this.bindingStore,acpAdapter:l instanceof A?l:null,globalConfigStore:this.globalConfigStore,agentName:this.name,log:u};if(n==="set_mode"){const m=String((e.params??{}).mode_id??""),b=await N(g,t,m);if(b.status==="failed")c.sendLocalActionResult(e.action_id,"failed",void 0,b.errorCode,b.errorMsg);else{const f=l instanceof A?l.buildToolbarContext(b.result?.outcome==="mode_set"?"mode_set":"mode_set_failed"):null,E=f?{...f,...b.result}:b.result;c.sendLocalActionResult(e.action_id,"ok",E)}}else{const m=String((e.params??{}).model_id??""),b=await W(g,t,m);if(b.status==="failed")c.sendLocalActionResult(e.action_id,"failed",void 0,b.errorCode,b.errorMsg);else{const f=l instanceof A?l.buildToolbarContext(b.result?.outcome==="model_set"?"model_set":"model_set_failed"):null,E=f?{...f,...b.result}:b.result;c.sendLocalActionResult(e.action_id,"ok",E)}}return}this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"unsupported_local_action",error_msg:`action type ${n} is not supported`})}async handleGetSessionUsage(e,n){if(!n){u.warn(this.name,`[usage] get_session_usage rejected: no session_id in params, action_id=${e.action_id}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"session_id_required",error_msg:"session_id is required for get_session_usage"});return}const t=this.config.adapterType??"acp",s=this.bindingStore.get(n),i=s?.cwd??this.config.agent.cwd??process.cwd();u.info(this.name,`[usage] get_session_usage action_id=${e.action_id} session_id=${n} adapterType=${t} hasBinding=${!!s} cwd=${i}`);let o=null,d,r;switch(t){case"claude":{if(d=s?.claudeSessionId,!d){u.warn(this.name,`[usage] no claude binding for session_id=${n}, action_id=${e.action_id}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No Claude session binding found"});return}u.info(this.name,`[usage] parsing claude usage: claudeSessionId=${d} cwd=${i}`),o=await ne(d,i),r="claude";break}case"agy":{this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{sessionId:n,adapterType:"agy",available:!1,reason:"agy print-mode adapter does not track token usage"}});return}case"acp":default:{if(d=s?.acpSessionId,!d){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No ACP session binding found"});return}o=await oe(d,i,this.config.aibot.clientType),r=t,(!r||r==="acp")&&(r="acp");break}case"codex":{if(d=s?.codexThreadId,!d){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No Codex thread binding found"});return}o=await re(d),r="codex";break}case"pi":{if(d=s?.piSessionPath,!d){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No Pi session path binding found"});return}o=await ue(d),r="pi";break}case"cursor":{const c=this.pool.getSlot(n)?.adapter;if(!(c instanceof L)){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No Cursor session found"});return}const l=c.getUsageSnapshot(n);if(!l){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"usage_not_found",error_msg:"No usage data found for this session"});return}this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{sessionId:n,adapterType:"cursor",models:[{modelId:this.bindingStore.get(n)?.modelId??"auto",turns:l.turns,input:l.total.input,output:l.total.output,cacheRead:l.total.cacheRead,cacheWrite:l.total.cacheWrite}],total:{input:l.total.input,output:l.total.output,cacheRead:l.total.cacheRead,cacheWrite:l.total.cacheWrite},turns:l.turns,sampledAt:l.sampledAt}});return}case"codewhale":{const c=this.pool.getSlot(n)?.adapter;if(!(c instanceof q)){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"no_binding",error_msg:"No CodeWhale session found"});return}const l=c.getUsageSnapshot();if(!l){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"usage_not_found",error_msg:"No usage data found for this session"});return}this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{sessionId:n,adapterType:"codewhale",models:[{modelId:this.bindingStore.get(n)?.modelId??"codewhale",turns:l.turns,input:l.total.input,output:l.total.output}],total:{input:l.total.input,output:l.total.output},turns:l.turns,sampledAt:l.sampledAt}});return}}if(!o){u.info(this.name,`[usage] no usage data found: session_id=${n} adapterSessionId=${d} adapterType=${r}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"usage_not_found",error_msg:"No usage data found for this session"});return}u.info(this.name,`[usage] result ok: session_id=${n} adapterSessionId=${d} turns=${o.turns} models=${o.models.length}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{sessionId:d,adapterType:r,models:o.models,total:o.total,turns:o.turns,sampledAt:new Date().toISOString()}})}async handleThreadCompact(e,n){if(!n){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"session_id_required",error_msg:"session_id is required for thread_compact"});return}if(!this.bindingStore.get(n)?.cwd){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.bindingMissing,error_msg:"session binding was not found"});return}try{await this.ensureSlotStarted(n);const i=this.pool.getSlot(n)?.adapter;if(!i?.execCommand){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.runtimeError,error_msg:"Agent does not support command execution"});return}u.info(this.name,`thread_compact session_id=${n} action_id=${e.action_id}`);const o=await i.execCommand("compact","",n);o.status==="ok"?this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{outcome:"compacted",message:o.message,data:o.data}}):this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.runtimeError,error_msg:o.message??"compact failed"})}catch(s){u.warn(this.name,`thread_compact error session_id=${n}: ${s instanceof Error?s.message:s}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:h.runtimeError,error_msg:s instanceof Error?s.message:String(s)})}}async handleGetRateLimits(e,n){const t=this.config.adapterType??"acp";if(this.config.aibot.clientType==="kiro"){const s=this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs),i=this.cachedAcpContextWindow;if(s&&this.cachedProviderQuota){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"acp",available:!0,cached:!0,sampledAt:this.cachedProviderQuotaSampledAtMs,rateLimits:null,contextWindow:i,tokenUsage:null,providerQuota:this.cachedProviderQuota}});return}try{const o=await B();this.cachedProviderQuota=o,this.cachedProviderQuotaSampledAtMs=Date.now(),u.info(this.name,`[rate-limits] kiro quota queried: success=${o.success}`+(o.balance?` balance=${o.balance.remaining} ${o.balance.unit}`:"")+(o.error?` error=${o.error}`:"")),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"acp",available:!0,cached:!1,sampledAt:this.cachedProviderQuotaSampledAtMs,rateLimits:null,contextWindow:i,tokenUsage:null,providerQuota:o}})}catch(o){u.warn(this.name,`[rate-limits] kiro quota query failed: ${o instanceof Error?o.message:String(o)}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"acp",available:!1,cached:!1,sampledAt:null,rateLimits:null,contextWindow:i,tokenUsage:null}})}return}switch(t){case"codex":{const s=this.maybeQueryProviderQuota(),i=this.getFreshCodexGlobalRateLimitCache();if(i.hasData&&!this.isRateLimitsCacheFresh(this.cachedRateLimitsSampledAtMs)&&!this.isRateLimitsCacheFresh(this.cachedCodexUsageSampledAtMs)){const l=this.pool.getAllSlots().find(g=>g.state==="ready"&&g.adapter);if(l&&(u.info(this.name,`[rate-limits] codex cache stale, refreshing from slot: session=${l.sessionId}`),(await l.adapter.handleLocalAction?.(e))?.handled))return}if(i.hasData){i.rateLimits?u.info(this.name,`[rate-limits] codex cached: primary=${i.rateLimits.primary.usedPercent.toFixed(1)}% resetsAt=${i.rateLimits.primary.resetsAt} secondary=${i.rateLimits.secondary.usedPercent.toFixed(1)}% resetsAt=${i.rateLimits.secondary.resetsAt}`):u.info(this.name,`[rate-limits] codex cached context/token only: hasContext=${!!i.contextWindow} hasToken=${!!i.tokenUsage}`);const l=await s;this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"codex",available:i.hasData||!!l,cached:!0,sampledAt:i.sampledAt,rateLimits:i.rateLimits,contextWindow:i.contextWindow,tokenUsage:i.tokenUsage,providerQuota:l}});return}const d=this.pool.getAllSlots().find(l=>l.state==="ready"&&l.adapter);if(d&&(u.info(this.name,`[rate-limits] codex reuse existing slot: session=${d.sessionId}`),(await d.adapter.handleLocalAction?.(e))?.handled))return;const r=this.resolveRateLimitWakeSessionId(n,t);if(r){const l=await this.wakeRateLimitSlot(r,t);if(l?.adapter&&(await l.adapter.handleLocalAction?.(e))?.handled)return}const a=await s,c=!!a;u.info(this.name,`[rate-limits] codex no native data, providerQuota=${a?a.provider:"none"} available=${c}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"codex",available:c,cached:!1,sampledAt:null,rateLimits:null,contextWindow:null,tokenUsage:null,providerQuota:a}});return}case"claude":{const s=this.maybeQueryProviderQuota(),i=this.getFreshClaudeRateLimitState();if(i){const c=i.rateLimits,l=await s;u.info(this.name,`[rate-limits] claude global cached: sampledAt=${i.sampledAt} hasRateLimits=${!!c}`+(l?` providerQuota=${l.provider}`:"")),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"claude",available:!0,cached:!0,sampledAt:i.sampledAt,rateLimits:i.rateLimits??null,contextWindow:i.contextWindow,tokenUsage:null,providerQuota:l}});return}let o=this.pool.getAllSlots().find(c=>c.state==="ready"&&c.adapter instanceof y)??null;if(!o){const c=this.resolveRateLimitWakeSessionId(n,t);c&&(o=await this.wakeRateLimitSlot(c,t))}u.info(this.name,`[rate-limits] handleGetRateLimits: session_id=${n} adapterType=claude hasSlot=${!!o} hasAdapter=${!!o?.adapter}`);const d=o?.adapter,r=d instanceof y?d.getSessionState():null,a=await s;if(r){(r.rateLimits?.fiveHour||r.rateLimits?.sevenDay||r.contextWindow?.usedPercentage!=null)&&(this.cachedClaudeRateLimitState=r);const c=this.getFreshClaudeRateLimitState(),l=c?.rateLimits&&!r.rateLimits?{...r,rateLimits:c.rateLimits}:r,g=l.rateLimits;u.info(this.name,`[rate-limits] claude global state: sampledAt=${l.sampledAt} hasRateLimits=${!!g}`+(g?` fiveHour=${g.fiveHour?.usedPercentage??"n/a"}% resetsAt=${g.fiveHour?.resetsAt??"n/a"} sevenDay=${g.sevenDay?.usedPercentage??"n/a"}% resetsAt=${g.sevenDay?.resetsAt??"n/a"}`:"")+(c?.rateLimits&&!r.rateLimits?" source=live+cached-fallback":" source=live")+(a?` providerQuota=${a.provider}`:"")),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"claude",available:!0,cached:!1,sampledAt:l.sampledAt,rateLimits:l.rateLimits??null,contextWindow:l.contextWindow,tokenUsage:null,providerQuota:a}})}else u.info(this.name,`[rate-limits] claude no global state: hasAdapter=${!!d} adapterType=${d?.constructor?.name??"n/a"}`+(a?` providerQuota=${a.provider}`:"")),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:"claude",available:!!a,cached:!1,sampledAt:null,rateLimits:null,contextWindow:null,tokenUsage:null,providerQuota:a}});return}case"cursor":{const i=(n?this.pool.getSlot(n):null)?.adapter,o=i instanceof L?i.getRateLimitsSnapshot():{adapterType:"cursor",available:!1,cached:!1,sampledAt:null,rateLimits:null,contextWindow:null,tokenUsage:null};this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:o});return}default:{const s=this.config.providerBaseUrl,i=this.config.providerApiKey;if(!s||!i){u.info(this.name,`[rate-limits] no provider config for adapterType=${t}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:t,available:!1,cached:!1,sampledAt:null,rateLimits:null,contextWindow:null,tokenUsage:null}});return}if(this.isRateLimitsCacheFresh(this.cachedProviderQuotaSampledAtMs)&&this.cachedProviderQuota){u.info(this.name,`[rate-limits] provider quota cached: provider=${this.cachedProviderQuota.provider} success=${this.cachedProviderQuota.success}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:t,available:!0,cached:!0,sampledAt:this.cachedProviderQuotaSampledAtMs,rateLimits:null,contextWindow:null,tokenUsage:null,providerQuota:this.cachedProviderQuota}});return}try{const d=await z(s,i);this.cachedProviderQuota=d,this.cachedProviderQuotaSampledAtMs=Date.now(),u.info(this.name,`[rate-limits] provider quota queried: provider=${d.provider} success=${d.success}`+(d.tiers.length>0?` tiers=${d.tiers.map(r=>`${r.name}=${r.usedPercent}%`).join(",")}`:"")+(d.balance?` balance=${d.balance.remaining} ${d.balance.unit}`:"")+(d.error?` error=${d.error}`:"")),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:t,available:!0,cached:!1,sampledAt:this.cachedProviderQuotaSampledAtMs,rateLimits:null,contextWindow:null,tokenUsage:null,providerQuota:d}})}catch(d){u.warn(this.name,`[rate-limits] provider quota query failed: ${d instanceof Error?d.message:String(d)}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{adapterType:t,available:!1,cached:!1,sampledAt:null,rateLimits:null,contextWindow:null,tokenUsage:null}})}return}}}resolveRateLimitWakeSessionId(e,n){const t=String(e??"").trim(),s=this.bindingStore.getMostRecentlyUpdatedSessionId({requireCwd:!0});return s?t&&t===s?t:n==="codex"||n==="claude"?(t&&t!==s&&u.info(this.name,`[rate-limits] ${n} remap wake session: requested=${t} use_recent=${s}`),s):t&&t!==s?(u.info(this.name,`[rate-limits] skip wake slot: session=${t} adapterType=${n} reason=not_recent_session recent=${s}`),null):t||null:(u.info(this.name,`[rate-limits] skip wake slot: adapterType=${n} reason=no_recent_binding`),null)}async wakeRateLimitSlot(e,n){const t=String(e??"").trim();if(!t)return null;if(!this.bindingStore.get(t)?.cwd)return u.info(this.name,`[rate-limits] skip wake slot: session=${t} adapterType=${n} reason=binding_cwd_missing`),null;try{const i=this.pool.getOrCreateSlot(t);if(!i)return u.info(this.name,`[rate-limits] skip wake slot: session=${t} adapterType=${n} reason=slot_unavailable`),null;if(i.startPromise){const o=n==="claude"?6e4:2e4;await Promise.race([i.startPromise,new Promise((d,r)=>setTimeout(()=>r(new Error(`wake rate-limits slot timeout (${o}ms)`)),o))])}return u.info(this.name,`[rate-limits] wake slot success: session=${t} adapterType=${n}`),this.pool.getSlot(t)??i}catch(i){return u.warn(this.name,`[rate-limits] wake slot failed: session=${t} adapterType=${n} err=${i instanceof Error?i.message:String(i)}`),null}}sessionControlCtx(e){const n=this.pool.getSlot(e),t=n?.adapter instanceof A?n.adapter:null,s=(this.config.adapterType??"acp")==="acp",i={bindingStore:this.bindingStore,acpAdapter:t,globalConfigStore:this.globalConfigStore,agentName:this.name,log:u};return{getCwd:()=>this.bindingStore.get(e)?.cwd??this.config.agent.cwd??process.cwd(),getSessionBindings:()=>{if(t)return t.getSessionBindings();const o=this.sessionBindings;if(e&&!o.has(e)){const d=this.bindingStore.get(e);d?.cwd&&o.set(e,d.cwd)}return o},getStatus:()=>this.getStatus(),isAcpAlive:!!t?.isAlive(),getAcpSessionOptions:()=>t?.acpSessionOptions??null,setMode:o=>t?t.setMode(o):Promise.resolve(!1),setModel:o=>t?t.setModel(o):Promise.resolve(!1),acpSetMode:s?(o,d)=>N(i,o,d):void 0,acpSetModel:s?(o,d)=>W(i,o,d):void 0,getPendingApproval:o=>{const d=t?.pendingApprovalEntries.get(o);return d?{requestId:d}:void 0},deletePendingApproval:o=>t?.pendingApprovalEntries.delete(o)??!1,respondPermission:(o,d)=>(t&&t.respondToPermission(o,d),Promise.resolve()),onSessionBound:(o,d)=>{this.bindingStore.set(o,d)},onSessionUnbound:o=>{const r=this.bindingStore.get(o)?.cwd??"";this.bindingStore.delete(o),this.sessionBindings.delete(o),this.claudeWorkerStatus.delete(o),this.deferredMgr.clearSession(o),r&&this.aibotHandle.sendUpdateBindingCard({session_id:o,worker_status:"stopped",cwd:r})},cancelActiveRun:()=>(this.config.adapterType??"acp")==="agy"&&n?.adapter instanceof O?(n.adapter.cancelCurrentRun(),Promise.resolve()):n?.adapter?.cancel("")??Promise.resolve(),onModeSet:o=>{const d=this.config.adapterType??"acp";T.has(d)||this.globalConfigStore?.set(this.name,{acpInitialMode:o})},onModelSet:o=>{const d=this.config.adapterType??"acp";T.has(d)||this.globalConfigStore?.set(this.name,{modelId:o})}}}sessionControlSenders(){return{sendEventAck:(e,n)=>this.aibotHandle.sendEventAck({event_id:e,session_id:n,received_at:Date.now()}),sendEventResult:(e,n,t)=>this.aibotHandle.sendEventResult({event_id:e,status:n,...t?.msg?{msg:t.msg}:{},...t?.code?{code:t.code}:{},updated_at:Date.now()}),sendLocalActionResult:(e,n,t,s,i)=>this.aibotHandle.sendLocalActionResult({action_id:e,status:n,...t?{result:t}:{},...s?{error_code:s}:{},...i?{error_msg:i}:{}})}}resolveBindingChannelKey(e){switch(e){case"claude":return"grix-claude";case"codex":return"codex";case"cursor":return"cursor";case"pi":return"pi";case"openhuman":return"openhuman";case"codewhale":return"codewhale";case"opencode":return"opencode";case"agy":return"acp";case"acp":return this.config.aibot.clientType==="qwen"?"qwen":"acp";default:return e}}finalizeThinking(e,n){this.sendCtrl.finalizeThinking(e,n)}logInboundConversation(e){const n=String(e.session_id??"").trim();n&&this.conversationLog?.logInbound(n,{event_id:e.event_id,msg_id:e.msg_id,sender_id:e.sender_id,msg_type:e.msg_type,content:e.content??""})}buildInboundEvent(e,n){const t=Te(this.sendCtrl.getGlobalRuntimeConfig(),n);return{event_id:e.event_id,session_id:e.session_id,thread_id:e.thread_id,sender_id:e.sender_id,msg_id:e.msg_id,msg_type:e.msg_type,content:e.content??"",quoted_message_id:e.quoted_message_id,context_messages_json:e.context_messages?JSON.stringify(e.context_messages):void 0,extra_json:e.extra?JSON.stringify(e.extra):void 0,connector_runtime_config:{response_delivery:t.responseDelivery,tool_events:t.toolEvents,thinking_events:t.thinkingEvents},session_type:e.session_type,created_at:e.created_at}}isStaleClaudeEvent(e){const n=Number(e.created_at);return!Number.isFinite(n)||n<=0?!1:Date.now()-n>Ue}async handleConnectorRollback(e){const n=String((e.params??{}).target_version??"").trim(),t=String((e.params??{}).reason??"server_initiated");if(!n){this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"MISSING_TARGET_VERSION",error_msg:"target_version is required for connector_rollback"});return}u.info(this.name,`connector_rollback: target=${n} reason=${t}`);try{const{npmInstall:s,writePending:i,removePending:o,upgradeLog:d}=await import("../core/upgrade/npm-upgrader.js"),{resolveClientVersion:r}=await import("../core/util/client-version.js"),a=r();d(`server rollback: ${a} -> ${n} reason=${t}`),i(a,n),await s("grix-connector",n),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"ok",result:{rolled_back_to:n}}),process.kill(process.pid,"SIGTERM")}catch(s){try{const{removePending:o}=await import("../core/upgrade/npm-upgrader.js");o()}catch{}const i=s instanceof Error?s.message:String(s);u.error(this.name,`connector_rollback failed: ${i}`),this.aibotHandle.sendLocalActionResult({action_id:e.action_id,status:"failed",error_code:"ROLLBACK_FAILED",error_msg:i})}}setUpgradeTrigger(e){this.upgradeTrigger=e}}export{Tt as AgentInstance};