@hydra-acp/cli 0.1.68 → 0.1.69

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -1,25 +1,25 @@
1
- import*as gi from"fs";import*as pe from"fs/promises";import bc from"fastify";import Ic from"@fastify/websocket";var jn="acp.v1";function Hn(n,e){return n.has(jn)?jn:!1}import mi from"pino";import Sc from"pino-roll";import*as Ln from"fs/promises";import{homedir as Wt}from"os";import{z as b}from"zod";import*as R from"path";import*as Jt from"os";var hi="HYDRA_ACP_HOME";function Ie(n){let e=Jt.homedir();return e?n===e?"~":n.startsWith(e+"/")?"~"+n.slice(e.length):n:n}function T(){let n=process.env[hi];if(n&&n.length>0)return R.resolve(n);if(process.env.VITEST)throw new Error("HYDRA_ACP_HOME is unset under VITEST; vitest.setup.ts must run first");return R.join(Jt.homedir(),".hydra-acp")}var w={home:T,config:()=>R.join(T(),"config.json"),authToken:()=>R.join(T(),"auth-token"),remotes:()=>R.join(T(),"remotes.json"),pidFile:()=>R.join(T(),"daemon.pid"),logFile:()=>R.join(T(),"daemon.log"),currentLogFile:()=>R.join(T(),"current.log"),registryCache:()=>R.join(T(),"registry.json"),agentsDir:()=>R.join(T(),"agents"),agentLogFile:n=>R.join(T(),"agents","logs",`${n}.log`),agentInstallDir:(n,e,t)=>R.join(T(),"agents",e,n,t),agentNpmInstallDir:(n,e,t)=>R.join(T(),"agents",e,n,t,`node${process.versions.modules}`),sessionsDir:()=>R.join(T(),"sessions"),sessionDir:n=>R.join(T(),"sessions",n),sessionFile:n=>R.join(T(),"sessions",n,"meta.json"),historyFile:n=>R.join(T(),"sessions",n,"history.jsonl"),toolsDir:n=>R.join(T(),"sessions",n,"tools"),toolBlobFile:(n,e)=>R.join(T(),"sessions",n,"tools",e),queueFile:n=>R.join(T(),"sessions",n,"queue.ndjson"),tombstonesDir:()=>R.join(T(),"sessions",".tombstones"),tombstoneAgentDir:n=>R.join(T(),"sessions",".tombstones",encodeURIComponent(n)),tombstoneFile:(n,e)=>R.join(T(),"sessions",".tombstones",encodeURIComponent(n),encodeURIComponent(e)),extensionsDir:()=>R.join(T(),"extensions"),extensionLogFile:n=>R.join(T(),"extensions",`${n}.log`),extensionPidFile:n=>R.join(T(),"extensions",`${n}.pid`),transformersDir:()=>R.join(T(),"transformers"),transformerLogFile:n=>R.join(T(),"transformers",`${n}.log`),transformerPidFile:n=>R.join(T(),"transformers",`${n}.pid`),transformerState:(n,e)=>R.join(T(),"sessions",n,"transformer-state",e),tuiHistoryFile:n=>R.join(T(),"sessions",n,"prompt-history"),globalTuiHistoryFile:()=>R.join(T(),"prompt-history"),tuiLogFile:()=>R.join(T(),"tui.log"),shimWireLogFile:()=>R.join(T(),"shim-wire.log")};import*as Se from"fs/promises";function rt(){let n=new Uint8Array(32);crypto.getRandomValues(n);let e="";for(let t of n)e+=t.toString(16).padStart(2,"0");return`hydra_token_${e}`}async function Bn(){try{let e=(await Se.readFile(w.authToken(),"utf8")).trim();return e.length>0?e:void 0}catch(n){if(n.code==="ENOENT")return;throw n}}async function yi(){let n=await Bn();if(!n)throw new Error(`No service token found at ${w.authToken()}. Run \`hydra-acp init\` to create one.`);return n}async function zt(n){await Se.mkdir(w.home(),{recursive:!0}),await Se.writeFile(w.authToken(),n+`
2
- `,{encoding:"utf8",mode:384})}async function wi(){let n=await Bn();if(n)return n;let e=rt();return await zt(e),process.stderr.write(`hydra-acp: initialized ${w.authToken()} with a fresh service token.
3
- `),e}import*as ee from"fs/promises";import*as Un from"fs";import{randomBytes as vi}from"crypto";async function K(n,e,t={}){let r=(t.pretty??!0?JSON.stringify(e,null,2):JSON.stringify(e))+`
4
- `;await bi(n,r,t)}async function bi(n,e,t={}){let s=Ii(n);await ee.mkdir(s,{recursive:!0});let r=`${n}.tmp-${process.pid}-${Si()}`;try{let i={encoding:"utf8"};t.mode!==void 0&&(i.mode=t.mode),await ee.writeFile(r,e,i),await ee.rename(r,n)}catch(i){throw await ee.unlink(r).catch(()=>{}),i}if(t.mode!==void 0)try{Un.chmodSync(n,t.mode)}catch{}}async function te(n){let e;try{e=await ee.readFile(n,"utf8")}catch(t){if(t.code==="ENOENT")return;throw t}if(e.trim().length!==0)try{return JSON.parse(e)}catch{return}}function Ii(n){let e=n.lastIndexOf("/");return e<=0?".":n.slice(0,e)}function Si(){return vi(4).toString("hex")}var qn="https://cdn.agentclientprotocol.com/registry/v1/latest/registry.json",ki=b.object({cert:b.string(),key:b.string()}),Jn=55514,Ai=b.object({host:b.string().default("127.0.0.1"),port:b.number().int().positive().default(Jn),logLevel:b.enum(["debug","info","warn","error"]).default("info"),tls:ki.optional(),sessionIdleTimeoutSeconds:b.number().int().nonnegative().default(3600),sessionHistoryMaxEntries:b.number().int().positive().default(1e4),agentStderrTailBytes:b.number().int().positive().default(4096),publicHost:b.string().optional(),agentSyncIntervalMinutes:b.number().nonnegative().default(60),sessionGcIntervalMinutes:b.number().nonnegative().default(60),sessionGcMaxAgeDays:b.number().positive().default(2)}),xi=b.object({url:b.string().url().default(qn),ttlHours:b.number().positive().default(24),pinned:b.boolean().default(!1)}),Ci=b.object({name:b.string().optional(),description:b.string().optional(),command:b.string().optional(),args:b.array(b.string()).optional(),env:b.record(b.string()).optional()}),Mi=b.object({packageSpec:b.string().optional()}),Ri=b.object({repaintThrottleMs:b.number().int().nonnegative().default(1e3),maxScrollbackLines:b.number().int().positive().default(1e4),mouse:b.boolean().default(!1),inAppSelection:b.boolean().optional(),selectionClipboard:b.enum(["primary","clipboard","both"]).default("both"),logMaxBytes:b.number().int().positive().default(5*1024*1024),cwdColumnMaxWidth:b.number().int().positive().default(32),progressIndicator:b.boolean().default(!0),defaultEnterAction:b.enum(["enqueue","amend"]).default("amend"),showThoughts:b.boolean().default(!0),ambiguousWidth:b.enum(["narrow","wide"]).default("wide"),toolContent:b.enum(["inline","references"]).default("references"),diffContextLines:b.number().int().min(0).default(3),promptHistoryMaxEntries:b.number().int().positive().default(2e3),maxToolItems:b.number().int().nonnegative().default(5),maxPlanItems:b.number().int().nonnegative().default(5),showFileUpdates:b.enum(["none","edit","diff"]).default("edit"),sessionColumns:b.array(b.enum(["session","upstream","host","state","agent","model","age","cwd","title","cost"])).nonempty().optional()}),Dn=b.string().min(1).regex(/^[A-Za-z0-9._-]+$/,"extension name must be filename-safe"),Ei=b.object({command:b.array(b.string()).default([]),args:b.array(b.string()).default([]),env:b.record(b.string()).default({}),enabled:b.boolean().default(!0)}),Pi=b.object({command:b.array(b.string()).default([]),args:b.array(b.string()).default([]),env:b.record(b.string()).default({}),enabled:b.boolean().default(!0)}),zn=b.object({daemon:Ai.default({}),registry:xi.default({url:qn,ttlHours:24,pinned:!1}),agents:b.record(b.string(),Ci).default({}),agentOverrides:b.record(b.string(),Mi).default({}),defaultAgent:b.string().default("opencode"),defaultModels:b.record(b.string(),b.string()).default({}),synopsisAgent:b.string().optional(),synopsisModel:b.string().optional(),synopsisOnClose:b.boolean().default(!1),defaultCwd:b.string().default("~"),compressToolContent:b.boolean().default(!0),sessionListColdLimit:b.number().int().nonnegative().default(20),extensions:b.record(Dn,Ei).default({}),transformers:b.record(Dn,Pi).default({}),defaultTransformers:b.array(b.string()).default([]),npmRegistry:b.string().url().optional(),tui:Ri.default({repaintThrottleMs:1e3,maxScrollbackLines:1e4,mouse:!1,logMaxBytes:5*1024*1024,cwdColumnMaxWidth:32,progressIndicator:!0,defaultEnterAction:"amend",showThoughts:!0,ambiguousWidth:"wide",toolContent:"references",diffContextLines:3,promptHistoryMaxEntries:2e3,maxToolItems:5,maxPlanItems:5,showFileUpdates:"edit",selectionClipboard:"both"})});function Wn(n){return Object.entries(n.extensions).map(([e,t])=>({name:e,...t}))}function Qn(n){return Object.entries(n.transformers).map(([e,t])=>({name:e,...t}))}async function Vn(){return await te(w.config())??{}}async function Ti(){let n=await Vn(),e=n.daemon,t=e&&typeof e.authToken=="string"?e.authToken:void 0;if(!t)return;let s=!1;try{await Ln.access(w.authToken()),s=!0}catch(r){if(r.code!=="ENOENT")throw r}if(s)throw new Error(`Auth token present in both ${w.authToken()} and ${w.config()} (daemon.authToken). Remove daemon.authToken from config.json to resolve.`);await zt(t),delete e.authToken,Object.keys(e).length===0&&delete n.daemon,await K(w.config(),n,{mode:384}),process.stderr.write(`hydra-acp: migrated auth token from ${w.config()} to ${w.authToken()}.
5
- `)}async function Gn(){return await Ti(),zn.parse(await Vn())}async function _i(n){await K(w.config(),n,{mode:384})}function $i(){return zn.parse({})}function He(n){return n==="~"||n==="$HOME"?Wt():n.startsWith("~/")?Wt()+n.slice(1):n.startsWith("$HOME/")?Wt()+n.slice(5):n}import*as L from"fs/promises";import{createHash as Fi}from"crypto";import{gzip as Oi,gunzip as Ni}from"zlib";import{promisify as Yn}from"util";var Kn=Yn(Oi),ji=Yn(Ni),Xn=/^[A-Za-z0-9_-]+$/,Hi=/^[a-f0-9]{64}$/,Qt=!0;function Zn(n){Qt=n}function it(n,e){return Xn.test(n)?e===void 0||Hi.test(e):!1}function ot(n,e){return`${w.toolBlobFile(n,e)}.gz`}async function Vt(n,e){if(!it(n))return null;let t=Fi("sha256").update(e,"utf8").digest("hex"),s=ot(n,t),r=w.toolBlobFile(n,t);for(let a of[s,r])try{return await L.access(a),t}catch{}await L.mkdir(w.toolsDir(n),{recursive:!0});let i=Qt?s:r,o=Qt?await Kn(Buffer.from(e,"utf8")):Buffer.from(e,"utf8");return await L.writeFile(i,o,{mode:384,flag:"wx"}).catch(async a=>{if(a.code!=="EEXIST")throw a}),t}async function at(n,e){if(!it(n,e))return null;try{let t=await L.readFile(ot(n,e));return(await ji(t)).toString("utf8")}catch{}try{return await L.readFile(w.toolBlobFile(n,e),"utf8")}catch{return null}}async function es(n,e){if(!it(n,e))return null;try{return await L.readFile(ot(n,e))}catch{}try{let t=await L.readFile(w.toolBlobFile(n,e));return await Kn(t)}catch{return null}}async function ts(n,e,t){if(!it(n,e))return;let s=ot(n,e);try{await L.access(s);return}catch{}await L.mkdir(w.toolsDir(n),{recursive:!0}),await L.writeFile(s,t,{mode:384,flag:"wx"}).catch(r=>{if(r.code!=="EEXIST")throw r})}async function ns(n){Xn.test(n)&&await L.rm(w.toolsDir(n),{recursive:!0,force:!0}).catch(()=>{})}import*as ps from"fs/promises";import*as dt from"path";import{z as x}from"zod";import*as rs from"fs";import*as J from"fs/promises";import*as ue from"path";import{spawn as is}from"child_process";import{Readable as Bi}from"stream";function ne(){let n=process.platform==="darwin"?"darwin":process.platform==="linux"?"linux":process.platform==="win32"?"windows":void 0,e=process.arch==="arm64"?"aarch64":process.arch==="x64"?"x86_64":void 0;if(!(!n||!e))return`${n}-${e}`}function Kt(n,e=ne()){if(e)return n[e]}var ke=n=>{process.stderr.write(n+`
6
- `)};function Xt(n){ke=n??(e=>process.stderr.write(e+`
7
- `))}async function os(n){if(!n.target.archive)throw new Error(`Agent ${n.agentId} has no archive URL for ${ne()??"this platform"}`);if(!n.target.cmd)throw new Error(`Agent ${n.agentId} has no cmd in its binary target`);let e=ne();if(!e)throw new Error(`Agent ${n.agentId}: cannot determine platform key for ${process.platform}/${process.arch}`);let t=w.agentInstallDir(n.agentId,e,n.version),s=ue.resolve(t,n.target.cmd);if(await Yt(s))return s;if(await Ui({agentId:n.agentId,version:n.version,archiveUrl:n.target.archive,installDir:t,onProgress:n.onProgress}),!await Yt(s))throw new Error(`Agent ${n.agentId}: extracted archive did not contain ${n.target.cmd} (looked in ${t})`);return process.platform!=="win32"&&await J.chmod(s,493).catch(()=>{}),s}async function Ui(n){await J.mkdir(ue.dirname(n.installDir),{recursive:!0});let e=await J.mkdtemp(`${n.installDir}.partial-`);try{ke(`hydra-acp: downloading ${n.agentId} from ${n.archiveUrl}`);let t=await Di({url:n.archiveUrl,dir:e,agentId:n.agentId,version:n.version,onProgress:n.onProgress});ke(`hydra-acp: extracting ${n.agentId}`),Ae(n.onProgress,{phase:"extract",agentId:n.agentId,version:n.version}),await qi(t,e),await J.unlink(t).catch(()=>{});try{await J.rename(e,n.installDir)}catch(s){let r=s;if((r.code==="EEXIST"||r.code==="ENOTEMPTY")&&await Yt(n.installDir)){await J.rm(e,{recursive:!0,force:!0}).catch(()=>{}),Ae(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version});return}throw s}ke(`hydra-acp: installed ${n.agentId} to ${n.installDir}`),Ae(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version})}catch(t){throw await J.rm(e,{recursive:!0,force:!0}).catch(()=>{}),t}}function Ae(n,e){if(n)try{n(e)}catch{}}async function Di(n){let e=Li(n.url),t=ue.join(n.dir,e),s=await fetch(n.url,{redirect:"follow"});if(!s.ok||!s.body)throw new Error(`Failed to download ${n.url}: HTTP ${s.status} ${s.statusText}`);let r=Number(s.headers.get("content-length")??"0"),i=rs.createWriteStream(t),o=Bi.fromWeb(s.body);Ae(n.onProgress,{phase:"download_start",agentId:n.agentId,version:n.version,totalBytes:r});let a=0,d=Date.now(),c=0,f=2e3,l=150;return o.on("data",u=>{a+=u.length;let p=Date.now();p-c>=l&&(c=p,Ae(n.onProgress,{phase:"download_progress",agentId:n.agentId,version:n.version,receivedBytes:a,totalBytes:r})),p-d>=f&&(d=p,ke(ss(n.agentId,a,r)))}),await new Promise((u,p)=>{o.on("error",p),i.on("error",p),i.on("finish",()=>u()),o.pipe(i)}),ke(ss(n.agentId,a,r,!0)),Ae(n.onProgress,{phase:"download_done",agentId:n.agentId,version:n.version,receivedBytes:a,totalBytes:r}),t}function ss(n,e,t,s=!1){let r=(e/1e6).toFixed(1);if(t>0){let o=(t/1e6).toFixed(1),a=Math.min(100,Math.floor(e/t*100));return`hydra-acp: ${s?"downloaded":"downloading"} ${n} ${r}/${o} MB (${a}%)`}return`hydra-acp: ${s?"downloaded":"downloading"} ${n} ${r} MB`}function Li(n){let e=new URL(n);return ue.posix.basename(e.pathname)||"archive"}async function qi(n,e){let t=n.toLowerCase();if(t.endsWith(".tar.gz")||t.endsWith(".tgz")||t.endsWith(".tar")){await Gt("tar",["-xf",n,"-C",e]);return}if(t.endsWith(".zip")){if(await Ji("unzip")){await Gt("unzip",["-q",n,"-d",e]);return}await Gt("tar",["-xf",n,"-C",e]);return}throw new Error(`Unsupported archive format: ${n}`)}function Gt(n,e){return new Promise((t,s)=>{let r=is(n,e,{stdio:["ignore","ignore","inherit"]});r.on("error",s),r.on("exit",(i,o)=>{if(i===0){t();return}s(new Error(`${n} ${e.join(" ")} exited with ${i!==null?`code ${i}`:`signal ${o}`}`))})})}async function Ji(n){return new Promise(e=>{let t=process.platform==="win32"?"where":"which",s=is(t,[n],{stdio:"ignore"});s.on("error",()=>e(!1)),s.on("exit",r=>e(r===0))})}async function Yt(n){try{return await J.access(n),!0}catch{return!1}}import*as G from"fs/promises";import*as se from"path";import{spawn as zi}from"child_process";var en=n=>{process.stderr.write(n+`
1
+ import*as hi from"fs";import*as me from"fs/promises";import Ic from"fastify";import Sc from"@fastify/websocket";var jn="acp.v1";function Hn(n,e){return n.has(jn)?jn:!1}import gi from"pino";import kc from"pino-roll";import*as Ln from"fs/promises";import{homedir as Wt}from"os";import{z as b}from"zod";import*as R from"path";import*as Jt from"os";var yi="HYDRA_ACP_HOME";function Se(n){let e=Jt.homedir();return e?n===e?"~":n.startsWith(e+"/")?"~"+n.slice(e.length):n:n}function T(){let n=process.env[yi];if(n&&n.length>0)return R.resolve(n);if(process.env.VITEST)throw new Error("HYDRA_ACP_HOME is unset under VITEST; vitest.setup.ts must run first");return R.join(Jt.homedir(),".hydra-acp")}var w={home:T,config:()=>R.join(T(),"config.json"),authToken:()=>R.join(T(),"auth-token"),remotes:()=>R.join(T(),"remotes.json"),pidFile:()=>R.join(T(),"daemon.pid"),logFile:()=>R.join(T(),"daemon.log"),currentLogFile:()=>R.join(T(),"current.log"),registryCache:()=>R.join(T(),"registry.json"),agentsDir:()=>R.join(T(),"agents"),agentLogFile:n=>R.join(T(),"agents","logs",`${n}.log`),agentInstallDir:(n,e,t)=>R.join(T(),"agents",e,n,t),agentNpmInstallDir:(n,e,t)=>R.join(T(),"agents",e,n,t,`node${process.versions.modules}`),sessionsDir:()=>R.join(T(),"sessions"),sessionDir:n=>R.join(T(),"sessions",n),sessionFile:n=>R.join(T(),"sessions",n,"meta.json"),historyFile:n=>R.join(T(),"sessions",n,"history.jsonl"),toolsDir:n=>R.join(T(),"sessions",n,"tools"),toolBlobFile:(n,e)=>R.join(T(),"sessions",n,"tools",e),queueFile:n=>R.join(T(),"sessions",n,"queue.ndjson"),tombstonesDir:()=>R.join(T(),"sessions",".tombstones"),tombstoneAgentDir:n=>R.join(T(),"sessions",".tombstones",encodeURIComponent(n)),tombstoneFile:(n,e)=>R.join(T(),"sessions",".tombstones",encodeURIComponent(n),encodeURIComponent(e)),extensionsDir:()=>R.join(T(),"extensions"),extensionLogFile:n=>R.join(T(),"extensions",`${n}.log`),extensionPidFile:n=>R.join(T(),"extensions",`${n}.pid`),transformersDir:()=>R.join(T(),"transformers"),transformerLogFile:n=>R.join(T(),"transformers",`${n}.log`),transformerPidFile:n=>R.join(T(),"transformers",`${n}.pid`),transformerState:(n,e)=>R.join(T(),"sessions",n,"transformer-state",e),tuiHistoryFile:n=>R.join(T(),"sessions",n,"prompt-history"),globalTuiHistoryFile:()=>R.join(T(),"prompt-history"),tuiLogFile:()=>R.join(T(),"tui.log"),shimWireLogFile:()=>R.join(T(),"shim-wire.log")};import*as ke from"fs/promises";function rt(){let n=new Uint8Array(32);crypto.getRandomValues(n);let e="";for(let t of n)e+=t.toString(16).padStart(2,"0");return`hydra_token_${e}`}async function Bn(){try{let e=(await ke.readFile(w.authToken(),"utf8")).trim();return e.length>0?e:void 0}catch(n){if(n.code==="ENOENT")return;throw n}}async function wi(){let n=await Bn();if(!n)throw new Error(`No service token found at ${w.authToken()}. Run \`hydra-acp init\` to create one.`);return n}async function zt(n){await ke.mkdir(w.home(),{recursive:!0}),await ke.writeFile(w.authToken(),n+`
2
+ `,{encoding:"utf8",mode:384})}async function vi(){let n=await Bn();if(n)return n;let e=rt();return await zt(e),process.stderr.write(`hydra-acp: initialized ${w.authToken()} with a fresh service token.
3
+ `),e}import*as ee from"fs/promises";import*as Un from"fs";import{randomBytes as bi}from"crypto";async function K(n,e,t={}){let r=(t.pretty??!0?JSON.stringify(e,null,2):JSON.stringify(e))+`
4
+ `;await Ii(n,r,t)}async function Ii(n,e,t={}){let s=Si(n);await ee.mkdir(s,{recursive:!0});let r=`${n}.tmp-${process.pid}-${ki()}`;try{let i={encoding:"utf8"};t.mode!==void 0&&(i.mode=t.mode),await ee.writeFile(r,e,i),await ee.rename(r,n)}catch(i){throw await ee.unlink(r).catch(()=>{}),i}if(t.mode!==void 0)try{Un.chmodSync(n,t.mode)}catch{}}async function te(n){let e;try{e=await ee.readFile(n,"utf8")}catch(t){if(t.code==="ENOENT")return;throw t}if(e.trim().length!==0)try{return JSON.parse(e)}catch{return}}function Si(n){let e=n.lastIndexOf("/");return e<=0?".":n.slice(0,e)}function ki(){return bi(4).toString("hex")}var qn="https://cdn.agentclientprotocol.com/registry/v1/latest/registry.json",Ai=b.object({cert:b.string(),key:b.string()}),Jn=55514,xi=b.object({host:b.string().default("127.0.0.1"),port:b.number().int().positive().default(Jn),logLevel:b.enum(["debug","info","warn","error"]).default("info"),tls:Ai.optional(),sessionIdleTimeoutSeconds:b.number().int().nonnegative().default(3600),sessionHistoryMaxEntries:b.number().int().positive().default(1e4),agentStderrTailBytes:b.number().int().positive().default(4096),publicHost:b.string().optional(),agentSyncIntervalMinutes:b.number().nonnegative().default(60),sessionGcIntervalMinutes:b.number().nonnegative().default(60),sessionGcMaxAgeDays:b.number().positive().default(2)}),Ci=b.object({url:b.string().url().default(qn),ttlHours:b.number().positive().default(24),pinned:b.boolean().default(!1)}),Mi=b.object({name:b.string().optional(),description:b.string().optional(),command:b.string().optional(),args:b.array(b.string()).optional(),env:b.record(b.string()).optional()}),Ri=b.object({packageSpec:b.string().optional()}),Ei=b.object({repaintThrottleMs:b.number().int().nonnegative().default(1e3),maxScrollbackLines:b.number().int().positive().default(1e4),mouse:b.boolean().default(!1),inAppSelection:b.boolean().optional(),selectionClipboard:b.enum(["primary","clipboard","both"]).default("both"),logMaxBytes:b.number().int().positive().default(5*1024*1024),cwdColumnMaxWidth:b.number().int().positive().default(32),progressIndicator:b.boolean().default(!0),defaultEnterAction:b.enum(["enqueue","amend"]).default("amend"),showThoughts:b.boolean().default(!0),ambiguousWidth:b.enum(["narrow","wide"]).default("wide"),toolContent:b.enum(["inline","references"]).default("references"),diffContextLines:b.number().int().min(0).default(3),promptHistoryMaxEntries:b.number().int().positive().default(2e3),maxToolItems:b.number().int().nonnegative().default(5),maxPlanItems:b.number().int().nonnegative().default(5),showFileUpdates:b.enum(["none","edit","diff"]).default("edit"),sessionColumns:b.array(b.enum(["session","upstream","host","state","agent","model","age","cwd","title","cost"])).nonempty().optional()}),Dn=b.string().min(1).regex(/^[A-Za-z0-9._-]+$/,"extension name must be filename-safe"),Pi=b.object({command:b.array(b.string()).default([]),args:b.array(b.string()).default([]),env:b.record(b.string()).default({}),enabled:b.boolean().default(!0)}),Ti=b.object({command:b.array(b.string()).default([]),args:b.array(b.string()).default([]),env:b.record(b.string()).default({}),enabled:b.boolean().default(!0)}),zn=b.object({daemon:xi.default({}),registry:Ci.default({url:qn,ttlHours:24,pinned:!1}),agents:b.record(b.string(),Mi).default({}),agentOverrides:b.record(b.string(),Ri).default({}),defaultAgent:b.string().default("opencode"),defaultModels:b.record(b.string(),b.string()).default({}),synopsisAgent:b.string().optional(),synopsisModel:b.string().optional(),synopsisOnClose:b.boolean().default(!1),defaultCwd:b.string().default("~"),compressToolContent:b.boolean().default(!0),sessionListColdLimit:b.number().int().nonnegative().default(20),extensions:b.record(Dn,Pi).default({}),transformers:b.record(Dn,Ti).default({}),defaultTransformers:b.array(b.string()).default([]),npmRegistry:b.string().url().optional(),tui:Ei.default({repaintThrottleMs:1e3,maxScrollbackLines:1e4,mouse:!1,logMaxBytes:5*1024*1024,cwdColumnMaxWidth:32,progressIndicator:!0,defaultEnterAction:"amend",showThoughts:!0,ambiguousWidth:"wide",toolContent:"references",diffContextLines:3,promptHistoryMaxEntries:2e3,maxToolItems:5,maxPlanItems:5,showFileUpdates:"edit",selectionClipboard:"both"})});function Wn(n){return Object.entries(n.extensions).map(([e,t])=>({name:e,...t}))}function Qn(n){return Object.entries(n.transformers).map(([e,t])=>({name:e,...t}))}async function Vn(){return await te(w.config())??{}}async function _i(){let n=await Vn(),e=n.daemon,t=e&&typeof e.authToken=="string"?e.authToken:void 0;if(!t)return;let s=!1;try{await Ln.access(w.authToken()),s=!0}catch(r){if(r.code!=="ENOENT")throw r}if(s)throw new Error(`Auth token present in both ${w.authToken()} and ${w.config()} (daemon.authToken). Remove daemon.authToken from config.json to resolve.`);await zt(t),delete e.authToken,Object.keys(e).length===0&&delete n.daemon,await K(w.config(),n,{mode:384}),process.stderr.write(`hydra-acp: migrated auth token from ${w.config()} to ${w.authToken()}.
5
+ `)}async function Gn(){return await _i(),zn.parse(await Vn())}async function $i(n){await K(w.config(),n,{mode:384})}function Oi(){return zn.parse({})}function He(n){return n==="~"||n==="$HOME"?Wt():n.startsWith("~/")?Wt()+n.slice(1):n.startsWith("$HOME/")?Wt()+n.slice(5):n}import*as L from"fs/promises";import{createHash as Fi}from"crypto";import{gzip as Ni,gunzip as ji}from"zlib";import{promisify as Yn}from"util";var Kn=Yn(Ni),Hi=Yn(ji),Xn=/^[A-Za-z0-9_-]+$/,Bi=/^[a-f0-9]{64}$/,Qt=!0;function Zn(n){Qt=n}function it(n,e){return Xn.test(n)?e===void 0||Bi.test(e):!1}function ot(n,e){return`${w.toolBlobFile(n,e)}.gz`}async function Vt(n,e){if(!it(n))return null;let t=Fi("sha256").update(e,"utf8").digest("hex"),s=ot(n,t),r=w.toolBlobFile(n,t);for(let a of[s,r])try{return await L.access(a),t}catch{}await L.mkdir(w.toolsDir(n),{recursive:!0});let i=Qt?s:r,o=Qt?await Kn(Buffer.from(e,"utf8")):Buffer.from(e,"utf8");return await L.writeFile(i,o,{mode:384,flag:"wx"}).catch(async a=>{if(a.code!=="EEXIST")throw a}),t}async function at(n,e){if(!it(n,e))return null;try{let t=await L.readFile(ot(n,e));return(await Hi(t)).toString("utf8")}catch{}try{return await L.readFile(w.toolBlobFile(n,e),"utf8")}catch{return null}}async function es(n,e){if(!it(n,e))return null;try{return await L.readFile(ot(n,e))}catch{}try{let t=await L.readFile(w.toolBlobFile(n,e));return await Kn(t)}catch{return null}}async function ts(n,e,t){if(!it(n,e))return;let s=ot(n,e);try{await L.access(s);return}catch{}await L.mkdir(w.toolsDir(n),{recursive:!0}),await L.writeFile(s,t,{mode:384,flag:"wx"}).catch(r=>{if(r.code!=="EEXIST")throw r})}async function ns(n){Xn.test(n)&&await L.rm(w.toolsDir(n),{recursive:!0,force:!0}).catch(()=>{})}import*as ps from"fs/promises";import*as dt from"path";import{z as x}from"zod";import*as rs from"fs";import*as J from"fs/promises";import*as ue from"path";import{spawn as is}from"child_process";import{Readable as Ui}from"stream";function ne(){let n=process.platform==="darwin"?"darwin":process.platform==="linux"?"linux":process.platform==="win32"?"windows":void 0,e=process.arch==="arm64"?"aarch64":process.arch==="x64"?"x86_64":void 0;if(!(!n||!e))return`${n}-${e}`}function Kt(n,e=ne()){if(e)return n[e]}var Ae=n=>{process.stderr.write(n+`
6
+ `)};function Xt(n){Ae=n??(e=>process.stderr.write(e+`
7
+ `))}async function os(n){if(!n.target.archive)throw new Error(`Agent ${n.agentId} has no archive URL for ${ne()??"this platform"}`);if(!n.target.cmd)throw new Error(`Agent ${n.agentId} has no cmd in its binary target`);let e=ne();if(!e)throw new Error(`Agent ${n.agentId}: cannot determine platform key for ${process.platform}/${process.arch}`);let t=w.agentInstallDir(n.agentId,e,n.version),s=ue.resolve(t,n.target.cmd);if(await Yt(s))return s;if(await Di({agentId:n.agentId,version:n.version,archiveUrl:n.target.archive,installDir:t,onProgress:n.onProgress}),!await Yt(s))throw new Error(`Agent ${n.agentId}: extracted archive did not contain ${n.target.cmd} (looked in ${t})`);return process.platform!=="win32"&&await J.chmod(s,493).catch(()=>{}),s}async function Di(n){await J.mkdir(ue.dirname(n.installDir),{recursive:!0});let e=await J.mkdtemp(`${n.installDir}.partial-`);try{Ae(`hydra-acp: downloading ${n.agentId} from ${n.archiveUrl}`);let t=await Li({url:n.archiveUrl,dir:e,agentId:n.agentId,version:n.version,onProgress:n.onProgress});Ae(`hydra-acp: extracting ${n.agentId}`),xe(n.onProgress,{phase:"extract",agentId:n.agentId,version:n.version}),await Ji(t,e),await J.unlink(t).catch(()=>{});try{await J.rename(e,n.installDir)}catch(s){let r=s;if((r.code==="EEXIST"||r.code==="ENOTEMPTY")&&await Yt(n.installDir)){await J.rm(e,{recursive:!0,force:!0}).catch(()=>{}),xe(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version});return}throw s}Ae(`hydra-acp: installed ${n.agentId} to ${n.installDir}`),xe(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version})}catch(t){throw await J.rm(e,{recursive:!0,force:!0}).catch(()=>{}),t}}function xe(n,e){if(n)try{n(e)}catch{}}async function Li(n){let e=qi(n.url),t=ue.join(n.dir,e),s=await fetch(n.url,{redirect:"follow"});if(!s.ok||!s.body)throw new Error(`Failed to download ${n.url}: HTTP ${s.status} ${s.statusText}`);let r=Number(s.headers.get("content-length")??"0"),i=rs.createWriteStream(t),o=Ui.fromWeb(s.body);xe(n.onProgress,{phase:"download_start",agentId:n.agentId,version:n.version,totalBytes:r});let a=0,d=Date.now(),c=0,p=2e3,l=150;return o.on("data",u=>{a+=u.length;let f=Date.now();f-c>=l&&(c=f,xe(n.onProgress,{phase:"download_progress",agentId:n.agentId,version:n.version,receivedBytes:a,totalBytes:r})),f-d>=p&&(d=f,Ae(ss(n.agentId,a,r)))}),await new Promise((u,f)=>{o.on("error",f),i.on("error",f),i.on("finish",()=>u()),o.pipe(i)}),Ae(ss(n.agentId,a,r,!0)),xe(n.onProgress,{phase:"download_done",agentId:n.agentId,version:n.version,receivedBytes:a,totalBytes:r}),t}function ss(n,e,t,s=!1){let r=(e/1e6).toFixed(1);if(t>0){let o=(t/1e6).toFixed(1),a=Math.min(100,Math.floor(e/t*100));return`hydra-acp: ${s?"downloaded":"downloading"} ${n} ${r}/${o} MB (${a}%)`}return`hydra-acp: ${s?"downloaded":"downloading"} ${n} ${r} MB`}function qi(n){let e=new URL(n);return ue.posix.basename(e.pathname)||"archive"}async function Ji(n,e){let t=n.toLowerCase();if(t.endsWith(".tar.gz")||t.endsWith(".tgz")||t.endsWith(".tar")){await Gt("tar",["-xf",n,"-C",e]);return}if(t.endsWith(".zip")){if(await zi("unzip")){await Gt("unzip",["-q",n,"-d",e]);return}await Gt("tar",["-xf",n,"-C",e]);return}throw new Error(`Unsupported archive format: ${n}`)}function Gt(n,e){return new Promise((t,s)=>{let r=is(n,e,{stdio:["ignore","ignore","inherit"]});r.on("error",s),r.on("exit",(i,o)=>{if(i===0){t();return}s(new Error(`${n} ${e.join(" ")} exited with ${i!==null?`code ${i}`:`signal ${o}`}`))})})}async function zi(n){return new Promise(e=>{let t=process.platform==="win32"?"where":"which",s=is(t,[n],{stdio:"ignore"});s.on("error",()=>e(!1)),s.on("exit",r=>e(r===0))})}async function Yt(n){try{return await J.access(n),!0}catch{return!1}}import*as G from"fs/promises";import*as se from"path";import{spawn as Wi}from"child_process";var en=n=>{process.stderr.write(n+`
8
8
  `)};function tn(n){en=n??(e=>process.stderr.write(e+`
9
- `))}async function ds(n){let e=ne();if(!e)throw new Error(`Agent ${n.agentId}: cannot determine platform key for ${process.platform}/${process.arch}`);let t=w.agentNpmInstallDir(n.agentId,e,n.version),s=Yi(n.packageSpec),r=Ki(s),i={installDir:t,packageName:s,hint:n.bin,basename:r},o=se.join(t,"node_modules",".bin",n.bin);if(await ge(o))return o;if(await ge(t)){let l=await as(i);if(l)return l.binPath}await Wi({agentId:n.agentId,version:n.version,packageSpec:n.packageSpec,installDir:t,registry:n.registry,onProgress:n.onProgress});let a=await as(i);if(a)return a.binPath;let d=await us(t,s),c=typeof d=="object"&&d!==null?Object.keys(d):typeof d=="string"?[r]:[],f=c.length>0?` (package declares bins: ${c.join(", ")})`:"";throw new Error(`Agent ${n.agentId}: npm install of ${n.packageSpec} did not produce bin ${n.bin} (looked in ${t}/node_modules/.bin/)${f}`)}async function Wi(n){await G.mkdir(se.dirname(n.installDir),{recursive:!0});let e=await G.mkdtemp(`${n.installDir}.partial-`);try{en(`hydra-acp: installing ${n.packageSpec} for ${n.agentId} into ${e}`),Zt(n.onProgress,{phase:"install_start",agentId:n.agentId,version:n.version,packageSpec:n.packageSpec}),await Gi({packageSpec:n.packageSpec,cwd:e,registry:n.registry});try{await G.rename(e,n.installDir)}catch(t){let s=t;if((s.code==="EEXIST"||s.code==="ENOTEMPTY")&&await ge(n.installDir)){await G.rm(e,{recursive:!0,force:!0}).catch(()=>{}),Zt(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version});return}throw t}en(`hydra-acp: installed ${n.agentId} to ${n.installDir}`),Zt(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version})}catch(t){throw await G.rm(e,{recursive:!0,force:!0}).catch(()=>{}),t}}function Zt(n,e){if(n)try{n(e)}catch{}}var Qi=5,Vi=25;function Gi(n){return cs(n,0)}async function cs(n,e){try{await new Promise((t,s)=>{let r=n.registry?["--registry",n.registry]:[],i;try{i=zi("npm",["install","--no-audit","--no-fund","--silent",...r,n.packageSpec],{cwd:n.cwd,stdio:["ignore","pipe","pipe"]})}catch(a){s(a);return}let o="";i.stdout?.on("data",a=>{}),i.stderr?.setEncoding("utf8"),i.stderr?.on("data",a=>{o=(o+a).slice(-4096)}),i.on("error",a=>{if(a.code==="ENOENT"){s(new Error("npm not found on PATH (install Node.js / npm, or use a binary-distributed agent)"));return}s(a)}),i.on("exit",(a,d)=>{if(a===0){t();return}let c=a!==null?`exit code ${a}`:`signal ${d??"unknown"}`,f=o.trim();s(new Error(f?`npm install ${n.packageSpec} failed (${c})
10
- stderr: ${f}`:`npm install ${n.packageSpec} failed (${c})`))})})}catch(t){if(t.code==="ETXTBSY"&&e<Qi)return await new Promise(r=>setTimeout(r,Vi*(e+1))),cs(n,e+1);throw t}}function Yi(n){if(n.startsWith("@")){let t=n.indexOf("/");if(t===-1)return n;let r=n.slice(t+1).indexOf("@");return r===-1?n:n.slice(0,t+1+r)}let e=n.indexOf("@");return e<=0?n:n.slice(0,e)}function Ki(n){let e=n.lastIndexOf("/");return e===-1?n:n.slice(e+1)}async function us(n,e){let t=se.join(n,"node_modules",e,"package.json");try{let s=await G.readFile(t,"utf8"),r=JSON.parse(s);return typeof r.bin=="string"||typeof r.bin=="object"&&r.bin!==null&&!Array.isArray(r.bin)?r.bin:void 0}catch{return}}async function as(n){let e=se.join(n.installDir,"node_modules",".bin"),t=se.join(e,n.hint);if(await ge(t))return{binName:n.hint,binPath:t};let s=await us(n.installDir,n.packageName);if(typeof s=="object"&&s!==null){let r=Object.keys(s);if(r.length===1){let i=r[0],o=se.join(e,i);if(await ge(o))return{binName:i,binPath:o}}else if(r.length>1){for(let i of[n.hint,n.basename])if(r.includes(i)){let o=se.join(e,i);if(await ge(o))return{binName:i,binPath:o}}}}if(n.basename!==n.hint){let r=se.join(e,n.basename);if(await ge(r))return{binName:n.basename,binPath:r}}}async function ge(n){try{return await G.access(n),!0}catch{return!1}}var Xi=x.object({package:x.string(),bin:x.string().optional(),args:x.array(x.string()).optional(),env:x.record(x.string()).optional()}),xe=x.object({archive:x.string().url().optional(),cmd:x.string().optional(),args:x.array(x.string()).optional(),env:x.record(x.string()).optional()}),Zi=x.object({"darwin-aarch64":xe.optional(),"darwin-x86_64":xe.optional(),"linux-aarch64":xe.optional(),"linux-x86_64":xe.optional(),"windows-x86_64":xe.optional(),"windows-aarch64":xe.optional()}),eo=x.object({package:x.string(),args:x.array(x.string()).optional(),env:x.record(x.string()).optional()}),to=x.object({command:x.string(),args:x.array(x.string()).optional(),env:x.record(x.string()).optional()}),no=x.object({npx:Xi.optional(),binary:Zi.optional(),uvx:eo.optional(),exec:to.optional()}),so=x.object({id:x.string(),name:x.string(),version:x.string().optional(),description:x.string().optional(),authors:x.array(x.string()).optional(),license:x.string().optional(),icon:x.string().optional(),repository:x.string().optional(),website:x.string().optional(),distribution:no}),ls=x.object({version:x.string(),agents:x.array(so),extensions:x.array(x.unknown()).optional()}),Be=class{constructor(e,t={}){this.config=e;this.options=t}config;options;cache;async load(){if(this.cache&&(this.isPinned()||this.isFresh(this.cache.fetchedAt)))return this.cache.data;let e=await this.readDiskCache();if(e&&(this.isPinned()||this.isFresh(e.fetchedAt)))return this.cache=e,e.data;try{let t=await this.fetchFromNetwork();return this.cache=t,await this.writeDiskCache(t),t.data}catch(t){if(e)return this.cache=e,e.data;throw t}}async refresh(){let e=await this.fetchFromNetwork();return this.cache=e,await this.writeDiskCache(e),e.data}lastFetchedAt(){return this.cache?.fetchedAt}async getAgent(e){let t=this.localAgents().find(o=>o.id===e);if(t)return t;let s=await this.load(),r=s.agents.find(o=>o.id===e);if(r)return this.applyOverride(r);let i=s.agents.find(o=>nn(o)===e);return i?this.applyOverride(i):void 0}localAgents(){return Object.entries(this.config.agents??{}).map(([e,t])=>({id:e,name:t.name??e,description:t.description,version:"local",distribution:{exec:{command:t.command??e,args:t.args,env:t.env}}}))}applyOverride(e){let t=this.config.agentOverrides?.[e.id];return!t?.packageSpec||!e.distribution.npx?e:{...e,version:ro(t.packageSpec),distribution:{...e.distribution,npx:{...e.distribution.npx,package:t.packageSpec}}}}isPinned(){return this.config.registry?.pinned===!0}isFresh(e){let t=Date.now()-e,s=this.config.registry.ttlHours*60*60*1e3;return t<s}async fetchFromNetwork(){let e=await fetch(this.config.registry.url);if(!e.ok)throw new Error(`Registry fetch failed: HTTP ${e.status}`);let t=await e.json(),s=ls.parse(t),r={fetchedAt:Date.now(),raw:t,data:s},i=this.options.onFetched;return i&&Promise.resolve().then(()=>i(s)).catch(()=>{}),r}async readDiskCache(){let e=await te(w.registryCache());if(!(!e||typeof e.fetchedAt!="number"||e.data===void 0))try{let t=ls.parse(e.data);return{fetchedAt:e.fetchedAt,raw:e.data,data:t}}catch{return}}async writeDiskCache(e){await K(w.registryCache(),{fetchedAt:e.fetchedAt,data:e.raw})}};function ro(n){let e=n.lastIndexOf("@"),s=(e>0?n.slice(e+1):"").replace(/[^a-zA-Z0-9._-]/g,"_");return s.length>0?`pin-${s}`:"pinned"}function nn(n){let e=n.distribution.npx?.package;if(!e)return;let t=e.lastIndexOf("/"),s=t===-1?e:e.slice(t+1),r=s.lastIndexOf("@");return r<=0?s:s.slice(0,r)}async function Ce(n){let e=typeof n.localAgents=="function"?n.localAgents():[],t;try{t=await n.load()}catch(o){if(e.length===0)throw o;t={version:"local-only",agents:[]}}let s=new Set(e.map(o=>o.id)),r=[...e,...t.agents.filter(o=>!s.has(o.id))],i=await Promise.all(r.map(async o=>({id:o.id,name:o.name,version:o.version,description:o.description,distributions:Object.keys(o.distribution),installed:await sn(o),source:s.has(o.id)?"local":"registry"})));return{version:t.version,fetchedAt:n.lastFetchedAt(),agents:i}}async function sn(n){let e=ne();if(!e)return"no";let t=n.version??"current";if(n.distribution.exec)return"yes";if(n.distribution.binary){let s=Kt(n.distribution.binary,e);if(s?.cmd){let r=dt.resolve(w.agentInstallDir(n.id,e,t),s.cmd);if(await fs(r))return"yes"}}if(n.distribution.npx){let s=n.distribution.npx,r=s.bin??nn(n)??s.package,i=w.agentNpmInstallDir(n.id,e,t),o=dt.join(i,"node_modules",".bin",r);if(await fs(o))return"yes"}return!n.distribution.npx&&!n.distribution.binary&&n.distribution.uvx?"lazy":"no"}async function fs(n){try{return await ps.access(n),!0}catch{return!1}}async function oe(n,e=[],t={}){let s=n.version??"current";if(n.distribution.npx){let r=n.distribution.npx,i=e.length>0?e:r.args??[];if(process.env.HYDRA_ACP_SKIP_NPM_PREFETCH)return{command:"npx",args:["-y",r.package,...i],env:r.env??{},version:s};let o=r.bin??nn(n)??r.package,a=t.onInstallProgress;return{command:await ds({agentId:n.id,version:s,packageSpec:r.package,bin:o,registry:t.npmRegistry,onProgress:a?c=>a({source:"npm",...c}):void 0}),args:i,env:r.env??{},version:s}}if(n.distribution.binary){let r=Kt(n.distribution.binary);if(!r)throw new Error(`Agent ${n.id} has no binary distribution for ${ne()??"this platform"}.`);let i=t.onInstallProgress,o=await os({agentId:n.id,version:s,target:r,onProgress:i?d=>i({source:"binary",...d}):void 0}),a=e.length>0?e:r.args??[];return{command:o,args:a,env:r.env??{},version:s}}if(n.distribution.uvx){let r=n.distribution.uvx,i=e.length>0?e:r.args??[];return{command:"uvx",args:[r.package,...i],env:r.env??{},version:s}}if(n.distribution.exec){let r=n.distribution.exec,i=e.length>0?e:r.args??[];return{command:r.command,args:i,env:r.env??{},version:s}}throw new Error(`Agent ${n.id} has no usable distribution method.`)}import{spawn as uo}from"child_process";import*as ft from"fs";import*as Ms from"path";import{z as h}from"zod";var le=1,I={ParseError:-32700,InvalidRequest:-32600,MethodNotFound:-32601,InvalidParams:-32602,InternalError:-32603,SessionNotFound:-32001,NotAuthorisedToAttach:-32002,MultiClientNotSupported:-32003,AgentNotInstalled:-32005,BundleAlreadyImported:-32010,PermissionDenied:-32011,AlreadyAttached:-32012,StreamNotEnabled:-32013,SessionClosing:-32014},ms=h.object({protocolVersion:h.number().optional(),clientCapabilities:h.record(h.unknown()).optional(),clientInfo:h.object({name:h.string(),version:h.string().optional()}).optional()}),io=h.enum(["full","pending_only","none","after_message"]),gs=h.object({cwd:h.string(),mcpServers:h.array(h.unknown()).optional(),_meta:h.record(h.unknown()).optional()}),oo=h.object({upstreamSessionId:h.string(),agentId:h.string(),cwd:h.string(),title:h.string().optional(),agentArgs:h.array(h.string()).optional()}),hs=h.object({sessionId:h.string(),historyPolicy:io.default("full"),afterMessageId:h.string().optional(),clientId:h.string().optional(),clientInfo:h.object({name:h.string(),version:h.string().optional()}).optional(),_meta:h.record(h.unknown()).optional()}),Ue="hydra-acp";function De(n){if(!n)return{};let e=n[Ue];if(!e||typeof e!="object"||Array.isArray(e))return{};let t=e,s={};if(typeof t.upstreamSessionId=="string"&&(s.upstreamSessionId=t.upstreamSessionId),typeof t.agentId=="string"&&(s.agentId=t.agentId),typeof t.cwd=="string"&&(s.cwd=t.cwd),typeof t.clientId=="string"&&(s.clientId=t.clientId),typeof t.readonly=="boolean"&&(s.readonly=t.readonly),(t.replayMode==="instant"||t.replayMode==="drip")&&(s.replayMode=t.replayMode),typeof t.dripSpeed=="number"&&t.dripSpeed>0&&(s.dripSpeed=t.dripSpeed),(t.toolContent==="inline"||t.toolContent==="references")&&(s.toolContent=t.toolContent),t.detachStatus==="detached"&&(s.detachStatus=t.detachStatus),typeof t.title=="string"&&(s.title=t.title),Array.isArray(t.agentArgs)&&t.agentArgs.every(r=>typeof r=="string")&&(s.agentArgs=t.agentArgs),Array.isArray(t.transformers)&&t.transformers.every(r=>typeof r=="string")&&(s.transformers=t.transformers),t.resume){let r=oo.safeParse(t.resume);r.success&&(s.resume=r.data)}if(typeof t.model=="string"&&(s.model=t.model),typeof t.currentModel=="string"&&(s.currentModel=t.currentModel),typeof t.currentMode=="string"&&(s.currentMode=t.currentMode),t.currentUsage){let r=vs.safeParse(t.currentUsage);r.success&&(s.currentUsage=r.data)}if(typeof t.turnStartedAt=="number"&&t.turnStartedAt>0&&(s.turnStartedAt=t.turnStartedAt),Array.isArray(t.availableCommands)){let r=[];for(let i of t.availableCommands){if(!i||typeof i!="object"||Array.isArray(i))continue;let o=i;if(typeof o.name!="string")continue;let a={name:o.name};typeof o.description=="string"&&(a.description=o.description),r.push(a)}r.length>0&&(s.availableCommands=r)}if(t.prompt&&typeof t.prompt=="object"&&!Array.isArray(t.prompt)){let r=t.prompt,i={};typeof r.queueing=="boolean"&&(i.queueing=r.queueing),typeof r.cancelling=="boolean"&&(i.cancelling=r.cancelling),typeof r.updating=="boolean"&&(i.updating=r.updating),typeof r.amending=="boolean"&&(i.amending=r.amending),typeof r.pipelining=="boolean"&&(i.pipelining=r.pipelining),s.prompt=i}if(t.agents&&typeof t.agents=="object"&&!Array.isArray(t.agents)){let r=t.agents,i={};typeof r.list=="boolean"&&(i.list=r.list),typeof r.installProgress=="boolean"&&(i.installProgress=r.installProgress),s.agents=i}if(typeof t.mcpStdin=="boolean"&&(s.mcpStdin=t.mcpStdin),typeof t.interactive=="boolean"&&(s.interactive=t.interactive),typeof t.ancillary=="boolean"&&(s.ancillary=t.ancillary),Array.isArray(t.queue)){let r=[];for(let i of t.queue){if(!i||typeof i!="object"||Array.isArray(i))continue;let o=i,a=o.originator;if(typeof o.messageId!="string"||!a||typeof a.clientId!="string"||!Array.isArray(o.prompt)||typeof o.position!="number"||typeof o.enqueuedAt!="number")continue;let d={clientId:a.clientId};typeof a.name=="string"&&(d.name=a.name),typeof a.version=="string"&&(d.version=a.version),r.push({messageId:o.messageId,originator:d,prompt:o.prompt,position:o.position,enqueuedAt:o.enqueuedAt})}r.length>0&&(s.queue=r)}if(Array.isArray(t.availableModes)){let r=[];for(let i of t.availableModes){if(!i||typeof i!="object"||Array.isArray(i))continue;let o=i;if(typeof o.id!="string")continue;let a={id:o.id};typeof o.name=="string"&&(a.name=o.name),typeof o.description=="string"&&(a.description=o.description),r.push(a)}r.length>0&&(s.availableModes=r)}if(Array.isArray(t.availableModels)){let r=[];for(let i of t.availableModels){if(!i||typeof i!="object"||Array.isArray(i))continue;let o=i;if(typeof o.modelId!="string")continue;let a={modelId:o.modelId};typeof o.name=="string"&&(a.name=o.name),typeof o.description=="string"&&(a.description=o.description),r.push(a)}r.length>0&&(s.availableModels=r)}if((t.status==="live"||t.status==="cold")&&(s.status=t.status),typeof t.busy=="boolean"&&(s.busy=t.busy),typeof t.awaitingInput=="boolean"&&(s.awaitingInput=t.awaitingInput),typeof t.attachedClients=="number"&&(s.attachedClients=t.attachedClients),typeof t.importedFromMachine=="string"&&(s.importedFromMachine=t.importedFromMachine),typeof t.importedFromUpstreamSessionId=="string"&&(s.importedFromUpstreamSessionId=t.importedFromUpstreamSessionId),typeof t.parentSessionId=="string"&&(s.parentSessionId=t.parentSessionId),typeof t.forkedFromSessionId=="string"&&(s.forkedFromSessionId=t.forkedFromSessionId),typeof t.forkedFromMessageId=="string"&&(s.forkedFromMessageId=t.forkedFromMessageId),t.originatingClient&&typeof t.originatingClient=="object"&&!Array.isArray(t.originatingClient)&&typeof t.originatingClient.name=="string"){let r=t.originatingClient;s.originatingClient={name:r.name,...typeof r.version=="string"?{version:r.version}:{}}}return t.agentCapabilities!==void 0&&(s.agentCapabilities=t.agentCapabilities),s}function ct(n,e){return{...n??{},[Ue]:e}}var ys=h.object({sessionId:h.string()}),ws=h.object({cwd:h.string().optional(),cursor:h.string().optional()}),vs=h.object({used:h.number().optional(),size:h.number().optional(),costAmount:h.number().optional(),costCurrency:h.string().optional()}),ou=h.object({sessionId:h.string(),upstreamSessionId:h.string().optional(),cwd:h.string(),title:h.string().optional(),agentId:h.string().optional(),currentModel:h.string().optional(),currentUsage:vs.optional(),importedFromMachine:h.string().optional(),importedFromUpstreamSessionId:h.string().optional(),parentSessionId:h.string().optional(),forkedFromSessionId:h.string().optional(),forkedFromMessageId:h.string().optional(),originatingClient:h.object({name:h.string(),version:h.string().optional()}).optional(),interactive:h.boolean().optional(),priority:h.number().int().nonnegative().optional(),updatedAt:h.string(),attachedClients:h.number().int().nonnegative(),status:h.enum(["live","cold"]).default("live"),busy:h.boolean().default(!1),awaitingInput:h.boolean().default(!1),_meta:h.record(h.unknown()).optional()}),ao=h.object({sessionId:h.string(),cwd:h.string(),title:h.string().optional(),updatedAt:h.string().optional(),_meta:h.record(h.unknown()).optional()}),au=h.object({sessions:h.array(ao),nextCursor:h.string().optional()});function ut(n,e){let t={attachedClients:n.attachedClients,status:n.status,busy:n.busy,awaitingInput:n.awaitingInput};return n.cwd!==void 0&&(t.cwd=n.cwd),n.title!==void 0&&(t.title=n.title),n.agentId!==void 0&&(t.agentId=n.agentId),n.upstreamSessionId!==void 0&&(t.upstreamSessionId=n.upstreamSessionId),n.currentModel!==void 0&&(t.currentModel=n.currentModel),n.currentUsage!==void 0&&(t.currentUsage=n.currentUsage),n.importedFromMachine!==void 0&&(t.importedFromMachine=n.importedFromMachine),n.importedFromUpstreamSessionId!==void 0&&(t.importedFromUpstreamSessionId=n.importedFromUpstreamSessionId),n.parentSessionId!==void 0&&(t.parentSessionId=n.parentSessionId),n.forkedFromSessionId!==void 0&&(t.forkedFromSessionId=n.forkedFromSessionId),n.forkedFromMessageId!==void 0&&(t.forkedFromMessageId=n.forkedFromMessageId),n.originatingClient!==void 0&&(t.originatingClient=n.originatingClient),n.interactive!==void 0&&(t.interactive=n.interactive),n.priority!==void 0&&n.priority>0&&(t.priority=n.priority),e&&(e.clientId!==void 0&&(t.clientId=e.clientId),e.currentMode!==void 0&&(t.currentMode=e.currentMode),e.agentArgs!==void 0&&e.agentArgs.length>0&&(t.agentArgs=e.agentArgs),e.availableCommands!==void 0&&e.availableCommands.length>0&&(t.availableCommands=e.availableCommands),e.availableModes!==void 0&&e.availableModes.length>0&&(t.availableModes=e.availableModes),e.availableModels!==void 0&&e.availableModels.length>0&&(t.availableModels=e.availableModels),e.turnStartedAt!==void 0&&(t.turnStartedAt=e.turnStartedAt),e.agentCapabilities!==void 0&&(t.agentCapabilities=e.agentCapabilities),e.queue!==void 0&&e.queue.length>0&&(t.queue=e.queue)),t}function bs(n){let e={sessionId:n.sessionId,cwd:n.cwd,updatedAt:n.updatedAt,_meta:ct(n._meta,ut(n))};return n.title!==void 0&&(e.title=n.title),e}var Is=h.object({sessionId:h.string(),prompt:h.array(h.unknown()),_meta:h.record(h.unknown()).optional()}),lt=h.object({sessionId:h.string()}),Ss=h.object({clientId:h.string(),name:h.string().optional(),version:h.string().optional()}),du=h.object({sessionId:h.string(),messageId:h.string(),originator:Ss,prompt:h.array(h.unknown()),position:h.number().int().nonnegative(),queueDepth:h.number().int().positive(),enqueuedAt:h.number()}),cu=h.object({sessionId:h.string(),messageId:h.string(),prompt:h.array(h.unknown())}),uu=h.object({sessionId:h.string(),messageId:h.string(),reason:h.enum(["started","cancelled","abandoned"])}),ks=h.object({sessionId:h.string(),messageId:h.string()}),lu=h.object({cancelled:h.boolean(),reason:h.enum(["ok","not_found","already_running"])}),As=h.object({sessionId:h.string(),messageId:h.string(),prompt:h.array(h.unknown())}),fu=h.object({updated:h.boolean(),reason:h.enum(["ok","not_found","already_running"])}),xs=h.object({sessionId:h.string(),targetMessageId:h.string(),prompt:h.array(h.unknown()),replaceQueue:h.boolean().optional(),onTargetCompleted:h.enum(["reject","send_anyway"]).optional()}),pu=h.object({amended:h.boolean(),reason:h.enum(["ok","target_completed","target_cancelled","target_not_found"]),messageId:h.string().optional()}),mu=h.object({sessionId:h.string(),cancelledMessageId:h.string(),newMessageId:h.string(),prompt:h.array(h.unknown()),originator:Ss,amendedAt:h.number()}),gu=h.object({agentId:h.string(),version:h.string(),source:h.enum(["binary","npm"]),phase:h.enum(["download_start","download_progress","download_done","extract","install_start","installed"]),receivedBytes:h.number().optional(),totalBytes:h.number().optional(),packageSpec:h.string().optional()}),Cs="hydra-acp/agents/install_progress";function rn(n,e){let t="",s=[],r=[],i=!1,o=a=>{if(!i){i=!0;for(let d of r)d(a)}};return n.setEncoding("utf8"),n.on("data",a=>{t+=a;let d=t.indexOf(`
11
- `);for(;d!==-1;){let c=t.slice(0,d).trim();if(t=t.slice(d+1),c.length>0)try{let f=JSON.parse(c);for(let l of s)l(f)}catch(f){for(let l of s)l({jsonrpc:"2.0",id:0,error:{code:I.ParseError,message:`Failed to parse ndjson line: ${f.message}`}})}d=t.indexOf(`
9
+ `))}async function ds(n){let e=ne();if(!e)throw new Error(`Agent ${n.agentId}: cannot determine platform key for ${process.platform}/${process.arch}`);let t=w.agentNpmInstallDir(n.agentId,e,n.version),s=Ki(n.packageSpec),r=Xi(s),i={installDir:t,packageName:s,hint:n.bin,basename:r},o=se.join(t,"node_modules",".bin",n.bin);if(await he(o))return o;if(await he(t)){let l=await as(i);if(l)return l.binPath}await Qi({agentId:n.agentId,version:n.version,packageSpec:n.packageSpec,installDir:t,registry:n.registry,onProgress:n.onProgress});let a=await as(i);if(a)return a.binPath;let d=await us(t,s),c=typeof d=="object"&&d!==null?Object.keys(d):typeof d=="string"?[r]:[],p=c.length>0?` (package declares bins: ${c.join(", ")})`:"";throw new Error(`Agent ${n.agentId}: npm install of ${n.packageSpec} did not produce bin ${n.bin} (looked in ${t}/node_modules/.bin/)${p}`)}async function Qi(n){await G.mkdir(se.dirname(n.installDir),{recursive:!0});let e=await G.mkdtemp(`${n.installDir}.partial-`);try{en(`hydra-acp: installing ${n.packageSpec} for ${n.agentId} into ${e}`),Zt(n.onProgress,{phase:"install_start",agentId:n.agentId,version:n.version,packageSpec:n.packageSpec}),await Yi({packageSpec:n.packageSpec,cwd:e,registry:n.registry});try{await G.rename(e,n.installDir)}catch(t){let s=t;if((s.code==="EEXIST"||s.code==="ENOTEMPTY")&&await he(n.installDir)){await G.rm(e,{recursive:!0,force:!0}).catch(()=>{}),Zt(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version});return}throw t}en(`hydra-acp: installed ${n.agentId} to ${n.installDir}`),Zt(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version})}catch(t){throw await G.rm(e,{recursive:!0,force:!0}).catch(()=>{}),t}}function Zt(n,e){if(n)try{n(e)}catch{}}var Vi=5,Gi=25;function Yi(n){return cs(n,0)}async function cs(n,e){try{await new Promise((t,s)=>{let r=n.registry?["--registry",n.registry]:[],i;try{i=Wi("npm",["install","--no-audit","--no-fund","--silent",...r,n.packageSpec],{cwd:n.cwd,stdio:["ignore","pipe","pipe"]})}catch(a){s(a);return}let o="";i.stdout?.on("data",a=>{}),i.stderr?.setEncoding("utf8"),i.stderr?.on("data",a=>{o=(o+a).slice(-4096)}),i.on("error",a=>{if(a.code==="ENOENT"){s(new Error("npm not found on PATH (install Node.js / npm, or use a binary-distributed agent)"));return}s(a)}),i.on("exit",(a,d)=>{if(a===0){t();return}let c=a!==null?`exit code ${a}`:`signal ${d??"unknown"}`,p=o.trim();s(new Error(p?`npm install ${n.packageSpec} failed (${c})
10
+ stderr: ${p}`:`npm install ${n.packageSpec} failed (${c})`))})})}catch(t){if(t.code==="ETXTBSY"&&e<Vi)return await new Promise(r=>setTimeout(r,Gi*(e+1))),cs(n,e+1);throw t}}function Ki(n){if(n.startsWith("@")){let t=n.indexOf("/");if(t===-1)return n;let r=n.slice(t+1).indexOf("@");return r===-1?n:n.slice(0,t+1+r)}let e=n.indexOf("@");return e<=0?n:n.slice(0,e)}function Xi(n){let e=n.lastIndexOf("/");return e===-1?n:n.slice(e+1)}async function us(n,e){let t=se.join(n,"node_modules",e,"package.json");try{let s=await G.readFile(t,"utf8"),r=JSON.parse(s);return typeof r.bin=="string"||typeof r.bin=="object"&&r.bin!==null&&!Array.isArray(r.bin)?r.bin:void 0}catch{return}}async function as(n){let e=se.join(n.installDir,"node_modules",".bin"),t=se.join(e,n.hint);if(await he(t))return{binName:n.hint,binPath:t};let s=await us(n.installDir,n.packageName);if(typeof s=="object"&&s!==null){let r=Object.keys(s);if(r.length===1){let i=r[0],o=se.join(e,i);if(await he(o))return{binName:i,binPath:o}}else if(r.length>1){for(let i of[n.hint,n.basename])if(r.includes(i)){let o=se.join(e,i);if(await he(o))return{binName:i,binPath:o}}}}if(n.basename!==n.hint){let r=se.join(e,n.basename);if(await he(r))return{binName:n.basename,binPath:r}}}async function he(n){try{return await G.access(n),!0}catch{return!1}}var Zi=x.object({package:x.string(),bin:x.string().optional(),args:x.array(x.string()).optional(),env:x.record(x.string()).optional()}),Ce=x.object({archive:x.string().url().optional(),cmd:x.string().optional(),args:x.array(x.string()).optional(),env:x.record(x.string()).optional()}),eo=x.object({"darwin-aarch64":Ce.optional(),"darwin-x86_64":Ce.optional(),"linux-aarch64":Ce.optional(),"linux-x86_64":Ce.optional(),"windows-x86_64":Ce.optional(),"windows-aarch64":Ce.optional()}),to=x.object({package:x.string(),args:x.array(x.string()).optional(),env:x.record(x.string()).optional()}),no=x.object({command:x.string(),args:x.array(x.string()).optional(),env:x.record(x.string()).optional()}),so=x.object({npx:Zi.optional(),binary:eo.optional(),uvx:to.optional(),exec:no.optional()}),ro=x.object({id:x.string(),name:x.string(),version:x.string().optional(),description:x.string().optional(),authors:x.array(x.string()).optional(),license:x.string().optional(),icon:x.string().optional(),repository:x.string().optional(),website:x.string().optional(),distribution:so}),ls=x.object({version:x.string(),agents:x.array(ro),extensions:x.array(x.unknown()).optional()}),Be=class{constructor(e,t={}){this.config=e;this.options=t}config;options;cache;async load(){if(this.cache&&(this.isPinned()||this.isFresh(this.cache.fetchedAt)))return this.cache.data;let e=await this.readDiskCache();if(e&&(this.isPinned()||this.isFresh(e.fetchedAt)))return this.cache=e,e.data;try{let t=await this.fetchFromNetwork();return this.cache=t,await this.writeDiskCache(t),t.data}catch(t){if(e)return this.cache=e,e.data;throw t}}async refresh(){let e=await this.fetchFromNetwork();return this.cache=e,await this.writeDiskCache(e),e.data}lastFetchedAt(){return this.cache?.fetchedAt}async getAgent(e){let t=this.localAgents().find(o=>o.id===e);if(t)return t;let s=await this.load(),r=s.agents.find(o=>o.id===e);if(r)return this.applyOverride(r);let i=s.agents.find(o=>nn(o)===e);return i?this.applyOverride(i):void 0}localAgents(){return Object.entries(this.config.agents??{}).map(([e,t])=>({id:e,name:t.name??e,description:t.description,version:"local",distribution:{exec:{command:t.command??e,args:t.args,env:t.env}}}))}applyOverride(e){let t=this.config.agentOverrides?.[e.id];return!t?.packageSpec||!e.distribution.npx?e:{...e,version:io(t.packageSpec),distribution:{...e.distribution,npx:{...e.distribution.npx,package:t.packageSpec}}}}isPinned(){return this.config.registry?.pinned===!0}isFresh(e){let t=Date.now()-e,s=this.config.registry.ttlHours*60*60*1e3;return t<s}async fetchFromNetwork(){let e=await fetch(this.config.registry.url);if(!e.ok)throw new Error(`Registry fetch failed: HTTP ${e.status}`);let t=await e.json(),s=ls.parse(t),r={fetchedAt:Date.now(),raw:t,data:s},i=this.options.onFetched;return i&&Promise.resolve().then(()=>i(s)).catch(()=>{}),r}async readDiskCache(){let e=await te(w.registryCache());if(!(!e||typeof e.fetchedAt!="number"||e.data===void 0))try{let t=ls.parse(e.data);return{fetchedAt:e.fetchedAt,raw:e.data,data:t}}catch{return}}async writeDiskCache(e){await K(w.registryCache(),{fetchedAt:e.fetchedAt,data:e.raw})}};function io(n){let e=n.lastIndexOf("@"),s=(e>0?n.slice(e+1):"").replace(/[^a-zA-Z0-9._-]/g,"_");return s.length>0?`pin-${s}`:"pinned"}function nn(n){let e=n.distribution.npx?.package;if(!e)return;let t=e.lastIndexOf("/"),s=t===-1?e:e.slice(t+1),r=s.lastIndexOf("@");return r<=0?s:s.slice(0,r)}async function Me(n){let e=typeof n.localAgents=="function"?n.localAgents():[],t;try{t=await n.load()}catch(o){if(e.length===0)throw o;t={version:"local-only",agents:[]}}let s=new Set(e.map(o=>o.id)),r=[...e,...t.agents.filter(o=>!s.has(o.id))],i=await Promise.all(r.map(async o=>({id:o.id,name:o.name,version:o.version,description:o.description,distributions:Object.keys(o.distribution),installed:await sn(o),source:s.has(o.id)?"local":"registry"})));return{version:t.version,fetchedAt:n.lastFetchedAt(),agents:i}}async function sn(n){let e=ne();if(!e)return"no";let t=n.version??"current";if(n.distribution.exec)return"yes";if(n.distribution.binary){let s=Kt(n.distribution.binary,e);if(s?.cmd){let r=dt.resolve(w.agentInstallDir(n.id,e,t),s.cmd);if(await fs(r))return"yes"}}if(n.distribution.npx){let s=n.distribution.npx,r=s.bin??nn(n)??s.package,i=w.agentNpmInstallDir(n.id,e,t),o=dt.join(i,"node_modules",".bin",r);if(await fs(o))return"yes"}return!n.distribution.npx&&!n.distribution.binary&&n.distribution.uvx?"lazy":"no"}async function fs(n){try{return await ps.access(n),!0}catch{return!1}}async function oe(n,e=[],t={}){let s=n.version??"current";if(n.distribution.npx){let r=n.distribution.npx,i=e.length>0?e:r.args??[];if(process.env.HYDRA_ACP_SKIP_NPM_PREFETCH)return{command:"npx",args:["-y",r.package,...i],env:r.env??{},version:s};let o=r.bin??nn(n)??r.package,a=t.onInstallProgress;return{command:await ds({agentId:n.id,version:s,packageSpec:r.package,bin:o,registry:t.npmRegistry,onProgress:a?c=>a({source:"npm",...c}):void 0}),args:i,env:r.env??{},version:s}}if(n.distribution.binary){let r=Kt(n.distribution.binary);if(!r)throw new Error(`Agent ${n.id} has no binary distribution for ${ne()??"this platform"}.`);let i=t.onInstallProgress,o=await os({agentId:n.id,version:s,target:r,onProgress:i?d=>i({source:"binary",...d}):void 0}),a=e.length>0?e:r.args??[];return{command:o,args:a,env:r.env??{},version:s}}if(n.distribution.uvx){let r=n.distribution.uvx,i=e.length>0?e:r.args??[];return{command:"uvx",args:[r.package,...i],env:r.env??{},version:s}}if(n.distribution.exec){let r=n.distribution.exec,i=e.length>0?e:r.args??[];return{command:r.command,args:i,env:r.env??{},version:s}}throw new Error(`Agent ${n.id} has no usable distribution method.`)}import{spawn as lo}from"child_process";import*as ft from"fs";import*as Ms from"path";import{z as h}from"zod";var le=1,I={ParseError:-32700,InvalidRequest:-32600,MethodNotFound:-32601,InvalidParams:-32602,InternalError:-32603,SessionNotFound:-32001,NotAuthorisedToAttach:-32002,MultiClientNotSupported:-32003,AgentNotInstalled:-32005,BundleAlreadyImported:-32010,PermissionDenied:-32011,AlreadyAttached:-32012,StreamNotEnabled:-32013,SessionClosing:-32014},ms=h.object({protocolVersion:h.number().optional(),clientCapabilities:h.record(h.unknown()).optional(),clientInfo:h.object({name:h.string(),version:h.string().optional()}).optional()}),oo=h.enum(["full","pending_only","none","after_message"]),gs=h.object({cwd:h.string(),mcpServers:h.array(h.unknown()).optional(),_meta:h.record(h.unknown()).optional()}),ao=h.object({upstreamSessionId:h.string(),agentId:h.string(),cwd:h.string(),title:h.string().optional(),agentArgs:h.array(h.string()).optional()}),hs=h.object({sessionId:h.string(),historyPolicy:oo.default("full"),afterMessageId:h.string().optional(),clientId:h.string().optional(),clientInfo:h.object({name:h.string(),version:h.string().optional()}).optional(),_meta:h.record(h.unknown()).optional()}),Ue="hydra-acp";function De(n){if(!n)return{};let e=n[Ue];if(!e||typeof e!="object"||Array.isArray(e))return{};let t=e,s={};if(typeof t.upstreamSessionId=="string"&&(s.upstreamSessionId=t.upstreamSessionId),typeof t.agentId=="string"&&(s.agentId=t.agentId),typeof t.cwd=="string"&&(s.cwd=t.cwd),typeof t.clientId=="string"&&(s.clientId=t.clientId),typeof t.readonly=="boolean"&&(s.readonly=t.readonly),(t.replayMode==="instant"||t.replayMode==="drip")&&(s.replayMode=t.replayMode),typeof t.dripSpeed=="number"&&t.dripSpeed>0&&(s.dripSpeed=t.dripSpeed),(t.toolContent==="inline"||t.toolContent==="references")&&(s.toolContent=t.toolContent),t.detachStatus==="detached"&&(s.detachStatus=t.detachStatus),typeof t.title=="string"&&(s.title=t.title),Array.isArray(t.agentArgs)&&t.agentArgs.every(r=>typeof r=="string")&&(s.agentArgs=t.agentArgs),Array.isArray(t.transformers)&&t.transformers.every(r=>typeof r=="string")&&(s.transformers=t.transformers),t.resume){let r=ao.safeParse(t.resume);r.success&&(s.resume=r.data)}if(typeof t.model=="string"&&(s.model=t.model),typeof t.currentModel=="string"&&(s.currentModel=t.currentModel),typeof t.currentMode=="string"&&(s.currentMode=t.currentMode),t.currentUsage){let r=vs.safeParse(t.currentUsage);r.success&&(s.currentUsage=r.data)}if(typeof t.turnStartedAt=="number"&&t.turnStartedAt>0&&(s.turnStartedAt=t.turnStartedAt),Array.isArray(t.availableCommands)){let r=[];for(let i of t.availableCommands){if(!i||typeof i!="object"||Array.isArray(i))continue;let o=i;if(typeof o.name!="string")continue;let a={name:o.name};typeof o.description=="string"&&(a.description=o.description),r.push(a)}r.length>0&&(s.availableCommands=r)}if(t.prompt&&typeof t.prompt=="object"&&!Array.isArray(t.prompt)){let r=t.prompt,i={};typeof r.queueing=="boolean"&&(i.queueing=r.queueing),typeof r.cancelling=="boolean"&&(i.cancelling=r.cancelling),typeof r.updating=="boolean"&&(i.updating=r.updating),typeof r.amending=="boolean"&&(i.amending=r.amending),typeof r.pipelining=="boolean"&&(i.pipelining=r.pipelining),s.prompt=i}if(t.agents&&typeof t.agents=="object"&&!Array.isArray(t.agents)){let r=t.agents,i={};typeof r.list=="boolean"&&(i.list=r.list),typeof r.installProgress=="boolean"&&(i.installProgress=r.installProgress),s.agents=i}if(typeof t.mcpStdin=="boolean"&&(s.mcpStdin=t.mcpStdin),typeof t.interactive=="boolean"&&(s.interactive=t.interactive),typeof t.ancillary=="boolean"&&(s.ancillary=t.ancillary),Array.isArray(t.queue)){let r=[];for(let i of t.queue){if(!i||typeof i!="object"||Array.isArray(i))continue;let o=i,a=o.originator;if(typeof o.messageId!="string"||!a||typeof a.clientId!="string"||!Array.isArray(o.prompt)||typeof o.position!="number"||typeof o.enqueuedAt!="number")continue;let d={clientId:a.clientId};typeof a.name=="string"&&(d.name=a.name),typeof a.version=="string"&&(d.version=a.version),r.push({messageId:o.messageId,originator:d,prompt:o.prompt,position:o.position,enqueuedAt:o.enqueuedAt})}r.length>0&&(s.queue=r)}if(Array.isArray(t.availableModes)){let r=[];for(let i of t.availableModes){if(!i||typeof i!="object"||Array.isArray(i))continue;let o=i;if(typeof o.id!="string")continue;let a={id:o.id};typeof o.name=="string"&&(a.name=o.name),typeof o.description=="string"&&(a.description=o.description),r.push(a)}r.length>0&&(s.availableModes=r)}if(Array.isArray(t.availableModels)){let r=[];for(let i of t.availableModels){if(!i||typeof i!="object"||Array.isArray(i))continue;let o=i;if(typeof o.modelId!="string")continue;let a={modelId:o.modelId};typeof o.name=="string"&&(a.name=o.name),typeof o.description=="string"&&(a.description=o.description),r.push(a)}r.length>0&&(s.availableModels=r)}if((t.status==="live"||t.status==="cold")&&(s.status=t.status),typeof t.busy=="boolean"&&(s.busy=t.busy),typeof t.awaitingInput=="boolean"&&(s.awaitingInput=t.awaitingInput),typeof t.attachedClients=="number"&&(s.attachedClients=t.attachedClients),typeof t.importedFromMachine=="string"&&(s.importedFromMachine=t.importedFromMachine),typeof t.importedFromUpstreamSessionId=="string"&&(s.importedFromUpstreamSessionId=t.importedFromUpstreamSessionId),typeof t.parentSessionId=="string"&&(s.parentSessionId=t.parentSessionId),typeof t.forkedFromSessionId=="string"&&(s.forkedFromSessionId=t.forkedFromSessionId),typeof t.forkedFromMessageId=="string"&&(s.forkedFromMessageId=t.forkedFromMessageId),t.originatingClient&&typeof t.originatingClient=="object"&&!Array.isArray(t.originatingClient)&&typeof t.originatingClient.name=="string"){let r=t.originatingClient;s.originatingClient={name:r.name,...typeof r.version=="string"?{version:r.version}:{}}}return t.agentCapabilities!==void 0&&(s.agentCapabilities=t.agentCapabilities),s}function ct(n,e){return{...n??{},[Ue]:e}}var ys=h.object({sessionId:h.string()}),ws=h.object({cwd:h.string().optional(),cursor:h.string().optional()}),vs=h.object({used:h.number().optional(),size:h.number().optional(),costAmount:h.number().optional(),costCurrency:h.string().optional()}),au=h.object({sessionId:h.string(),upstreamSessionId:h.string().optional(),cwd:h.string(),title:h.string().optional(),agentId:h.string().optional(),currentModel:h.string().optional(),currentUsage:vs.optional(),importedFromMachine:h.string().optional(),importedFromUpstreamSessionId:h.string().optional(),parentSessionId:h.string().optional(),forkedFromSessionId:h.string().optional(),forkedFromMessageId:h.string().optional(),originatingClient:h.object({name:h.string(),version:h.string().optional()}).optional(),interactive:h.boolean().optional(),priority:h.number().int().nonnegative().optional(),updatedAt:h.string(),attachedClients:h.number().int().nonnegative(),status:h.enum(["live","cold"]).default("live"),busy:h.boolean().default(!1),awaitingInput:h.boolean().default(!1),_meta:h.record(h.unknown()).optional()}),co=h.object({sessionId:h.string(),cwd:h.string(),title:h.string().optional(),updatedAt:h.string().optional(),_meta:h.record(h.unknown()).optional()}),du=h.object({sessions:h.array(co),nextCursor:h.string().optional()});function ut(n,e){let t={attachedClients:n.attachedClients,status:n.status,busy:n.busy,awaitingInput:n.awaitingInput};return n.cwd!==void 0&&(t.cwd=n.cwd),n.title!==void 0&&(t.title=n.title),n.agentId!==void 0&&(t.agentId=n.agentId),n.upstreamSessionId!==void 0&&(t.upstreamSessionId=n.upstreamSessionId),n.currentModel!==void 0&&(t.currentModel=n.currentModel),n.currentUsage!==void 0&&(t.currentUsage=n.currentUsage),n.importedFromMachine!==void 0&&(t.importedFromMachine=n.importedFromMachine),n.importedFromUpstreamSessionId!==void 0&&(t.importedFromUpstreamSessionId=n.importedFromUpstreamSessionId),n.parentSessionId!==void 0&&(t.parentSessionId=n.parentSessionId),n.forkedFromSessionId!==void 0&&(t.forkedFromSessionId=n.forkedFromSessionId),n.forkedFromMessageId!==void 0&&(t.forkedFromMessageId=n.forkedFromMessageId),n.originatingClient!==void 0&&(t.originatingClient=n.originatingClient),n.interactive!==void 0&&(t.interactive=n.interactive),n.priority!==void 0&&n.priority>0&&(t.priority=n.priority),e&&(e.clientId!==void 0&&(t.clientId=e.clientId),e.currentMode!==void 0&&(t.currentMode=e.currentMode),e.agentArgs!==void 0&&e.agentArgs.length>0&&(t.agentArgs=e.agentArgs),e.availableCommands!==void 0&&e.availableCommands.length>0&&(t.availableCommands=e.availableCommands),e.availableModes!==void 0&&e.availableModes.length>0&&(t.availableModes=e.availableModes),e.availableModels!==void 0&&e.availableModels.length>0&&(t.availableModels=e.availableModels),e.turnStartedAt!==void 0&&(t.turnStartedAt=e.turnStartedAt),e.agentCapabilities!==void 0&&(t.agentCapabilities=e.agentCapabilities),e.queue!==void 0&&e.queue.length>0&&(t.queue=e.queue)),t}function bs(n){let e={sessionId:n.sessionId,cwd:n.cwd,updatedAt:n.updatedAt,_meta:ct(n._meta,ut(n))};return n.title!==void 0&&(e.title=n.title),e}var Is=h.object({sessionId:h.string(),prompt:h.array(h.unknown()),_meta:h.record(h.unknown()).optional()}),lt=h.object({sessionId:h.string()}),Ss=h.object({clientId:h.string(),name:h.string().optional(),version:h.string().optional()}),cu=h.object({sessionId:h.string(),messageId:h.string(),originator:Ss,prompt:h.array(h.unknown()),position:h.number().int().nonnegative(),queueDepth:h.number().int().positive(),enqueuedAt:h.number()}),uu=h.object({sessionId:h.string(),messageId:h.string(),prompt:h.array(h.unknown())}),lu=h.object({sessionId:h.string(),messageId:h.string(),reason:h.enum(["started","cancelled","abandoned"])}),ks=h.object({sessionId:h.string(),messageId:h.string()}),fu=h.object({cancelled:h.boolean(),reason:h.enum(["ok","not_found","already_running"])}),As=h.object({sessionId:h.string(),messageId:h.string(),prompt:h.array(h.unknown())}),pu=h.object({updated:h.boolean(),reason:h.enum(["ok","not_found","already_running"])}),xs=h.object({sessionId:h.string(),targetMessageId:h.string(),prompt:h.array(h.unknown()),replaceQueue:h.boolean().optional(),onTargetCompleted:h.enum(["reject","send_anyway"]).optional()}),mu=h.object({amended:h.boolean(),reason:h.enum(["ok","target_completed","target_cancelled","target_not_found"]),messageId:h.string().optional()}),gu=h.object({sessionId:h.string(),cancelledMessageId:h.string(),newMessageId:h.string(),prompt:h.array(h.unknown()),originator:Ss,amendedAt:h.number()}),hu=h.object({agentId:h.string(),version:h.string(),source:h.enum(["binary","npm"]),phase:h.enum(["download_start","download_progress","download_done","extract","install_start","installed"]),receivedBytes:h.number().optional(),totalBytes:h.number().optional(),packageSpec:h.string().optional()}),Cs="hydra-acp/agents/install_progress";function rn(n,e){let t="",s=[],r=[],i=!1,o=a=>{if(!i){i=!0;for(let d of r)d(a)}};return n.setEncoding("utf8"),n.on("data",a=>{t+=a;let d=t.indexOf(`
11
+ `);for(;d!==-1;){let c=t.slice(0,d).trim();if(t=t.slice(d+1),c.length>0)try{let p=JSON.parse(c);for(let l of s)l(p)}catch(p){for(let l of s)l({jsonrpc:"2.0",id:0,error:{code:I.ParseError,message:`Failed to parse ndjson line: ${p.message}`}})}d=t.indexOf(`
12
12
  `)}}),n.on("end",()=>o()),n.on("error",a=>o(a)),e.on("error",a=>o(a)),{async send(a){if(i)throw new Error("stream is closed");let d=JSON.stringify(a)+`
13
- `;await new Promise((c,f)=>{e.write(d,l=>{if(l){f(l);return}c()})})},onMessage(a){s.push(a)},onClose(a){r.push(a)},async close(){e.end(),o()}}}import{nanoid as co}from"nanoid";var he=class n{constructor(e){this.stream=e;this.stream.onMessage(t=>this.handleIncoming(t)),this.stream.onClose(t=>this.handleClose(t))}stream;requestHandlers=new Map;defaultRequestHandler;notificationHandlers=new Map;bufferedNotifications=new Map;static MAX_BUFFERED_PER_METHOD=64;pending=new Map;closed=!1;closeHandlers=[];orphanErrorHandlers=[];onRequest(e,t){this.requestHandlers.set(e,t)}setDefaultHandler(e){this.defaultRequestHandler=e}onNotification(e,t){this.notificationHandlers.set(e,t);let s=this.bufferedNotifications.get(e);if(s){this.bufferedNotifications.delete(e);for(let r of s)try{t(r.params,r.method)}catch{}}}drainBuffered(e){let s=this.bufferedNotifications.get(e)?.length??0;return this.bufferedNotifications.delete(e),s}onClose(e){this.closeHandlers.push(e)}onOrphanError(e){this.orphanErrorHandlers.push(e)}async request(e,t){return this.requestWithId(e,t).response}requestWithId(e,t){if(this.closed)return{id:"",response:Promise.reject(new Error("connection is closed"))};let s=co(),r={jsonrpc:"2.0",id:s,method:e,params:t},i=new Promise((o,a)=>{this.pending.set(s,{resolve:d=>o(d),reject:a}),this.stream.send(r).catch(d=>{this.pending.delete(s),a(d)})});return{id:s,response:i}}notify(e,t){if(this.closed)return Promise.resolve();let s={jsonrpc:"2.0",method:e,params:t};return this.stream.send(s)}async close(){this.closed||await this.stream.close()}fail(e){this.handleClose(e)}handleIncoming(e){"method"in e?"id"in e&&e.id!==void 0?this.handleRequest(e).catch(()=>{}):this.handleNotification(e):"id"in e?this.handleResponse(e):"error"in e&&this.handleResponse(e)}async handleRequest(e){let t=this.requestHandlers.get(e.method)??this.defaultRequestHandler;if(!t){await this.sendError(e.id,{code:I.MethodNotFound,message:`Method not found: ${e.method}`}).catch(()=>{});return}try{let s=await t(e.params,e.method),r={jsonrpc:"2.0",id:e.id,result:s};await this.stream.send(r).catch(()=>{})}catch(s){let r=s;await this.sendError(e.id,{code:r.code??I.InternalError,message:r.message,data:r.data}).catch(()=>{})}}handleNotification(e){let t=this.notificationHandlers.get(e.method);if(t){t(e.params,e.method);return}let s=this.bufferedNotifications.get(e.method);s||(s=[],this.bufferedNotifications.set(e.method,s)),s.push(e),s.length>n.MAX_BUFFERED_PER_METHOD&&s.shift()}handleResponse(e){let t=this.pending.get(e.id);if(!t){if(e.error)for(let s of this.orphanErrorHandlers)try{s({code:e.error.code,message:e.error.message,data:e.error.data})}catch{}return}if(this.pending.delete(e.id),e.error){let s=new Error(e.error.message);s.code=e.error.code,s.data=e.error.data,t.reject(s)}else t.resolve(e.result)}async sendError(e,t){let s={jsonrpc:"2.0",id:e,error:t};await this.stream.send(s)}handleClose(e){if(!this.closed){this.closed=!0;for(let t of this.pending.values())t.reject(e??new Error("connection closed"));this.pending.clear();for(let t of this.closeHandlers)t(e)}}};var lo=4096,ae=class n{agentId;version;cwd;connection;child;exited=!1;killed=!1;stderrTail="";stderrTailBytes;killEscalationMs;logger;fileLog;exitHandlers=[];constructor(e,t){if(this.agentId=e.agentId,this.version=e.plan.version,this.cwd=e.cwd,this.child=t,this.stderrTailBytes=e.stderrTailBytes??lo,this.killEscalationMs=e.killEscalationMs??fo,this.logger=e.logger,this.fileLog=po(e.agentId),this.writeLog(`--- spawn pid=${t.pid} version=${e.plan.version} cwd=${e.cwd} cmd=${e.plan.command} args=${JSON.stringify(e.plan.args)} time=${new Date().toISOString()} ---
14
- `),!t.stdout||!t.stdin)throw new Error("agent subprocess missing stdio");let s=rn(t.stdout,t.stdin);this.connection=new he(s),t.stderr?.setEncoding("utf8"),t.stderr?.on("data",r=>{if(this.stderrTail=(this.stderrTail+r).slice(-this.stderrTailBytes),this.writeLog(r),this.logger)for(let i of r.split(/\r?\n/))i.length>0&&this.logger.info(`[${e.agentId}] ${i}`);else process.stderr.write(`[${e.agentId}] ${r}`)}),t.on("error",r=>{let i=this.formatFailure(r.message);this.writeLog(`--- spawn error: ${r.message} time=${new Date().toISOString()} ---
13
+ `;await new Promise((c,p)=>{e.write(d,l=>{if(l){p(l);return}c()})})},onMessage(a){s.push(a)},onClose(a){r.push(a)},async close(){e.end(),o()}}}import{nanoid as uo}from"nanoid";var ye=class n{constructor(e){this.stream=e;this.stream.onMessage(t=>this.handleIncoming(t)),this.stream.onClose(t=>this.handleClose(t))}stream;requestHandlers=new Map;defaultRequestHandler;notificationHandlers=new Map;bufferedNotifications=new Map;static MAX_BUFFERED_PER_METHOD=64;pending=new Map;closed=!1;closeHandlers=[];orphanErrorHandlers=[];onRequest(e,t){this.requestHandlers.set(e,t)}setDefaultHandler(e){this.defaultRequestHandler=e}onNotification(e,t){this.notificationHandlers.set(e,t);let s=this.bufferedNotifications.get(e);if(s){this.bufferedNotifications.delete(e);for(let r of s)try{t(r.params,r.method)}catch{}}}drainBuffered(e){let s=this.bufferedNotifications.get(e)?.length??0;return this.bufferedNotifications.delete(e),s}onClose(e){this.closeHandlers.push(e)}onOrphanError(e){this.orphanErrorHandlers.push(e)}async request(e,t){return this.requestWithId(e,t).response}requestWithId(e,t){if(this.closed)return{id:"",response:Promise.reject(new Error("connection is closed"))};let s=uo(),r={jsonrpc:"2.0",id:s,method:e,params:t},i=new Promise((o,a)=>{this.pending.set(s,{resolve:d=>o(d),reject:a}),this.stream.send(r).catch(d=>{this.pending.delete(s),a(d)})});return{id:s,response:i}}notify(e,t){if(this.closed)return Promise.resolve();let s={jsonrpc:"2.0",method:e,params:t};return this.stream.send(s)}async close(){this.closed||await this.stream.close()}fail(e){this.handleClose(e)}handleIncoming(e){"method"in e?"id"in e&&e.id!==void 0?this.handleRequest(e).catch(()=>{}):this.handleNotification(e):"id"in e?this.handleResponse(e):"error"in e&&this.handleResponse(e)}async handleRequest(e){let t=this.requestHandlers.get(e.method)??this.defaultRequestHandler;if(!t){await this.sendError(e.id,{code:I.MethodNotFound,message:`Method not found: ${e.method}`}).catch(()=>{});return}try{let s=await t(e.params,e.method),r={jsonrpc:"2.0",id:e.id,result:s};await this.stream.send(r).catch(()=>{})}catch(s){let r=s;await this.sendError(e.id,{code:r.code??I.InternalError,message:r.message,data:r.data}).catch(()=>{})}}handleNotification(e){let t=this.notificationHandlers.get(e.method);if(t){t(e.params,e.method);return}let s=this.bufferedNotifications.get(e.method);s||(s=[],this.bufferedNotifications.set(e.method,s)),s.push(e),s.length>n.MAX_BUFFERED_PER_METHOD&&s.shift()}handleResponse(e){let t=this.pending.get(e.id);if(!t){if(e.error)for(let s of this.orphanErrorHandlers)try{s({code:e.error.code,message:e.error.message,data:e.error.data})}catch{}return}if(this.pending.delete(e.id),e.error){let s=new Error(e.error.message);s.code=e.error.code,s.data=e.error.data,t.reject(s)}else t.resolve(e.result)}async sendError(e,t){let s={jsonrpc:"2.0",id:e,error:t};await this.stream.send(s)}handleClose(e){if(!this.closed){this.closed=!0;for(let t of this.pending.values())t.reject(e??new Error("connection closed"));this.pending.clear();for(let t of this.closeHandlers)t(e)}}};var fo=4096,ae=class n{agentId;version;cwd;connection;child;exited=!1;killed=!1;stderrTail="";stderrTailBytes;killEscalationMs;logger;fileLog;exitHandlers=[];constructor(e,t){if(this.agentId=e.agentId,this.version=e.plan.version,this.cwd=e.cwd,this.child=t,this.stderrTailBytes=e.stderrTailBytes??fo,this.killEscalationMs=e.killEscalationMs??po,this.logger=e.logger,this.fileLog=mo(e.agentId),this.writeLog(`--- spawn pid=${t.pid} version=${e.plan.version} cwd=${e.cwd} cmd=${e.plan.command} args=${JSON.stringify(e.plan.args)} time=${new Date().toISOString()} ---
14
+ `),!t.stdout||!t.stdin)throw new Error("agent subprocess missing stdio");let s=rn(t.stdout,t.stdin);this.connection=new ye(s),t.stderr?.setEncoding("utf8"),t.stderr?.on("data",r=>{if(this.stderrTail=(this.stderrTail+r).slice(-this.stderrTailBytes),this.writeLog(r),this.logger)for(let i of r.split(/\r?\n/))i.length>0&&this.logger.info(`[${e.agentId}] ${i}`);else process.stderr.write(`[${e.agentId}] ${r}`)}),t.on("error",r=>{let i=this.formatFailure(r.message);this.writeLog(`--- spawn error: ${r.message} time=${new Date().toISOString()} ---
15
15
  `),this.connection.fail(new Error(i))}),t.on("exit",(r,i)=>{if(this.exited=!0,this.writeLog(`--- exit code=${r} signal=${i} ${this.killed?"(after kill) ":""}time=${new Date().toISOString()} ---
16
16
  `),this.fileLog?.end(),this.fileLog=void 0,this.killed)this.logger?.info(`agent ${e.agentId} pid=${t.pid} exited after kill code=${r} signal=${i}`);else{let o=`agent ${e.agentId} exited before responding (code=${r} signal=${i})`;this.connection.fail(new Error(this.formatFailure(o))),this.logger?.warn(`agent ${e.agentId} pid=${t.pid} exited unexpectedly code=${r} signal=${i}`)}for(let o of this.exitHandlers)o(r,i)})}writeLog(e){if(this.fileLog)try{this.fileLog.write(e)}catch{}}formatFailure(e){let t=this.stderrTail.trim();return t?`${e}
17
- stderr: ${t}`:e}static spawn(e){let t={...process.env,...e.plan.env,...e.extraEnv??{}},s=uo(e.plan.command,e.plan.args,{cwd:e.cwd,env:t,stdio:["pipe","pipe","pipe"],detached:!0});return s.unref(),new n(e,s)}onExit(e){this.exitHandlers.push(e)}isAlive(){return!this.exited}async kill(e="SIGTERM"){this.exited||(this.killed=!0,this.writeLog(`--- kill requested signal=${e} time=${new Date().toISOString()} ---
17
+ stderr: ${t}`:e}static spawn(e){let t={...process.env,...e.plan.env,...e.extraEnv??{}},s=lo(e.plan.command,e.plan.args,{cwd:e.cwd,env:t,stdio:["pipe","pipe","pipe"],detached:!0});return s.unref(),new n(e,s)}onExit(e){this.exitHandlers.push(e)}isAlive(){return!this.exited}async kill(e="SIGTERM"){this.exited||(this.killed=!0,this.writeLog(`--- kill requested signal=${e} time=${new Date().toISOString()} ---
18
18
  `),this.logger?.info(`agent ${this.agentId} pid=${this.child.pid} kill requested signal=${e}`),await this.connection.close().catch(()=>{}),this.signalProcessGroup(e),await this.waitForExit(this.killEscalationMs),!this.exited&&(this.writeLog(`--- kill escalating signal=SIGKILL time=${new Date().toISOString()} ---
19
- `),this.logger?.warn(`agent ${this.agentId} pid=${this.child.pid} did not exit after ${e}; sending SIGKILL`),this.signalProcessGroup("SIGKILL"),await this.waitForExit(this.killEscalationMs)))}signalProcessGroup(e){let t=this.child.pid;if(t!==void 0)try{process.kill(-t,e)}catch{try{this.child.kill(e)}catch{}}}waitForExit(e){return this.exited?Promise.resolve():new Promise(t=>{let s=setTimeout(()=>{this.child.off("exit",r),t()},e),r=()=>{clearTimeout(s),t()};this.child.once("exit",r)})}},fo=2e3;function po(n){try{let e=w.agentLogFile(n);ft.mkdirSync(Ms.dirname(e),{recursive:!0});let t=ft.createWriteStream(e,{flags:"a"});return t.on("error",()=>{}),t}catch{return}}import*as ve from"fs/promises";import*as kt from"os";import*as Ve from"path";import{customAlphabet as xa}from"nanoid";import{customAlphabet as Os}from"nanoid";import*as Es from"fs/promises";var mo=64*1024*1024,go=1*1024*1024,Me=64*1024,ho=6e4,yo=100,wo=1e3,vo=64*1024,bo=256*1024,Rs=20;function Io(n){return n.replace(/[.*+?^${}()|[\]\\]/g,"\\$&")}var pt=class{storage;maxCapacityBytes;currentCapacityBytes;writeCursor=0;closed=!1;waiters=[];filePath;fileCapBytes;fileBytesWritten=0;fileCapReached=!1;onFileCapReached;logWriteError;fileWriteChain=Promise.resolve();constructor(e={}){if(this.maxCapacityBytes=e.capacityBytes??mo,this.maxCapacityBytes<=0)throw new Error("capacityBytes must be > 0");this.currentCapacityBytes=Math.min(go,this.maxCapacityBytes),this.storage=Buffer.alloc(this.currentCapacityBytes),this.filePath=e.filePath,this.fileCapBytes=e.fileCapBytes??Number.POSITIVE_INFINITY,this.onFileCapReached=e.onFileCapReached,this.logWriteError=e.logWriteError}get capacity(){return this.maxCapacityBytes}get allocatedBytes(){return this.currentCapacityBytes}get writeCursorPos(){return this.writeCursor}get oldestAvailable(){return Math.max(0,this.writeCursor-this.currentCapacityBytes)}get isClosed(){return this.closed}append(e){this.closed||e.length===0||(this.writeRing(e),this.writeCursor+=e.length,this.filePath!==void 0&&this.scheduleFileWrite(e),this.wakeWaiters("data"))}close(){this.closed||(this.closed=!0,this.wakeWaiters("eof"))}read(e,t){let s=Math.max(0,Math.min(t,Me));if(s===0){let l={bytes:Buffer.alloc(0),nextCursor:e};return this.closed&&e>=this.writeCursor&&(l.eof=!0),l}let r=e,i=0,o=this.oldestAvailable;r<o&&(i=o-r,r=o);let a=this.writeCursor-r;if(a<=0){let l={bytes:Buffer.alloc(0),nextCursor:r};return i>0&&(l.gap=i),this.closed&&(l.eof=!0),l}let d=Math.min(a,s),f={bytes:this.sliceFromRing(r,d),nextCursor:r+d};return i>0&&(f.gap=i),this.closed&&r+d>=this.writeCursor&&(f.eof=!0),f}tail(e){let t=Math.max(0,Math.min(e,Me)),s=this.oldestAvailable,r=this.writeCursor-t,i=Math.max(s,r),o=r<s;return{bytes:this.sliceFromRing(i,this.writeCursor-i),startCursor:i,endCursor:this.writeCursor,truncated:o}}head(e){let t=Math.max(0,Math.min(e,Me)),s=this.oldestAvailable,r=s>0,i=s,o=Math.min(this.writeCursor,i+t);return{bytes:this.sliceFromRing(i,o-i),startCursor:i,endCursor:o,truncated:r}}waitForData(e,t){if(e<this.writeCursor)return Promise.resolve("data");if(this.closed)return Promise.resolve("eof");let s=Math.max(0,Math.min(t,ho));return s===0?Promise.resolve("timeout"):new Promise(r=>{let i={resolve:o=>{i.timer!==void 0&&(clearTimeout(i.timer),i.timer=void 0),r(o)},timer:setTimeout(()=>{let o=this.waiters.indexOf(i);o>=0&&this.waiters.splice(o,1),i.timer=void 0,r("timeout")},s)};this.waiters.push(i)})}grep(e){let t=this.oldestAvailable,s=e.cursor,r=s??t,i=0;s!==void 0&&s<t&&(i=t-s,r=t),r>this.writeCursor&&(r=this.writeCursor);let o=this.sliceFromRing(r,this.writeCursor-r),a=e.regex??!0,d=e.caseInsensitive===!0?"i":"",c=a?new RegExp(e.pattern,d):new RegExp(Io(e.pattern),d),f=e.invert??!1,l=Math.max(1,Math.min(e.maxMatches??yo,wo)),u=Math.max(1,Math.min(e.maxBytes??vo,bo)),p=Math.max(0,Math.min(e.contextBefore??0,Rs)),g=Math.max(0,Math.min(e.contextAfter??0,Rs)),m=[],y=[],v=[],S=0,C=!1,k=0,E=0,N=(M,P)=>{for(let D of v)D.remaining>0&&(D.match.after===void 0&&(D.match.after=[]),D.match.after.push({cursor:M,line:P}),D.remaining--,S+=P.length);for(;v.length>0&&v[0].remaining===0;)v.shift();if(c.test(P)!==f&&m.length<l){let D={cursor:M,line:P};if(p>0&&y.length>0){D.before=y.slice();for(let je of D.before)S+=je.line.length}S+=P.length,m.push(D),g>0&&v.push({match:D,remaining:g})}if(p>0)for(y.push({cursor:M,line:P});y.length>p;)y.shift();let F=m.length>=l&&v.length===0,U=S>=u;return F||U};for(let M=0;M<o.length;M++){if(o[M]!==10)continue;let P=o.subarray(k,M).toString("utf8"),B=r+k;if(k=M+1,E=k,N(B,P)){C=!0;break}}if(!C&&k<o.length&&this.closed){let M=o.subarray(k).toString("utf8"),P=r+k;N(P,M)&&(C=!0),E=o.length}let V=Math.min(r+E,this.writeCursor),W={matches:m,truncated:C,nextCursor:V,scannedBytes:E};return i>0&&(W.gap=i),this.closed&&V>=this.writeCursor&&(W.eof=!0),W}wakeWaiters(e){if(this.waiters.length===0)return;let t=this.waiters;this.waiters=[];for(let s of t)s.resolve(e)}growIfNeeded(e){if(this.currentCapacityBytes>=this.maxCapacityBytes)return;let t=this.writeCursor+e;if(t<=this.currentCapacityBytes)return;let s=this.currentCapacityBytes;for(;s<t&&s<this.maxCapacityBytes;)s=Math.min(this.maxCapacityBytes,s*2);if(s===this.currentCapacityBytes)return;let r=Buffer.alloc(s);this.storage.copy(r,0,0,this.writeCursor),this.storage=r,this.currentCapacityBytes=s}writeRing(e){let t=e.length;if(this.growIfNeeded(t),t>=this.currentCapacityBytes){let i=t-this.currentCapacityBytes;e.copy(this.storage,0,i,t);return}let s=this.writeCursor%this.currentCapacityBytes,r=this.currentCapacityBytes-s;t<=r?e.copy(this.storage,s,0,t):(e.copy(this.storage,s,0,r),e.copy(this.storage,0,r,t))}sliceFromRing(e,t){if(t<=0)return Buffer.alloc(0);let s=Buffer.alloc(t),r=e%this.currentCapacityBytes,i=Math.min(t,this.currentCapacityBytes-r);return this.storage.copy(s,0,r,r+i),i<t&&this.storage.copy(s,i,0,t-i),s}scheduleFileWrite(e){let t=this.filePath;if(t===void 0||this.fileCapReached)return;let s=this.fileCapBytes-this.fileBytesWritten;if(s<=0){this.fileCapReached=!0,this.onFileCapReached?.();return}let r=e.length<=s?e:e.subarray(0,s);this.fileBytesWritten+=r.length;let i=this.fileBytesWritten>=this.fileCapBytes;this.fileWriteChain=this.fileWriteChain.then(()=>Es.appendFile(t,r)).catch(o=>{this.logWriteError?.(o)}),i&&!this.fileCapReached&&(this.fileCapReached=!0,this.onFileCapReached?.())}async drainFileWrites(){await this.fileWriteChain.catch(()=>{})}};var mt=[{verb:"title",name:"hydra title",description:"Regenerate the session title + synopsis via the agent (or set title manually with an arg)"},{verb:"agent",name:"hydra agent",argsHint:"<agent>",description:"Swap the agent backing this session, preserving context"},{verb:"kill",name:"hydra kill",description:"Close this session (kills the agent; record is kept so it can be resumed later)"},{verb:"restart",name:"hydra restart",description:"Restart the agent with a fresh session/new while preserving conversation history (useful when the proxy has changed available models)"}],Ru=new Map(mt.map(n=>[n.verb,n]));function Ps(n){let e=n.lastIndexOf("/");return(e===-1?n:n.slice(e+1)).toLowerCase()}function Re(n,e){if(e.length===0)return{kind:"none",requested:n};if(e.some(r=>r.modelId===n))return{kind:"exact",modelId:n};let t=Ps(n),s=e.map(r=>r.modelId).filter(r=>Ps(r)===t);return s.length===1?{kind:"resolved",modelId:s[0],requested:n}:s.length>1?{kind:"ambiguous",requested:n,candidates:s}:{kind:"unknown",requested:n}}function Ts(n){if(n.length===0)return n;let e=new Map,t=new Map,s=new Map;for(let a=0;a<n.length;a++){let d=n[a];if(d===void 0)continue;let c=on(d);if(c?.sessionUpdate!=="tool_call_update")continue;let f=typeof c.toolCallId=="string"?c.toolCallId:void 0;if(f!==void 0&&(e.set(f,a),c.rawInput&&typeof c.rawInput=="object"&&!Array.isArray(c.rawInput)&&Object.keys(c.rawInput).length>0&&s.set(f,c.rawInput),Array.isArray(c.content)&&c.content.length>0)){let l=t.get(f);l?l.push(...c.content):t.set(f,[...c.content])}}let r=[],i=null,o=null;for(let a=0;a<n.length;a++){let d=n[a];if(d===void 0)continue;let c=on(d);if(!c||typeof c.sessionUpdate!="string"){r.push(d),i=null;continue}let f=c.sessionUpdate;if(f==="agent_message_chunk"||f==="agent_thought_chunk"||f==="user_message_chunk"){i&&i.kind===f?ko(r,i.outIndex,So(c.content)):(r.push(d),i={outIndex:r.length-1,kind:f});continue}if(i=null,f==="tool_call_update"){let l=typeof c.toolCallId=="string"?c.toolCallId:void 0;if(l!==void 0&&e.get(l)!==a)continue;let u=l!==void 0&&t.has(l)?Ao(d,t.get(l)??[]):d;l!==void 0&&s.has(l)&&!xo(u)&&(u=Co(u,s.get(l))),r.push(u);continue}if(f==="plan"){o!==null?r[o]=d:(r.push(d),o=r.length-1);continue}(f==="prompt_received"||f==="turn_complete")&&(o=null),r.push(d)}return r}function on(n){if(n.method!=="session/update")return;let t=n.params?.update;if(!(!t||typeof t!="object"||Array.isArray(t)))return t}function So(n){if(!n||typeof n!="object")return"";let e=n;return typeof e.text=="string"?e.text:""}function ko(n,e,t){if(t.length===0)return;let s=n[e];if(s===void 0)return;let r=s.params??{},i=r.update??{},o=i.content??{},a=typeof o.text=="string"?o.text:"";n[e]={...s,params:{...r,update:{...i,content:{...o,text:a+t}}}}}function Ao(n,e){let t=n.params??{},s=t.update??{};return{...n,params:{...t,update:{...s,content:e}}}}function xo(n){let t=on(n)?.rawInput;return!!t&&typeof t=="object"&&!Array.isArray(t)&&Object.keys(t).length>0}function Co(n,e){let t=n.params??{},s=t.update??{};return{...n,params:{...t,update:{...s,rawInput:e}}}}import*as re from"fs/promises";async function Le(n,e){let t=w.queueFile(n);if(e.length===0){await re.unlink(t).catch(()=>{});return}await re.mkdir(w.sessionDir(n),{recursive:!0});let s=e.map(r=>JSON.stringify(r)).join(`
19
+ `),this.logger?.warn(`agent ${this.agentId} pid=${this.child.pid} did not exit after ${e}; sending SIGKILL`),this.signalProcessGroup("SIGKILL"),await this.waitForExit(this.killEscalationMs)))}signalProcessGroup(e){let t=this.child.pid;if(t!==void 0)try{process.kill(-t,e)}catch{try{this.child.kill(e)}catch{}}}waitForExit(e){return this.exited?Promise.resolve():new Promise(t=>{let s=setTimeout(()=>{this.child.off("exit",r),t()},e),r=()=>{clearTimeout(s),t()};this.child.once("exit",r)})}},po=2e3;function mo(n){try{let e=w.agentLogFile(n);ft.mkdirSync(Ms.dirname(e),{recursive:!0});let t=ft.createWriteStream(e,{flags:"a"});return t.on("error",()=>{}),t}catch{return}}import*as be from"fs/promises";import*as kt from"os";import*as Ve from"path";import{customAlphabet as Ca}from"nanoid";import{customAlphabet as Ns}from"nanoid";import*as Es from"fs/promises";var go=64*1024*1024,ho=1*1024*1024,Re=64*1024,yo=6e4,wo=100,vo=1e3,bo=64*1024,Io=256*1024,Rs=20;function So(n){return n.replace(/[.*+?^${}()|[\]\\]/g,"\\$&")}var pt=class{storage;maxCapacityBytes;currentCapacityBytes;writeCursor=0;closed=!1;waiters=[];filePath;fileCapBytes;fileBytesWritten=0;fileCapReached=!1;onFileCapReached;logWriteError;fileWriteChain=Promise.resolve();constructor(e={}){if(this.maxCapacityBytes=e.capacityBytes??go,this.maxCapacityBytes<=0)throw new Error("capacityBytes must be > 0");this.currentCapacityBytes=Math.min(ho,this.maxCapacityBytes),this.storage=Buffer.alloc(this.currentCapacityBytes),this.filePath=e.filePath,this.fileCapBytes=e.fileCapBytes??Number.POSITIVE_INFINITY,this.onFileCapReached=e.onFileCapReached,this.logWriteError=e.logWriteError}get capacity(){return this.maxCapacityBytes}get allocatedBytes(){return this.currentCapacityBytes}get writeCursorPos(){return this.writeCursor}get oldestAvailable(){return Math.max(0,this.writeCursor-this.currentCapacityBytes)}get isClosed(){return this.closed}append(e){this.closed||e.length===0||(this.writeRing(e),this.writeCursor+=e.length,this.filePath!==void 0&&this.scheduleFileWrite(e),this.wakeWaiters("data"))}close(){this.closed||(this.closed=!0,this.wakeWaiters("eof"))}read(e,t){let s=Math.max(0,Math.min(t,Re));if(s===0){let l={bytes:Buffer.alloc(0),nextCursor:e};return this.closed&&e>=this.writeCursor&&(l.eof=!0),l}let r=e,i=0,o=this.oldestAvailable;r<o&&(i=o-r,r=o);let a=this.writeCursor-r;if(a<=0){let l={bytes:Buffer.alloc(0),nextCursor:r};return i>0&&(l.gap=i),this.closed&&(l.eof=!0),l}let d=Math.min(a,s),p={bytes:this.sliceFromRing(r,d),nextCursor:r+d};return i>0&&(p.gap=i),this.closed&&r+d>=this.writeCursor&&(p.eof=!0),p}tail(e){let t=Math.max(0,Math.min(e,Re)),s=this.oldestAvailable,r=this.writeCursor-t,i=Math.max(s,r),o=r<s;return{bytes:this.sliceFromRing(i,this.writeCursor-i),startCursor:i,endCursor:this.writeCursor,truncated:o}}head(e){let t=Math.max(0,Math.min(e,Re)),s=this.oldestAvailable,r=s>0,i=s,o=Math.min(this.writeCursor,i+t);return{bytes:this.sliceFromRing(i,o-i),startCursor:i,endCursor:o,truncated:r}}waitForData(e,t){if(e<this.writeCursor)return Promise.resolve("data");if(this.closed)return Promise.resolve("eof");let s=Math.max(0,Math.min(t,yo));return s===0?Promise.resolve("timeout"):new Promise(r=>{let i={resolve:o=>{i.timer!==void 0&&(clearTimeout(i.timer),i.timer=void 0),r(o)},timer:setTimeout(()=>{let o=this.waiters.indexOf(i);o>=0&&this.waiters.splice(o,1),i.timer=void 0,r("timeout")},s)};this.waiters.push(i)})}grep(e){let t=this.oldestAvailable,s=e.cursor,r=s??t,i=0;s!==void 0&&s<t&&(i=t-s,r=t),r>this.writeCursor&&(r=this.writeCursor);let o=this.sliceFromRing(r,this.writeCursor-r),a=e.regex??!0,d=e.caseInsensitive===!0?"i":"",c=a?new RegExp(e.pattern,d):new RegExp(So(e.pattern),d),p=e.invert??!1,l=Math.max(1,Math.min(e.maxMatches??wo,vo)),u=Math.max(1,Math.min(e.maxBytes??bo,Io)),f=Math.max(0,Math.min(e.contextBefore??0,Rs)),g=Math.max(0,Math.min(e.contextAfter??0,Rs)),m=[],y=[],v=[],S=0,C=!1,k=0,E=0,N=(M,P)=>{for(let D of v)D.remaining>0&&(D.match.after===void 0&&(D.match.after=[]),D.match.after.push({cursor:M,line:P}),D.remaining--,S+=P.length);for(;v.length>0&&v[0].remaining===0;)v.shift();if(c.test(P)!==p&&m.length<l){let D={cursor:M,line:P};if(f>0&&y.length>0){D.before=y.slice();for(let je of D.before)S+=je.line.length}S+=P.length,m.push(D),g>0&&v.push({match:D,remaining:g})}if(f>0)for(y.push({cursor:M,line:P});y.length>f;)y.shift();let O=m.length>=l&&v.length===0,U=S>=u;return O||U};for(let M=0;M<o.length;M++){if(o[M]!==10)continue;let P=o.subarray(k,M).toString("utf8"),B=r+k;if(k=M+1,E=k,N(B,P)){C=!0;break}}if(!C&&k<o.length&&this.closed){let M=o.subarray(k).toString("utf8"),P=r+k;N(P,M)&&(C=!0),E=o.length}let V=Math.min(r+E,this.writeCursor),W={matches:m,truncated:C,nextCursor:V,scannedBytes:E};return i>0&&(W.gap=i),this.closed&&V>=this.writeCursor&&(W.eof=!0),W}wakeWaiters(e){if(this.waiters.length===0)return;let t=this.waiters;this.waiters=[];for(let s of t)s.resolve(e)}growIfNeeded(e){if(this.currentCapacityBytes>=this.maxCapacityBytes)return;let t=this.writeCursor+e;if(t<=this.currentCapacityBytes)return;let s=this.currentCapacityBytes;for(;s<t&&s<this.maxCapacityBytes;)s=Math.min(this.maxCapacityBytes,s*2);if(s===this.currentCapacityBytes)return;let r=Buffer.alloc(s);this.storage.copy(r,0,0,this.writeCursor),this.storage=r,this.currentCapacityBytes=s}writeRing(e){let t=e.length;if(this.growIfNeeded(t),t>=this.currentCapacityBytes){let i=t-this.currentCapacityBytes;e.copy(this.storage,0,i,t);return}let s=this.writeCursor%this.currentCapacityBytes,r=this.currentCapacityBytes-s;t<=r?e.copy(this.storage,s,0,t):(e.copy(this.storage,s,0,r),e.copy(this.storage,0,r,t))}sliceFromRing(e,t){if(t<=0)return Buffer.alloc(0);let s=Buffer.alloc(t),r=e%this.currentCapacityBytes,i=Math.min(t,this.currentCapacityBytes-r);return this.storage.copy(s,0,r,r+i),i<t&&this.storage.copy(s,i,0,t-i),s}scheduleFileWrite(e){let t=this.filePath;if(t===void 0||this.fileCapReached)return;let s=this.fileCapBytes-this.fileBytesWritten;if(s<=0){this.fileCapReached=!0,this.onFileCapReached?.();return}let r=e.length<=s?e:e.subarray(0,s);this.fileBytesWritten+=r.length;let i=this.fileBytesWritten>=this.fileCapBytes;this.fileWriteChain=this.fileWriteChain.then(()=>Es.appendFile(t,r)).catch(o=>{this.logWriteError?.(o)}),i&&!this.fileCapReached&&(this.fileCapReached=!0,this.onFileCapReached?.())}async drainFileWrites(){await this.fileWriteChain.catch(()=>{})}};var mt=[{verb:"title",name:"hydra title",description:"Regenerate the session title + synopsis via the agent (or set title manually with an arg)"},{verb:"agent",name:"hydra agent",argsHint:"<agent>",description:"Swap the agent backing this session, preserving context"},{verb:"kill",name:"hydra kill",description:"Close this session (kills the agent; record is kept so it can be resumed later)"},{verb:"restart",name:"hydra restart",description:"Restart the agent with a fresh session/new while preserving conversation history (useful when the proxy has changed available models)"},{verb:"config",name:"hydra config",argsHint:"[<id> [<value>]]",description:"List or set an agent-advertised configOption (model, mode, effort, etc.). With no args, lists all options; with <id>, shows its choices; with <id> <value>, applies it."}],Eu=new Map(mt.map(n=>[n.verb,n]));function Ps(n){let e=n.lastIndexOf("/");return(e===-1?n:n.slice(e+1)).toLowerCase()}function Ee(n,e){if(e.length===0)return{kind:"none",requested:n};if(e.some(r=>r.modelId===n))return{kind:"exact",modelId:n};let t=Ps(n),s=e.map(r=>r.modelId).filter(r=>Ps(r)===t);return s.length===1?{kind:"resolved",modelId:s[0],requested:n}:s.length>1?{kind:"ambiguous",requested:n,candidates:s}:{kind:"unknown",requested:n}}function Ts(n){if(n.length===0)return n;let e=new Map,t=new Map,s=new Map;for(let a=0;a<n.length;a++){let d=n[a];if(d===void 0)continue;let c=on(d);if(c?.sessionUpdate!=="tool_call_update")continue;let p=typeof c.toolCallId=="string"?c.toolCallId:void 0;if(p!==void 0&&(e.set(p,a),c.rawInput&&typeof c.rawInput=="object"&&!Array.isArray(c.rawInput)&&Object.keys(c.rawInput).length>0&&s.set(p,c.rawInput),Array.isArray(c.content)&&c.content.length>0)){let l=t.get(p);l?l.push(...c.content):t.set(p,[...c.content])}}let r=[],i=null,o=null;for(let a=0;a<n.length;a++){let d=n[a];if(d===void 0)continue;let c=on(d);if(!c||typeof c.sessionUpdate!="string"){r.push(d),i=null;continue}let p=c.sessionUpdate;if(p==="agent_message_chunk"||p==="agent_thought_chunk"||p==="user_message_chunk"){i&&i.kind===p?Ao(r,i.outIndex,ko(c.content)):(r.push(d),i={outIndex:r.length-1,kind:p});continue}if(i=null,p==="tool_call_update"){let l=typeof c.toolCallId=="string"?c.toolCallId:void 0;if(l!==void 0&&e.get(l)!==a)continue;let u=l!==void 0&&t.has(l)?xo(d,t.get(l)??[]):d;l!==void 0&&s.has(l)&&!Co(u)&&(u=Mo(u,s.get(l))),r.push(u);continue}if(p==="plan"){o!==null?r[o]=d:(r.push(d),o=r.length-1);continue}(p==="prompt_received"||p==="turn_complete")&&(o=null),r.push(d)}return r}function on(n){if(n.method!=="session/update")return;let t=n.params?.update;if(!(!t||typeof t!="object"||Array.isArray(t)))return t}function ko(n){if(!n||typeof n!="object")return"";let e=n;return typeof e.text=="string"?e.text:""}function Ao(n,e,t){if(t.length===0)return;let s=n[e];if(s===void 0)return;let r=s.params??{},i=r.update??{},o=i.content??{},a=typeof o.text=="string"?o.text:"";n[e]={...s,params:{...r,update:{...i,content:{...o,text:a+t}}}}}function xo(n,e){let t=n.params??{},s=t.update??{};return{...n,params:{...t,update:{...s,content:e}}}}function Co(n){let t=on(n)?.rawInput;return!!t&&typeof t=="object"&&!Array.isArray(t)&&Object.keys(t).length>0}function Mo(n,e){let t=n.params??{},s=t.update??{};return{...n,params:{...t,update:{...s,rawInput:e}}}}import*as re from"fs/promises";async function Le(n,e){let t=w.queueFile(n);if(e.length===0){await re.unlink(t).catch(()=>{});return}await re.mkdir(w.sessionDir(n),{recursive:!0});let s=e.map(r=>JSON.stringify(r)).join(`
20
20
  `)+`
21
21
  `;await re.writeFile(t,s,"utf8")}async function _s(n){let e=w.queueFile(n),t;try{t=await re.readFile(e,"utf8")}catch(r){if(r.code==="ENOENT")return[];throw r}let s=[];for(let r of t.split(`
22
- `))if(r.trim())try{let i=JSON.parse(r);i&&typeof i.messageId=="string"&&Array.isArray(i.prompt)&&typeof i.enqueuedAt=="number"&&s.push(i)}catch{}return s}async function $s(n){let e=w.queueFile(n);await re.unlink(e).catch(()=>{})}import*as Bs from"fs/promises";var Ns="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz",js=Os(Ns,16),Fs=Os(Ns,16),de="hydra_session_";function Ee(){return`m_${js()}`}function Hs(n){return n.startsWith(de)?n.slice(de.length):n}var Mo=1e3,an=300*1e3,Ro=64,ye=class n{sessionId;cwd;agentId;agent;upstreamSessionId;agentMeta;agentCapabilities;agentArgs;parentSessionId;forkedFromSessionId;forkedFromMessageId;originatingClient;_interactive;get interactive(){return this._interactive}_priority;get priority(){return this._priority}title;currentModel;currentMode;_currentUsage;updatedAt;createdAt;clients=new Map;historyStore;promptQueue=[];currentEntry;promptInFlight=!1;queueWriteChain=Promise.resolve();closed=!1;closing=!1;closeInFlight;closeHandlers=[];titleHandlers=[];scheduleSynopsisHook;broadcastHandlers=[];lastCancelAt=0;static CANCEL_ERROR_WINDOW_MS=2e3;forceCancelling=!1;_firstPromptSeeded=!1;get firstPromptSeeded(){return this._firstPromptSeeded}promptStartedAt;appendCount=0;historyMaxEntries;compactEvery;inFlightPermissions=new Set;internalPromptCapture;idleTimeoutMs;idleTimer;idleEventTimer;idleEventTimeoutMs;lastRecordedAt;spawnReplacementAgent;listSessions;logger;transformChain;extensionCommands;extensionCommandsUnsub;pendingClaims=new Map;agentChangeHandlers=[];agentAdvertisedCommands=[];agentAdvertisedModes=[];agentAdvertisedModels=[];agentCommandsHandlers=[];agentModesHandlers=[];agentModelsHandlers=[];availableAgentsFn;modelHandlers=[];modeHandlers=[];interactiveHandlers=[];priorityHandlers=[];usageHandlers=[];cumulativeCost=0;get currentUsage(){if(!this._currentUsage&&!this.cumulativeCost)return;let e=this._currentUsage??{},t=this.cumulativeCost+(e.costAmount??0);return{...e,costAmount:t||void 0,cumulativeCost:void 0}}amendInProgress;inFlightExtensionDispatches=new Map;recentlyTerminal=new Map;streamBuffer;streamFilePath;constructor(e){this.sessionId=e.sessionId??`${de}${js()}`,this.cwd=e.cwd,this.agentId=e.agentId,this.agent=e.agent,this.upstreamSessionId=e.upstreamSessionId,this.agentMeta=e.agentMeta,this.agentCapabilities=e.agentCapabilities,this.agentArgs=e.agentArgs,this.parentSessionId=e.parentSessionId,this.forkedFromSessionId=e.forkedFromSessionId,this.forkedFromMessageId=e.forkedFromMessageId,this.originatingClient=e.originatingClient,this.title=e.title,this.scheduleSynopsisHook=e.scheduleSynopsis,this.currentModel=e.currentModel,this.currentMode=e.currentMode,this._currentUsage=e.currentUsage,this.cumulativeCost=e.currentUsage?.cumulativeCost??0,e.agentCommands&&e.agentCommands.length>0&&(this.agentAdvertisedCommands=[...e.agentCommands]),e.agentModes&&e.agentModes.length>0&&(this.agentAdvertisedModes=[...e.agentModes]),e.agentModels&&e.agentModels.length>0&&(this.agentAdvertisedModels=[...e.agentModels]),this.idleTimeoutMs=e.idleTimeoutMs??0,this.idleEventTimeoutMs=e.idleEventTimeoutMs??3e4,this.spawnReplacementAgent=e.spawnReplacementAgent,this.availableAgentsFn=e.availableAgents,this.listSessions=e.listSessions,this.logger=e.logger,this.transformChain=e.transformChain??[],this.extensionCommands=e.extensionCommands,this.extensionCommands&&(this.extensionCommandsUnsub=this.extensionCommands.onChange(()=>{this.closed||this.broadcastMergedCommands()})),e.firstPromptSeeded&&(this._firstPromptSeeded=!0),this._interactive=e.interactive,this._priority=e.priority,this.historyStore=e.historyStore,this.historyMaxEntries=e.historyMaxEntries??Mo,this.compactEvery=Math.max(1,Math.floor(this.historyMaxEntries*.2)),this.updatedAt=Date.now(),this.createdAt=e.createdAt??this.updatedAt,this.lastRecordedAt=this.updatedAt,this.wireAgent(this.agent),this.scheduleIdleCheck(),this.notifyChain("session.opened",{})}broadcastMergedCommands(){this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"available_commands_update",availableCommands:this.mergedAvailableCommands()}})}broadcastAvailableModes(){this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"available_modes_update",availableModes:this.agentAdvertisedModes}})}broadcastAvailableModels(){let e={sessionUpdate:"current_model_update",availableModels:[...this.agentAdvertisedModels]};this.currentModel!==void 0&&this.currentModel.length>0&&(e.currentModel=this.currentModel),this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:e})}wireAgent(e){e.connection.onNotification("session/update",t=>{if(this.internalPromptCapture){Oo(this.internalPromptCapture,t);return}this.runResponseChain(t)}),e.connection.onRequest("session/request_permission",async t=>this.handlePermissionRequest(t)),typeof e.connection.onOrphanError=="function"&&e.connection.onOrphanError(t=>{this.handleOrphanError(t)}),e.onExit(()=>{this.agent===e&&this.markClosed({deleteRecord:!1})})}handleOrphanError(e){let t=Date.now()-this.lastCancelAt;if(this.lastCancelAt===0||t>n.CANCEL_ERROR_WINDOW_MS){this.logger?.warn(`agent ${this.agentId} sent uncorrelated error frame code=${e.code} message=${e.message}`);return}this.lastCancelAt=0,this.logger?.warn(`agent ${this.agentId} rejected session/cancel code=${e.code} message=${e.message}`),this.broadcastQueueNotification("hydra-acp/cancel_failed",{sessionId:this.sessionId,code:e.code,message:e.message})}async runResponseChain(e,t=new Set,s=0){let r=e,i=this.injectCumulativeCost(e);for(let d=s;d<this.transformChain.length;d++){let c=this.transformChain[d];if(t.has(c.name)||!c.intercepts.has("response:session/update"))continue;let f=`t_${Fs()}`,l;try{l=await c.connection.request("hydra-acp/transformer/message",{token:f,phase:"response",method:"session/update",direction:"agent\u2192client",sessionId:this.sessionId,envelope:i})}catch(p){this.logger?.warn(`transformer ${c.name} error on response:session/update: ${p.message}`);continue}let u=l?.action??"continue";if(u==="stop")return;if(u==="processing"){let p=d,g=i,m=new Set(t);await new Promise(y=>{let v=setTimeout(()=>{this.pendingClaims.delete(f)&&(this.broadcastQueueNotification("hydra-acp/transformer/abandoned_request",{sessionId:this.sessionId,token:f,transformerName:c.name}),this.runResponseChain(g,new Set([...m,c.name]),p+1).then(y))},an);typeof v.unref=="function"&&v.unref(),this.pendingClaims.set(f,{resolve:()=>y(),timer:v,transformerName:c.name,method:"session/update",envelope:g,chainIdx:p,originatedBy:m,side:"response"})});return}t.add(c.name)}let o=No(i);if(o!==null){this.setAgentAdvertisedCommands(o);return}let a=Fo(i);if(a!==null){this.setAgentAdvertisedModes(a);return}if(this.maybeApplyAgentModel(i)){this.recordAndBroadcast("session/update",i);return}if(this.maybeApplyAgentMode(i)){this.recordAndBroadcast("session/update",i);return}if(this.maybeApplyAgentConfigOption(i)){this.recordAndBroadcast("session/update",this.mergeAgentOptionIntoEnvelope(i));return}if(this.maybeApplyAgentUsage(r)){this.recordAndBroadcast("session/update",i);return}this.maybeApplyAgentSessionInfo(i),this.recordAndBroadcast("session/update",i)}onAgentChange(e){this.agentChangeHandlers.push(e)}get attachedCount(){return this.clients.size}connectedClients(e){let t=[];for(let s of this.clients.values()){if(e&&s.clientId===e)continue;let r={clientId:s.clientId};s.clientInfo?.name&&(r.name=s.clientInfo.name),s.clientInfo?.version&&(r.version=s.clientInfo.version),t.push(r)}return t}get turnStartedAt(){return this.promptStartedAt}get awaitingInput(){return this.inFlightPermissions.size>0}async getHistorySnapshot(e="inline"){return this.historyStore?this.historyStore.load(this.sessionId,{tools:e}).catch(()=>[]):[]}onBroadcast(e){return this.broadcastHandlers.push(e),()=>{let t=this.broadcastHandlers.indexOf(e);t>=0&&this.broadcastHandlers.splice(t,1)}}attach(e,t,s={}){if(this.closed)throw Y(new Error("session is closed"),I.SessionNotFound);if(this.clients.has(e.clientId))throw Y(new Error(`client ${e.clientId} is already attached`),I.AlreadyAttached);return this.clients.set(e.clientId,e),this.updatedAt=Date.now(),t==="none"?Promise.resolve({entries:[],appliedPolicy:t}):t==="pending_only"?Promise.resolve({entries:this.buildStateSnapshotReplay(),appliedPolicy:t}):this.loadReplay(t,s)}async loadReplay(e,t){let s=o=>t.raw?o:Ts(o),r=await this.getHistorySnapshot(t.toolContent??"inline"),i=this.buildStateSnapshotReplay();if(e==="after_message"){let o=t.afterMessageId?cn(r,t.afterMessageId):-1;return o<0?{entries:[...i,...s(r)],appliedPolicy:"full"}:{entries:[...i,...s(r.slice(o+1))],appliedPolicy:"after_message"}}return{entries:[...i,...s(r)],appliedPolicy:"full"}}buildStateSnapshotReplay(){let e=[],t=this.sessionId,s=Date.now();if(this.title!==void 0&&this.title.length>0&&e.push({method:"session/update",params:{sessionId:t,update:{sessionUpdate:"session_info_update",title:this.title}},recordedAt:s}),this.currentModel!==void 0&&this.currentModel.length>0||this.agentAdvertisedModels.length>0){let i={sessionUpdate:"current_model_update"};this.currentModel!==void 0&&this.currentModel.length>0&&(i.currentModel=this.currentModel),this.agentAdvertisedModels.length>0&&(i.availableModels=[...this.agentAdvertisedModels]),e.push({method:"session/update",params:{sessionId:t,update:i},recordedAt:s})}this.currentMode!==void 0&&this.currentMode.length>0&&e.push({method:"session/update",params:{sessionId:t,update:{sessionUpdate:"current_mode_update",currentModeId:this.currentMode}},recordedAt:s});let r=this.mergedAvailableCommands();if(r.length>0&&e.push({method:"session/update",params:{sessionId:t,update:{sessionUpdate:"available_commands_update",availableCommands:r}},recordedAt:s}),this.agentAdvertisedModes.length>0&&e.push({method:"session/update",params:{sessionId:t,update:{sessionUpdate:"available_modes_update",availableModes:[...this.agentAdvertisedModes]}},recordedAt:s}),this.currentUsage!==void 0){let i=this.currentUsage,o={sessionUpdate:"usage_update"};if(typeof i.used=="number"&&(o.used=i.used),typeof i.size=="number"&&(o.size=i.size),typeof i.costAmount=="number"||typeof i.costCurrency=="string"){let a={};typeof i.costAmount=="number"&&(a.amount=i.costAmount),typeof i.costCurrency=="string"&&(a.currency=i.costCurrency),o.cost=a}Object.keys(o).length>1&&e.push({method:"session/update",params:{sessionId:t,update:o},recordedAt:s})}return e}replayPendingPermissions(e){for(let t of this.inFlightPermissions)t.addClient(e)}detach(e){let t=this.clients.get(e);t&&(this.clients.delete(e),this.updatedAt=Date.now(),this.broadcastClientDisconnected(t))}broadcastClientDisconnected(e){let t={clientId:e.clientId};e.clientInfo?.name&&(t.name=e.clientInfo.name),e.clientInfo?.version&&(t.version=e.clientInfo.version);let s={sessionUpdate:"client_disconnected",client:t,timestamp:new Date().toISOString()};for(let r of this.clients.values())r.connection.notify("session/update",{sessionId:this.sessionId,update:s}).catch(()=>{})}async prompt(e,t){let s=this.clients.get(e);if(!s)throw Y(new Error("client not attached"),I.SessionNotFound);if(this.closing)throw Y(new Error("session is closing; new prompts cannot be accepted"),I.SessionClosing);let r=Ee();if(this.maybeSeedTitleFromPrompt(t),this._firstPromptSeeded=!0,!(De((t??{})._meta).ancillary===!0)&&this._interactive!==!0){this._interactive=!0;for(let o of this.interactiveHandlers)try{o(!0)}catch{}}return this.enqueueUserPrompt(s,t,r)}broadcastPromptReceived(e){let t={clientId:e.originator.clientId};e.originator.name&&(t.name=e.originator.name),e.originator.version&&(t.version=e.originator.version),this.promptStartedAt=Date.now(),this.recordAndBroadcast("session/update",{sessionId:this.sessionId,update:{sessionUpdate:"prompt_received",messageId:e.messageId,prompt:e.prompt,sentBy:t}},e.clientId);let s=Pe(e.prompt);s.length>0&&this.recordAndBroadcast("session/update",{sessionId:this.sessionId,update:{sessionUpdate:"user_message_chunk",content:{type:"text",text:s},_meta:{"hydra-acp":{compatFor:"prompt_received"}}}},e.clientId)}broadcastTurnComplete(e,t,s,r){let i=t&&typeof t=="object"&&"stopReason"in t&&typeof t.stopReason=="string"?t.stopReason:void 0,o={sessionUpdate:"turn_complete",messageId:Ee()};i!==void 0&&(o.stopReason=i);let a=this.amendInProgress;a&&s!==void 0&&a.cancelledMessageId===s&&(o._meta={"hydra-acp":{amended:{cancelledMessageId:a.cancelledMessageId,newMessageId:a.newMessageId}}}),this.promptStartedAt=void 0,s!==void 0&&i!==void 0&&this.recordTerminal(s,i),this.recordAndBroadcast("session/update",{sessionId:this.sessionId,update:o},r?void 0:e),a&&s!==void 0&&a.cancelledMessageId===s&&this.broadcastPromptAmended(a)}recordTerminal(e,t){for(this.recentlyTerminal.set(e,{stopReason:t,terminatedAt:Date.now()});this.recentlyTerminal.size>Ro;){let s=this.recentlyTerminal.keys().next().value;if(s===void 0)break;this.recentlyTerminal.delete(s)}}broadcastPromptAmended(e){let t=this.findUserEntry(e.newMessageId);if(!t)return;let s={sessionId:this.sessionId,cancelledMessageId:e.cancelledMessageId,newMessageId:e.newMessageId,prompt:t.prompt,originator:t.originator,amendedAt:Date.now()};this.broadcastQueueNotification("hydra-acp/prompt/amended",s)}findUserEntry(e){if(this.currentEntry?.messageId===e&&this.currentEntry.kind==="user")return this.currentEntry;let t=this.promptQueue.find(s=>s.messageId===e&&s.kind==="user");return t?.kind==="user"?t:void 0}visibleQueueDepth(){let e=this.currentEntry?.kind==="user"&&!this.currentEntry.cancelled?1:0;for(let t of this.promptQueue)t.kind==="user"&&!t.cancelled&&(e+=1);return e}broadcastQueueAdded(e,t){let s=this.visibleQueueDepth(),r=t?.position??Math.max(0,s-1),i={sessionId:this.sessionId,messageId:e.messageId,originator:e.originator,prompt:e.prompt,position:r,queueDepth:s,enqueuedAt:e.enqueuedAt};t?.amending!==void 0&&(i._meta={"hydra-acp":{amending:t.amending}}),this.broadcastQueueNotification("hydra-acp/prompt_queue/added",i)}broadcastQueueUpdated(e,t){this.broadcastQueueNotification("hydra-acp/prompt_queue/updated",{sessionId:this.sessionId,messageId:e,prompt:t})}broadcastQueueRemoved(e,t){this.broadcastQueueNotification("hydra-acp/prompt_queue/removed",{sessionId:this.sessionId,messageId:e,reason:t})}broadcastQueueNotification(e,t){for(let s of this.clients.values())s.connection.notify(e,t).catch(()=>{})}queueSnapshot(){let e=[],t=0;this.currentEntry?.kind==="user"&&!this.currentEntry.cancelled&&e.push({messageId:this.currentEntry.messageId,originator:this.currentEntry.originator,prompt:this.currentEntry.prompt,position:t++,enqueuedAt:this.currentEntry.enqueuedAt});for(let s of this.promptQueue)s.kind!=="user"||s.cancelled||e.push({messageId:s.messageId,originator:s.originator,prompt:s.prompt,position:t++,enqueuedAt:s.enqueuedAt});return e}async flushPersistWrites(){await this.queueWriteChain.catch(()=>{})}replayPersistedQueue(e){for(let t of e){let s={clientId:`hydra-resurrected_${t.messageId}`};t.originator.clientInfo.name!==void 0&&(s.name=t.originator.clientInfo.name),t.originator.clientInfo.version!==void 0&&(s.version=t.originator.clientInfo.version);let r={kind:"user",messageId:t.messageId,originator:s,clientId:s.clientId,prompt:t.prompt,enqueuedAt:t.enqueuedAt,cancelled:!1,resolve:()=>{},reject:()=>{}};this.promptQueue.push(r),this.broadcastQueueAdded(r)}this.drainQueue()}cancelQueuedPrompt(e){if(this.currentEntry?.messageId===e)return{cancelled:!1,reason:"already_running"};let t=this.promptQueue.findIndex(r=>r.messageId===e);if(t<0)return{cancelled:!1,reason:"not_found"};let s=this.promptQueue[t];return s.cancelled=!0,this.promptQueue.splice(t,1),s.kind==="user"&&(this.broadcastQueueRemoved(e,"cancelled"),this.persistRewrite()),this.amendInProgress?.newMessageId===e&&(this.amendInProgress=void 0),s.resolve({stopReason:"cancelled"}),{cancelled:!0,reason:"ok"}}updateQueuedPrompt(e,t){if(this.currentEntry?.messageId===e)return{updated:!1,reason:"already_running"};let s=this.promptQueue.find(r=>r.messageId===e);return!s||s.kind!=="user"?{updated:!1,reason:"not_found"}:(s.prompt=t,this.broadcastQueueUpdated(e,t),this.persistRewrite(),{updated:!0,reason:"ok"})}amendPrompt(e,t){let s=this.clients.get(e);if(!s)throw Y(new Error("client not attached"),I.SessionNotFound);let{targetMessageId:r,prompt:i,replaceQueue:o,onTargetCompleted:a}=t;if(this.currentEntry?.messageId===r&&this.currentEntry.kind==="user"&&!this.currentEntry.cancelled&&this.amendInProgress===void 0)return this.amendOnHead(s,i,r,o);let d=this.promptQueue.find(f=>f.messageId===r&&f.kind==="user");if(d&&d.kind==="user"&&!d.cancelled)return d.prompt=i,this.broadcastQueueUpdated(r,i),this.persistRewrite(),{amended:!0,reason:"ok",messageId:r};let c=this.recentlyTerminal.get(r);return c?c.stopReason==="cancelled"?{amended:!1,reason:"target_cancelled"}:a==="send_anyway"?{amended:!1,reason:"target_completed",messageId:this.enqueueAmendmentAsFollowUp(s,i)}:{amended:!1,reason:"target_completed"}:{amended:!1,reason:"target_not_found"}}amendOnHead(e,t,s,r){let i=Ee(),o={clientId:e.clientId};if(e.clientInfo?.name&&(o.name=e.clientInfo.name),e.clientInfo?.version&&(o.version=e.clientInfo.version),r){let d=[];for(let c of this.promptQueue){if(c.kind==="user"&&!c.cancelled){c.cancelled=!0,this.broadcastQueueRemoved(c.messageId,"cancelled"),c.resolve({stopReason:"cancelled"});continue}d.push(c)}this.promptQueue=d}let a={kind:"user",messageId:i,originator:o,clientId:e.clientId,prompt:t,enqueuedAt:Date.now(),cancelled:!1,wasAmend:!0,resolve:()=>{},reject:()=>{}};return this.promptQueue.unshift(a),this.persistRewrite(),this.broadcastQueueAdded(a,{amending:s,position:1}),this.amendInProgress={cancelledMessageId:s,newMessageId:i},this.cancelExtensionDispatch(s,"amended"),this.agent.connection.notify("session/cancel",{sessionId:this.upstreamSessionId}).catch(()=>{}),{amended:!0,reason:"ok",messageId:i}}enqueueAmendmentAsFollowUp(e,t){let s=Ee(),r={clientId:e.clientId};e.clientInfo?.name&&(r.name=e.clientInfo.name),e.clientInfo?.version&&(r.version=e.clientInfo.version);let i={kind:"user",messageId:s,originator:r,clientId:e.clientId,prompt:t,enqueuedAt:Date.now(),cancelled:!1,resolve:()=>{},reject:()=>{}};return this.promptQueue.push(i),this.persistRewrite(),this.broadcastQueueAdded(i),this.drainQueue(),s}async cancel(e){if(!this.clients.get(e))throw Y(new Error("client not attached"),I.SessionNotFound);this.lastCancelAt=Date.now(),this.currentEntry?.kind==="user"&&this.currentEntry.messageId!==void 0&&this.cancelExtensionDispatch(this.currentEntry.messageId,"cancelled"),await this.forwardRequest("session/cancel",{sessionId:this.sessionId},new Set,0,"notification")}addTransformer(e){let t=this.transformChain.findIndex(s=>s.name===e.name);t>=0?this.transformChain[t]=e:this.transformChain.push(e),e.intercepts.has("lifecycle:session.opened")&&e.connection.notify("hydra-acp/transformer/session_event",{event:"session.opened",sessionId:this.sessionId}).catch(()=>{})}async forwardRequest(e,t,s=new Set,r=0,i="request"){let o=this.rewriteForAgent(t);for(let a=r;a<this.transformChain.length;a++){let d=this.transformChain[a];if(s.has(d.name))continue;let c=`request:${e}`;if(!d.intercepts.has(c))continue;let f=`t_${Fs()}`,l;try{l=await d.connection.request("hydra-acp/transformer/message",{token:f,phase:"request",method:e,direction:"client\u2192agent",sessionId:this.sessionId,envelope:o})}catch(p){this.logger?.warn(`transformer ${d.name} error on ${c}: ${p.message}`);continue}let u=l?.action??"continue";if(u==="stop")return i==="notification"?void 0:l?.payload??dn(e);if(u==="processing"){let p=a,g=o,m=new Set(s);return new Promise(y=>{let v=setTimeout(()=>{this.pendingClaims.delete(f)&&(this.broadcastQueueNotification("hydra-acp/transformer/abandoned_request",{sessionId:this.sessionId,token:f,transformerName:d.name}),this.forwardRequest(e,g,new Set([...m,d.name]),p+1,i).then(y).catch(()=>y(i==="notification"?void 0:dn(e))))},an);typeof v.unref=="function"&&v.unref(),this.pendingClaims.set(f,{resolve:y,timer:v,transformerName:d.name,method:e,envelope:g,chainIdx:p,originatedBy:m,side:"request",tailKind:i})})}s.add(d.name)}if(i==="notification"){await this.agent.connection.notify(e,o);return}return this.agent.connection.request(e,o)}dischargeClaim(e,t){let s=this.pendingClaims.get(e);return s?(clearTimeout(s.timer),this.pendingClaims.delete(e),s.resolve(t),!0):!1}keepAliveClaim(e,t){let s=this.pendingClaims.get(e);if(!s)return!1;clearTimeout(s.timer);let r=typeof t=="number"&&t>0?Math.min(t*1.5,1800*1e3):an,i=setTimeout(()=>{if(this.pendingClaims.delete(e))if(this.broadcastQueueNotification("hydra-acp/transformer/abandoned_request",{sessionId:this.sessionId,token:e,transformerName:s.transformerName}),s.side==="response")this.runResponseChain(s.envelope,new Set([...s.originatedBy,s.transformerName]),s.chainIdx+1).then(()=>s.resolve(void 0));else{let o=s.tailKind??"request";this.forwardRequest(s.method,s.envelope,new Set([...s.originatedBy,s.transformerName]),s.chainIdx+1,o).then(s.resolve).catch(()=>s.resolve(o==="notification"?void 0:dn(s.method)))}},r);return typeof i.unref=="function"&&i.unref(),s.timer=i,!0}async emitToChain(e,t,s){let r=this.transformChain.findIndex(a=>a.name===e),i=r>=0?r+1:0,o=new Set([e]);if(t==="session/update"){await this.runResponseChain(s,o,i);return}return this.forwardRequest(t,s,o,i)}rewriteForAgent(e){if(e&&typeof e=="object"&&!Array.isArray(e)){let t=e;if(t.sessionId===this.sessionId)return{...t,sessionId:this.upstreamSessionId}}return e}async close(e={}){if(!this.closed)return this.closeInFlight?this.closeInFlight:(this.closing=!0,this.closeInFlight=this.doClose(e),this.closeInFlight)}async doClose(e){this.logger?.info(`session ${this.sessionId} closing deleteRecord=${e.deleteRecord??!1}`),this.cancelIdleTimer(),await this.agent.kill().catch(()=>{}),this.markClosed({deleteRecord:e.deleteRecord??!1})}onClose(e){this.closeHandlers.push(e)}onTitleChange(e){this.titleHandlers.push(e)}retitle(e){return this.runTitleCommand(e)}retitleFromAgent(){return this.runTitleCommand("")}setTitle(e){let t=e.trim();if(!(!t||t===this.title)){this.title=t,this.recordAndBroadcast("session/update",{sessionId:this.sessionId,update:{sessionUpdate:"session_info_update",title:t,updatedAt:new Date().toISOString()}});for(let s of this.titleHandlers)try{s(t)}catch{}}}maybeSeedTitleFromPrompt(e){if(this.firstPromptSeeded)return;let s=Pe((e??{}).prompt),r=un(s,200);r&&(r.startsWith("/")||(this._firstPromptSeeded=!0,this.setTitle(r)))}maybeApplyAgentModel(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="current_model_update")return!1;let r=we(s.availableModels);r.length>0&&this.setAgentAdvertisedModels(r);let i=typeof s.currentModel=="string"?s.currentModel:typeof s.model=="string"?s.model:void 0;if(i===void 0)return!0;let o=i.trim();if(!o||o===this.currentModel)return!0;this.logger?.info(`live current_model_update: sessionId=${this.sessionId} ${JSON.stringify(this.currentModel)} \u2192 ${JSON.stringify(o)}`),this.currentModel=o;for(let a of this.modelHandlers)try{a(o)}catch{}return this.broadcastConfigOptions(),!0}maybeApplyAgentConfigOption(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="config_option_update")return!1;let r=s.configOptions;if(!Array.isArray(r))return!0;for(let i of r){if(!i||typeof i!="object")continue;let o=i;if(o.id==="model"){let a=we(o.options);a.length>0&&this.setAgentAdvertisedModels(a);let d=o.currentValue;if(typeof d=="string"){let c=d.trim();c&&c!==this.currentModel&&(this.logger?.info(`live config_option_update(model): sessionId=${this.sessionId} ${JSON.stringify(this.currentModel)} \u2192 ${JSON.stringify(c)}`),this.applyModelChange(c))}}else if(o.id==="mode"){let a=Te(o.options);a.length>0&&this.setAgentAdvertisedModes(a);let d=o.currentValue;if(typeof d=="string"){let c=d.trim();c&&c!==this.currentMode&&(this.logger?.info(`live config_option_update(mode): sessionId=${this.sessionId} ${JSON.stringify(this.currentMode)} \u2192 ${JSON.stringify(c)}`),this.applyModeChange(c))}}}return!0}maybeApplyAgentMode(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="current_mode_update")return!1;let r=typeof s.currentModeId=="string"?s.currentModeId:typeof s.currentMode=="string"?s.currentMode:typeof s.mode=="string"?s.mode:void 0;if(r===void 0)return!0;let i=r.trim();if(!i||i===this.currentMode)return!0;this.logger?.info(`current_mode_update: sessionId=${this.sessionId} ${JSON.stringify(this.currentMode)} \u2192 ${JSON.stringify(i)}`),this.currentMode=i;for(let o of this.modeHandlers)try{o(i)}catch{}return this.broadcastConfigOptions(),!0}maybeApplyAgentUsage(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="usage_update")return!1;let r={...this._currentUsage??{}},i=!1;if(typeof s.used=="number"&&r.used!==s.used&&(r.used=s.used,i=!0),typeof s.size=="number"&&r.size!==s.size&&(r.size=s.size,i=!0),s.cost&&typeof s.cost=="object"){let a=s.cost;typeof a.amount=="number"&&r.costAmount!==a.amount&&(r.costAmount=a.amount,i=!0),typeof a.currency=="string"&&r.costCurrency!==a.currency&&(r.costCurrency=a.currency,i=!0)}if(!i)return!0;this._currentUsage=r;let o=this.currentUsage??r;for(let a of this.usageHandlers)try{a(o)}catch{}return!0}accumulateAndResetCost(){let e=this._currentUsage?.costAmount;if(!e)return;this.cumulativeCost+=e;let t={...this._currentUsage??{},cumulativeCost:this.cumulativeCost,costAmount:void 0};this._currentUsage=t;for(let s of this.usageHandlers)try{s(t)}catch{}}injectCumulativeCost(e){if(!this.cumulativeCost||!e||typeof e!="object")return e;let t=e;if(!t.update||typeof t.update!="object")return e;let s=t.update;if(s.sessionUpdate!=="usage_update"||!s.cost||typeof s.cost!="object")return e;let r=s.cost;return typeof r.amount!="number"?e:{...t,update:{...s,cost:{...r,amount:this.cumulativeCost+r.amount}}}}get totalUsage(){return this.currentUsage}setAgentAdvertisedCommands(e){if(To(this.agentAdvertisedCommands,e)){this.broadcastMergedCommands();return}this.agentAdvertisedCommands=e;for(let t of this.agentCommandsHandlers)try{t(e)}catch{}this.broadcastMergedCommands()}setAgentAdvertisedModes(e){if(_o(this.agentAdvertisedModes,e)){this.broadcastAvailableModes();return}this.agentAdvertisedModes=e;for(let t of this.agentModesHandlers)try{t(e)}catch{}this.broadcastAvailableModes()}setAgentAdvertisedModels(e){if(this.logger?.info(`setAgentAdvertisedModels: sessionId=${this.sessionId} currentModel=${JSON.stringify(this.currentModel)} newList=[${e.map(t=>t.modelId).join(",")}]`),$o(this.agentAdvertisedModels,e)){this.broadcastAvailableModels();return}this.agentAdvertisedModels=e;for(let t of this.agentModelsHandlers)try{t(e)}catch{}this.broadcastAvailableModels()}onAgentCommandsChange(e){this.agentCommandsHandlers.push(e)}onAgentModesChange(e){this.agentModesHandlers.push(e)}onAgentModelsChange(e){this.agentModelsHandlers.push(e)}onModelChange(e){this.modelHandlers.push(e)}onModeChange(e){this.modeHandlers.push(e)}onInteractiveChange(e){this.interactiveHandlers.push(e)}onPriorityChange(e){this.priorityHandlers.push(e)}setPriority(e){let t=e===void 0||e<=0?void 0:Math.floor(e);if(t!==this._priority){this._priority=t,this.updatedAt=Date.now();for(let s of this.priorityHandlers)try{s(t)}catch{}}}applyModelChange(e){let t=e.trim();if(!t)return;if(t!==this.currentModel){this.logger?.info(`applyModelChange: sessionId=${this.sessionId} ${JSON.stringify(this.currentModel)} \u2192 ${JSON.stringify(t)}`),this.currentModel=t;for(let r of this.modelHandlers)try{r(t)}catch{}}let s={sessionUpdate:"current_model_update",currentModel:t};this.agentAdvertisedModels.length>0&&(s.availableModels=[...this.agentAdvertisedModels]),this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:s}),this.broadcastConfigOptions()}applyModeChange(e){let t=e.trim();if(!t)return;if(t!==this.currentMode){this.logger?.info(`applyModeChange: sessionId=${this.sessionId} ${JSON.stringify(this.currentMode)} \u2192 ${JSON.stringify(t)}`),this.currentMode=t;for(let r of this.modeHandlers)try{r(t)}catch{}}let s={sessionUpdate:"current_mode_update",currentModeId:t};this.agentAdvertisedModes.length>0&&(s.availableModes=[...this.agentAdvertisedModes]),this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:s}),this.broadcastConfigOptions()}buildConfigOptions(){let e=[];if(this.agentAdvertisedModels.length>0){let r=this.agentAdvertisedModels.map(o=>({value:o.modelId,name:o.name??o.modelId,...o.description!==void 0?{description:o.description}:{}})),i=this.currentModel&&r.some(o=>o.value===this.currentModel)?this.currentModel:r[0].value;e.push({id:"model",name:"Model",category:"model",type:"select",currentValue:i,options:r})}if(this.agentAdvertisedModes.length>0){let r=this.agentAdvertisedModes.map(o=>({value:o.id,name:o.name??o.id,...o.description!==void 0?{description:o.description}:{}})),i=this.currentMode&&r.some(o=>o.value===this.currentMode)?this.currentMode:r[0].value;e.push({id:"mode",name:"Session Mode",category:"mode",type:"select",currentValue:i,options:r})}let s=(this.availableAgentsFn?.()??[]).map(r=>({value:r.id,name:r.name??r.id,...r.description!==void 0?{description:r.description}:{}}));return s.some(r=>r.value===this.agentId)||s.unshift({value:this.agentId,name:this.agentId}),e.push({id:"agent",name:"Agent",category:"_hydra_agent",type:"select",currentValue:this.agentId,options:s}),e}broadcastConfigOptions(){this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"config_option_update",configOptions:this.buildConfigOptions()}})}mergeAgentOptionIntoEnvelope(e){if(!e||typeof e!="object")return e;let t=e;if(!t.update||typeof t.update!="object")return e;let s=t.update,r=Array.isArray(s.configOptions)?[...s.configOptions]:[];if(r.some(a=>a&&typeof a=="object"&&a.id==="agent"))return e;let o=this.buildConfigOptions().find(a=>a.id==="agent");return o?{...t,update:{...s,configOptions:[...r,o]}}:e}onUsageChange(e){this.usageHandlers.push(e)}mergedAvailableCommands(){let e=[{name:"hydra",description:"Hydra session command (kill, restart, title, agent <agent>)"},{name:"model",description:"Switch model; omit arg to list available models"},{name:"mode",description:"Switch mode; omit arg to list available modes"},{name:"sessions",description:"List all sessions"},{name:"help",description:"Show available commands"}];if(this.extensionCommands)for(let{name:t,command:s}of this.extensionCommands.list()){let r=s.verb?`hydra ${t} ${s.verb}`:`hydra ${t}`,o={name:s.argsHint?`${r} ${s.argsHint}`:r};if(s.description&&(o.description=s.description),e.push(o),t.startsWith("hydra-acp-")){let a=t.slice(10);if(a.length>0){let d=s.verb?`hydra ${a} ${s.verb}`:`hydra ${a}`,f={name:s.argsHint?`${d} ${s.argsHint}`:d};s.description&&(f.description=s.description),e.push(f)}}}return e.push(...this.agentAdvertisedCommands),e}agentOnlyAdvertisedCommands(){return[...this.agentAdvertisedCommands]}availableModes(){return[...this.agentAdvertisedModes]}availableModels(){return[...this.agentAdvertisedModels]}maybeApplyAgentSessionInfo(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="session_info_update"||typeof s.title!="string")return;let r=s.title.trim();if(!(!r||r===this.title)){this.title=r,this._firstPromptSeeded=!0;for(let i of this.titleHandlers)try{i(r)}catch{}}}async handleSlashCommand(e,t){let r=e.slice(6).trim().match(/^(\S+)(?:\s+([\s\S]*))?$/),i=r?.[1]??"",o=(r?.[2]??"").trim();if(i==="")return{stopReason:"end_turn"};let a=t!==void 0;if(mt.some(l=>l.verb===i))switch(i){case"title":return a?this.runTitleCommandInline(o):this.runTitleCommand(o);case"agent":return a?this.runAgentCommandInline(o):this.runAgentCommand(o);case"kill":return this.runKillCommand();case"restart":return a?this.runRestartCommandInline():this.runRestartCommand();default:{let l=new Error(`no dispatcher for /hydra verb ${i}`);throw l.code=I.InternalError,l}}if(this.extensionCommands?.has(i))return a?this.runExtensionCommandInline(i,o,t):this.runExtensionCommand(i,o);let d=`hydra-acp-${i}`;if(this.extensionCommands?.has(d))return a?this.runExtensionCommandInline(d,o,t):this.runExtensionCommand(d,o);let c=mt.map(l=>l.verb);if(this.extensionCommands){let l=new Set;for(let{name:u}of this.extensionCommands.list())l.has(u)||(c.push(u),l.add(u))}let f=new Error(`unknown /hydra verb: ${i} (known: ${c.join(", ")})`);throw f.code=I.InvalidParams,f}runExtensionCommand(e,t){return this.enqueuePrompt(()=>this.runExtensionCommandInline(e,t,void 0))}async runExtensionCommandInline(e,t,s){let r=this.extensionCommands?.get(e);if(!r)return this.emitExtensionReply(`extension "${e}" is no longer connected`);let i=t.match(/^(\S+)(?:\s+([\s\S]*))?$/),o=i?.[1]??"",a=(i?.[2]??"").trim();if(!r.commands.some(g=>g.verb===o)){let g=r.commands.map(m=>m.verb).join(", ");return this.emitExtensionReply(`/hydra ${e}${o?` ${o}`:""}: unknown verb (known: ${g||"(none)"})`)}let d={sessionId:this.sessionId,verb:o,args:a};s!==void 0&&(d.messageId=s);let c=()=>{},f=new Promise(g=>{c=g});s!==void 0&&this.inFlightExtensionDispatches.set(s,{extensionName:e,connection:r.connection,cancel:g=>c({stopReason:"cancelled",reason:g})});let l,u=!1;try{let g=r.connection.request("hydra-acp/commands/invoke",d),m=await Promise.race([g,f]);if(m&&typeof m=="object"&&m.stopReason==="cancelled"&&"reason"in m)return u=!0,this.logger?.info(`extension ${e} ${o} cancelled (cancelExtensionDispatch); abandoning in-flight commands/invoke`),{stopReason:"cancelled"};l=m}catch(g){return this.emitExtensionReply(`${e} ${o}: ${g.message}`)}finally{s!==void 0&&!u&&this.inFlightExtensionDispatches.delete(s)}let p=l&&typeof l=="object"&&typeof l.text=="string"?l.text:"";return p.length>0?this.emitExtensionReply(p):{stopReason:"end_turn"}}cancelExtensionDispatch(e,t){let s=this.inFlightExtensionDispatches.get(e);if(!s)return!1;this.inFlightExtensionDispatches.delete(e);try{s.connection.notify("hydra-acp/commands/cancel",{sessionId:this.sessionId,messageId:e,reason:t})}catch(r){this.logger?.warn(`notify commands/cancel to ${s.extensionName} failed: ${r.message}`)}return s.cancel(t),!0}emitExtensionReply(e){return this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"agent_message_chunk",content:{type:"text",text:`
22
+ `))if(r.trim())try{let i=JSON.parse(r);i&&typeof i.messageId=="string"&&Array.isArray(i.prompt)&&typeof i.enqueuedAt=="number"&&s.push(i)}catch{}return s}async function $s(n){let e=w.queueFile(n);await re.unlink(e).catch(()=>{})}import*as Us from"fs/promises";var js="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz",Hs=Ns(js,16),Os=Ns(js,16),de="hydra_session_";function Pe(){return`m_${Hs()}`}function Bs(n){return n.startsWith(de)?n.slice(de.length):n}var Ro=1e3,an=300*1e3,Eo=64,we=class n{sessionId;cwd;agentId;agent;upstreamSessionId;agentMeta;agentCapabilities;agentArgs;parentSessionId;forkedFromSessionId;forkedFromMessageId;originatingClient;_interactive;get interactive(){return this._interactive}_priority;get priority(){return this._priority}title;currentModel;currentMode;_currentUsage;updatedAt;createdAt;clients=new Map;historyStore;promptQueue=[];currentEntry;promptInFlight=!1;queueWriteChain=Promise.resolve();closed=!1;closing=!1;closeInFlight;closeHandlers=[];titleHandlers=[];scheduleSynopsisHook;broadcastHandlers=[];lastCancelAt=0;static CANCEL_ERROR_WINDOW_MS=2e3;forceCancelling=!1;_firstPromptSeeded=!1;get firstPromptSeeded(){return this._firstPromptSeeded}promptStartedAt;appendCount=0;historyMaxEntries;compactEvery;inFlightPermissions=new Set;internalPromptCapture;idleTimeoutMs;idleTimer;idleEventTimer;idleEventTimeoutMs;lastRecordedAt;spawnReplacementAgent;listSessions;logger;transformChain;extensionCommands;extensionCommandsUnsub;pendingClaims=new Map;agentChangeHandlers=[];agentAdvertisedCommands=[];agentAdvertisedModes=[];agentAdvertisedModels=[];agentAdvertisedConfigOptions=[];agentCommandsHandlers=[];agentModesHandlers=[];agentModelsHandlers=[];availableAgentsFn;modelHandlers=[];modeHandlers=[];interactiveHandlers=[];priorityHandlers=[];usageHandlers=[];cumulativeCost=0;get currentUsage(){if(!this._currentUsage&&!this.cumulativeCost)return;let e=this._currentUsage??{},t=this.cumulativeCost+(e.costAmount??0);return{...e,costAmount:t||void 0,cumulativeCost:void 0}}amendInProgress;inFlightExtensionDispatches=new Map;recentlyTerminal=new Map;streamBuffer;streamFilePath;constructor(e){this.sessionId=e.sessionId??`${de}${Hs()}`,this.cwd=e.cwd,this.agentId=e.agentId,this.agent=e.agent,this.upstreamSessionId=e.upstreamSessionId,this.agentMeta=e.agentMeta,this.agentCapabilities=e.agentCapabilities,this.agentArgs=e.agentArgs,this.parentSessionId=e.parentSessionId,this.forkedFromSessionId=e.forkedFromSessionId,this.forkedFromMessageId=e.forkedFromMessageId,this.originatingClient=e.originatingClient,this.title=e.title,this.scheduleSynopsisHook=e.scheduleSynopsis,this.currentModel=e.currentModel,this.currentMode=e.currentMode,this._currentUsage=e.currentUsage,this.cumulativeCost=e.currentUsage?.cumulativeCost??0,e.agentCommands&&e.agentCommands.length>0&&(this.agentAdvertisedCommands=[...e.agentCommands]),e.agentModes&&e.agentModes.length>0&&(this.agentAdvertisedModes=[...e.agentModes]),e.agentModels&&e.agentModels.length>0&&(this.agentAdvertisedModels=[...e.agentModels]),this.idleTimeoutMs=e.idleTimeoutMs??0,this.idleEventTimeoutMs=e.idleEventTimeoutMs??3e4,this.spawnReplacementAgent=e.spawnReplacementAgent,this.availableAgentsFn=e.availableAgents,this.listSessions=e.listSessions,this.logger=e.logger,this.transformChain=e.transformChain??[],this.extensionCommands=e.extensionCommands,this.extensionCommands&&(this.extensionCommandsUnsub=this.extensionCommands.onChange(()=>{this.closed||this.broadcastMergedCommands()})),e.firstPromptSeeded&&(this._firstPromptSeeded=!0),this._interactive=e.interactive,this._priority=e.priority,this.historyStore=e.historyStore,this.historyMaxEntries=e.historyMaxEntries??Ro,this.compactEvery=Math.max(1,Math.floor(this.historyMaxEntries*.2)),this.updatedAt=Date.now(),this.createdAt=e.createdAt??this.updatedAt,this.lastRecordedAt=this.updatedAt,this.wireAgent(this.agent),this.scheduleIdleCheck(),this.notifyChain("session.opened",{})}broadcastMergedCommands(){this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"available_commands_update",availableCommands:this.mergedAvailableCommands()}})}broadcastAvailableModes(){this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"available_modes_update",availableModes:this.agentAdvertisedModes}})}broadcastAvailableModels(){let e={sessionUpdate:"current_model_update",availableModels:[...this.agentAdvertisedModels]};this.currentModel!==void 0&&this.currentModel.length>0&&(e.currentModel=this.currentModel),this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:e})}wireAgent(e){e.connection.onNotification("session/update",t=>{if(this.internalPromptCapture){No(this.internalPromptCapture,t);return}this.runResponseChain(t)}),e.connection.onRequest("session/request_permission",async t=>this.handlePermissionRequest(t)),typeof e.connection.onOrphanError=="function"&&e.connection.onOrphanError(t=>{this.handleOrphanError(t)}),e.onExit(()=>{this.agent===e&&this.markClosed({deleteRecord:!1})})}handleOrphanError(e){let t=Date.now()-this.lastCancelAt;if(this.lastCancelAt===0||t>n.CANCEL_ERROR_WINDOW_MS){this.logger?.warn(`agent ${this.agentId} sent uncorrelated error frame code=${e.code} message=${e.message}`);return}this.lastCancelAt=0,this.logger?.warn(`agent ${this.agentId} rejected session/cancel code=${e.code} message=${e.message}`),this.broadcastQueueNotification("hydra-acp/cancel_failed",{sessionId:this.sessionId,code:e.code,message:e.message})}async runResponseChain(e,t=new Set,s=0){let r=e,i=this.injectCumulativeCost(e);for(let d=s;d<this.transformChain.length;d++){let c=this.transformChain[d];if(t.has(c.name)||!c.intercepts.has("response:session/update"))continue;let p=`t_${Os()}`,l;try{l=await c.connection.request("hydra-acp/transformer/message",{token:p,phase:"response",method:"session/update",direction:"agent\u2192client",sessionId:this.sessionId,envelope:i})}catch(f){this.logger?.warn(`transformer ${c.name} error on response:session/update: ${f.message}`);continue}let u=l?.action??"continue";if(u==="stop")return;if(u==="processing"){let f=d,g=i,m=new Set(t);await new Promise(y=>{let v=setTimeout(()=>{this.pendingClaims.delete(p)&&(this.broadcastQueueNotification("hydra-acp/transformer/abandoned_request",{sessionId:this.sessionId,token:p,transformerName:c.name}),this.runResponseChain(g,new Set([...m,c.name]),f+1).then(y))},an);typeof v.unref=="function"&&v.unref(),this.pendingClaims.set(p,{resolve:()=>y(),timer:v,transformerName:c.name,method:"session/update",envelope:g,chainIdx:f,originatedBy:m,side:"response"})});return}t.add(c.name)}let o=jo(i);if(o!==null){this.setAgentAdvertisedCommands(o);return}let a=Fo(i);if(a!==null){this.setAgentAdvertisedModes(a);return}if(this.maybeApplyAgentModel(i)){this.recordAndBroadcast("session/update",i);return}if(this.maybeApplyAgentMode(i)){this.recordAndBroadcast("session/update",i);return}if(this.maybeApplyAgentConfigOption(i)){this.recordAndBroadcast("session/update",this.mergeAgentOptionIntoEnvelope(i));return}if(this.maybeApplyAgentUsage(r)){this.recordAndBroadcast("session/update",i);return}this.maybeApplyAgentSessionInfo(i),this.recordAndBroadcast("session/update",i)}onAgentChange(e){this.agentChangeHandlers.push(e)}get attachedCount(){return this.clients.size}connectedClients(e){let t=[];for(let s of this.clients.values()){if(e&&s.clientId===e)continue;let r={clientId:s.clientId};s.clientInfo?.name&&(r.name=s.clientInfo.name),s.clientInfo?.version&&(r.version=s.clientInfo.version),t.push(r)}return t}get turnStartedAt(){return this.promptStartedAt}get awaitingInput(){return this.inFlightPermissions.size>0}async getHistorySnapshot(e="inline"){return this.historyStore?this.historyStore.load(this.sessionId,{tools:e}).catch(()=>[]):[]}onBroadcast(e){return this.broadcastHandlers.push(e),()=>{let t=this.broadcastHandlers.indexOf(e);t>=0&&this.broadcastHandlers.splice(t,1)}}attach(e,t,s={}){if(this.closed)throw Y(new Error("session is closed"),I.SessionNotFound);if(this.clients.has(e.clientId))throw Y(new Error(`client ${e.clientId} is already attached`),I.AlreadyAttached);return this.clients.set(e.clientId,e),this.updatedAt=Date.now(),t==="none"?Promise.resolve({entries:[],appliedPolicy:t}):t==="pending_only"?Promise.resolve({entries:this.buildStateSnapshotReplay(),appliedPolicy:t}):this.loadReplay(t,s)}async loadReplay(e,t){let s=o=>t.raw?o:Ts(o),r=await this.getHistorySnapshot(t.toolContent??"inline"),i=this.buildStateSnapshotReplay();if(e==="after_message"){let o=t.afterMessageId?cn(r,t.afterMessageId):-1;return o<0?{entries:[...i,...s(r)],appliedPolicy:"full"}:{entries:[...i,...s(r.slice(o+1))],appliedPolicy:"after_message"}}return{entries:[...i,...s(r)],appliedPolicy:"full"}}buildStateSnapshotReplay(){let e=[],t=this.sessionId,s=Date.now();if(this.title!==void 0&&this.title.length>0&&e.push({method:"session/update",params:{sessionId:t,update:{sessionUpdate:"session_info_update",title:this.title}},recordedAt:s}),this.currentModel!==void 0&&this.currentModel.length>0||this.agentAdvertisedModels.length>0){let i={sessionUpdate:"current_model_update"};this.currentModel!==void 0&&this.currentModel.length>0&&(i.currentModel=this.currentModel),this.agentAdvertisedModels.length>0&&(i.availableModels=[...this.agentAdvertisedModels]),e.push({method:"session/update",params:{sessionId:t,update:i},recordedAt:s})}this.currentMode!==void 0&&this.currentMode.length>0&&e.push({method:"session/update",params:{sessionId:t,update:{sessionUpdate:"current_mode_update",currentModeId:this.currentMode}},recordedAt:s});let r=this.mergedAvailableCommands();if(r.length>0&&e.push({method:"session/update",params:{sessionId:t,update:{sessionUpdate:"available_commands_update",availableCommands:r}},recordedAt:s}),this.agentAdvertisedModes.length>0&&e.push({method:"session/update",params:{sessionId:t,update:{sessionUpdate:"available_modes_update",availableModes:[...this.agentAdvertisedModes]}},recordedAt:s}),this.currentUsage!==void 0){let i=this.currentUsage,o={sessionUpdate:"usage_update"};if(typeof i.used=="number"&&(o.used=i.used),typeof i.size=="number"&&(o.size=i.size),typeof i.costAmount=="number"||typeof i.costCurrency=="string"){let a={};typeof i.costAmount=="number"&&(a.amount=i.costAmount),typeof i.costCurrency=="string"&&(a.currency=i.costCurrency),o.cost=a}Object.keys(o).length>1&&e.push({method:"session/update",params:{sessionId:t,update:o},recordedAt:s})}return e}replayPendingPermissions(e){for(let t of this.inFlightPermissions)t.addClient(e)}detach(e){let t=this.clients.get(e);t&&(this.clients.delete(e),this.updatedAt=Date.now(),this.broadcastClientDisconnected(t))}broadcastClientDisconnected(e){let t={clientId:e.clientId};e.clientInfo?.name&&(t.name=e.clientInfo.name),e.clientInfo?.version&&(t.version=e.clientInfo.version);let s={sessionUpdate:"client_disconnected",client:t,timestamp:new Date().toISOString()};for(let r of this.clients.values())r.connection.notify("session/update",{sessionId:this.sessionId,update:s}).catch(()=>{})}async prompt(e,t){let s=this.clients.get(e);if(!s)throw Y(new Error("client not attached"),I.SessionNotFound);if(this.closing)throw Y(new Error("session is closing; new prompts cannot be accepted"),I.SessionClosing);let r=Pe();if(this.maybeSeedTitleFromPrompt(t),this._firstPromptSeeded=!0,!(De((t??{})._meta).ancillary===!0)&&this._interactive!==!0){this._interactive=!0;for(let o of this.interactiveHandlers)try{o(!0)}catch{}}return this.enqueueUserPrompt(s,t,r)}broadcastPromptReceived(e){let t={clientId:e.originator.clientId};e.originator.name&&(t.name=e.originator.name),e.originator.version&&(t.version=e.originator.version),this.promptStartedAt=Date.now(),this.recordAndBroadcast("session/update",{sessionId:this.sessionId,update:{sessionUpdate:"prompt_received",messageId:e.messageId,prompt:e.prompt,sentBy:t}},e.clientId);let s=Te(e.prompt);s.length>0&&this.recordAndBroadcast("session/update",{sessionId:this.sessionId,update:{sessionUpdate:"user_message_chunk",content:{type:"text",text:s},_meta:{"hydra-acp":{compatFor:"prompt_received"}}}},e.clientId)}broadcastTurnComplete(e,t,s,r){let i=t&&typeof t=="object"&&"stopReason"in t&&typeof t.stopReason=="string"?t.stopReason:void 0,o={sessionUpdate:"turn_complete",messageId:Pe()};i!==void 0&&(o.stopReason=i);let a=this.amendInProgress;a&&s!==void 0&&a.cancelledMessageId===s&&(o._meta={"hydra-acp":{amended:{cancelledMessageId:a.cancelledMessageId,newMessageId:a.newMessageId}}}),this.promptStartedAt=void 0,s!==void 0&&i!==void 0&&this.recordTerminal(s,i),this.recordAndBroadcast("session/update",{sessionId:this.sessionId,update:o},r?void 0:e),a&&s!==void 0&&a.cancelledMessageId===s&&this.broadcastPromptAmended(a)}recordTerminal(e,t){for(this.recentlyTerminal.set(e,{stopReason:t,terminatedAt:Date.now()});this.recentlyTerminal.size>Eo;){let s=this.recentlyTerminal.keys().next().value;if(s===void 0)break;this.recentlyTerminal.delete(s)}}broadcastPromptAmended(e){let t=this.findUserEntry(e.newMessageId);if(!t)return;let s={sessionId:this.sessionId,cancelledMessageId:e.cancelledMessageId,newMessageId:e.newMessageId,prompt:t.prompt,originator:t.originator,amendedAt:Date.now()};this.broadcastQueueNotification("hydra-acp/prompt/amended",s)}findUserEntry(e){if(this.currentEntry?.messageId===e&&this.currentEntry.kind==="user")return this.currentEntry;let t=this.promptQueue.find(s=>s.messageId===e&&s.kind==="user");return t?.kind==="user"?t:void 0}visibleQueueDepth(){let e=this.currentEntry?.kind==="user"&&!this.currentEntry.cancelled?1:0;for(let t of this.promptQueue)t.kind==="user"&&!t.cancelled&&(e+=1);return e}broadcastQueueAdded(e,t){let s=this.visibleQueueDepth(),r=t?.position??Math.max(0,s-1),i={sessionId:this.sessionId,messageId:e.messageId,originator:e.originator,prompt:e.prompt,position:r,queueDepth:s,enqueuedAt:e.enqueuedAt};t?.amending!==void 0&&(i._meta={"hydra-acp":{amending:t.amending}}),this.broadcastQueueNotification("hydra-acp/prompt_queue/added",i)}broadcastQueueUpdated(e,t){this.broadcastQueueNotification("hydra-acp/prompt_queue/updated",{sessionId:this.sessionId,messageId:e,prompt:t})}broadcastQueueRemoved(e,t){this.broadcastQueueNotification("hydra-acp/prompt_queue/removed",{sessionId:this.sessionId,messageId:e,reason:t})}broadcastQueueNotification(e,t){for(let s of this.clients.values())s.connection.notify(e,t).catch(()=>{})}queueSnapshot(){let e=[],t=0;this.currentEntry?.kind==="user"&&!this.currentEntry.cancelled&&e.push({messageId:this.currentEntry.messageId,originator:this.currentEntry.originator,prompt:this.currentEntry.prompt,position:t++,enqueuedAt:this.currentEntry.enqueuedAt});for(let s of this.promptQueue)s.kind!=="user"||s.cancelled||e.push({messageId:s.messageId,originator:s.originator,prompt:s.prompt,position:t++,enqueuedAt:s.enqueuedAt});return e}async flushPersistWrites(){await this.queueWriteChain.catch(()=>{})}replayPersistedQueue(e){for(let t of e){let s={clientId:`hydra-resurrected_${t.messageId}`};t.originator.clientInfo.name!==void 0&&(s.name=t.originator.clientInfo.name),t.originator.clientInfo.version!==void 0&&(s.version=t.originator.clientInfo.version);let r={kind:"user",messageId:t.messageId,originator:s,clientId:s.clientId,prompt:t.prompt,enqueuedAt:t.enqueuedAt,cancelled:!1,resolve:()=>{},reject:()=>{}};this.promptQueue.push(r),this.broadcastQueueAdded(r)}this.drainQueue()}cancelQueuedPrompt(e){if(this.currentEntry?.messageId===e)return{cancelled:!1,reason:"already_running"};let t=this.promptQueue.findIndex(r=>r.messageId===e);if(t<0)return{cancelled:!1,reason:"not_found"};let s=this.promptQueue[t];return s.cancelled=!0,this.promptQueue.splice(t,1),s.kind==="user"&&(this.broadcastQueueRemoved(e,"cancelled"),this.persistRewrite()),this.amendInProgress?.newMessageId===e&&(this.amendInProgress=void 0),s.resolve({stopReason:"cancelled"}),{cancelled:!0,reason:"ok"}}updateQueuedPrompt(e,t){if(this.currentEntry?.messageId===e)return{updated:!1,reason:"already_running"};let s=this.promptQueue.find(r=>r.messageId===e);return!s||s.kind!=="user"?{updated:!1,reason:"not_found"}:(s.prompt=t,this.broadcastQueueUpdated(e,t),this.persistRewrite(),{updated:!0,reason:"ok"})}amendPrompt(e,t){let s=this.clients.get(e);if(!s)throw Y(new Error("client not attached"),I.SessionNotFound);let{targetMessageId:r,prompt:i,replaceQueue:o,onTargetCompleted:a}=t;if(this.currentEntry?.messageId===r&&this.currentEntry.kind==="user"&&!this.currentEntry.cancelled&&this.amendInProgress===void 0)return this.amendOnHead(s,i,r,o);let d=this.promptQueue.find(p=>p.messageId===r&&p.kind==="user");if(d&&d.kind==="user"&&!d.cancelled)return d.prompt=i,this.broadcastQueueUpdated(r,i),this.persistRewrite(),{amended:!0,reason:"ok",messageId:r};let c=this.recentlyTerminal.get(r);return c?c.stopReason==="cancelled"?{amended:!1,reason:"target_cancelled"}:a==="send_anyway"?{amended:!1,reason:"target_completed",messageId:this.enqueueAmendmentAsFollowUp(s,i)}:{amended:!1,reason:"target_completed"}:{amended:!1,reason:"target_not_found"}}amendOnHead(e,t,s,r){let i=Pe(),o={clientId:e.clientId};if(e.clientInfo?.name&&(o.name=e.clientInfo.name),e.clientInfo?.version&&(o.version=e.clientInfo.version),r){let d=[];for(let c of this.promptQueue){if(c.kind==="user"&&!c.cancelled){c.cancelled=!0,this.broadcastQueueRemoved(c.messageId,"cancelled"),c.resolve({stopReason:"cancelled"});continue}d.push(c)}this.promptQueue=d}let a={kind:"user",messageId:i,originator:o,clientId:e.clientId,prompt:t,enqueuedAt:Date.now(),cancelled:!1,wasAmend:!0,resolve:()=>{},reject:()=>{}};return this.promptQueue.unshift(a),this.persistRewrite(),this.broadcastQueueAdded(a,{amending:s,position:1}),this.amendInProgress={cancelledMessageId:s,newMessageId:i},this.cancelExtensionDispatch(s,"amended"),this.agent.connection.notify("session/cancel",{sessionId:this.upstreamSessionId}).catch(()=>{}),{amended:!0,reason:"ok",messageId:i}}enqueueAmendmentAsFollowUp(e,t){let s=Pe(),r={clientId:e.clientId};e.clientInfo?.name&&(r.name=e.clientInfo.name),e.clientInfo?.version&&(r.version=e.clientInfo.version);let i={kind:"user",messageId:s,originator:r,clientId:e.clientId,prompt:t,enqueuedAt:Date.now(),cancelled:!1,resolve:()=>{},reject:()=>{}};return this.promptQueue.push(i),this.persistRewrite(),this.broadcastQueueAdded(i),this.drainQueue(),s}async cancel(e){if(!this.clients.get(e))throw Y(new Error("client not attached"),I.SessionNotFound);this.lastCancelAt=Date.now(),this.currentEntry?.kind==="user"&&this.currentEntry.messageId!==void 0&&this.cancelExtensionDispatch(this.currentEntry.messageId,"cancelled"),await this.forwardRequest("session/cancel",{sessionId:this.sessionId},new Set,0,"notification")}addTransformer(e){let t=this.transformChain.findIndex(s=>s.name===e.name);t>=0?this.transformChain[t]=e:this.transformChain.push(e),e.intercepts.has("lifecycle:session.opened")&&e.connection.notify("hydra-acp/transformer/session_event",{event:"session.opened",sessionId:this.sessionId}).catch(()=>{})}async forwardRequest(e,t,s=new Set,r=0,i="request"){let o=this.rewriteForAgent(t);for(let a=r;a<this.transformChain.length;a++){let d=this.transformChain[a];if(s.has(d.name))continue;let c=`request:${e}`;if(!d.intercepts.has(c))continue;let p=`t_${Os()}`,l;try{l=await d.connection.request("hydra-acp/transformer/message",{token:p,phase:"request",method:e,direction:"client\u2192agent",sessionId:this.sessionId,envelope:o})}catch(f){this.logger?.warn(`transformer ${d.name} error on ${c}: ${f.message}`);continue}let u=l?.action??"continue";if(u==="stop")return i==="notification"?void 0:l?.payload??dn(e);if(u==="processing"){let f=a,g=o,m=new Set(s);return new Promise(y=>{let v=setTimeout(()=>{this.pendingClaims.delete(p)&&(this.broadcastQueueNotification("hydra-acp/transformer/abandoned_request",{sessionId:this.sessionId,token:p,transformerName:d.name}),this.forwardRequest(e,g,new Set([...m,d.name]),f+1,i).then(y).catch(()=>y(i==="notification"?void 0:dn(e))))},an);typeof v.unref=="function"&&v.unref(),this.pendingClaims.set(p,{resolve:y,timer:v,transformerName:d.name,method:e,envelope:g,chainIdx:f,originatedBy:m,side:"request",tailKind:i})})}s.add(d.name)}if(i==="notification"){await this.agent.connection.notify(e,o);return}return this.agent.connection.request(e,o)}dischargeClaim(e,t){let s=this.pendingClaims.get(e);return s?(clearTimeout(s.timer),this.pendingClaims.delete(e),s.resolve(t),!0):!1}keepAliveClaim(e,t){let s=this.pendingClaims.get(e);if(!s)return!1;clearTimeout(s.timer);let r=typeof t=="number"&&t>0?Math.min(t*1.5,1800*1e3):an,i=setTimeout(()=>{if(this.pendingClaims.delete(e))if(this.broadcastQueueNotification("hydra-acp/transformer/abandoned_request",{sessionId:this.sessionId,token:e,transformerName:s.transformerName}),s.side==="response")this.runResponseChain(s.envelope,new Set([...s.originatedBy,s.transformerName]),s.chainIdx+1).then(()=>s.resolve(void 0));else{let o=s.tailKind??"request";this.forwardRequest(s.method,s.envelope,new Set([...s.originatedBy,s.transformerName]),s.chainIdx+1,o).then(s.resolve).catch(()=>s.resolve(o==="notification"?void 0:dn(s.method)))}},r);return typeof i.unref=="function"&&i.unref(),s.timer=i,!0}async emitToChain(e,t,s){let r=this.transformChain.findIndex(a=>a.name===e),i=r>=0?r+1:0,o=new Set([e]);if(t==="session/update"){await this.runResponseChain(s,o,i);return}return this.forwardRequest(t,s,o,i)}rewriteForAgent(e){if(e&&typeof e=="object"&&!Array.isArray(e)){let t=e;if(t.sessionId===this.sessionId)return{...t,sessionId:this.upstreamSessionId}}return e}async close(e={}){if(!this.closed)return this.closeInFlight?this.closeInFlight:(this.closing=!0,this.closeInFlight=this.doClose(e),this.closeInFlight)}async doClose(e){this.logger?.info(`session ${this.sessionId} closing deleteRecord=${e.deleteRecord??!1}`),this.cancelIdleTimer(),await this.agent.kill().catch(()=>{}),this.markClosed({deleteRecord:e.deleteRecord??!1})}onClose(e){this.closeHandlers.push(e)}onTitleChange(e){this.titleHandlers.push(e)}retitle(e){return this.runTitleCommand(e)}retitleFromAgent(){return this.runTitleCommand("")}setTitle(e){let t=e.trim();if(!(!t||t===this.title)){this.title=t,this.recordAndBroadcast("session/update",{sessionId:this.sessionId,update:{sessionUpdate:"session_info_update",title:t,updatedAt:new Date().toISOString()}});for(let s of this.titleHandlers)try{s(t)}catch{}}}maybeSeedTitleFromPrompt(e){if(this.firstPromptSeeded)return;let s=Te((e??{}).prompt),r=un(s,200);r&&(r.startsWith("/")||(this._firstPromptSeeded=!0,this.setTitle(r)))}maybeApplyAgentModel(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="current_model_update")return!1;let r=fe(s.availableModels);r.length>0&&this.setAgentAdvertisedModels(r);let i=typeof s.currentModel=="string"?s.currentModel:typeof s.model=="string"?s.model:void 0;if(i===void 0)return!0;let o=i.trim();if(!o||o===this.currentModel)return!0;this.logger?.info(`live current_model_update: sessionId=${this.sessionId} ${JSON.stringify(this.currentModel)} \u2192 ${JSON.stringify(o)}`),this.currentModel=o;for(let a of this.modelHandlers)try{a(o)}catch{}return this.broadcastConfigOptions(),!0}maybeApplyAgentConfigOption(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="config_option_update")return!1;let r=s.configOptions;if(!Array.isArray(r))return!0;for(let i of r){if(!i||typeof i!="object")continue;let o=i;if(o.id==="model"){let a=fe(o.options);a.length>0&&this.setAgentAdvertisedModels(a);let d=o.currentValue;if(typeof d=="string"){let c=d.trim();c&&c!==this.currentModel&&(this.logger?.info(`live config_option_update(model): sessionId=${this.sessionId} ${JSON.stringify(this.currentModel)} \u2192 ${JSON.stringify(c)}`),this.applyModelChange(c))}}else if(o.id==="mode"){let a=ve(o.options);a.length>0&&this.setAgentAdvertisedModes(a);let d=o.currentValue;if(typeof d=="string"){let c=d.trim();c&&c!==this.currentMode&&(this.logger?.info(`live config_option_update(mode): sessionId=${this.sessionId} ${JSON.stringify(this.currentMode)} \u2192 ${JSON.stringify(c)}`),this.applyModeChange(c))}}else if(typeof o.id=="string"&&o.id.trim()){let a=o.id.trim(),d=fe(o.options);if(d.length>0){let c=ve(o.options),p=c.length>0?c.map(y=>({value:y.id,name:y.name??y.id})):d.map(y=>({value:y.modelId,name:y.name??y.modelId})),l=o.name,u=typeof l=="string"&&l.trim()?l.trim():a,f=o.currentValue,g=typeof f=="string"&&f.trim()?f.trim():p[0]?.value??"",m=this.agentAdvertisedConfigOptions.findIndex(y=>y.id===a);if(m>=0){let y={...this.agentAdvertisedConfigOptions[m]};y.options=p,typeof f=="string"&&f.trim()&&(y.currentValue=f.trim()),this.agentAdvertisedConfigOptions[m]=y}else this.agentAdvertisedConfigOptions.push({id:a,name:u,category:"other",type:"select",currentValue:g,options:p})}}}return!0}maybeApplyAgentMode(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="current_mode_update")return!1;let r=typeof s.currentModeId=="string"?s.currentModeId:typeof s.currentMode=="string"?s.currentMode:typeof s.mode=="string"?s.mode:void 0;if(r===void 0)return!0;let i=r.trim();if(!i||i===this.currentMode)return!0;this.logger?.info(`current_mode_update: sessionId=${this.sessionId} ${JSON.stringify(this.currentMode)} \u2192 ${JSON.stringify(i)}`),this.currentMode=i;for(let o of this.modeHandlers)try{o(i)}catch{}return this.broadcastConfigOptions(),!0}maybeApplyAgentUsage(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="usage_update")return!1;let r={...this._currentUsage??{}},i=!1;if(typeof s.used=="number"&&r.used!==s.used&&(r.used=s.used,i=!0),typeof s.size=="number"&&r.size!==s.size&&(r.size=s.size,i=!0),s.cost&&typeof s.cost=="object"){let a=s.cost;typeof a.amount=="number"&&r.costAmount!==a.amount&&(r.costAmount=a.amount,i=!0),typeof a.currency=="string"&&r.costCurrency!==a.currency&&(r.costCurrency=a.currency,i=!0)}if(!i)return!0;this._currentUsage=r;let o=this.currentUsage??r;for(let a of this.usageHandlers)try{a(o)}catch{}return!0}accumulateAndResetCost(){let e=this._currentUsage?.costAmount;if(!e)return;this.cumulativeCost+=e;let t={...this._currentUsage??{},cumulativeCost:this.cumulativeCost,costAmount:void 0};this._currentUsage=t;for(let s of this.usageHandlers)try{s(t)}catch{}}injectCumulativeCost(e){if(!this.cumulativeCost||!e||typeof e!="object")return e;let t=e;if(!t.update||typeof t.update!="object")return e;let s=t.update;if(s.sessionUpdate!=="usage_update"||!s.cost||typeof s.cost!="object")return e;let r=s.cost;return typeof r.amount!="number"?e:{...t,update:{...s,cost:{...r,amount:this.cumulativeCost+r.amount}}}}get totalUsage(){return this.currentUsage}setAgentAdvertisedCommands(e){if(_o(this.agentAdvertisedCommands,e)){this.broadcastMergedCommands();return}this.agentAdvertisedCommands=e;for(let t of this.agentCommandsHandlers)try{t(e)}catch{}this.broadcastMergedCommands()}setAgentAdvertisedModes(e){if($o(this.agentAdvertisedModes,e)){this.broadcastAvailableModes();return}this.agentAdvertisedModes=e;for(let t of this.agentModesHandlers)try{t(e)}catch{}this.broadcastAvailableModes()}setAgentAdvertisedModels(e){if(this.logger?.info(`setAgentAdvertisedModels: sessionId=${this.sessionId} currentModel=${JSON.stringify(this.currentModel)} newList=[${e.map(t=>t.modelId).join(",")}]`),Oo(this.agentAdvertisedModels,e)){this.broadcastAvailableModels();return}this.agentAdvertisedModels=e;for(let t of this.agentModelsHandlers)try{t(e)}catch{}this.broadcastAvailableModels()}onAgentCommandsChange(e){this.agentCommandsHandlers.push(e)}onAgentModesChange(e){this.agentModesHandlers.push(e)}onAgentModelsChange(e){this.agentModelsHandlers.push(e)}onModelChange(e){this.modelHandlers.push(e)}onModeChange(e){this.modeHandlers.push(e)}onInteractiveChange(e){this.interactiveHandlers.push(e)}onPriorityChange(e){this.priorityHandlers.push(e)}setPriority(e){let t=e===void 0||e<=0?void 0:Math.floor(e);if(t!==this._priority){this._priority=t,this.updatedAt=Date.now();for(let s of this.priorityHandlers)try{s(t)}catch{}}}applyModelChange(e){let t=e.trim();if(!t)return;if(t!==this.currentModel){this.logger?.info(`applyModelChange: sessionId=${this.sessionId} ${JSON.stringify(this.currentModel)} \u2192 ${JSON.stringify(t)}`),this.currentModel=t;for(let r of this.modelHandlers)try{r(t)}catch{}}let s={sessionUpdate:"current_model_update",currentModel:t};this.agentAdvertisedModels.length>0&&(s.availableModels=[...this.agentAdvertisedModels]),this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:s}),this.broadcastConfigOptions()}applyModeChange(e){let t=e.trim();if(!t)return;if(t!==this.currentMode){this.logger?.info(`applyModeChange: sessionId=${this.sessionId} ${JSON.stringify(this.currentMode)} \u2192 ${JSON.stringify(t)}`),this.currentMode=t;for(let r of this.modeHandlers)try{r(t)}catch{}}let s={sessionUpdate:"current_mode_update",currentModeId:t};this.agentAdvertisedModes.length>0&&(s.availableModes=[...this.agentAdvertisedModes]),this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:s}),this.broadcastConfigOptions()}buildConfigOptions(){let e=[];if(this.agentAdvertisedModels.length>0){let r=this.agentAdvertisedModels.map(o=>({value:o.modelId,name:o.name??o.modelId,...o.description!==void 0?{description:o.description}:{}})),i=this.currentModel&&r.some(o=>o.value===this.currentModel)?this.currentModel:r[0].value;e.push({id:"model",name:"Model",category:"model",type:"select",currentValue:i,options:r})}if(this.agentAdvertisedModes.length>0){let r=this.agentAdvertisedModes.map(o=>({value:o.id,name:o.name??o.id,...o.description!==void 0?{description:o.description}:{}})),i=this.currentMode&&r.some(o=>o.value===this.currentMode)?this.currentMode:r[0].value;e.push({id:"mode",name:"Session Mode",category:"mode",type:"select",currentValue:i,options:r})}let s=(this.availableAgentsFn?.()??[]).map(r=>({value:r.id,name:r.name??r.id,...r.description!==void 0?{description:r.description}:{}}));return s.some(r=>r.value===this.agentId)||s.unshift({value:this.agentId,name:this.agentId}),e.push({id:"agent",name:"Agent",category:"_hydra_agent",type:"select",currentValue:this.agentId,options:s}),e.push(...this.agentAdvertisedConfigOptions),e}broadcastConfigOptions(){this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"config_option_update",configOptions:this.buildConfigOptions()}})}mergeAgentOptionIntoEnvelope(e){if(!e||typeof e!="object")return e;let t=e;if(!t.update||typeof t.update!="object")return e;let s=t.update,r=Array.isArray(s.configOptions)?[...s.configOptions]:[];if(r.some(a=>a&&typeof a=="object"&&a.id==="agent"))return e;let o=this.buildConfigOptions().find(a=>a.id==="agent");return o?{...t,update:{...s,configOptions:[...r,o]}}:e}onUsageChange(e){this.usageHandlers.push(e)}mergedAvailableCommands(){let e=[{name:"hydra",description:"Hydra session command (kill, restart, title, agent <agent>)"},{name:"model",description:"Switch model; omit arg to list available models"},{name:"mode",description:"Switch mode; omit arg to list available modes"},{name:"sessions",description:"List all sessions"},{name:"help",description:"Show available commands"}];if(this.extensionCommands)for(let{name:t,command:s}of this.extensionCommands.list()){let r=s.verb?`hydra ${t} ${s.verb}`:`hydra ${t}`,o={name:s.argsHint?`${r} ${s.argsHint}`:r};if(s.description&&(o.description=s.description),e.push(o),t.startsWith("hydra-acp-")){let a=t.slice(10);if(a.length>0){let d=s.verb?`hydra ${a} ${s.verb}`:`hydra ${a}`,p={name:s.argsHint?`${d} ${s.argsHint}`:d};s.description&&(p.description=s.description),e.push(p)}}}return e.push(...this.agentAdvertisedCommands),e}agentOnlyAdvertisedCommands(){return[...this.agentAdvertisedCommands]}availableModes(){return[...this.agentAdvertisedModes]}availableModels(){return[...this.agentAdvertisedModels]}maybeApplyAgentSessionInfo(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="session_info_update"||typeof s.title!="string")return;let r=s.title.trim();if(!(!r||r===this.title)){this.title=r,this._firstPromptSeeded=!0;for(let i of this.titleHandlers)try{i(r)}catch{}}}async handleSlashCommand(e,t){let r=e.slice(6).trim().match(/^(\S+)(?:\s+([\s\S]*))?$/),i=r?.[1]??"",o=(r?.[2]??"").trim();if(i==="")return{stopReason:"end_turn"};let a=t!==void 0;if(mt.some(l=>l.verb===i))switch(i){case"title":return a?this.runTitleCommandInline(o):this.runTitleCommand(o);case"agent":return a?this.runAgentCommandInline(o):this.runAgentCommand(o);case"config":return this.handleConfigCommand(`/config ${o}`);case"kill":return this.runKillCommand();case"restart":return a?this.runRestartCommandInline():this.runRestartCommand();default:{let l=new Error(`no dispatcher for /hydra verb ${i}`);throw l.code=I.InternalError,l}}if(this.extensionCommands?.has(i))return a?this.runExtensionCommandInline(i,o,t):this.runExtensionCommand(i,o);let d=`hydra-acp-${i}`;if(this.extensionCommands?.has(d))return a?this.runExtensionCommandInline(d,o,t):this.runExtensionCommand(d,o);let c=mt.map(l=>l.verb);if(this.extensionCommands){let l=new Set;for(let{name:u}of this.extensionCommands.list())l.has(u)||(c.push(u),l.add(u))}let p=new Error(`unknown /hydra verb: ${i} (known: ${c.join(", ")})`);throw p.code=I.InvalidParams,p}runExtensionCommand(e,t){return this.enqueuePrompt(()=>this.runExtensionCommandInline(e,t,void 0))}async runExtensionCommandInline(e,t,s){let r=this.extensionCommands?.get(e);if(!r)return this.emitExtensionReply(`extension "${e}" is no longer connected`);let i=t.match(/^(\S+)(?:\s+([\s\S]*))?$/),o=i?.[1]??"",a=(i?.[2]??"").trim();if(!r.commands.some(g=>g.verb===o)){let g=r.commands.map(m=>m.verb).join(", ");return this.emitExtensionReply(`/hydra ${e}${o?` ${o}`:""}: unknown verb (known: ${g||"(none)"})`)}let d={sessionId:this.sessionId,verb:o,args:a};s!==void 0&&(d.messageId=s);let c=()=>{},p=new Promise(g=>{c=g});s!==void 0&&this.inFlightExtensionDispatches.set(s,{extensionName:e,connection:r.connection,cancel:g=>c({stopReason:"cancelled",reason:g})});let l,u=!1;try{let g=r.connection.request("hydra-acp/commands/invoke",d),m=await Promise.race([g,p]);if(m&&typeof m=="object"&&m.stopReason==="cancelled"&&"reason"in m)return u=!0,this.logger?.info(`extension ${e} ${o} cancelled (cancelExtensionDispatch); abandoning in-flight commands/invoke`),{stopReason:"cancelled"};l=m}catch(g){return this.emitExtensionReply(`${e} ${o}: ${g.message}`)}finally{s!==void 0&&!u&&this.inFlightExtensionDispatches.delete(s)}let f=l&&typeof l=="object"&&typeof l.text=="string"?l.text:"";return f.length>0?this.emitExtensionReply(f):{stopReason:"end_turn"}}cancelExtensionDispatch(e,t){let s=this.inFlightExtensionDispatches.get(e);if(!s)return!1;this.inFlightExtensionDispatches.delete(e);try{s.connection.notify("hydra-acp/commands/cancel",{sessionId:this.sessionId,messageId:e,reason:t})}catch(r){this.logger?.warn(`notify commands/cancel to ${s.extensionName} failed: ${r.message}`)}return s.cancel(t),!0}emitExtensionReply(e){return this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"agent_message_chunk",content:{type:"text",text:`
23
23
  ${e}
24
24
  `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"}}async handleSessionsCommand(){let e;if(!this.listSessions)e="_(session listing not available)_";else{let t=await this.listSessions();if(t.length===0)e="_(no sessions)_";else{let s=t.map(r=>{let i=r.sessionId.replace(/^hydra_session_/,"").slice(0,12),o=r.currentModel?` \xB7 ${r.currentModel}`:"",a=r.sessionId===this.sessionId?" \u25C0":"",d=r.title?` ${r.title}`:"";return`\`${i}\` ${r.cwd}${o}${a}${d}`});e=`Sessions (${t.length}):
25
25
  ${s.join(`
@@ -29,20 +29,40 @@ ${e}
29
29
  ${t.join(`
30
30
  `)}`:"_(no commands advertised)_";return this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"agent_message_chunk",content:{type:"text",text:`
31
31
  ${s}
32
- `},_meta:{"hydra-acp":{synthetic:!0}}}}),Promise.resolve({stopReason:"end_turn"})}async handleModelCommand(e){let t=e.slice(6).trim();if(t===""){let i=this.agentAdvertisedModels,o=this.currentModel,a;if(i.length===0)a=o?`Current model: ${o}`:"_(no models advertised yet)_";else{let d=o?i.some(f=>f.modelId===o):!0,c=i.map(f=>{let l=f.modelId===o?"\u25B6 ":" ",u=f.name&&f.name!==f.modelId?` ${f.name}`:"";return`${l}${f.modelId}${u}`});!d&&o&&c.unshift(`\u25B6 ${o}`),a=c.join(`
32
+ `},_meta:{"hydra-acp":{synthetic:!0}}}}),Promise.resolve({stopReason:"end_turn"})}async handleModelCommand(e){let t=e.slice(6).trim();if(t===""){let i=this.agentAdvertisedModels,o=this.currentModel,a;if(i.length===0)a=o?`Current model: ${o}`:"_(no models advertised yet)_";else{let d=o?i.some(p=>p.modelId===o):!0,c=i.map(p=>{let l=p.modelId===o?"\u25B6 ":" ",u=p.name&&p.name!==p.modelId?` ${p.name}`:"";return`${l}${p.modelId}${u}`});!d&&o&&c.unshift(`\u25B6 ${o}`),a=c.join(`
33
33
  `)}return this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"agent_message_chunk",content:{type:"text",text:`
34
34
  ${a}
35
- `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"}}let s=Re(t,this.agentAdvertisedModels),r=t;if(s.kind==="resolved")r=s.modelId;else if(s.kind==="ambiguous"||s.kind==="unknown"){let i=this.agentAdvertisedModels.map(a=>a.modelId).join(`
35
+ `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"}}let s=Ee(t,this.agentAdvertisedModels),r=t;if(s.kind==="resolved")r=s.modelId;else if(s.kind==="ambiguous"||s.kind==="unknown"){let i=this.agentAdvertisedModels.map(a=>a.modelId).join(`
36
36
  `),o=s.kind==="ambiguous"?`"${t}" matches multiple models: ${s.candidates.join(", ")}`:`"${t}" is not an available model`;return this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"agent_message_chunk",content:{type:"text",text:`
37
37
  ${o}.
38
38
  Available models:
39
39
  ${i}
40
- `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"}}return await this.forwardRequest("session/set_model",{sessionId:this.sessionId,modelId:r}),{stopReason:"end_turn"}}async handleModeCommand(e){let t=e.slice(5).trim();if(t===""){let r=this.agentAdvertisedModes,i=this.currentMode,o;if(r.length===0)o=i?`Current mode: ${i}`:"_(no modes advertised yet)_";else{let a=i?r.some(c=>c.id===i):!0,d=r.map(c=>{let f=c.id===i?"\u25B6 ":" ",l=c.name&&c.name!==c.id?` ${c.name}`:"";return`${f}${c.id}${l}`});!a&&i&&d.unshift(`\u25B6 ${i}`),o=d.join(`
40
+ `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"}}return await this.forwardRequest("session/set_model",{sessionId:this.sessionId,modelId:r}),{stopReason:"end_turn"}}async handleModeCommand(e){let t=e.slice(5).trim();if(t===""){let r=this.agentAdvertisedModes,i=this.currentMode,o;if(r.length===0)o=i?`Current mode: ${i}`:"_(no modes advertised yet)_";else{let a=i?r.some(c=>c.id===i):!0,d=r.map(c=>{let p=c.id===i?"\u25B6 ":" ",l=c.name&&c.name!==c.id?` ${c.name}`:"";return`${p}${c.id}${l}`});!a&&i&&d.unshift(`\u25B6 ${i}`),o=d.join(`
41
41
  `)}return this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"agent_message_chunk",content:{type:"text",text:`
42
42
  ${o}
43
- `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"}}let s=this.agentAdvertisedModes;if(s.length>0&&!s.some(r=>r.id===t)){let r=s.map(i=>i.id).join(", ");throw Y(new Error(`unknown mode: ${t} (known: ${r})`),I.InvalidParams)}return await this.forwardRequest("session/set_mode",{sessionId:this.sessionId,modeId:t}),this.applyModeChange(t),{stopReason:"end_turn"}}runTitleCommand(e){return this.enqueuePrompt(()=>this.runTitleCommandInline(e))}async runTitleCommandInline(e){return e?(this.setTitle(e),{stopReason:"end_turn"}):(this.scheduleSynopsisHook?.(),{stopReason:"end_turn"})}async runInternalPrompt(e){if(this.internalPromptCapture)throw new Error("internal prompt already in flight");let t={chunks:[]};this.internalPromptCapture=t;try{return await this.agent.connection.request("session/prompt",{sessionId:this.upstreamSessionId,prompt:[{type:"text",text:e}]}),t.chunks.join("")}finally{this.internalPromptCapture=void 0}}setAgent(e){return this.runAgentCommand(e)}runAgentCommand(e){return this.enqueuePrompt(()=>this.runAgentCommandInline(e))}async runAgentCommandInline(e){if(!e)throw Y(new Error("/hydra agent requires an agent id"),I.InvalidParams);if(e===this.agentId)throw Y(new Error(`already on agent ${e}`),I.InvalidParams);if(!this.spawnReplacementAgent)throw Y(new Error("agent switching not configured for this session"),I.InternalError);let t=this.spawnReplacementAgent,s=this.agentId,r=await this.buildSwitchTranscript(s),i=await t({agentId:e,cwd:this.cwd,agentArgs:this.agentArgs});this.accumulateAndResetCost(),this.wireAgent(i.agent);let o=this.agent;this.agent=i.agent,this.agentId=e,this.upstreamSessionId=i.upstreamSessionId,this.agentMeta=i.agentMeta,this.agentCapabilities=i.agentCapabilities,this.agentAdvertisedCommands=[],this.broadcastMergedCommands(),this.currentModel=i.initialModel,this.currentMode=i.initialMode,this.setAgentAdvertisedModels(i.initialModels??[]),this.setAgentAdvertisedModes(i.initialModes??[]),await o.kill().catch(()=>{}),r&&await this.runInternalPrompt(r).catch(()=>{}),this.broadcastAgentSwitch(s,e);let a={agentId:this.agentId,upstreamSessionId:this.upstreamSessionId};for(let d of this.agentChangeHandlers)try{d(a)}catch{}return{stopReason:"end_turn"}}async runKillCommand(){return await this.close({deleteRecord:!1}),{stopReason:"end_turn"}}runRestartCommand(){return this.enqueuePrompt(()=>this.runRestartCommandInline())}async runRestartCommandInline(){if(!this.spawnReplacementAgent)throw Y(new Error("agent restart not configured for this session"),I.InternalError);return await this.respawnAgent(),{stopReason:"end_turn"}}async forceCancel(){if(this.closed||this.closing)throw Y(new Error("session is closing"),I.SessionClosing);return this.lastCancelAt=0,this.forceCancelling=!0,await this.close({deleteRecord:!1}),{stopReason:"cancelled"}}async respawnAgent(){let e=this.spawnReplacementAgent,t=this.agentId,s=await this.buildSwitchTranscript(t),r=await e({agentId:t,cwd:this.cwd,agentArgs:this.agentArgs});this.accumulateAndResetCost(),this.wireAgent(r.agent);let i=this.agent;this.agent=r.agent,this.upstreamSessionId=r.upstreamSessionId,this.agentMeta=r.agentMeta,this.agentCapabilities=r.agentCapabilities,this.agentAdvertisedCommands=[],this.broadcastMergedCommands(),this.currentModel=r.initialModel,this.currentMode=r.initialMode,this.setAgentAdvertisedModels(r.initialModels??[]),this.setAgentAdvertisedModes(r.initialModes??[]),await i.kill().catch(()=>{}),s&&await this.runInternalPrompt(s).catch(()=>{}),this.broadcastAgentSwitch(t,t);let o={agentId:t,upstreamSessionId:this.upstreamSessionId};for(let a of this.agentChangeHandlers)try{a(o)}catch{}}async buildSwitchTranscript(e,t){let s=[],r=await this.getHistorySnapshot();for(let c of r){if(c.method!=="session/update")continue;let l=(c.params??{}).update;if(!l||l._meta?.["hydra-acp"]?.synthetic)continue;let p=l.sessionUpdate;if(p==="prompt_received"){let g=Pe(l.prompt);g&&s.push({speaker:"user",text:g})}else if(p==="agent_message_chunk"){let m=l.content?.text;m&&s.push({speaker:`agent: ${e}`,text:m})}else if(p==="tool_call"||p==="tool_call_update"){let g=l.status,m=l.title;(g==="completed"||g==="failed")&&s.push({speaker:"tool",text:`${m??"?"} ${g}`})}}if(s.length===0)return"";let i=[],o;for(let c of s)o&&o.speaker===c.speaker?o.text+=c.text:(o&&i.push(`<${o.speaker}>: ${o.text.trim()}`),o={speaker:c.speaker,text:c.text});o&&i.push(`<${o.speaker}>: ${o.text.trim()}`);let a=t?.intro??`You are taking over this conversation from ${e}. Below is the transcript so far.`,d=t?.followup??`Each line is prefixed with its speaker. Continue from where ${e} left off, responding to the user's most recent message.`;return[a,d,"","--- begin transcript ---",...i,"--- end transcript ---"].join(`
44
- `)}async seedFromImport(){await this.enqueuePrompt(async()=>{let e=await this.buildSwitchTranscript(this.agentId,{intro:"You are continuing a conversation that was imported from another hydra. Below is the transcript so far.",followup:"Each line is prefixed with its speaker. Treat this as context for the next user message; do not re-respond to earlier turns."});e&&await this.runInternalPrompt(e).catch(()=>{})})}broadcastAgentSwitch(e,t){this.recordAndBroadcast("session/update",{sessionId:this.sessionId,update:{sessionUpdate:"session_info_update",_meta:{"hydra-acp":{synthetic:!0,agentId:t}}}}),this.broadcastConfigOptions()}hasStreamBuffer(){return this.streamBuffer!==void 0}get streamPath(){return this.streamFilePath}openStream(e){if(this.streamBuffer!==void 0)throw new Error(`stream buffer already open for session ${this.sessionId}`);let s=(e.mode??"memory")==="file"&&e.filePathFor!==void 0?e.filePathFor(this.sessionId):void 0,r={};e.capacityBytes!==void 0&&(r.capacityBytes=e.capacityBytes),s!==void 0&&(r.filePath=s),e.fileCapBytes!==void 0&&(r.fileCapBytes=e.fileCapBytes),r.onFileCapReached=()=>{this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"stream_truncated",...s!==void 0?{filePath:s}:{},fileCapBytes:e.fileCapBytes}})};let i=new pt(r);this.streamBuffer=i,this.streamFilePath=s;let o={capacityBytes:i.capacity};return s!==void 0&&(o.filePath=s),e.fileCapBytes!==void 0&&(o.fileCapBytes=e.fileCapBytes),o}streamWrite(e,t){let s=this.requireStreamBuffer();if(e.length>0){let r=Buffer.from(e,"base64");s.append(r)}return t===!0&&s.close(),{writeCursor:s.writeCursorPos}}async streamRead(e,t,s){let r=this.requireStreamBuffer(),i=Math.max(0,Math.min(t??Me,Me)),o=r.read(e,i);if(o.bytes.length===0&&o.eof!==!0&&s!==void 0&&s>0){let d=await r.waitForData(o.nextCursor,s);(d==="data"||d==="eof")&&(o=r.read(o.nextCursor,i))}let a={bytes:o.bytes.toString("base64"),nextCursor:o.nextCursor};return o.gap!==void 0&&(a.gap=o.gap),o.eof===!0&&(a.eof=!0),a}streamTail(e){let s=this.requireStreamBuffer().tail(e);return{bytes:s.bytes.toString("base64"),startCursor:s.startCursor,endCursor:s.endCursor,truncated:s.truncated}}streamHead(e){let s=this.requireStreamBuffer().head(e);return{bytes:s.bytes.toString("base64"),startCursor:s.startCursor,endCursor:s.endCursor,truncated:s.truncated}}async streamWaitFor(e,t){return this.requireStreamBuffer().waitForData(e,t)}streamGrep(e){return this.requireStreamBuffer().grep(e)}streamInfo(){let e=this.requireStreamBuffer();return{writeCursor:e.writeCursorPos,oldestAvailable:e.oldestAvailable,capacity:e.capacity,closed:e.isClosed}}requireStreamBuffer(){if(this.streamBuffer===void 0){let e=new Error(`session ${this.sessionId} has no stream buffer; POST /v1/sessions/:id/stdin/open first`);throw e.code=I.StreamNotEnabled,e}return this.streamBuffer}markClosed(e){if(this.closed)return;this.closing=!0,this.closed=!0,this.cancelIdleTimer(),this.extensionCommandsUnsub&&(this.extensionCommandsUnsub(),this.extensionCommandsUnsub=void 0);for(let r of[...this.inFlightExtensionDispatches.keys()])this.cancelExtensionDispatch(r,"abandoned");this.currentEntry?.kind==="user"&&!this.recentlyTerminal.has(this.currentEntry.messageId)&&this.broadcastTurnComplete(this.currentEntry.clientId,{stopReason:"interrupted"},this.currentEntry.messageId,this.currentEntry.wasAmend),this.currentEntry=void 0;let t=this.promptQueue;this.promptQueue=[];for(let r of t){r.cancelled=!0,r.kind==="user"&&this.broadcastQueueRemoved(r.messageId,"abandoned");try{r.resolve({stopReason:"cancelled"})}catch{}}this.notifyChain("session.closed",{});let s=this.sessionId;this.queueWriteChain=this.queueWriteChain.catch(()=>{}).then(()=>$s(s).catch(()=>{}));for(let r of this.clients.values())r.connection.notify("hydra-acp/session/closed",{sessionId:this.sessionId}).catch(()=>{});if(this.clients.clear(),this.streamBuffer!==void 0){let r=this.streamBuffer,i=this.streamFilePath;this.streamBuffer=void 0,this.streamFilePath=void 0,r.close(),i!==void 0&&r.drainFileWrites().then(()=>Bs.unlink(i).catch(()=>{}))}for(let r of this.closeHandlers)r(e)}get lastActivityAt(){return this.lastRecordedAt}scheduleIdleCheck(){if(this.closed||this.idleTimeoutMs<=0)return;let e=this.lastActivityAt+this.idleTimeoutMs;this.armIdleTimer(Math.max(0,e-Date.now()))}armIdleTimer(e){this.idleTimer&&clearTimeout(this.idleTimer),this.idleTimer=setTimeout(()=>{this.idleTimer=void 0,this.checkIdle()},e),typeof this.idleTimer.unref=="function"&&this.idleTimer.unref()}checkIdle(){if(this.closed||this.idleTimeoutMs<=0)return;if(this.turnStartedAt!==void 0||this.inFlightPermissions.size>0||this.promptQueue.length>0){this.armIdleTimer(this.idleTimeoutMs);return}let e=Date.now()-this.lastActivityAt;if(e<this.idleTimeoutMs){this.armIdleTimer(this.idleTimeoutMs-e);return}let t=this.firstPromptSeeded?{deleteRecord:!1}:{deleteRecord:!0},s=Math.round(e/1e3);this.logger?.info(`session ${this.sessionId} idle timeout fired after ${s}s (window=${Math.round(this.idleTimeoutMs/1e3)}s) \u2014 closing`),this.close(t).catch(()=>{})}cancelIdleTimer(){this.idleTimer&&(clearTimeout(this.idleTimer),this.idleTimer=void 0),this.cancelIdleEventTimer()}scheduleIdleEvent(){this.closed||this.idleEventTimeoutMs<=0||this.transformChain.length===0||(this.idleEventTimer&&clearTimeout(this.idleEventTimer),this.idleEventTimer=setTimeout(()=>{this.idleEventTimer=void 0,this.notifyChain("session.idle",{})},this.idleEventTimeoutMs),typeof this.idleEventTimer.unref=="function"&&this.idleEventTimer.unref())}cancelIdleEventTimer(){this.idleEventTimer&&(clearTimeout(this.idleEventTimer),this.idleEventTimer=void 0)}notifyChain(e,t){let s=`lifecycle:${e}`;for(let r of this.transformChain)r.intercepts.has(s)&&r.connection.notify("hydra-acp/transformer/session_event",{event:e,sessionId:this.sessionId,payload:t}).catch(()=>{})}rewriteForClient(e){if(e&&typeof e=="object"&&!Array.isArray(e)){let t=e;if(t.sessionId===this.upstreamSessionId)return{...t,sessionId:this.sessionId}}return e}recordAndBroadcast(e,t,s){let r=this.rewriteForClient(t),i=!Po(e,r),o=i?jo(e,r):r;if(i){let a={method:e,params:o,recordedAt:Date.now()};if(this.lastRecordedAt=a.recordedAt,this.appendCount+=1,this.historyStore){let d=this.historyStore;d.append(this.sessionId,a).catch(()=>{}),this.appendCount>=this.compactEvery&&(this.appendCount=0,d.compact(this.sessionId,this.historyMaxEntries).catch(()=>{}))}for(let d of this.broadcastHandlers)try{d(a)}catch{}this.scheduleIdleCheck(),this.scheduleIdleEvent()}this.updatedAt=Date.now();for(let a of this.clients.values())s&&a.clientId===s||a.connection.notify(e,o).catch(()=>{})}async requestPermissionFromClients(e){return this.handlePermissionRequest(e)}async handlePermissionRequest(e){let t=[...this.clients.values()];if(t.length===0)throw Y(new Error("no clients attached to handle permission request"),I.PermissionDenied);let s=this.rewriteForClient(e),r=Ho(s);return new Promise((i,o)=>{let a=!1,d=[],c={addClient:u};this.inFlightPermissions.add(c);let f=this.sessionId,l=p=>{a||(a=!0,this.inFlightPermissions.delete(c),p())};function u(p){if(a)return;let g=p.connection.request("session/request_permission",s);d.push({client:p}),g.then(m=>{l(()=>{let y=Bo({toolCallId:r,result:m,resolver:p});for(let v of d)v.client.clientId!==p.clientId&&v.client.connection.notify("session/update",{sessionId:f,update:y}).catch(()=>{});i(m)})}).catch(m=>{l(()=>o(m))})}for(let p of t)u(p)})}async enqueuePrompt(e){return new Promise((t,s)=>{let r={kind:"internal",messageId:Ee(),enqueuedAt:Date.now(),cancelled:!1,task:e,resolve:t,reject:s};this.promptQueue.push(r),this.drainQueue()})}async enqueueUserPrompt(e,t,s){let r=(t??{}).prompt??[],i={clientId:e.clientId};e.clientInfo?.name&&(i.name=e.clientInfo.name),e.clientInfo?.version&&(i.version=e.clientInfo.version);let o=this.parseQueuePosition(t);return new Promise((a,d)=>{let c={kind:"user",messageId:s,originator:i,clientId:e.clientId,prompt:r,enqueuedAt:Date.now(),cancelled:!1,resolve:a,reject:d},f=this.insertEntryAt(c,o);this.persistRewrite(),this.broadcastQueueAdded(c,{position:f}),this.drainQueue()})}parseQueuePosition(e){if(!e||typeof e!="object")return"tail";let t=e._meta;if(!t||typeof t!="object"||Array.isArray(t))return"tail";let s=t["hydra-acp"];if(!s||typeof s!="object"||Array.isArray(s))return"tail";let r=s.queuePosition;if(r==="head"||r==="tail")return r;if(r&&typeof r=="object"&&!Array.isArray(r)){let i=r.afterMessageId;if(typeof i=="string"&&i.length>0)return{afterMessageId:i}}return"tail"}insertEntryAt(e,t){let s=this.currentEntry?.kind==="user"&&!this.currentEntry.cancelled?1:0;if(t==="head")return this.promptQueue.unshift(e),s;if(typeof t=="object"){let r=t.afterMessageId,i=this.promptQueue.findIndex(o=>o.messageId===r);if(i>=0)return this.promptQueue.splice(i+1,0,e),s+i+1}return this.promptQueue.push(e),s+this.promptQueue.length-1}persistRewrite(){let e=[];for(let s of this.promptQueue)s.kind!=="user"||s.cancelled||e.push(this.persistedFromEntry(s));let t=this.sessionId;this.queueWriteChain=this.queueWriteChain.catch(()=>{}).then(()=>Le(t,e).catch(()=>{}))}persistedFromEntry(e){return{messageId:e.messageId,originator:{clientInfo:{...e.originator.name!==void 0?{name:e.originator.name}:{},...e.originator.version!==void 0?{version:e.originator.version}:{}}},prompt:e.prompt,enqueuedAt:e.enqueuedAt}}async drainQueue(){if(!this.promptInFlight){this.promptInFlight=!0,await new Promise(e=>setImmediate(e));try{for(;this.promptQueue.length>0&&!this.closing;){let e=this.promptQueue.shift();if(!e)break;if(!e.cancelled){this.currentEntry=e,e.kind==="user"&&this.persistRewrite(),e.kind==="user"&&this.broadcastQueueRemoved(e.messageId,"started");try{let t=await this.runQueueEntry(e);e.resolve(t),await new Promise(s=>setImmediate(s))}catch(t){e.reject(t)}finally{this.currentEntry=void 0}}}}finally{this.promptInFlight=!1}}}async runQueueEntry(e){if(e.kind==="internal")return e.task();this.broadcastPromptReceived(e);let t=Pe(e.prompt).replace(/\s+$/,""),s=!t.includes(`
45
- `)&&t.startsWith("/")?t.split(/\s+/)[0]:"";if(s==="/model"||s==="/mode"||s==="/sessions"||s==="/help"||s==="/hydra"){let i;return s==="/sessions"?i=await this.handleSessionsCommand():s==="/help"?i=await this.handleHelpCommand():s==="/mode"?i=await this.handleModeCommand(t):s==="/model"?i=await this.handleModelCommand(t):i=await this.handleSlashCommand(t,e.messageId),this.closed||this.broadcastTurnComplete(e.clientId,i,e.messageId,e.wasAmend),this.clearAmendIfMatches(e.messageId),i}let r;try{r=await this.agent.connection.request("session/prompt",{sessionId:this.upstreamSessionId,prompt:e.prompt})}catch(i){if(this.forceCancelling)return this.clearAmendIfMatches(e.messageId),{stopReason:"cancelled"};throw this.closed||this.broadcastTurnComplete(e.clientId,{stopReason:"error"},e.messageId,e.wasAmend),this.clearAmendIfMatches(e.messageId),i}return this.closed||this.broadcastTurnComplete(e.clientId,r,e.messageId,e.wasAmend),this.clearAmendIfMatches(e.messageId),r}clearAmendIfMatches(e){this.amendInProgress?.cancelledMessageId===e&&(this.amendInProgress=void 0)}};function Y(n,e){return n.code=e,n}var Eo=new Set(["session_info_update","current_model_update","current_mode_update","available_commands_update","available_modes_update","usage_update","config_option_update"]);function Po(n,e){if(n!=="session/update")return!1;let s=(e??{}).update?.sessionUpdate;return typeof s=="string"&&Eo.has(s)}function To(n,e){if(n.length!==e.length)return!1;for(let t=0;t<n.length;t++)if(n[t]?.name!==e[t]?.name||n[t]?.description!==e[t]?.description)return!1;return!0}function _o(n,e){if(n.length!==e.length)return!1;for(let t=0;t<n.length;t++)if(n[t]?.id!==e[t]?.id||n[t]?.name!==e[t]?.name||n[t]?.description!==e[t]?.description)return!1;return!0}function $o(n,e){if(n.length!==e.length)return!1;for(let t=0;t<n.length;t++)if(n[t]?.modelId!==e[t]?.modelId||n[t]?.name!==e[t]?.name||n[t]?.description!==e[t]?.description)return!1;return!0}function we(n){if(!Array.isArray(n))return[];let e=[];for(let t of n){if(!t||typeof t!="object"||Array.isArray(t))continue;let s=t,r=typeof s.modelId=="string"&&s.modelId.trim()||typeof s.value=="string"&&s.value.trim()||typeof s.id=="string"&&s.id.trim()||void 0;if(!r)continue;let i={modelId:r};typeof s.name=="string"&&s.name.length>0&&(i.name=s.name),typeof s.description=="string"&&s.description.length>0&&(i.description=s.description),e.push(i)}return e}function Te(n){if(!Array.isArray(n))return[];let e=[];for(let t of n){if(!t||typeof t!="object"||Array.isArray(t))continue;let s=t,r=typeof s.value=="string"&&s.value.trim()||typeof s.id=="string"&&s.id.trim()||void 0;if(!r)continue;let i={id:r};typeof s.name=="string"&&s.name.length>0&&(i.name=s.name),typeof s.description=="string"&&s.description.length>0&&(i.description=s.description),e.push(i)}return e}function Fo(n){let t=(n??{}).update??{};if(t.sessionUpdate!=="available_modes_update")return null;let s=t.availableModes;if(!Array.isArray(s))return[];let r=[];for(let i of s){if(!i||typeof i!="object")continue;let o=i;if(typeof o.id!="string"||o.id.length===0)continue;let a={id:o.id};typeof o.name=="string"&&(a.name=o.name),typeof o.description=="string"&&(a.description=o.description),r.push(a)}return r}function Oo(n,e){let s=(e??{}).update??{};if(s.sessionUpdate!=="agent_message_chunk")return;let r=s.content??{};typeof r.text=="string"&&n.chunks.push(r.text)}function No(n){let t=(n??{}).update??{};if(t.sessionUpdate!=="available_commands_update")return null;let s=t.availableCommands??t.commands;if(!Array.isArray(s))return[];let r=[];for(let i of s){if(!i||typeof i!="object")continue;let o=i;if(typeof o.name!="string"||o.name.length===0)continue;let a={name:o.name};typeof o.description=="string"&&(a.description=o.description),r.push(a)}return r}function jo(n,e){if(n!=="session/update"||!e||typeof e!="object")return e;let t=e;return!t.update||typeof t.update!="object"||Array.isArray(t.update)||typeof t.update.messageId=="string"?e:{...e,update:{...t.update,messageId:Ee()}}}function cn(n,e){for(let t=0;t<n.length;t++){let s=n[t];if(!s||s.method!=="session/update")continue;if(s.params?.update?.messageId===e)return t}return-1}function Ho(n){if(!n||typeof n!="object")return;let e=n.toolCall;if(!e||typeof e!="object")return;let t=e.toolCallId;return typeof t=="string"?t:void 0}function Bo(n){let e=Uo(n.result),t={sessionUpdate:"permission_resolved"};return n.toolCallId!==void 0&&(t.toolCallId=n.toolCallId),e&&(t.outcome=e,e.kind==="selected"&&typeof e.optionId=="string"&&(t.chosenOptionId=e.optionId)),t.resolvedBy=Do(n.resolver),t}function Uo(n){if(!n||typeof n!="object")return;let e=n.outcome;if(!e||typeof e!="object")return;let t=e.kind;if(typeof t!="string")return;let s={kind:t},r=e.optionId;typeof r=="string"&&(s.optionId=r);let i=e.reason;return typeof i=="string"&&(s.reason=i),s}function Do(n){let e={clientId:n.clientId};return n.clientInfo?.name&&(e.name=n.clientInfo.name),n.clientInfo?.version&&(e.version=n.clientInfo.version),e}function Pe(n){return typeof n=="string"?n:Array.isArray(n)?n.map(e=>e&&typeof e=="object"&&typeof e.text=="string"?e.text:"").join(""):""}function dn(n){return n==="session/prompt"?{stopReason:"stopped"}:{}}function un(n,e){for(let t of n.split(/\r?\n/)){let s=t.trim();if(s)return s.length>e?`${s.slice(0,e)}\u2026`:s}}import*as _e from"fs/promises";import*as Ju from"path";import{customAlphabet as Jo}from"nanoid";import{z as A}from"zod";import{z as X}from"zod";var qe=X.object({goal:X.string().optional(),outcome:X.string().optional(),files_touched:X.array(X.string()).optional(),tools_used:X.array(X.string()).optional(),rejected_approaches:X.array(X.string()).optional(),open_threads:X.array(X.string()).optional()}),Lo=200,ln=`Reply with ONLY a JSON object with exactly these keys, no prose, no markdown, no code fences:
43
+ `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"}}let s=this.agentAdvertisedModes;if(s.length>0&&!s.some(r=>r.id===t)){let r=s.map(i=>i.id).join(", ");throw Y(new Error(`unknown mode: ${t} (known: ${r})`),I.InvalidParams)}return await this.forwardRequest("session/set_mode",{sessionId:this.sessionId,modeId:t}),this.applyModeChange(t),{stopReason:"end_turn"}}async handleConfigCommand(e){let t=e.slice(7).trim().match(/^(\S+)(?:\s+([\s\S]*))?$/),s=t?.[1],r=t?.[2]??"";if(!s){let a=this.buildConfigOptions();if(a.length===0)return this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"agent_message_chunk",content:{type:"text",text:`
44
+ _(no config options advertised yet)_
45
+ `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"};let d=a.map(c=>Fs(c));return this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"agent_message_chunk",content:{type:"text",text:`
46
+ ${d.join(`
47
+
48
+ `)}
49
+ `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"}}let i=this.buildConfigOptions(),o=i.find(a=>a.id===s);if(!o){let a=i.map(d=>d.id).join(`
50
+ `);return this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"agent_message_chunk",content:{type:"text",text:`
51
+ "${s}" is not a known config option.
52
+ Available options:
53
+ ${a}
54
+ `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"}}if(!r)return this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"agent_message_chunk",content:{type:"text",text:`
55
+ ${Fs(o)}
56
+ `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"};if(!o.options.some(a=>a.value===r)){let a=o.options.map(d=>d.value).join(`
57
+ `);return this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"agent_message_chunk",content:{type:"text",text:`
58
+ "${r}" is not a valid value for "${s}".
59
+ Valid values:
60
+ ${a}
61
+ `},_meta:{"hydra-acp":{synthetic:!0}}}}),{stopReason:"end_turn"}}return await this.forwardRequest("session/set_config_option",{sessionId:this.sessionId,configId:s,value:r}),{stopReason:"end_turn"}}runTitleCommand(e){return this.enqueuePrompt(()=>this.runTitleCommandInline(e))}async runTitleCommandInline(e){return e?(this.setTitle(e),{stopReason:"end_turn"}):(this.scheduleSynopsisHook?.(),{stopReason:"end_turn"})}async runInternalPrompt(e){if(this.internalPromptCapture)throw new Error("internal prompt already in flight");let t={chunks:[]};this.internalPromptCapture=t;try{return await this.agent.connection.request("session/prompt",{sessionId:this.upstreamSessionId,prompt:[{type:"text",text:e}]}),t.chunks.join("")}finally{this.internalPromptCapture=void 0}}setAgent(e){return this.runAgentCommand(e)}runAgentCommand(e){return this.enqueuePrompt(()=>this.runAgentCommandInline(e))}async runAgentCommandInline(e){if(!e)throw Y(new Error("/hydra agent requires an agent id"),I.InvalidParams);if(e===this.agentId)throw Y(new Error(`already on agent ${e}`),I.InvalidParams);if(!this.spawnReplacementAgent)throw Y(new Error("agent switching not configured for this session"),I.InternalError);let t=this.spawnReplacementAgent,s=this.agentId,r=await this.buildSwitchTranscript(s),i=await t({agentId:e,cwd:this.cwd,agentArgs:this.agentArgs});this.accumulateAndResetCost(),this.wireAgent(i.agent);let o=this.agent;this.agent=i.agent,this.agentId=e,this.upstreamSessionId=i.upstreamSessionId,this.agentMeta=i.agentMeta,this.agentCapabilities=i.agentCapabilities,this.agentAdvertisedCommands=[],this.broadcastMergedCommands(),this.currentModel=i.initialModel,this.currentMode=i.initialMode,this.setAgentAdvertisedModels(i.initialModels??[]),this.setAgentAdvertisedModes(i.initialModes??[]),await o.kill().catch(()=>{}),r&&await this.runInternalPrompt(r).catch(()=>{}),this.broadcastAgentSwitch(s,e);let a={agentId:this.agentId,upstreamSessionId:this.upstreamSessionId};for(let d of this.agentChangeHandlers)try{d(a)}catch{}return{stopReason:"end_turn"}}async runKillCommand(){return await this.close({deleteRecord:!1}),{stopReason:"end_turn"}}runRestartCommand(){return this.enqueuePrompt(()=>this.runRestartCommandInline())}async runRestartCommandInline(){if(!this.spawnReplacementAgent)throw Y(new Error("agent restart not configured for this session"),I.InternalError);return await this.respawnAgent(),{stopReason:"end_turn"}}async forceCancel(){if(this.closed||this.closing)throw Y(new Error("session is closing"),I.SessionClosing);return this.lastCancelAt=0,this.forceCancelling=!0,await this.close({deleteRecord:!1}),{stopReason:"cancelled"}}async respawnAgent(){let e=this.spawnReplacementAgent,t=this.agentId,s=await this.buildSwitchTranscript(t),r=await e({agentId:t,cwd:this.cwd,agentArgs:this.agentArgs});this.accumulateAndResetCost(),this.wireAgent(r.agent);let i=this.agent;this.agent=r.agent,this.upstreamSessionId=r.upstreamSessionId,this.agentMeta=r.agentMeta,this.agentCapabilities=r.agentCapabilities,this.agentAdvertisedCommands=[],this.broadcastMergedCommands(),this.currentModel=r.initialModel,this.currentMode=r.initialMode,this.setAgentAdvertisedModels(r.initialModels??[]),this.setAgentAdvertisedModes(r.initialModes??[]),await i.kill().catch(()=>{}),s&&await this.runInternalPrompt(s).catch(()=>{}),this.broadcastAgentSwitch(t,t);let o={agentId:t,upstreamSessionId:this.upstreamSessionId};for(let a of this.agentChangeHandlers)try{a(o)}catch{}}async buildSwitchTranscript(e,t){let s=[],r=await this.getHistorySnapshot();for(let c of r){if(c.method!=="session/update")continue;let l=(c.params??{}).update;if(!l||l._meta?.["hydra-acp"]?.synthetic)continue;let f=l.sessionUpdate;if(f==="prompt_received"){let g=Te(l.prompt);g&&s.push({speaker:"user",text:g})}else if(f==="agent_message_chunk"){let m=l.content?.text;m&&s.push({speaker:`agent: ${e}`,text:m})}else if(f==="tool_call"||f==="tool_call_update"){let g=l.status,m=l.title;(g==="completed"||g==="failed")&&s.push({speaker:"tool",text:`${m??"?"} ${g}`})}}if(s.length===0)return"";let i=[],o;for(let c of s)o&&o.speaker===c.speaker?o.text+=c.text:(o&&i.push(`<${o.speaker}>: ${o.text.trim()}`),o={speaker:c.speaker,text:c.text});o&&i.push(`<${o.speaker}>: ${o.text.trim()}`);let a=t?.intro??`You are taking over this conversation from ${e}. Below is the transcript so far.`,d=t?.followup??`Each line is prefixed with its speaker. Continue from where ${e} left off, responding to the user's most recent message.`;return[a,d,"","--- begin transcript ---",...i,"--- end transcript ---"].join(`
62
+ `)}async seedFromImport(){await this.enqueuePrompt(async()=>{let e=await this.buildSwitchTranscript(this.agentId,{intro:"You are continuing a conversation that was imported from another hydra. Below is the transcript so far.",followup:"Each line is prefixed with its speaker. Treat this as context for the next user message; do not re-respond to earlier turns."});e&&await this.runInternalPrompt(e).catch(()=>{})})}broadcastAgentSwitch(e,t){this.recordAndBroadcast("session/update",{sessionId:this.sessionId,update:{sessionUpdate:"session_info_update",_meta:{"hydra-acp":{synthetic:!0,agentId:t}}}}),this.broadcastConfigOptions()}hasStreamBuffer(){return this.streamBuffer!==void 0}get streamPath(){return this.streamFilePath}openStream(e){if(this.streamBuffer!==void 0)throw new Error(`stream buffer already open for session ${this.sessionId}`);let s=(e.mode??"memory")==="file"&&e.filePathFor!==void 0?e.filePathFor(this.sessionId):void 0,r={};e.capacityBytes!==void 0&&(r.capacityBytes=e.capacityBytes),s!==void 0&&(r.filePath=s),e.fileCapBytes!==void 0&&(r.fileCapBytes=e.fileCapBytes),r.onFileCapReached=()=>{this.recordAndBroadcast("session/update",{sessionId:this.upstreamSessionId,update:{sessionUpdate:"stream_truncated",...s!==void 0?{filePath:s}:{},fileCapBytes:e.fileCapBytes}})};let i=new pt(r);this.streamBuffer=i,this.streamFilePath=s;let o={capacityBytes:i.capacity};return s!==void 0&&(o.filePath=s),e.fileCapBytes!==void 0&&(o.fileCapBytes=e.fileCapBytes),o}streamWrite(e,t){let s=this.requireStreamBuffer();if(e.length>0){let r=Buffer.from(e,"base64");s.append(r)}return t===!0&&s.close(),{writeCursor:s.writeCursorPos}}async streamRead(e,t,s){let r=this.requireStreamBuffer(),i=Math.max(0,Math.min(t??Re,Re)),o=r.read(e,i);if(o.bytes.length===0&&o.eof!==!0&&s!==void 0&&s>0){let d=await r.waitForData(o.nextCursor,s);(d==="data"||d==="eof")&&(o=r.read(o.nextCursor,i))}let a={bytes:o.bytes.toString("base64"),nextCursor:o.nextCursor};return o.gap!==void 0&&(a.gap=o.gap),o.eof===!0&&(a.eof=!0),a}streamTail(e){let s=this.requireStreamBuffer().tail(e);return{bytes:s.bytes.toString("base64"),startCursor:s.startCursor,endCursor:s.endCursor,truncated:s.truncated}}streamHead(e){let s=this.requireStreamBuffer().head(e);return{bytes:s.bytes.toString("base64"),startCursor:s.startCursor,endCursor:s.endCursor,truncated:s.truncated}}async streamWaitFor(e,t){return this.requireStreamBuffer().waitForData(e,t)}streamGrep(e){return this.requireStreamBuffer().grep(e)}streamInfo(){let e=this.requireStreamBuffer();return{writeCursor:e.writeCursorPos,oldestAvailable:e.oldestAvailable,capacity:e.capacity,closed:e.isClosed}}requireStreamBuffer(){if(this.streamBuffer===void 0){let e=new Error(`session ${this.sessionId} has no stream buffer; POST /v1/sessions/:id/stdin/open first`);throw e.code=I.StreamNotEnabled,e}return this.streamBuffer}markClosed(e){if(this.closed)return;this.closing=!0,this.closed=!0,this.cancelIdleTimer(),this.extensionCommandsUnsub&&(this.extensionCommandsUnsub(),this.extensionCommandsUnsub=void 0);for(let r of[...this.inFlightExtensionDispatches.keys()])this.cancelExtensionDispatch(r,"abandoned");this.currentEntry?.kind==="user"&&!this.recentlyTerminal.has(this.currentEntry.messageId)&&this.broadcastTurnComplete(this.currentEntry.clientId,{stopReason:"interrupted"},this.currentEntry.messageId,this.currentEntry.wasAmend),this.currentEntry=void 0;let t=this.promptQueue;this.promptQueue=[];for(let r of t){r.cancelled=!0,r.kind==="user"&&this.broadcastQueueRemoved(r.messageId,"abandoned");try{r.resolve({stopReason:"cancelled"})}catch{}}this.notifyChain("session.closed",{});let s=this.sessionId;this.queueWriteChain=this.queueWriteChain.catch(()=>{}).then(()=>$s(s).catch(()=>{}));for(let r of this.clients.values())r.connection.notify("hydra-acp/session/closed",{sessionId:this.sessionId}).catch(()=>{});if(this.clients.clear(),this.streamBuffer!==void 0){let r=this.streamBuffer,i=this.streamFilePath;this.streamBuffer=void 0,this.streamFilePath=void 0,r.close(),i!==void 0&&r.drainFileWrites().then(()=>Us.unlink(i).catch(()=>{}))}for(let r of this.closeHandlers)r(e)}get lastActivityAt(){return this.lastRecordedAt}scheduleIdleCheck(){if(this.closed||this.idleTimeoutMs<=0)return;let e=this.lastActivityAt+this.idleTimeoutMs;this.armIdleTimer(Math.max(0,e-Date.now()))}armIdleTimer(e){this.idleTimer&&clearTimeout(this.idleTimer),this.idleTimer=setTimeout(()=>{this.idleTimer=void 0,this.checkIdle()},e),typeof this.idleTimer.unref=="function"&&this.idleTimer.unref()}checkIdle(){if(this.closed||this.idleTimeoutMs<=0)return;if(this.turnStartedAt!==void 0||this.inFlightPermissions.size>0||this.promptQueue.length>0){this.armIdleTimer(this.idleTimeoutMs);return}let e=Date.now()-this.lastActivityAt;if(e<this.idleTimeoutMs){this.armIdleTimer(this.idleTimeoutMs-e);return}let t=this.firstPromptSeeded?{deleteRecord:!1}:{deleteRecord:!0},s=Math.round(e/1e3);this.logger?.info(`session ${this.sessionId} idle timeout fired after ${s}s (window=${Math.round(this.idleTimeoutMs/1e3)}s) \u2014 closing`),this.close(t).catch(()=>{})}cancelIdleTimer(){this.idleTimer&&(clearTimeout(this.idleTimer),this.idleTimer=void 0),this.cancelIdleEventTimer()}scheduleIdleEvent(){this.closed||this.idleEventTimeoutMs<=0||this.transformChain.length===0||(this.idleEventTimer&&clearTimeout(this.idleEventTimer),this.idleEventTimer=setTimeout(()=>{this.idleEventTimer=void 0,this.notifyChain("session.idle",{})},this.idleEventTimeoutMs),typeof this.idleEventTimer.unref=="function"&&this.idleEventTimer.unref())}cancelIdleEventTimer(){this.idleEventTimer&&(clearTimeout(this.idleEventTimer),this.idleEventTimer=void 0)}notifyChain(e,t){let s=`lifecycle:${e}`;for(let r of this.transformChain)r.intercepts.has(s)&&r.connection.notify("hydra-acp/transformer/session_event",{event:e,sessionId:this.sessionId,payload:t}).catch(()=>{})}rewriteForClient(e){if(e&&typeof e=="object"&&!Array.isArray(e)){let t=e;if(t.sessionId===this.upstreamSessionId)return{...t,sessionId:this.sessionId}}return e}recordAndBroadcast(e,t,s){let r=this.rewriteForClient(t),i=!To(e,r),o=i?Ho(e,r):r;if(i){let a={method:e,params:o,recordedAt:Date.now()};if(this.lastRecordedAt=a.recordedAt,this.appendCount+=1,this.historyStore){let d=this.historyStore;d.append(this.sessionId,a).catch(()=>{}),this.appendCount>=this.compactEvery&&(this.appendCount=0,d.compact(this.sessionId,this.historyMaxEntries).catch(()=>{}))}for(let d of this.broadcastHandlers)try{d(a)}catch{}this.scheduleIdleCheck(),this.scheduleIdleEvent()}this.updatedAt=Date.now();for(let a of this.clients.values())s&&a.clientId===s||a.connection.notify(e,o).catch(()=>{})}async requestPermissionFromClients(e){return this.handlePermissionRequest(e)}async handlePermissionRequest(e){let t=[...this.clients.values()];if(t.length===0)throw Y(new Error("no clients attached to handle permission request"),I.PermissionDenied);let s=this.rewriteForClient(e),r=Bo(s);return new Promise((i,o)=>{let a=!1,d=[],c={addClient:u};this.inFlightPermissions.add(c);let p=this.sessionId,l=f=>{a||(a=!0,this.inFlightPermissions.delete(c),f())};function u(f){if(a)return;let g=f.connection.request("session/request_permission",s);d.push({client:f}),g.then(m=>{l(()=>{let y=Uo({toolCallId:r,result:m,resolver:f});for(let v of d)v.client.clientId!==f.clientId&&v.client.connection.notify("session/update",{sessionId:p,update:y}).catch(()=>{});i(m)})}).catch(m=>{l(()=>o(m))})}for(let f of t)u(f)})}async enqueuePrompt(e){return new Promise((t,s)=>{let r={kind:"internal",messageId:Pe(),enqueuedAt:Date.now(),cancelled:!1,task:e,resolve:t,reject:s};this.promptQueue.push(r),this.drainQueue()})}async enqueueUserPrompt(e,t,s){let r=(t??{}).prompt??[],i={clientId:e.clientId};e.clientInfo?.name&&(i.name=e.clientInfo.name),e.clientInfo?.version&&(i.version=e.clientInfo.version);let o=this.parseQueuePosition(t);return new Promise((a,d)=>{let c={kind:"user",messageId:s,originator:i,clientId:e.clientId,prompt:r,enqueuedAt:Date.now(),cancelled:!1,resolve:a,reject:d},p=this.insertEntryAt(c,o);this.persistRewrite(),this.broadcastQueueAdded(c,{position:p}),this.drainQueue()})}parseQueuePosition(e){if(!e||typeof e!="object")return"tail";let t=e._meta;if(!t||typeof t!="object"||Array.isArray(t))return"tail";let s=t["hydra-acp"];if(!s||typeof s!="object"||Array.isArray(s))return"tail";let r=s.queuePosition;if(r==="head"||r==="tail")return r;if(r&&typeof r=="object"&&!Array.isArray(r)){let i=r.afterMessageId;if(typeof i=="string"&&i.length>0)return{afterMessageId:i}}return"tail"}insertEntryAt(e,t){let s=this.currentEntry?.kind==="user"&&!this.currentEntry.cancelled?1:0;if(t==="head")return this.promptQueue.unshift(e),s;if(typeof t=="object"){let r=t.afterMessageId,i=this.promptQueue.findIndex(o=>o.messageId===r);if(i>=0)return this.promptQueue.splice(i+1,0,e),s+i+1}return this.promptQueue.push(e),s+this.promptQueue.length-1}persistRewrite(){let e=[];for(let s of this.promptQueue)s.kind!=="user"||s.cancelled||e.push(this.persistedFromEntry(s));let t=this.sessionId;this.queueWriteChain=this.queueWriteChain.catch(()=>{}).then(()=>Le(t,e).catch(()=>{}))}persistedFromEntry(e){return{messageId:e.messageId,originator:{clientInfo:{...e.originator.name!==void 0?{name:e.originator.name}:{},...e.originator.version!==void 0?{version:e.originator.version}:{}}},prompt:e.prompt,enqueuedAt:e.enqueuedAt}}async drainQueue(){if(!this.promptInFlight){this.promptInFlight=!0,await new Promise(e=>setImmediate(e));try{for(;this.promptQueue.length>0&&!this.closing;){let e=this.promptQueue.shift();if(!e)break;if(!e.cancelled){this.currentEntry=e,e.kind==="user"&&this.persistRewrite(),e.kind==="user"&&this.broadcastQueueRemoved(e.messageId,"started");try{let t=await this.runQueueEntry(e);e.resolve(t),await new Promise(s=>setImmediate(s))}catch(t){e.reject(t)}finally{this.currentEntry=void 0}}}}finally{this.promptInFlight=!1}}}async runQueueEntry(e){if(e.kind==="internal")return e.task();this.broadcastPromptReceived(e);let t=Te(e.prompt).replace(/\s+$/,""),s=!t.includes(`
63
+ `)&&t.startsWith("/")?t.split(/\s+/)[0]:"";if(s==="/model"||s==="/mode"||s==="/sessions"||s==="/help"||s==="/hydra"){let i;return s==="/sessions"?i=await this.handleSessionsCommand():s==="/help"?i=await this.handleHelpCommand():s==="/mode"?i=await this.handleModeCommand(t):s==="/model"?i=await this.handleModelCommand(t):i=await this.handleSlashCommand(t,e.messageId),this.closed||this.broadcastTurnComplete(e.clientId,i,e.messageId,e.wasAmend),this.clearAmendIfMatches(e.messageId),i}let r;try{r=await this.agent.connection.request("session/prompt",{sessionId:this.upstreamSessionId,prompt:e.prompt})}catch(i){if(this.forceCancelling)return this.clearAmendIfMatches(e.messageId),{stopReason:"cancelled"};throw this.closed||this.broadcastTurnComplete(e.clientId,{stopReason:"error"},e.messageId,e.wasAmend),this.clearAmendIfMatches(e.messageId),i}return this.closed||this.broadcastTurnComplete(e.clientId,r,e.messageId,e.wasAmend),this.clearAmendIfMatches(e.messageId),r}clearAmendIfMatches(e){this.amendInProgress?.cancelledMessageId===e&&(this.amendInProgress=void 0)}};function Y(n,e){return n.code=e,n}var Po=new Set(["session_info_update","current_model_update","current_mode_update","available_commands_update","available_modes_update","usage_update","config_option_update"]);function To(n,e){if(n!=="session/update")return!1;let s=(e??{}).update?.sessionUpdate;return typeof s=="string"&&Po.has(s)}function _o(n,e){if(n.length!==e.length)return!1;for(let t=0;t<n.length;t++)if(n[t]?.name!==e[t]?.name||n[t]?.description!==e[t]?.description)return!1;return!0}function $o(n,e){if(n.length!==e.length)return!1;for(let t=0;t<n.length;t++)if(n[t]?.id!==e[t]?.id||n[t]?.name!==e[t]?.name||n[t]?.description!==e[t]?.description)return!1;return!0}function Oo(n,e){if(n.length!==e.length)return!1;for(let t=0;t<n.length;t++)if(n[t]?.modelId!==e[t]?.modelId||n[t]?.name!==e[t]?.name||n[t]?.description!==e[t]?.description)return!1;return!0}function fe(n){if(!Array.isArray(n))return[];let e=[];for(let t of n){if(!t||typeof t!="object"||Array.isArray(t))continue;let s=t,r=typeof s.modelId=="string"&&s.modelId.trim()||typeof s.value=="string"&&s.value.trim()||typeof s.id=="string"&&s.id.trim()||void 0;if(!r)continue;let i={modelId:r};typeof s.name=="string"&&s.name.length>0&&(i.name=s.name),typeof s.description=="string"&&s.description.length>0&&(i.description=s.description),e.push(i)}return e}function ve(n){if(!Array.isArray(n))return[];let e=[];for(let t of n){if(!t||typeof t!="object"||Array.isArray(t))continue;let s=t,r=typeof s.value=="string"&&s.value.trim()||typeof s.id=="string"&&s.id.trim()||void 0;if(!r)continue;let i={id:r};typeof s.name=="string"&&s.name.length>0&&(i.name=s.name),typeof s.description=="string"&&s.description.length>0&&(i.description=s.description),e.push(i)}return e}function Fo(n){let t=(n??{}).update??{};if(t.sessionUpdate!=="available_modes_update")return null;let s=t.availableModes;if(!Array.isArray(s))return[];let r=[];for(let i of s){if(!i||typeof i!="object")continue;let o=i;if(typeof o.id!="string"||o.id.length===0)continue;let a={id:o.id};typeof o.name=="string"&&(a.name=o.name),typeof o.description=="string"&&(a.description=o.description),r.push(a)}return r}function No(n,e){let s=(e??{}).update??{};if(s.sessionUpdate!=="agent_message_chunk")return;let r=s.content??{};typeof r.text=="string"&&n.chunks.push(r.text)}function jo(n){let t=(n??{}).update??{};if(t.sessionUpdate!=="available_commands_update")return null;let s=t.availableCommands??t.commands;if(!Array.isArray(s))return[];let r=[];for(let i of s){if(!i||typeof i!="object")continue;let o=i;if(typeof o.name!="string"||o.name.length===0)continue;let a={name:o.name};typeof o.description=="string"&&(a.description=o.description),r.push(a)}return r}function Ho(n,e){if(n!=="session/update"||!e||typeof e!="object")return e;let t=e;return!t.update||typeof t.update!="object"||Array.isArray(t.update)||typeof t.update.messageId=="string"?e:{...e,update:{...t.update,messageId:Pe()}}}function cn(n,e){for(let t=0;t<n.length;t++){let s=n[t];if(!s||s.method!=="session/update")continue;if(s.params?.update?.messageId===e)return t}return-1}function Bo(n){if(!n||typeof n!="object")return;let e=n.toolCall;if(!e||typeof e!="object")return;let t=e.toolCallId;return typeof t=="string"?t:void 0}function Uo(n){let e=Do(n.result),t={sessionUpdate:"permission_resolved"};return n.toolCallId!==void 0&&(t.toolCallId=n.toolCallId),e&&(t.outcome=e,e.kind==="selected"&&typeof e.optionId=="string"&&(t.chosenOptionId=e.optionId)),t.resolvedBy=Lo(n.resolver),t}function Do(n){if(!n||typeof n!="object")return;let e=n.outcome;if(!e||typeof e!="object")return;let t=e.kind;if(typeof t!="string")return;let s={kind:t},r=e.optionId;typeof r=="string"&&(s.optionId=r);let i=e.reason;return typeof i=="string"&&(s.reason=i),s}function Lo(n){let e={clientId:n.clientId};return n.clientInfo?.name&&(e.name=n.clientInfo.name),n.clientInfo?.version&&(e.version=n.clientInfo.version),e}function Te(n){return typeof n=="string"?n:Array.isArray(n)?n.map(e=>e&&typeof e=="object"&&typeof e.text=="string"?e.text:"").join(""):""}function dn(n){return n==="session/prompt"?{stopReason:"stopped"}:{}}function Fs(n){let e=n.name&&n.name!==n.id?`${n.id} (${n.name}) \u2014 \u25B6 ${n.currentValue}`:`${n.id} \u2014 \u25B6 ${n.currentValue}`;if(n.options.length===0)return e;let t=Math.max(...n.options.map(r=>r.value.length)),s=n.options.map(r=>{let i=r.value===n.currentValue?"\u25B6":"\xB7",o=r.name&&r.name!==r.value,a=o?r.value.padEnd(t):r.value;return o?` ${i} ${a} ${r.name}`:` ${i} ${a}`});return`${e}
64
+ ${s.join(`
65
+ `)}`}function un(n,e){for(let t of n.split(/\r?\n/)){let s=t.trim();if(s)return s.length>e?`${s.slice(0,e)}\u2026`:s}}import*as _e from"fs/promises";import*as zu from"path";import{customAlphabet as zo}from"nanoid";import{z as A}from"zod";import{z as X}from"zod";var qe=X.object({goal:X.string().optional(),outcome:X.string().optional(),files_touched:X.array(X.string()).optional(),tools_used:X.array(X.string()).optional(),rejected_approaches:X.array(X.string()).optional(),open_threads:X.array(X.string()).optional()}),qo=200,ln=`Reply with ONLY a JSON object with exactly these keys, no prose, no markdown, no code fences:
46
66
  {
47
67
  "title": "short summary, max 80 chars",
48
68
  "synopsis": {
@@ -52,41 +72,41 @@ ${o}
52
72
  "open_threads": ["work started but not finished"]
53
73
  }
54
74
  }
55
- Use empty arrays/strings where a field doesn't apply.`;function Ds(n){let e=n.trim();if(e.length===0)return;let t=Us(e);if(t===void 0){let i=e.indexOf("{"),o=e.lastIndexOf("}");if(i<0||o<=i||(t=Us(e.slice(i,o+1)),t===void 0))return}if(t===null||typeof t!="object")return;let s=t,r={};if(typeof s.title=="string"){let i=s.title.trim();i.length>0&&(r.title=i.slice(0,Lo))}if(s.synopsis!==void 0&&s.synopsis!==null){let i=qe.safeParse(s.synopsis);i.success&&qo(i.data)&&(r.synopsis=i.data)}if(!(r.title===void 0&&r.synopsis===void 0))return r}function Us(n){try{return JSON.parse(n)}catch{return}}function qo(n){return n.goal!==void 0&&n.goal.trim().length>0||n.outcome!==void 0&&n.outcome.trim().length>0||n.files_touched!==void 0&&n.files_touched.length>0||n.tools_used!==void 0&&n.tools_used.length>0||n.rejected_approaches!==void 0&&n.rejected_approaches.length>0||n.open_threads!==void 0&&n.open_threads.length>0}var zo="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz",Wo=Jo(zo,16),Qo="hydra_lineage_";function Je(){return`${Qo}${Wo()}`}var pn=A.object({name:A.string(),description:A.string().optional()}),mn=A.object({id:A.string(),name:A.string().optional(),description:A.string().optional()}),Vo=A.object({modelId:A.string(),name:A.string().optional(),description:A.string().optional()}),gn=A.object({used:A.number().optional(),size:A.number().optional(),costAmount:A.number().optional(),costCurrency:A.string().optional(),cumulativeCost:A.number().optional()}),hn=A.object({name:A.string(),version:A.string().optional()}),Go=A.object({version:A.literal(1),sessionId:A.string(),lineageId:A.string().optional(),upstreamSessionId:A.string(),importedFromSessionId:A.string().optional(),importedFromUpstreamSessionId:A.string().optional(),importedFromMachine:A.string().optional(),agentId:A.string(),cwd:A.string(),title:A.string().optional(),synopsis:qe.optional(),summarizedThroughEntry:A.number().int().nonnegative().optional(),agentArgs:A.array(A.string()).optional(),currentModel:A.string().optional(),currentMode:A.string().optional(),currentUsage:gn.optional(),agentCommands:A.array(pn).optional(),agentModes:A.array(mn).optional(),agentModels:A.array(Vo).optional(),pendingHistorySync:A.boolean().optional(),parentSessionId:A.string().optional(),forkedFromSessionId:A.string().optional(),forkedFromMessageId:A.string().optional(),originatingClient:hn.optional(),interactive:A.boolean().optional(),priority:A.number().int().nonnegative().optional(),createdAt:A.string(),updatedAt:A.string()}),fn=/^[A-Za-z0-9_-]+$/;function Yo(n){if(!fn.test(n))throw new Error(`unsafe session id: ${n}`)}var gt=class{async write(e){Yo(e.sessionId);let t={version:1,...e};await K(w.sessionFile(e.sessionId),t,{mode:384})}async read(e){if(!fn.test(e))return;let t=await te(w.sessionFile(e));if(t!==void 0)try{return Go.parse(t)}catch{return}}async delete(e){if(fn.test(e)){try{await _e.unlink(w.sessionFile(e))}catch(t){if(t.code!=="ENOENT")throw t}try{await _e.rmdir(w.sessionDir(e))}catch(t){let s=t;if(s.code!=="ENOENT"&&s.code!=="ENOTEMPTY")throw t}}}async findByLineageId(e){if(e.length===0)return;let t=await this.list().catch(()=>[]);for(let s of t)if(s.lineageId===e)return s}async list(){let e;try{e=await _e.readdir(w.sessionsDir())}catch(r){if(r.code==="ENOENT")return[];throw r}let t=await Promise.all(e.map(r=>this.read(r).catch(()=>{}))),s=[];for(let r of t)r&&s.push(r);return s}};function yn(n){let e=new Date().toISOString();return{sessionId:n.sessionId,lineageId:n.lineageId,upstreamSessionId:n.upstreamSessionId,importedFromSessionId:n.importedFromSessionId,importedFromUpstreamSessionId:n.importedFromUpstreamSessionId,importedFromMachine:n.importedFromMachine,agentId:n.agentId,cwd:n.cwd,title:n.title,synopsis:n.synopsis,summarizedThroughEntry:n.summarizedThroughEntry,agentArgs:n.agentArgs,currentModel:n.currentModel,currentMode:n.currentMode,currentUsage:n.currentUsage,agentCommands:n.agentCommands,agentModes:n.agentModes,agentModels:n.agentModels,pendingHistorySync:n.pendingHistorySync,parentSessionId:n.parentSessionId,forkedFromSessionId:n.forkedFromSessionId,forkedFromMessageId:n.forkedFromMessageId,originatingClient:n.originatingClient,interactive:n.interactive,priority:n.priority,createdAt:n.createdAt??e,updatedAt:n.updatedAt??e}}import*as ce from"fs/promises";import{z as ie}from"zod";var Ko=ie.object({version:ie.literal(1),agentId:ie.string(),upstreamSessionId:ie.string(),deletedAt:ie.string(),upstreamUpdatedAt:ie.string().optional(),cwd:ie.string().optional(),title:ie.string().optional(),reason:ie.enum(["user","expired"]).optional(),interactive:ie.boolean().optional()}),ht=class{async add(e){let t={version:1,...e};await K(w.tombstoneFile(e.agentId,e.upstreamSessionId),t,{mode:384})}async has(e,t){try{return await ce.access(w.tombstoneFile(e,t)),!0}catch{return!1}}async read(e,t){let s=w.tombstoneFile(e,t),r=await te(s);if(r===void 0)return await this.has(e,t)?{version:1,agentId:e,upstreamSessionId:t,deletedAt:new Date(0).toISOString()}:void 0;try{return Ko.parse(r)}catch{return{version:1,agentId:e,upstreamSessionId:t,deletedAt:new Date(0).toISOString()}}}async remove(e,t){try{await ce.unlink(w.tombstoneFile(e,t))}catch(s){if(s.code!=="ENOENT")throw s}try{await ce.rmdir(w.tombstoneAgentDir(e))}catch(s){let r=s;if(r.code!=="ENOENT"&&r.code!=="ENOTEMPTY")throw s}}async list(e){if(e!==void 0)return this.listForAgent(e);let t;try{t=await ce.readdir(w.tombstonesDir())}catch(r){if(r.code==="ENOENT")return[];throw r}let s=[];for(let r of t){let i;try{i=decodeURIComponent(r)}catch{continue}s.push(...await this.listForAgent(i))}return s}async listForAgent(e){let t;try{t=await ce.readdir(w.tombstoneAgentDir(e))}catch(r){if(r.code==="ENOENT")return[];throw r}let s=[];for(let r of t){let i;try{i=decodeURIComponent(r)}catch{continue}let o=await this.read(e,i);o&&s.push(o)}return s}};function Ls(n,e){return e===void 0?!1:n.upstreamUpdatedAt===void 0?!0:e>n.upstreamUpdatedAt}import*as Xs from"fs/promises";import{fileURLToPath as Xo}from"url";import*as ze from"path";import*as yt from"fs";function Zo(){try{let n=ze.dirname(Xo(import.meta.url));for(let e=0;e<8;e+=1){let t=ze.join(n,"package.json");if(yt.existsSync(t)){let r=JSON.parse(yt.readFileSync(t,"utf8"));if(typeof r.version=="string"&&r.version.length>0&&(typeof r.name!="string"||r.name.includes("hydra-acp")))return r.version}let s=ze.dirname(n);if(s===n)break;n=s}}catch{}return"0.0.0"}var Q=Zo(),wt="hydra-acp-cat";var qs=`[older history truncated]
56
- `,ea=["file_path","path","command","pattern","query"];function Js(n,e={}){let t=e.maxChars??4e5,s=[],r="",i=()=>{r.length!==0&&(s.push(`Assistant: ${r}`),r="")};for(let o of n){if(o.method!=="session/update")continue;let d=o.params?.update;if(!d||typeof d.sessionUpdate!="string")continue;let c=d.sessionUpdate;if(c==="prompt_received"){i();let f=ra(d.prompt).trim();f.length>0&&s.push(`User: ${f}`)}else if(c==="agent_message_chunk"){let f=ia(d.content);f.length>0&&(r+=f)}else c==="tool_call"?(i(),s.push(ta(d))):c==="turn_complete"&&i()}return i(),aa(s.join(`
57
- `),t)}function ta(n){let e=na(n),t=sa(n.rawInput);return t.length===0?`Tool: ${e}`:`Tool: ${e}(${t.join(", ")})`}function na(n){return typeof n.name=="string"&&n.name.length>0?n.name:typeof n.title=="string"&&n.title.length>0?n.title:"(unnamed)"}function sa(n){if(!n||typeof n!="object"||Array.isArray(n))return[];let e=n,t=[];for(let s of ea){let r=e[s];typeof r=="string"&&r.length>0&&t.push(`${s}=${oa(r,200)}`)}return t}function ra(n){return typeof n=="string"?n:Array.isArray(n)?n.map(e=>{if(e&&typeof e=="object"){let t=e.text;if(typeof t=="string")return t}return""}).join(""):""}function ia(n){if(!n||typeof n!="object")return"";let e=n.text;return typeof e=="string"?e:""}function oa(n,e){return n.length<=e?n:n.slice(0,e-1)+"\u2026"}function aa(n,e){if(n.length<=e)return n;let t=n.split(`
58
- `);for(;t.length>0;){let s=qs+t.join(`
59
- `);if(s.length<=e)return s;t.shift()}return qs}var da=12e4;async function Ws(n){let e=n.timeoutMs??da,t,s,r=!1;try{let i=(async()=>{t=ae.spawn({agentId:n.agentId,cwd:n.cwd,plan:n.plan,logger:n.logger});let o=await t.connection.request("initialize",{protocolVersion:le,clientCapabilities:{},clientInfo:{name:"hydra-synopsis",version:Q}}),a=await t.connection.request("session/new",{cwd:n.cwd,mcpServers:[]}),d=a.sessionId;if(typeof d!="string"){n.logger?.warn(`synopsis: agent ${n.agentId} returned non-string sessionId from session/new`);return}if(n.modelId){let g=ua(a);if(g.size===0||g.has(n.modelId))try{await t.connection.request("session/set_model",{sessionId:d,modelId:n.modelId})}catch(m){n.logger?.warn(`synopsis: agent ${n.agentId} rejected set_model ${JSON.stringify(n.modelId)}: ${m.message}; continuing on default`)}else n.logger?.warn(`synopsis: model ${JSON.stringify(n.modelId)} not advertised by agent ${n.agentId} (have [${[...g].join(", ")}]); continuing on default`)}let c=[];t.connection.onNotification("session/update",g=>{let m=ca(g);m.length>0&&c.push(m)});let f=Js(n.history,{maxChars:n.maxTranscriptChars}),l=f.length>0?`${f}
75
+ Use empty arrays/strings where a field doesn't apply.`;function Ls(n){let e=n.trim();if(e.length===0)return;let t=Ds(e);if(t===void 0){let i=e.indexOf("{"),o=e.lastIndexOf("}");if(i<0||o<=i||(t=Ds(e.slice(i,o+1)),t===void 0))return}if(t===null||typeof t!="object")return;let s=t,r={};if(typeof s.title=="string"){let i=s.title.trim();i.length>0&&(r.title=i.slice(0,qo))}if(s.synopsis!==void 0&&s.synopsis!==null){let i=qe.safeParse(s.synopsis);i.success&&Jo(i.data)&&(r.synopsis=i.data)}if(!(r.title===void 0&&r.synopsis===void 0))return r}function Ds(n){try{return JSON.parse(n)}catch{return}}function Jo(n){return n.goal!==void 0&&n.goal.trim().length>0||n.outcome!==void 0&&n.outcome.trim().length>0||n.files_touched!==void 0&&n.files_touched.length>0||n.tools_used!==void 0&&n.tools_used.length>0||n.rejected_approaches!==void 0&&n.rejected_approaches.length>0||n.open_threads!==void 0&&n.open_threads.length>0}var Wo="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz",Qo=zo(Wo,16),Vo="hydra_lineage_";function Je(){return`${Vo}${Qo()}`}var pn=A.object({name:A.string(),description:A.string().optional()}),mn=A.object({id:A.string(),name:A.string().optional(),description:A.string().optional()}),Go=A.object({modelId:A.string(),name:A.string().optional(),description:A.string().optional()}),gn=A.object({used:A.number().optional(),size:A.number().optional(),costAmount:A.number().optional(),costCurrency:A.string().optional(),cumulativeCost:A.number().optional()}),hn=A.object({name:A.string(),version:A.string().optional()}),Yo=A.object({version:A.literal(1),sessionId:A.string(),lineageId:A.string().optional(),upstreamSessionId:A.string(),importedFromSessionId:A.string().optional(),importedFromUpstreamSessionId:A.string().optional(),importedFromMachine:A.string().optional(),agentId:A.string(),cwd:A.string(),title:A.string().optional(),synopsis:qe.optional(),summarizedThroughEntry:A.number().int().nonnegative().optional(),agentArgs:A.array(A.string()).optional(),currentModel:A.string().optional(),currentMode:A.string().optional(),currentUsage:gn.optional(),agentCommands:A.array(pn).optional(),agentModes:A.array(mn).optional(),agentModels:A.array(Go).optional(),pendingHistorySync:A.boolean().optional(),parentSessionId:A.string().optional(),forkedFromSessionId:A.string().optional(),forkedFromMessageId:A.string().optional(),originatingClient:hn.optional(),interactive:A.boolean().optional(),priority:A.number().int().nonnegative().optional(),createdAt:A.string(),updatedAt:A.string()}),fn=/^[A-Za-z0-9_-]+$/;function Ko(n){if(!fn.test(n))throw new Error(`unsafe session id: ${n}`)}var gt=class{async write(e){Ko(e.sessionId);let t={version:1,...e};await K(w.sessionFile(e.sessionId),t,{mode:384})}async read(e){if(!fn.test(e))return;let t=await te(w.sessionFile(e));if(t!==void 0)try{return Yo.parse(t)}catch{return}}async delete(e){if(fn.test(e)){try{await _e.unlink(w.sessionFile(e))}catch(t){if(t.code!=="ENOENT")throw t}try{await _e.rmdir(w.sessionDir(e))}catch(t){let s=t;if(s.code!=="ENOENT"&&s.code!=="ENOTEMPTY")throw t}}}async findByLineageId(e){if(e.length===0)return;let t=await this.list().catch(()=>[]);for(let s of t)if(s.lineageId===e)return s}async list(){let e;try{e=await _e.readdir(w.sessionsDir())}catch(r){if(r.code==="ENOENT")return[];throw r}let t=await Promise.all(e.map(r=>this.read(r).catch(()=>{}))),s=[];for(let r of t)r&&s.push(r);return s}};function yn(n){let e=new Date().toISOString();return{sessionId:n.sessionId,lineageId:n.lineageId,upstreamSessionId:n.upstreamSessionId,importedFromSessionId:n.importedFromSessionId,importedFromUpstreamSessionId:n.importedFromUpstreamSessionId,importedFromMachine:n.importedFromMachine,agentId:n.agentId,cwd:n.cwd,title:n.title,synopsis:n.synopsis,summarizedThroughEntry:n.summarizedThroughEntry,agentArgs:n.agentArgs,currentModel:n.currentModel,currentMode:n.currentMode,currentUsage:n.currentUsage,agentCommands:n.agentCommands,agentModes:n.agentModes,agentModels:n.agentModels,pendingHistorySync:n.pendingHistorySync,parentSessionId:n.parentSessionId,forkedFromSessionId:n.forkedFromSessionId,forkedFromMessageId:n.forkedFromMessageId,originatingClient:n.originatingClient,interactive:n.interactive,priority:n.priority,createdAt:n.createdAt??e,updatedAt:n.updatedAt??e}}import*as ce from"fs/promises";import{z as ie}from"zod";var Xo=ie.object({version:ie.literal(1),agentId:ie.string(),upstreamSessionId:ie.string(),deletedAt:ie.string(),upstreamUpdatedAt:ie.string().optional(),cwd:ie.string().optional(),title:ie.string().optional(),reason:ie.enum(["user","expired"]).optional(),interactive:ie.boolean().optional()}),ht=class{async add(e){let t={version:1,...e};await K(w.tombstoneFile(e.agentId,e.upstreamSessionId),t,{mode:384})}async has(e,t){try{return await ce.access(w.tombstoneFile(e,t)),!0}catch{return!1}}async read(e,t){let s=w.tombstoneFile(e,t),r=await te(s);if(r===void 0)return await this.has(e,t)?{version:1,agentId:e,upstreamSessionId:t,deletedAt:new Date(0).toISOString()}:void 0;try{return Xo.parse(r)}catch{return{version:1,agentId:e,upstreamSessionId:t,deletedAt:new Date(0).toISOString()}}}async remove(e,t){try{await ce.unlink(w.tombstoneFile(e,t))}catch(s){if(s.code!=="ENOENT")throw s}try{await ce.rmdir(w.tombstoneAgentDir(e))}catch(s){let r=s;if(r.code!=="ENOENT"&&r.code!=="ENOTEMPTY")throw s}}async list(e){if(e!==void 0)return this.listForAgent(e);let t;try{t=await ce.readdir(w.tombstonesDir())}catch(r){if(r.code==="ENOENT")return[];throw r}let s=[];for(let r of t){let i;try{i=decodeURIComponent(r)}catch{continue}s.push(...await this.listForAgent(i))}return s}async listForAgent(e){let t;try{t=await ce.readdir(w.tombstoneAgentDir(e))}catch(r){if(r.code==="ENOENT")return[];throw r}let s=[];for(let r of t){let i;try{i=decodeURIComponent(r)}catch{continue}let o=await this.read(e,i);o&&s.push(o)}return s}};function qs(n,e){return e===void 0?!1:n.upstreamUpdatedAt===void 0?!0:e>n.upstreamUpdatedAt}import*as Zs from"fs/promises";import{fileURLToPath as Zo}from"url";import*as ze from"path";import*as yt from"fs";function ea(){try{let n=ze.dirname(Zo(import.meta.url));for(let e=0;e<8;e+=1){let t=ze.join(n,"package.json");if(yt.existsSync(t)){let r=JSON.parse(yt.readFileSync(t,"utf8"));if(typeof r.version=="string"&&r.version.length>0&&(typeof r.name!="string"||r.name.includes("hydra-acp")))return r.version}let s=ze.dirname(n);if(s===n)break;n=s}}catch{}return"0.0.0"}var Q=ea(),wt="hydra-acp-cat";var Js=`[older history truncated]
76
+ `,ta=["file_path","path","command","pattern","query"];function zs(n,e={}){let t=e.maxChars??4e5,s=[],r="",i=()=>{r.length!==0&&(s.push(`Assistant: ${r}`),r="")};for(let o of n){if(o.method!=="session/update")continue;let d=o.params?.update;if(!d||typeof d.sessionUpdate!="string")continue;let c=d.sessionUpdate;if(c==="prompt_received"){i();let p=ia(d.prompt).trim();p.length>0&&s.push(`User: ${p}`)}else if(c==="agent_message_chunk"){let p=oa(d.content);p.length>0&&(r+=p)}else c==="tool_call"?(i(),s.push(na(d))):c==="turn_complete"&&i()}return i(),da(s.join(`
77
+ `),t)}function na(n){let e=sa(n),t=ra(n.rawInput);return t.length===0?`Tool: ${e}`:`Tool: ${e}(${t.join(", ")})`}function sa(n){return typeof n.name=="string"&&n.name.length>0?n.name:typeof n.title=="string"&&n.title.length>0?n.title:"(unnamed)"}function ra(n){if(!n||typeof n!="object"||Array.isArray(n))return[];let e=n,t=[];for(let s of ta){let r=e[s];typeof r=="string"&&r.length>0&&t.push(`${s}=${aa(r,200)}`)}return t}function ia(n){return typeof n=="string"?n:Array.isArray(n)?n.map(e=>{if(e&&typeof e=="object"){let t=e.text;if(typeof t=="string")return t}return""}).join(""):""}function oa(n){if(!n||typeof n!="object")return"";let e=n.text;return typeof e=="string"?e:""}function aa(n,e){return n.length<=e?n:n.slice(0,e-1)+"\u2026"}function da(n,e){if(n.length<=e)return n;let t=n.split(`
78
+ `);for(;t.length>0;){let s=Js+t.join(`
79
+ `);if(s.length<=e)return s;t.shift()}return Js}var ca=12e4;async function Qs(n){let e=n.timeoutMs??ca,t,s,r=!1;try{let i=(async()=>{t=ae.spawn({agentId:n.agentId,cwd:n.cwd,plan:n.plan,logger:n.logger});let o=await t.connection.request("initialize",{protocolVersion:le,clientCapabilities:{},clientInfo:{name:"hydra-synopsis",version:Q}}),a=await t.connection.request("session/new",{cwd:n.cwd,mcpServers:[]}),d=a.sessionId;if(typeof d!="string"){n.logger?.warn(`synopsis: agent ${n.agentId} returned non-string sessionId from session/new`);return}if(n.modelId){let g=la(a);if(g.size===0||g.has(n.modelId))try{await t.connection.request("session/set_model",{sessionId:d,modelId:n.modelId})}catch(m){n.logger?.warn(`synopsis: agent ${n.agentId} rejected set_model ${JSON.stringify(n.modelId)}: ${m.message}; continuing on default`)}else n.logger?.warn(`synopsis: model ${JSON.stringify(n.modelId)} not advertised by agent ${n.agentId} (have [${[...g].join(", ")}]); continuing on default`)}let c=[];t.connection.onNotification("session/update",g=>{let m=ua(g);m.length>0&&c.push(m)});let p=zs(n.history,{maxChars:n.maxTranscriptChars}),l=p.length>0?`${p}
60
80
 
61
- ${ln}`:ln;await t.connection.request("session/prompt",{sessionId:d,prompt:[{type:"text",text:l}]});let u=c.join(""),p=Ds(u);return p||n.logger?.warn(`synopsis: agent ${n.agentId} reply did not parse as snapshot JSON (replyLen=${u.length} preview=${JSON.stringify(u.slice(0,200))})`),p})();return await new Promise((o,a)=>{s=setTimeout(()=>{r=!0,n.logger?.warn(`synopsis: agent ${n.agentId} timed out after ${e}ms`),o(void 0)},e),s.unref?.(),i.then(d=>{s&&clearTimeout(s),r||o(d)},d=>{s&&clearTimeout(s),r||a(d)})})}catch(i){n.logger?.warn(`synopsis: agent ${n.agentId} failed: ${i.message}`);return}finally{s&&clearTimeout(s),t&&await t.kill().catch(()=>{})}}function ca(n){if(!n||typeof n!="object")return"";let e=n.update;if(!e||typeof e!="object")return"";let t=e;if(t.sessionUpdate!=="agent_message_chunk")return"";let s=t.content;return s&&typeof s.text=="string"?s.text:""}function ua(n){let e=new Set;zs(e,n.availableModels);let t=n.models;return t&&typeof t=="object"&&!Array.isArray(t)&&zs(e,t.availableModels),e}function zs(n,e){if(Array.isArray(e)){for(let t of e)if(t&&typeof t=="object"){let s=t.modelId??t.value??t.id;typeof s=="string"&&s.length>0&&n.add(s)}}}function la(n){let e=new Map;for(let t of Ks(n).values())e.set(t.toolName,(e.get(t.toolName)??0)+1);return[...e.entries()].map(([t,s])=>({name:t,count:s})).sort((t,s)=>s.count-t.count||t.name.localeCompare(s.name))}function Gs(n){return la(n).map(e=>e.name)}function fa(n){let e=new Map;for(let t of Ks(n).values())for(let s of t.paths){let r=e.get(s);r===void 0&&(r=new Map,e.set(s,r)),r.set(t.toolName,(r.get(t.toolName)??0)+1)}return[...e.entries()].map(([t,s])=>{let r=[...s.entries()].map(([o,a])=>({name:o,count:a})).sort((o,a)=>a.count-o.count||o.name.localeCompare(a.name)),i=r.reduce((o,a)=>o+a.count,0);return{path:t,count:i,byTool:r}}).sort((t,s)=>s.count-t.count||t.path.localeCompare(s.path))}function Ys(n){return fa(n).map(e=>e.path)}function Ks(n){let e=new Map,t=0;for(let s of n){let i=s.params?.update;if(!i)continue;let o=i.sessionUpdate;if(o!=="tool_call"&&o!=="tool_call_update")continue;if(o==="tool_call"){let d=typeof i.toolCallId=="string"&&i.toolCallId.length>0?i.toolCallId:`__synth_${t++}`,c=e.get(d);c===void 0?(c={toolName:Qs(i),paths:new Set},e.set(d,c)):c.toolName=Qs(i);for(let f of Vs(i.rawInput,i.locations))c.paths.add(f);continue}if(typeof i.toolCallId!="string"||i.toolCallId.length===0)continue;let a=e.get(i.toolCallId);if(a!==void 0)for(let d of Vs(i.rawInput,i.locations))a.paths.add(d)}return e}function Qs(n){return typeof n.name=="string"&&n.name.length>0?n.name:typeof n.title=="string"&&n.title.length>0?n.title:"(unnamed)"}function Vs(n,e){let t=new Set;if(n&&typeof n=="object"&&!Array.isArray(n)){let s=n;typeof s.file_path=="string"?t.add(s.file_path):typeof s.path=="string"&&t.add(s.path);let r=s.edits;if(Array.isArray(r)){for(let i of r)if(i&&typeof i=="object"){let o=i.file_path;typeof o=="string"&&t.add(o)}}}if(Array.isArray(e)){for(let s of e)if(s&&typeof s=="object"){let r=s.path;typeof r=="string"&&t.add(r)}}return t}var pa=2,vt=class{constructor(e){this.opts=e;this.maxConcurrent=e.maxConcurrent??pa}opts;queued=new Set;inflight=new Map;stopped=!1;maxConcurrent;schedule(e){this.stopped||this.queued.has(e)||this.inflight.has(e)||(this.queued.add(e),this.drain())}size(){return{queued:this.queued.size,inflight:this.inflight.size}}async flush(e){let t=Date.now()+e;for(;this.queued.size>0||this.inflight.size>0;){let s=t-Date.now();if(s<=0)return;let r=[...this.inflight.values()];if(r.length===0){await new Promise(i=>{setTimeout(i,25).unref?.()});continue}await Promise.race([Promise.race(r),new Promise(i=>{setTimeout(i,s).unref?.()})])}}async shutdown(){this.stopped=!0,this.queued.clear(),await Promise.allSettled([...this.inflight.values()])}drain(){if(!this.stopped)for(;this.inflight.size<this.maxConcurrent&&this.queued.size>0;){let e=this.queued.values().next().value;if(!e)return;this.queued.delete(e);let t=this.runOne(e).finally(()=>{this.inflight.delete(e),this.drain()});this.inflight.set(e,t)}}async runOne(e){try{let t=await this.opts.store.read(e);if(!t){this.opts.logger?.info(`synopsis: session ${e} record missing; skipping`);return}let s=await this.opts.histories.load(e),r=t.summarizedThroughEntry;if(r!==void 0&&s.length<=r){this.opts.logger?.info(`synopsis: skip ${e} (history unchanged at ${s.length})`);return}let i=this.opts.synopsisAgent??t.agentId,o=await this.opts.registry.getAgent(i);if(!o){this.opts.logger?.warn(`synopsis: agent ${i} not in registry for session ${e}; skipping`);return}let a=await oe(o,[],{npmRegistry:this.opts.npmRegistry}),d=this.opts.synopsisModel,c=w.sessionDir(e);await Xs.mkdir(c,{recursive:!0}).catch(()=>{}),this.opts.logger?.info(`synopsis: start sessionId=${e} agentId=${i} historyLen=${s.length} model=${JSON.stringify(d??"(default)")} cwd=${c}`);let f=await Ws({agentId:i,cwd:c,plan:a,history:s,modelId:d,logger:this.opts.logger,timeoutMs:this.opts.generateTimeoutMs});if(!f){this.opts.logger?.warn(`synopsis: sessionId=${e} no parseable result; not persisting`);return}f.title&&await this.opts.persistTitle(e,f.title);let l=ma(f.synopsis,s);l&&ga(l)?(await this.opts.persistSynopsis(e,l,s.length),this.opts.logger?.info(`synopsis: persisted sessionId=${e} title=${JSON.stringify(!!f.title)} fields=${ha(l)}`)):f.title&&this.opts.logger?.info(`synopsis: persisted title only sessionId=${e}`)}catch(t){this.opts.logger?.warn(`synopsis: sessionId=${e} failed: ${t.message}`)}}};function ma(n,e){let t=Ys(e),s=Gs(e);return n?{...n,files_touched:t.length>0?t:n.files_touched,tools_used:s.length>0?s:n.tools_used}:t.length===0&&s.length===0?void 0:{files_touched:t.length>0?t:void 0,tools_used:s.length>0?s:void 0}}function ga(n){return!!(n.goal&&n.goal.trim().length>0||n.outcome&&n.outcome.trim().length>0||n.files_touched&&n.files_touched.length>0||n.tools_used&&n.tools_used.length>0||n.rejected_approaches&&n.rejected_approaches.length>0||n.open_threads&&n.open_threads.length>0)}function ha(n){let e=[];return n.goal&&e.push("goal"),n.outcome&&e.push("outcome"),n.files_touched&&n.files_touched.length>0&&e.push(`files=${n.files_touched.length}`),n.tools_used&&n.tools_used.length>0&&e.push(`tools=${n.tools_used.length}`),n.rejected_approaches&&n.rejected_approaches.length>0&&e.push(`rejected=${n.rejected_approaches.length}`),n.open_threads&&n.open_threads.length>0&&e.push(`open=${n.open_threads.length}`),`[${e.join(",")}]`}import*as z from"fs/promises";function Zs(n){return n==="summary"?"summary":"inline"}var wn=256;function er(n,e){return e!=="summary"?n:n.map(ya)}function ya(n){if(n.method!=="session/update")return n;let e=n.params;if(!e||typeof e!="object"||Array.isArray(e))return n;let t=e,s=t.update;if(!s||typeof s!="object"||Array.isArray(s))return n;let r=s;if(r.sessionUpdate!=="tool_call"&&r.sessionUpdate!=="tool_call_update")return n;let i={...r};Array.isArray(r.content)&&(i.content=r.content.map(tr));let o=r.rawOutput;if(o&&typeof o=="object"&&!Array.isArray(o)){let a=o,d={};a.error!==void 0&&(d.error=vn(a.error)),a.metadata!==void 0&&(d.metadata=a.metadata),i.rawOutput=d}return{...n,params:{...t,update:i}}}function wa(n){return!!n&&typeof n=="object"&&!Array.isArray(n)&&n.type==="diff"}function tr(n){if(wa(n)){let s=n,r={type:"diff",oldText:"",newText:""};return typeof s.path=="string"&&(r.path=s.path),r}if(!n||typeof n!="object"||Array.isArray(n))return n;let e=n,t={...e};return typeof e.text=="string"&&(t.text=vn(e.text)),typeof e.content=="string"?t.content=vn(e.content):e.content&&typeof e.content=="object"&&(t.content=tr(e.content)),t}function vn(n){if(typeof n=="string"&&n.length>wn){let e=n.length-wn;return`${n.slice(0,wn)}\u2026[+${e} chars omitted from summary export]`}return n}var va=2048;function nr(n){let e=new Set,t=s=>{if(sr(s)){e.add(s.__hydraBlob);return}if(Array.isArray(s)){for(let r of s)t(r);return}if(s&&typeof s=="object")for(let r of Object.values(s))t(r)};for(let s of n)t(s.params);return e}function sr(n){return!!n&&typeof n=="object"&&!Array.isArray(n)&&typeof n.__hydraBlob=="string"}function ba(n){if(n.method!=="session/update")return!1;let e=n.params;if(!e||typeof e!="object"||Array.isArray(e))return!1;let t=e.update;if(!t||typeof t!="object"||Array.isArray(t))return!1;let s=t.sessionUpdate;return s==="tool_call"||s==="tool_call_update"}async function Sn(n,e){if(!ba(n))return n;let t=n.params,s=t.update,r=await bn(s,e);return{...n,params:{...t,update:r}}}async function bn(n,e){if(typeof n=="string"){if(n.length<=va)return n;let t=await e(n);return t===null?n:{__hydraBlob:t,bytes:n.length}}if(Array.isArray(n)){let t=[];for(let s of n)t.push(await bn(s,e));return t}if(n&&typeof n=="object"){let t={};for(let[s,r]of Object.entries(n))t[s]=await bn(r,e);return t}return n}async function rr(n,e){let t=n.params;if(!t||typeof t!="object"||Array.isArray(t))return n;let s=await In(t,e);return s===t?n:{...n,params:s}}async function In(n,e){if(sr(n))return await e(n.__hydraBlob)??"";if(Array.isArray(n)){let t=!1,s=[];for(let r of n){let i=await In(r,e);i!==r&&(t=!0),s.push(i)}return t?s:n}if(n&&typeof n=="object"){let t=!1,s={};for(let[r,i]of Object.entries(n)){let o=await In(i,e);o!==i&&(t=!0),s[r]=o}return t?s:n}return n}var We=/^[A-Za-z0-9_-]+$/,Ia=1e3,bt=class{writeQueues=new Map;maxEntries;constructor(e={}){this.maxEntries=e.maxEntries??Ia}async append(e,t){if(We.test(e))return this.enqueue(e,async()=>{await z.mkdir(w.sessionDir(e),{recursive:!0});let s=await Sn(t,i=>Vt(e,i)),r=JSON.stringify(s)+`
81
+ ${ln}`:ln;await t.connection.request("session/prompt",{sessionId:d,prompt:[{type:"text",text:l}]});let u=c.join(""),f=Ls(u);return f||n.logger?.warn(`synopsis: agent ${n.agentId} reply did not parse as snapshot JSON (replyLen=${u.length} preview=${JSON.stringify(u.slice(0,200))})`),f})();return await new Promise((o,a)=>{s=setTimeout(()=>{r=!0,n.logger?.warn(`synopsis: agent ${n.agentId} timed out after ${e}ms`),o(void 0)},e),s.unref?.(),i.then(d=>{s&&clearTimeout(s),r||o(d)},d=>{s&&clearTimeout(s),r||a(d)})})}catch(i){n.logger?.warn(`synopsis: agent ${n.agentId} failed: ${i.message}`);return}finally{s&&clearTimeout(s),t&&await t.kill().catch(()=>{})}}function ua(n){if(!n||typeof n!="object")return"";let e=n.update;if(!e||typeof e!="object")return"";let t=e;if(t.sessionUpdate!=="agent_message_chunk")return"";let s=t.content;return s&&typeof s.text=="string"?s.text:""}function la(n){let e=new Set;Ws(e,n.availableModels);let t=n.models;return t&&typeof t=="object"&&!Array.isArray(t)&&Ws(e,t.availableModels),e}function Ws(n,e){if(Array.isArray(e)){for(let t of e)if(t&&typeof t=="object"){let s=t.modelId??t.value??t.id;typeof s=="string"&&s.length>0&&n.add(s)}}}function fa(n){let e=new Map;for(let t of Xs(n).values())e.set(t.toolName,(e.get(t.toolName)??0)+1);return[...e.entries()].map(([t,s])=>({name:t,count:s})).sort((t,s)=>s.count-t.count||t.name.localeCompare(s.name))}function Ys(n){return fa(n).map(e=>e.name)}function pa(n){let e=new Map;for(let t of Xs(n).values())for(let s of t.paths){let r=e.get(s);r===void 0&&(r=new Map,e.set(s,r)),r.set(t.toolName,(r.get(t.toolName)??0)+1)}return[...e.entries()].map(([t,s])=>{let r=[...s.entries()].map(([o,a])=>({name:o,count:a})).sort((o,a)=>a.count-o.count||o.name.localeCompare(a.name)),i=r.reduce((o,a)=>o+a.count,0);return{path:t,count:i,byTool:r}}).sort((t,s)=>s.count-t.count||t.path.localeCompare(s.path))}function Ks(n){return pa(n).map(e=>e.path)}function Xs(n){let e=new Map,t=0;for(let s of n){let i=s.params?.update;if(!i)continue;let o=i.sessionUpdate;if(o!=="tool_call"&&o!=="tool_call_update")continue;if(o==="tool_call"){let d=typeof i.toolCallId=="string"&&i.toolCallId.length>0?i.toolCallId:`__synth_${t++}`,c=e.get(d);c===void 0?(c={toolName:Vs(i),paths:new Set},e.set(d,c)):c.toolName=Vs(i);for(let p of Gs(i.rawInput,i.locations))c.paths.add(p);continue}if(typeof i.toolCallId!="string"||i.toolCallId.length===0)continue;let a=e.get(i.toolCallId);if(a!==void 0)for(let d of Gs(i.rawInput,i.locations))a.paths.add(d)}return e}function Vs(n){return typeof n.name=="string"&&n.name.length>0?n.name:typeof n.title=="string"&&n.title.length>0?n.title:"(unnamed)"}function Gs(n,e){let t=new Set;if(n&&typeof n=="object"&&!Array.isArray(n)){let s=n;typeof s.file_path=="string"?t.add(s.file_path):typeof s.path=="string"&&t.add(s.path);let r=s.edits;if(Array.isArray(r)){for(let i of r)if(i&&typeof i=="object"){let o=i.file_path;typeof o=="string"&&t.add(o)}}}if(Array.isArray(e)){for(let s of e)if(s&&typeof s=="object"){let r=s.path;typeof r=="string"&&t.add(r)}}return t}var ma=2,vt=class{constructor(e){this.opts=e;this.maxConcurrent=e.maxConcurrent??ma}opts;queued=new Set;inflight=new Map;stopped=!1;maxConcurrent;schedule(e){this.stopped||this.queued.has(e)||this.inflight.has(e)||(this.queued.add(e),this.drain())}size(){return{queued:this.queued.size,inflight:this.inflight.size}}async flush(e){let t=Date.now()+e;for(;this.queued.size>0||this.inflight.size>0;){let s=t-Date.now();if(s<=0)return;let r=[...this.inflight.values()];if(r.length===0){await new Promise(i=>{setTimeout(i,25).unref?.()});continue}await Promise.race([Promise.race(r),new Promise(i=>{setTimeout(i,s).unref?.()})])}}async shutdown(){this.stopped=!0,this.queued.clear(),await Promise.allSettled([...this.inflight.values()])}drain(){if(!this.stopped)for(;this.inflight.size<this.maxConcurrent&&this.queued.size>0;){let e=this.queued.values().next().value;if(!e)return;this.queued.delete(e);let t=this.runOne(e).finally(()=>{this.inflight.delete(e),this.drain()});this.inflight.set(e,t)}}async runOne(e){try{let t=await this.opts.store.read(e);if(!t){this.opts.logger?.info(`synopsis: session ${e} record missing; skipping`);return}let s=await this.opts.histories.load(e),r=t.summarizedThroughEntry;if(r!==void 0&&s.length<=r){this.opts.logger?.info(`synopsis: skip ${e} (history unchanged at ${s.length})`);return}let i=this.opts.synopsisAgent??t.agentId,o=await this.opts.registry.getAgent(i);if(!o){this.opts.logger?.warn(`synopsis: agent ${i} not in registry for session ${e}; skipping`);return}let a=await oe(o,[],{npmRegistry:this.opts.npmRegistry}),d=this.opts.synopsisModel,c=w.sessionDir(e);await Zs.mkdir(c,{recursive:!0}).catch(()=>{}),this.opts.logger?.info(`synopsis: start sessionId=${e} agentId=${i} historyLen=${s.length} model=${JSON.stringify(d??"(default)")} cwd=${c}`);let p=await Qs({agentId:i,cwd:c,plan:a,history:s,modelId:d,logger:this.opts.logger,timeoutMs:this.opts.generateTimeoutMs});if(!p){this.opts.logger?.warn(`synopsis: sessionId=${e} no parseable result; not persisting`);return}p.title&&await this.opts.persistTitle(e,p.title);let l=ga(p.synopsis,s);l&&ha(l)?(await this.opts.persistSynopsis(e,l,s.length),this.opts.logger?.info(`synopsis: persisted sessionId=${e} title=${JSON.stringify(!!p.title)} fields=${ya(l)}`)):p.title&&this.opts.logger?.info(`synopsis: persisted title only sessionId=${e}`)}catch(t){this.opts.logger?.warn(`synopsis: sessionId=${e} failed: ${t.message}`)}}};function ga(n,e){let t=Ks(e),s=Ys(e);return n?{...n,files_touched:t.length>0?t:n.files_touched,tools_used:s.length>0?s:n.tools_used}:t.length===0&&s.length===0?void 0:{files_touched:t.length>0?t:void 0,tools_used:s.length>0?s:void 0}}function ha(n){return!!(n.goal&&n.goal.trim().length>0||n.outcome&&n.outcome.trim().length>0||n.files_touched&&n.files_touched.length>0||n.tools_used&&n.tools_used.length>0||n.rejected_approaches&&n.rejected_approaches.length>0||n.open_threads&&n.open_threads.length>0)}function ya(n){let e=[];return n.goal&&e.push("goal"),n.outcome&&e.push("outcome"),n.files_touched&&n.files_touched.length>0&&e.push(`files=${n.files_touched.length}`),n.tools_used&&n.tools_used.length>0&&e.push(`tools=${n.tools_used.length}`),n.rejected_approaches&&n.rejected_approaches.length>0&&e.push(`rejected=${n.rejected_approaches.length}`),n.open_threads&&n.open_threads.length>0&&e.push(`open=${n.open_threads.length}`),`[${e.join(",")}]`}import*as z from"fs/promises";function er(n){return n==="summary"?"summary":"inline"}var wn=256;function tr(n,e){return e!=="summary"?n:n.map(wa)}function wa(n){if(n.method!=="session/update")return n;let e=n.params;if(!e||typeof e!="object"||Array.isArray(e))return n;let t=e,s=t.update;if(!s||typeof s!="object"||Array.isArray(s))return n;let r=s;if(r.sessionUpdate!=="tool_call"&&r.sessionUpdate!=="tool_call_update")return n;let i={...r};Array.isArray(r.content)&&(i.content=r.content.map(nr));let o=r.rawOutput;if(o&&typeof o=="object"&&!Array.isArray(o)){let a=o,d={};a.error!==void 0&&(d.error=vn(a.error)),a.metadata!==void 0&&(d.metadata=a.metadata),i.rawOutput=d}return{...n,params:{...t,update:i}}}function va(n){return!!n&&typeof n=="object"&&!Array.isArray(n)&&n.type==="diff"}function nr(n){if(va(n)){let s=n,r={type:"diff",oldText:"",newText:""};return typeof s.path=="string"&&(r.path=s.path),r}if(!n||typeof n!="object"||Array.isArray(n))return n;let e=n,t={...e};return typeof e.text=="string"&&(t.text=vn(e.text)),typeof e.content=="string"?t.content=vn(e.content):e.content&&typeof e.content=="object"&&(t.content=nr(e.content)),t}function vn(n){if(typeof n=="string"&&n.length>wn){let e=n.length-wn;return`${n.slice(0,wn)}\u2026[+${e} chars omitted from summary export]`}return n}var ba=2048;function sr(n){let e=new Set,t=s=>{if(rr(s)){e.add(s.__hydraBlob);return}if(Array.isArray(s)){for(let r of s)t(r);return}if(s&&typeof s=="object")for(let r of Object.values(s))t(r)};for(let s of n)t(s.params);return e}function rr(n){return!!n&&typeof n=="object"&&!Array.isArray(n)&&typeof n.__hydraBlob=="string"}function Ia(n){if(n.method!=="session/update")return!1;let e=n.params;if(!e||typeof e!="object"||Array.isArray(e))return!1;let t=e.update;if(!t||typeof t!="object"||Array.isArray(t))return!1;let s=t.sessionUpdate;return s==="tool_call"||s==="tool_call_update"}async function Sn(n,e){if(!Ia(n))return n;let t=n.params,s=t.update,r=await bn(s,e);return{...n,params:{...t,update:r}}}async function bn(n,e){if(typeof n=="string"){if(n.length<=ba)return n;let t=await e(n);return t===null?n:{__hydraBlob:t,bytes:n.length}}if(Array.isArray(n)){let t=[];for(let s of n)t.push(await bn(s,e));return t}if(n&&typeof n=="object"){let t={};for(let[s,r]of Object.entries(n))t[s]=await bn(r,e);return t}return n}async function ir(n,e){let t=n.params;if(!t||typeof t!="object"||Array.isArray(t))return n;let s=await In(t,e);return s===t?n:{...n,params:s}}async function In(n,e){if(rr(n))return await e(n.__hydraBlob)??"";if(Array.isArray(n)){let t=!1,s=[];for(let r of n){let i=await In(r,e);i!==r&&(t=!0),s.push(i)}return t?s:n}if(n&&typeof n=="object"){let t=!1,s={};for(let[r,i]of Object.entries(n)){let o=await In(i,e);o!==i&&(t=!0),s[r]=o}return t?s:n}return n}var We=/^[A-Za-z0-9_-]+$/,Sa=1e3,bt=class{writeQueues=new Map;maxEntries;constructor(e={}){this.maxEntries=e.maxEntries??Sa}async append(e,t){if(We.test(e))return this.enqueue(e,async()=>{await z.mkdir(w.sessionDir(e),{recursive:!0});let s=await Sn(t,i=>Vt(e,i)),r=JSON.stringify(s)+`
62
82
  `;await z.appendFile(w.historyFile(e),r,{encoding:"utf8",mode:384})})}async rewrite(e,t){if(We.test(e))return this.enqueue(e,async()=>{await z.mkdir(w.sessionDir(e),{recursive:!0});let s=[];for(let i of t)s.push(await Sn(i,o=>Vt(e,o)));let r=s.length===0?"":s.map(i=>JSON.stringify(i)).join(`
63
83
  `)+`
64
84
  `;await z.writeFile(w.historyFile(e),r,{encoding:"utf8",mode:384})})}async compact(e,t){if(We.test(e))return this.enqueue(e,async()=>{let s;try{s=await z.readFile(w.historyFile(e),"utf8")}catch(o){if(o.code==="ENOENT")return;throw o}let r=s.split(`
65
85
  `).filter(o=>o.length>0);if(r.length<=t)return;let i=r.slice(-t);await z.writeFile(w.historyFile(e),i.join(`
66
86
  `)+`
67
87
  `,{encoding:"utf8",mode:384})})}async load(e,t={}){if(!We.test(e))return[];let s=(t.tools??"inline")==="inline",r=this.writeQueues.get(e);r&&await r;let i;try{i=await z.readFile(w.historyFile(e),"utf8")}catch(l){if(l.code==="ENOENT")return[];throw l}let o=[];for(let l of i.split(`
68
- `)){if(l.length===0)continue;let u;try{u=JSON.parse(l)}catch{continue}if(!u||typeof u!="object"||Array.isArray(u))continue;let p=u;typeof p.method=="string"&&typeof p.recordedAt=="number"&&o.push({method:p.method,params:p.params,recordedAt:p.recordedAt})}let a=o.length>this.maxEntries?o.slice(-this.maxEntries):o;if(!s)return a;let d=new Map,c=async l=>{let u=d.get(l);if(u!==void 0)return u;let p=await at(e,l);return d.set(l,p),p},f=[];for(let l of a)f.push(await rr(l,c));return f}async flushAll(){let e=[...this.writeQueues.values()];e.length!==0&&await Promise.allSettled(e)}async delete(e){if(We.test(e))return this.enqueue(e,async()=>{try{await z.unlink(w.historyFile(e))}catch(t){if(t.code!=="ENOENT")throw t}await ns(e);try{await z.rmdir(w.sessionDir(e))}catch(t){let s=t;if(s.code!=="ENOENT"&&s.code!=="ENOTEMPTY")throw t}})}enqueue(e,t){let r=(this.writeQueues.get(e)??Promise.resolve()).then(t,t),i=r.catch(()=>{});return this.writeQueues.set(e,i),i.finally(()=>{this.writeQueues.get(e)===i&&this.writeQueues.delete(e)}),r}};import{promises as ir}from"fs";import*as or from"path";async function ar(n,e){await ir.mkdir(or.dirname(n),{recursive:!0});let t=e.map(s=>JSON.stringify(s));await ir.writeFile(n,t.length>0?t.join(`
88
+ `)){if(l.length===0)continue;let u;try{u=JSON.parse(l)}catch{continue}if(!u||typeof u!="object"||Array.isArray(u))continue;let f=u;typeof f.method=="string"&&typeof f.recordedAt=="number"&&o.push({method:f.method,params:f.params,recordedAt:f.recordedAt})}let a=o.length>this.maxEntries?o.slice(-this.maxEntries):o;if(!s)return a;let d=new Map,c=async l=>{let u=d.get(l);if(u!==void 0)return u;let f=await at(e,l);return d.set(l,f),f},p=[];for(let l of a)p.push(await ir(l,c));return p}async flushAll(){let e=[...this.writeQueues.values()];e.length!==0&&await Promise.allSettled(e)}async delete(e){if(We.test(e))return this.enqueue(e,async()=>{try{await z.unlink(w.historyFile(e))}catch(t){if(t.code!=="ENOENT")throw t}await ns(e);try{await z.rmdir(w.sessionDir(e))}catch(t){let s=t;if(s.code!=="ENOENT"&&s.code!=="ENOTEMPTY")throw t}})}enqueue(e,t){let r=(this.writeQueues.get(e)??Promise.resolve()).then(t,t),i=r.catch(()=>{});return this.writeQueues.set(e,i),i.finally(()=>{this.writeQueues.get(e)===i&&this.writeQueues.delete(e)}),r}};import{promises as or}from"fs";import*as ar from"path";async function dr(n,e){await or.mkdir(ar.dirname(n),{recursive:!0});let t=e.map(s=>JSON.stringify(s));await or.writeFile(n,t.length>0?t.join(`
69
89
  `)+`
70
- `:"")}import{z as _}from"zod";var Sa=_.object({method:_.string(),params:_.unknown(),recordedAt:_.number()}),ka=_.object({sessionId:_.string(),lineageId:_.string(),upstreamSessionId:_.string().optional(),agentId:_.string(),cwd:_.string(),title:_.string().optional(),synopsis:qe.optional(),summarizedThroughEntry:_.number().int().nonnegative().optional(),currentModel:_.string().optional(),currentMode:_.string().optional(),currentUsage:gn.optional(),agentCommands:_.array(pn).optional(),agentModes:_.array(mn).optional(),interactive:_.boolean().optional(),originatingClient:hn.optional(),priority:_.number().int().nonnegative().optional(),createdAt:_.string(),updatedAt:_.string()}),Aa=_.object({version:_.literal(1),exportedAt:_.string(),exportedFrom:_.object({hydraVersion:_.string(),machine:_.string(),hydraHost:_.string().optional()}),session:ka,history:_.array(Sa),promptHistory:_.array(_.string()).optional(),toolBlobs:_.record(_.string()).optional()});function Qe(n){let e={version:1,exportedAt:new Date().toISOString(),exportedFrom:{hydraVersion:n.hydraVersion,machine:n.machine,...n.hydraHost!==void 0&&n.hydraHost.length>0?{hydraHost:n.hydraHost}:{}},session:{sessionId:n.record.sessionId,lineageId:n.record.lineageId,...n.record.upstreamSessionId?{upstreamSessionId:n.record.upstreamSessionId}:{},agentId:n.record.agentId,cwd:n.record.cwd,...n.record.title!==void 0?{title:n.record.title}:{},...n.record.synopsis!==void 0?{synopsis:n.record.synopsis}:{},...n.record.summarizedThroughEntry!==void 0?{summarizedThroughEntry:n.record.summarizedThroughEntry}:{},...n.record.currentModel!==void 0?{currentModel:n.record.currentModel}:{},...n.record.currentMode!==void 0?{currentMode:n.record.currentMode}:{},...n.record.currentUsage!==void 0?{currentUsage:n.record.currentUsage}:{},...n.record.agentCommands!==void 0?{agentCommands:n.record.agentCommands}:{},...n.record.agentModes!==void 0?{agentModes:n.record.agentModes}:{},...n.record.interactive!==void 0?{interactive:n.record.interactive}:{},...n.record.originatingClient!==void 0?{originatingClient:n.record.originatingClient}:{},...n.record.priority!==void 0&&n.record.priority>0?{priority:n.record.priority}:{},createdAt:n.record.createdAt,updatedAt:n.record.updatedAt},history:n.history};return n.promptHistory!==void 0&&(e.promptHistory=n.promptHistory),n.toolBlobs!==void 0&&Object.keys(n.toolBlobs).length>0&&(e.toolBlobs=n.toolBlobs),e}function dr(n){return Aa.parse(n)}var cr=900*1e3,Ca="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz",It=xa(Ca,16),Ge=class n{constructor(e,t,s,r={}){this.registry=e;this.spawner=t??(i=>ae.spawn(i)),this.store=s??new gt,this.tombstones=r.tombstones??new ht,this.sessionHistoryMaxEntries=r.sessionHistoryMaxEntries??1e3,this.histories=new bt({maxEntries:this.sessionHistoryMaxEntries}),this.idleTimeoutMs=r.idleTimeoutMs??0,this.idleEventTimeoutMs=r.idleEventTimeoutMs??3e4,this.defaultModels=r.defaultModels??{},this.synopsisAgent=r.synopsisAgent,this.synopsisModel=r.synopsisModel,this.synopsisOnClose=r.synopsisOnClose??!1,this.defaultTransformers=r.defaultTransformers??[],this.logger=r.logger,this.npmRegistry=r.npmRegistry,this.extensionCommands=r.extensionCommands,this.defaultCwd=r.defaultCwd??"~",this.synopsisCoordinator=new vt({registry:this.registry,store:this.store,histories:this.histories,synopsisAgent:this.synopsisAgent,synopsisModel:this.synopsisModel,persistTitle:async(i,o)=>{let a=this.get(i);if(a){await a.retitle(o);return}await this.persistTitle(i,o)},persistSynopsis:(i,o,a)=>this.persistSynopsis(i,o,a),logger:this.logger,npmRegistry:this.npmRegistry}),this.refreshAgentCatalog()}registry;sessions=new Map;resurrectionInflight=new Map;spawner;store;tombstones;histories;idleTimeoutMs;defaultModels;synopsisAgent;synopsisModel;synopsisOnClose;defaultTransformers;idleEventTimeoutMs;sessionHistoryMaxEntries;metaWriteQueues=new Map;listCache=new Map;static LIST_CACHE_TTL_MS=500;logger;npmRegistry;extensionCommands;defaultCwd;synopsisCoordinator;agentCatalog=[];async refreshAgentCatalog(){try{let{agents:e}=await Ce(this.registry);this.agentCatalog=e.map(t=>({id:t.id,name:t.name,...t.description!==void 0?{description:t.description}:{}}))}catch{}}async create(e){let t=await this.bootstrapAgent({agentId:e.agentId,cwd:e.cwd,agentArgs:e.agentArgs,mcpServers:e.mcpServers,model:e.model,onInstallProgress:e.onInstallProgress});if(e.transformChain&&e.transformChain.length>0){let r={...t.agentCapabilities??{}};for(let i of e.transformChain)if(i.intercepts.has("agent:initialize"))try{let o=await i.connection.request("hydra-acp/transformer/message",{token:`t_${It()}`,phase:"response",method:"initialize",direction:"agent\u2192daemon",sessionId:"(pre-session)",envelope:r});o.action==="stop"&&o.payload&&(r=o.payload)}catch{}t.agentCapabilities=r}let s=new ye({cwd:e.cwd,agentId:e.agentId,agent:t.agent,upstreamSessionId:t.upstreamSessionId,agentMeta:t.agentMeta,agentCapabilities:t.agentCapabilities,title:e.title,agentArgs:e.agentArgs,idleTimeoutMs:this.idleTimeoutMs,idleEventTimeoutMs:this.idleEventTimeoutMs,logger:this.logger,spawnReplacementAgent:r=>this.bootstrapAgent({...r,mcpServers:[]}),listSessions:()=>this.list(),availableAgents:()=>this.agentCatalog,historyStore:this.histories,historyMaxEntries:this.sessionHistoryMaxEntries,currentModel:t.initialModel,currentMode:t.initialMode,agentModes:t.initialModes,agentModels:t.initialModels,transformChain:e.transformChain,parentSessionId:e.parentSessionId,originatingClient:e.originatingClient,interactive:e.interactive,extensionCommands:this.extensionCommands,scheduleSynopsis:()=>this.synopsisCoordinator.schedule(s.sessionId)});return await this.attachManagerHooks(s),s}async resurrect(e){let t=this.sessions.get(e.hydraSessionId);if(t){if(t.upstreamSessionId!==e.upstreamSessionId){let i=new Error(`session ${e.hydraSessionId} already exists with a different upstream id`);throw i.code=I.AlreadyAttached,i}return t}let s=this.resurrectionInflight.get(e.hydraSessionId);if(s)return s;let r=this.doResurrect(e);this.resurrectionInflight.set(e.hydraSessionId,r);try{return await r}finally{this.resurrectionInflight.delete(e.hydraSessionId)}}async doResurrect(e){let t=this.sessions.get(e.hydraSessionId);if(t)return t;let s=await this.registry.getAgent(e.agentId);if(!s){let m=new Error(`agent ${e.agentId} not found in registry; cannot resurrect`);throw m.code=I.AgentNotInstalled,m}if(e.upstreamSessionId==="")return this.doResurrectFromImport(e);if(!await this.dirExists(e.cwd))return this.doResurrectFromImport(e);let r=await oe(s,e.agentArgs??[],{npmRegistry:this.npmRegistry,onInstallProgress:e.onInstallProgress}),i=this.spawner({agentId:e.agentId,cwd:e.cwd,plan:r}),o;try{o=(await i.connection.request("initialize",{protocolVersion:le,clientCapabilities:{},clientInfo:{name:"hydra",version:Q}})).agentCapabilities}catch(m){throw await i.kill().catch(()=>{}),m}let a;try{let m=Pa(e.agentId,e.currentModel);a=await i.connection.request("session/load",{sessionId:e.upstreamSessionId,cwd:e.cwd,mcpServers:e.mcpServers??[],...m&&{_meta:m}})}catch(m){return process.stderr.write(`session/load failed for upstream ${e.upstreamSessionId} on ${e.agentId} (${m.message}); recovering via import-reseed
71
- `),await i.kill().catch(()=>{}),this.doResurrectFromImport(e)}if(e.pendingHistorySync===!0)this.clearPendingHistorySync(e.hydraSessionId).catch(()=>{});else{let m=i.connection.drainBuffered("session/update");this.logger?.info(`resurrect: drain1 dropped ${m} buffered session/update(s) for sessionId=${e.hydraSessionId}`)}let d=mr(a??{}),c=e.agentModes??lr(pr(a??{}));this.logger?.info(`resurrect: sessionId=${e.hydraSessionId} persistedMode=${JSON.stringify(e.currentMode)} agentReportedMode=${JSON.stringify(d)} advertisedModes=${JSON.stringify(c?.map(m=>m.id))}`);let f=await gr({agent:i,upstreamSessionId:e.upstreamSessionId,persistedMode:e.currentMode,agentReportedMode:d,advertisedModes:c,logger:this.logger});this.logger?.info(`resurrect: effectiveMode=${JSON.stringify(f)} for sessionId=${e.hydraSessionId}`);let l=ur(a??{}),u=lr(fr(a??{}))??e.agentModels;if(this.logger?.info(`resurrect: sessionId=${e.hydraSessionId} persistedModel=${JSON.stringify(e.currentModel)} agentReportedModel=${JSON.stringify(l)} advertisedModels=${JSON.stringify(u?.map(m=>m.modelId))}`),e.pendingHistorySync!==!0){let m=i.connection.drainBuffered("session/update");this.logger?.info(`resurrect: drain2 (post-mode-restore) dropped ${m} buffered session/update(s) for sessionId=${e.hydraSessionId}`)}let p=await hr({agent:i,upstreamSessionId:e.upstreamSessionId,persistedModel:e.currentModel,agentReportedModel:l,logger:this.logger});if(e.pendingHistorySync!==!0){let m=i.connection.drainBuffered("session/update");this.logger?.info(`resurrect: drain3 (post-model-restore) dropped ${m} buffered session/update(s) for sessionId=${e.hydraSessionId}`)}let g=new ye({sessionId:e.hydraSessionId,cwd:e.cwd,agentId:e.agentId,agent:i,upstreamSessionId:e.upstreamSessionId,agentMeta:a?._meta,agentCapabilities:o,title:e.title,agentArgs:e.agentArgs,idleTimeoutMs:this.idleTimeoutMs,logger:this.logger,spawnReplacementAgent:m=>this.bootstrapAgent({...m,mcpServers:e.mcpServers??[]}),listSessions:()=>this.list(),availableAgents:()=>this.agentCatalog,historyStore:this.histories,historyMaxEntries:this.sessionHistoryMaxEntries,currentModel:p,currentMode:f,currentUsage:e.currentUsage,agentCommands:e.agentCommands,agentModes:c,agentModels:u,firstPromptSeeded:!!e.title,createdAt:e.createdAt?new Date(e.createdAt).getTime():void 0,originatingClient:e.originatingClient,interactive:e.interactive,priority:e.priority,forkedFromSessionId:e.forkedFromSessionId,forkedFromMessageId:e.forkedFromMessageId,extensionCommands:this.extensionCommands,scheduleSynopsis:()=>this.synopsisCoordinator.schedule(g.sessionId)});return await this.attachManagerHooks(g),g}async doResurrectFromImport(e){let t=await this.resolveResurrectCwd(e.cwd),s=await this.bootstrapAgent({agentId:e.agentId,cwd:t,agentArgs:e.agentArgs,mcpServers:e.mcpServers??[],onInstallProgress:e.onInstallProgress,model:e.currentModel}),r=e.agentModes??s.initialModes,i=await gr({agent:s.agent,upstreamSessionId:s.upstreamSessionId,persistedMode:e.currentMode,agentReportedMode:s.initialMode,advertisedModes:r,logger:this.logger}),o=e.agentModels??s.initialModels,a=await hr({agent:s.agent,upstreamSessionId:s.upstreamSessionId,persistedModel:e.currentModel,agentReportedModel:s.initialModel,logger:this.logger});s.agent.connection.drainBuffered("session/update");let d=new ye({sessionId:e.hydraSessionId,cwd:t,agentId:e.agentId,agent:s.agent,upstreamSessionId:s.upstreamSessionId,agentMeta:s.agentMeta,agentCapabilities:s.agentCapabilities,title:e.title,agentArgs:e.agentArgs,idleTimeoutMs:this.idleTimeoutMs,logger:this.logger,spawnReplacementAgent:c=>this.bootstrapAgent({...c,mcpServers:e.mcpServers??[]}),listSessions:()=>this.list(),availableAgents:()=>this.agentCatalog,historyStore:this.histories,historyMaxEntries:this.sessionHistoryMaxEntries,currentModel:a,currentMode:i,currentUsage:e.currentUsage,agentCommands:e.agentCommands,agentModes:r,agentModels:o,firstPromptSeeded:!!e.title,createdAt:e.createdAt?new Date(e.createdAt).getTime():void 0,originatingClient:e.originatingClient,interactive:e.interactive,priority:e.priority,forkedFromSessionId:e.forkedFromSessionId,forkedFromMessageId:e.forkedFromMessageId,extensionCommands:this.extensionCommands,scheduleSynopsis:()=>this.synopsisCoordinator.schedule(d.sessionId)});return await this.attachManagerHooks(d),d.seedFromImport().catch(()=>{}),d}async dirExists(e){try{return(await ve.stat(e)).isDirectory()}catch{return!1}}async reapIfOrphanedNonInteractive(e){let t=this.sessions.get(e);!t||t.attachedCount>0||t.interactive!==!0&&(this.logger?.info(`reaping orphaned non-interactive session ${e} (agent killed, cold record kept)`),await t.close({deleteRecord:!1}).catch(()=>{}))}async resolveResurrectCwd(e){return await this.dirExists(e)?e:He(this.defaultCwd)}async syncFromAgent(e){let t=await this.registry.getAgent(e);if(!t){let p=new Error(`agent ${e} not found in registry`);throw p.code=I.AgentNotInstalled,p}let s=await oe(t,[],{npmRegistry:this.npmRegistry}),r=this.spawner({agentId:e,cwd:kt.homedir(),plan:s}),i;try{i=await r.connection.request("initialize",{protocolVersion:le,clientCapabilities:{},clientInfo:{name:"hydra",version:Q}})}catch(p){throw await r.kill().catch(()=>{}),p}if((i.agentCapabilities??{}).sessionCapabilities?.list===void 0)throw await r.kill().catch(()=>{}),new Error(`agent ${e} does not advertise sessionCapabilities.list; cannot sync`);let a;try{a=await this.collectAgentSessions(r)}catch(p){throw await r.kill().catch(()=>{}),p}await r.kill().catch(()=>{});let d=new Set;for(let p of this.sessions.values())d.add(`${p.agentId}::${p.upstreamSessionId}`);let c=await this.store.list().catch(()=>[]);for(let p of c)d.add(`${p.agentId}::${p.upstreamSessionId}`);let f=w.sessionsDir(),l=[],u=0;for(let p of a){let g=`${e}::${p.sessionId}`;if(d.has(g)){u+=1;continue}if(Ma(p.cwd,f)){u+=1;continue}let m=await this.tombstones.read(e,p.sessionId).catch(()=>{});if(m){if(!Ls(m,p.updatedAt)){u+=1;continue}await this.tombstones.remove(e,p.sessionId).catch(()=>{}),this.logger?.info(`syncFromAgent: resurrecting tombstoned ${e}/${p.sessionId} (upstream updatedAt advanced past ${m.upstreamUpdatedAt??"<unset>"})`)}d.add(g);let y=`${de}${It()}`,v=new Date().toISOString(),S=p.updatedAt??v,C={sessionId:y,lineageId:Je(),upstreamSessionId:p.sessionId,agentId:e,cwd:p.cwd,pendingHistorySync:!0,interactive:!0,createdAt:S,updatedAt:S};p.title!==void 0&&(C.title=p.title);let k=yn(C);await this.store.write(k),l.push({version:1,...k})}return{synced:l,skipped:u}}async collectAgentSessions(e){let t=[],s;for(let r=0;r<100;r+=1){let i={};s!==void 0&&(i.cursor=s);let o=await e.connection.request("session/list",i),a=Array.isArray(o.sessions)?o.sessions:[];for(let d of a){if(typeof d.sessionId!="string"||typeof d.cwd!="string")continue;let c={sessionId:d.sessionId,cwd:d.cwd};typeof d.title=="string"&&(c.title=d.title),typeof d.updatedAt=="string"&&(c.updatedAt=d.updatedAt),t.push(c)}if(typeof o.nextCursor!="string"||o.nextCursor.length===0)break;s=o.nextCursor}return t}async applySeedModel(e,t,s,r){try{return await e.connection.request("session/set_model",{sessionId:t,modelId:s}),this.logger?.info(`${r}: session/set_model accepted`),!0}catch(i){return this.logger?.warn(`${r} rejected by agent (${i.message}); session will use the agent's own default`),!1}}async bootstrapAgent(e){let t=await this.registry.getAgent(e.agentId);if(!t){let i=new Error(`agent ${e.agentId} not found in registry`);throw i.code=I.AgentNotInstalled,i}let s=await oe(t,e.agentArgs??[],{npmRegistry:this.npmRegistry,onInstallProgress:e.onInstallProgress}),r=this.spawner({agentId:e.agentId,cwd:e.cwd,plan:s});try{let o=(await r.connection.request("initialize",{protocolVersion:le,clientCapabilities:{},clientInfo:{name:"hydra",version:Q}})).agentCapabilities,a=await r.connection.request("session/new",{cwd:e.cwd,mcpServers:e.mcpServers??[]}),d=a.sessionId;if(typeof d!="string")throw new Error(`agent ${e.agentId} returned a non-string sessionId from session/new`);let c=ur(a),f=fr(a),l=e.model??this.defaultModels[e.agentId];if(l&&l!==c){let g=Re(l,f),m=e.model!==void 0?`model=${JSON.stringify(l)}`:`defaultModels[${e.agentId}]=${JSON.stringify(l)}`;if(g.kind==="exact"||g.kind==="none")await this.applySeedModel(r,d,l,m)&&(c=l);else if(g.kind==="resolved")(g.modelId===c||await this.applySeedModel(r,d,g.modelId,`${m} resolved to ${JSON.stringify(g.modelId)}`))&&(c=g.modelId);else if(g.kind==="ambiguous")this.logger?.warn(`${m} is ambiguous (trailing-segment matches [${g.candidates.join(", ")}]); skipping session/set_model, session will use ${JSON.stringify(c)}`);else{let y=f.map(v=>v.modelId).join(", ");this.logger?.warn(`${m} not in agent's availableModels ([${y}]); skipping session/set_model, session will use ${JSON.stringify(c)}`)}}let u=pr(a),p=mr(a);return{agent:r,upstreamSessionId:d,agentMeta:a._meta,agentCapabilities:o,initialModel:c,initialModels:f.length>0?f:void 0,initialModes:u.length>0?u:void 0,initialMode:p}}catch(i){throw await r.kill().catch(()=>{}),i}}async attachManagerHooks(e){e.onClose(({deleteRecord:t})=>{if(this.sessions.delete(e.sessionId),this.invalidateListCache(),t){if(e.upstreamSessionId){let s=St({interactive:e.interactive,...e.originatingClient?{originatingClient:e.originatingClient}:{}},!0);this.tombstones.add({agentId:e.agentId,upstreamSessionId:e.upstreamSessionId,deletedAt:new Date().toISOString(),upstreamUpdatedAt:new Date(e.updatedAt).toISOString(),cwd:e.cwd,title:e.title,reason:"user",...s!==void 0?{interactive:s}:{}}).catch(()=>{})}this.store.delete(e.sessionId).catch(()=>{}),this.histories.delete(e.sessionId).catch(()=>{});return}e.firstPromptSeeded&&this.synopsisOnClose&&this.synopsisCoordinator.schedule(e.sessionId)}),e.onTitleChange(t=>{this.persistTitle(e.sessionId,t).catch(()=>{})}),e.onPriorityChange(t=>{this.persistPriority(e.sessionId,t).catch(()=>{})}),e.onAgentChange(({agentId:t,upstreamSessionId:s})=>{this.persistAgentChange(e.sessionId,t,s).catch(()=>{})}),e.onModelChange(t=>{this.persistSnapshot(e.sessionId,{currentModel:t}).catch(()=>{})}),e.onModeChange(t=>{this.persistSnapshot(e.sessionId,{currentMode:t}).catch(()=>{})}),e.onInteractiveChange(t=>{this.persistSnapshot(e.sessionId,{interactive:t}).catch(()=>{})}),e.onUsageChange(t=>{this.persistSnapshot(e.sessionId,{currentUsage:wr(t)}).catch(()=>{})}),e.onAgentCommandsChange(t=>{this.persistSnapshot(e.sessionId,{agentCommands:t.map(s=>({name:s.name,...s.description!==void 0?{description:s.description}:{}}))}).catch(()=>{})}),e.onAgentModesChange(t=>{this.persistSnapshot(e.sessionId,{agentModes:t.map(s=>({id:s.id,...s.name!==void 0?{name:s.name}:{},...s.description!==void 0?{description:s.description}:{}}))}).catch(()=>{})}),e.onAgentModelsChange(t=>{this.persistSnapshot(e.sessionId,{agentModels:t.map(s=>({modelId:s.modelId,...s.name!==void 0?{name:s.name}:{},...s.description!==void 0?{description:s.description}:{}}))}).catch(()=>{})}),this.sessions.set(e.sessionId,e),this.invalidateListCache(),await this.enqueueMetaWrite(e.sessionId,async()=>{let t=await this.store.read(e.sessionId),s=Ra(e,t);await this.store.write(s)}).catch(()=>{})}async getHistory(e){if(this.sessions.has(e))return this.histories.load(e).catch(()=>[]);if(await this.store.read(e))return this.histories.load(e).catch(()=>[])}async loadHistory(e){return this.histories.load(e)}async loadToolBlob(e,t){return at(e,t)}async loadFromDisk(e){let t=await this.store.read(e);if(!t)return;let s=t.title;return s||(s=await this.deriveTitleFromHistory(e)),{hydraSessionId:t.sessionId,upstreamSessionId:t.upstreamSessionId,agentId:t.agentId,cwd:t.cwd,title:s,synopsis:t.synopsis,summarizedThroughEntry:t.summarizedThroughEntry,agentArgs:t.agentArgs,currentModel:t.currentModel,currentMode:t.currentMode,currentUsage:Ea(t.currentUsage?{...t.currentUsage,cumulativeCost:(t.currentUsage.cumulativeCost??0)+(t.currentUsage.costAmount??0),costAmount:void 0}:void 0),agentCommands:t.agentCommands,agentModes:t.agentModes,agentModels:t.agentModels,createdAt:t.createdAt,pendingHistorySync:t.pendingHistorySync,originatingClient:t.originatingClient,interactive:t.interactive,priority:t.priority,forkedFromSessionId:t.forkedFromSessionId,forkedFromMessageId:t.forkedFromMessageId}}async clearPendingHistorySync(e){await this.enqueueMetaWrite(e,async()=>{let t=await this.store.read(e);if(!t||t.pendingHistorySync!==!0)return;let s={...t};delete s.pendingHistorySync,await this.store.write(s)})}async deriveTitleFromHistory(e){let t=await this.histories.load(e).catch(()=>[]);for(let s of t){let r=s.params;if(r?.update?.sessionUpdate!=="prompt_received")continue;let i=Pe(r.update.prompt),o=un(i,200);if(o)return o}}get(e){return this.sessions.get(e)}liveSessions(){return this.sessions.values()}activeAgentVersions(){let e=new Map;for(let t of this.sessions.values()){let s=t.agent.agentId,r=t.agent.version,i=e.get(s);i||(i=new Set,e.set(s,i)),i.add(r)}return e}async resolveCanonicalId(e){if(this.sessions.has(e)||await this.store.read(e))return e;if(e.startsWith(de))return;let t=de+e;if(this.sessions.has(t)||await this.store.read(t))return t}require(e){let t=this.sessions.get(e);if(!t){let s=new Error(`session ${e} not found`);throw s.code=I.SessionNotFound,s}return t}liveListEntry(e){return{sessionId:e.sessionId,upstreamSessionId:e.upstreamSessionId,cwd:e.cwd,title:e.title,agentId:e.agentId,currentModel:e.currentModel,currentUsage:e.currentUsage,parentSessionId:e.parentSessionId,forkedFromSessionId:e.forkedFromSessionId,forkedFromMessageId:e.forkedFromMessageId,originatingClient:e.originatingClient,interactive:e.interactive,updatedAt:new Date(e.updatedAt).toISOString(),attachedClients:e.attachedCount,status:"live",busy:e.turnStartedAt!==void 0,awaitingInput:e.awaitingInput}}async list(e={}){let t=`${e.cwd??""}|${e.includeNonInteractive?"1":"0"}`,s=Date.now(),r=this.listCache.get(t);if(r&&r.expiresAt>s)return r.promise;let i=this.listUncached(e);return this.listCache.set(t,{expiresAt:s+n.LIST_CACHE_TTL_MS,promise:i}),i.catch(()=>{let o=this.listCache.get(t);o&&o.promise===i&&this.listCache.delete(t)}),i}invalidateListCache(){this.listCache.clear()}async listUncached(e={}){let t=[],s=new Set,r=o=>e.includeNonInteractive?!0:o===!0;for(let o of this.sessions.values()){if(e.cwd&&o.cwd!==e.cwd)continue;s.add(o.sessionId);let a=await kn(o.sessionId),d=St({interactive:o.interactive,...o.originatingClient?{originatingClient:o.originatingClient}:{}},a.hasContent);if(!r(d))continue;let c=a.mtime??new Date(o.updatedAt).toISOString();t.push({sessionId:o.sessionId,upstreamSessionId:o.upstreamSessionId,cwd:o.cwd,title:o.title,agentId:o.agentId,currentModel:o.currentModel,currentUsage:o.currentUsage,parentSessionId:o.parentSessionId,forkedFromSessionId:o.forkedFromSessionId,forkedFromMessageId:o.forkedFromMessageId,originatingClient:o.originatingClient,interactive:d,priority:o.priority,updatedAt:c,attachedClients:o.attachedCount,status:"live",busy:o.turnStartedAt!==void 0,awaitingInput:o.awaitingInput})}let i=await this.store.list().catch(()=>[]);for(let o of i){if(s.has(o.sessionId)||e.cwd&&o.cwd!==e.cwd)continue;let a=await kn(o.sessionId),d=St(o,a.hasContent);if(!r(d))continue;let c=a.mtime??o.updatedAt;t.push({sessionId:o.sessionId,upstreamSessionId:o.upstreamSessionId,cwd:o.cwd,title:o.title,agentId:o.agentId,currentModel:o.currentModel,currentUsage:o.currentUsage?{...o.currentUsage,costAmount:(o.currentUsage.cumulativeCost??0)+(o.currentUsage.costAmount??0)||void 0}:void 0,importedFromMachine:o.importedFromMachine,importedFromUpstreamSessionId:o.importedFromUpstreamSessionId,parentSessionId:o.parentSessionId,forkedFromSessionId:o.forkedFromSessionId,forkedFromMessageId:o.forkedFromMessageId,originatingClient:o.originatingClient,interactive:d,priority:o.priority,updatedAt:c,attachedClients:0,status:"cold",busy:!1,awaitingInput:!1})}return t.sort((o,a)=>o.updatedAt<a.updatedAt?1:-1),t}async exportBundle(e,t={}){let s=await this.store.read(e);if(!s)return;let r;if(s.lineageId)r=s;else{let c=Je(),f={...s,lineageId:c};await this.enqueueMetaWrite(e,async()=>{let l=await this.store.read(e);l&&(l.lineageId||await this.store.write({...l,lineageId:c}))}).catch(()=>{}),r=f}let i=t.tools??"inline",o=await this.histories.load(e,i==="references"?{tools:"references"}:{}).catch(()=>[]),a=await yr(e);if(i!=="references")return{record:r,history:o,promptHistory:a};let d={};for(let c of nr(o)){let f=await es(e,c);f&&(d[c]=f.toString("base64"))}return{record:r,history:o,promptHistory:a,toolBlobs:d}}async importBundle(e,t={}){let s=await this.store.findByLineageId(e.session.lineageId);if(s){if(!t.replace){let o=new Error(`bundle already imported as ${s.sessionId}`);throw o.code=I.BundleAlreadyImported,o.existingSessionId=s.sessionId,o}let i=this.sessions.get(s.sessionId);return i&&await i.close({deleteRecord:!1}).catch(()=>{}),await this.writeImportedRecord({sessionId:s.sessionId,bundle:e,preservedCreatedAt:s.createdAt,cwd:t.cwd}),{sessionId:s.sessionId,importedFromSessionId:e.session.sessionId,replaced:!0}}let r=`${de}${It()}`;return await this.writeImportedRecord({sessionId:r,bundle:e,cwd:t.cwd}),{sessionId:r,importedFromSessionId:e.session.sessionId,replaced:!1}}async forkSession(e,t={}){let s=await this.store.read(e);if(!s){let g=new Error(`source session not found: ${e}`);throw g.code=I.SessionNotFound,g}let r=t.agentId??s.agentId,i=r!==s.agentId;if(i&&!await this.registry.getAgent(r)){let m=new Error(`agent ${r} not found in registry`);throw m.code=I.AgentNotInstalled,m}let o=await this.histories.load(e).catch(()=>[]),a,d;if(t.forkAt!==void 0){if(a=cn(o,t.forkAt),a<0){let g=new Error(`forkAt messageId not found in source history: ${t.forkAt}`);throw g.code=I.InvalidParams,g}d=t.forkAt}else{let g=Ta(o);if(!g){let m=new Error(`source session ${e} has no completed turns to fork from`);throw m.code=I.InvalidParams,m}a=g.index,d=g.messageId}let c=o.slice(0,a+1),f=await yr(e),l={...s,lineageId:Je(),agentId:r,...i?{currentModel:void 0,currentMode:void 0,currentUsage:void 0,agentCommands:void 0,agentModes:void 0,agentModels:void 0}:{}},u=Qe({record:l,history:c,promptHistory:f.length>0?f:void 0,hydraVersion:Q,machine:kt.hostname()}),p=`${de}${It()}`;return await this.writeImportedRecord({sessionId:p,bundle:u,cwd:t.cwd,forkedFromSessionId:e,forkedFromMessageId:d}),{sessionId:p,forkedFromSessionId:e,forkedAt:d}}async writeImportedRecord(e){if(await this.histories.rewrite(e.sessionId,e.bundle.history),e.bundle.toolBlobs)for(let[i,o]of Object.entries(e.bundle.toolBlobs))await ts(e.sessionId,i,Buffer.from(o,"base64")).catch(()=>{});let t=new Date(e.bundle.session.updatedAt);Number.isNaN(t.getTime())||await ve.utimes(w.historyFile(e.sessionId),t,t).catch(()=>{}),e.bundle.promptHistory&&e.bundle.promptHistory.length>0&&await ar(w.tuiHistoryFile(e.sessionId),e.bundle.promptHistory).catch(()=>{});let s=new Date().toISOString(),r=e.forkedFromSessionId!==void 0;await this.enqueueMetaWrite(e.sessionId,async()=>{await this.store.write({sessionId:e.sessionId,lineageId:e.bundle.session.lineageId,upstreamSessionId:"",...r?{forkedFromSessionId:e.forkedFromSessionId,forkedFromMessageId:e.forkedFromMessageId}:{importedFromSessionId:e.bundle.session.sessionId,importedFromUpstreamSessionId:e.bundle.session.upstreamSessionId,importedFromMachine:e.bundle.exportedFrom.machine},agentId:e.bundle.session.agentId,cwd:e.cwd??e.bundle.session.cwd,title:e.bundle.session.title,synopsis:e.bundle.session.synopsis,summarizedThroughEntry:e.bundle.session.summarizedThroughEntry,currentModel:e.bundle.session.currentModel,currentMode:e.bundle.session.currentMode,currentUsage:e.bundle.session.currentUsage,agentCommands:e.bundle.session.agentCommands,agentModes:e.bundle.session.agentModes,interactive:e.bundle.session.interactive,originatingClient:e.bundle.session.originatingClient,priority:e.bundle.session.priority,createdAt:e.preservedCreatedAt??s,updatedAt:e.bundle.session.updatedAt})})}async deleteRecord(e){let t=await this.store.read(e);if(!t)return!1;if(t.upstreamSessionId){let s=await kn(e),r=St(t,s.hasContent);await this.tombstones.add({agentId:t.agentId,upstreamSessionId:t.upstreamSessionId,deletedAt:new Date().toISOString(),upstreamUpdatedAt:t.updatedAt,cwd:t.cwd,title:t.title,reason:"user",...r!==void 0?{interactive:r}:{}}).catch(()=>{})}return await this.store.delete(e).catch(()=>{}),await this.histories.delete(e).catch(()=>{}),this.invalidateListCache(),!0}async hasRecord(e){return await this.store.read(e).catch(()=>{})!==void 0}async setPriority(e,t){let s=this.get(e);if(s)return s.setPriority(t),!0;if(!await this.hasRecord(e))return!1;let r=t===void 0||t<=0?void 0:Math.floor(t);return await this.persistPriority(e,r),!0}async persistPriority(e,t){await this.enqueueMetaWrite(e,async()=>{let s=await this.store.read(e);if(!s)return;let r={...s,updatedAt:new Date().toISOString()};t===void 0?delete r.priority:r.priority=t,await this.store.write(r)}),this.invalidateListCache()}async setTitle(e,t){let s=this.get(e);return s?(await s.retitle(t),!0):await this.hasRecord(e)?(await this.persistTitle(e,t),!0):!1}async persistTitle(e,t){await this.enqueueMetaWrite(e,async()=>{let s=await this.store.read(e);s&&await this.store.write({...s,title:t,updatedAt:new Date().toISOString()})})}async persistSynopsis(e,t,s){await this.enqueueMetaWrite(e,async()=>{let r=await this.store.read(e);r&&await this.store.write({...r,synopsis:t,summarizedThroughEntry:s,updatedAt:new Date().toISOString()})})}async persistAgentChange(e,t,s){await this.enqueueMetaWrite(e,async()=>{let r=await this.store.read(e);r&&await this.store.write({...r,agentId:t,upstreamSessionId:s,updatedAt:new Date().toISOString()})})}async persistSnapshot(e,t){await this.enqueueMetaWrite(e,async()=>{let s=await this.store.read(e);s&&await this.store.write({...s,...t.currentModel!==void 0?{currentModel:t.currentModel}:{},...t.currentMode!==void 0?{currentMode:t.currentMode}:{},...t.currentUsage!==void 0?{currentUsage:t.currentUsage}:{},...t.agentCommands!==void 0?{agentCommands:t.agentCommands}:{},...t.agentModes!==void 0?{agentModes:t.agentModes}:{},...t.agentModels!==void 0?{agentModels:t.agentModels}:{},...t.interactive!==void 0?{interactive:t.interactive}:{},...t.cwd!==void 0?{cwd:t.cwd}:{},updatedAt:new Date().toISOString()})})}enqueueMetaWrite(e,t){let r=(this.metaWriteQueues.get(e)??Promise.resolve()).then(t,t),i=r.catch(()=>{});return this.metaWriteQueues.set(e,i),i.finally(()=>{this.metaWriteQueues.get(e)===i&&this.metaWriteQueues.delete(e)}),r}async closeAll(){let e=[...this.sessions.values()];await Promise.allSettled(e.map(t=>t.close({deleteRecord:!1}))),this.sessions.clear()}async flushSynopsis(e){await this.synopsisCoordinator.flush(e)}async shutdownSynopsis(){await this.synopsisCoordinator.shutdown()}scheduleSynopsis(e){this.synopsisCoordinator.schedule(e)}async flushMetaWrites(){let e=[...this.metaWriteQueues.values()];e.length!==0&&await Promise.allSettled(e)}async flushHistoryWrites(){await this.histories.flushAll()}async resurrectPendingQueues(){let e=await this.store.list().catch(()=>[]);for(let t of e){let s=await _s(t.sessionId).catch(()=>[]);if(s.length===0)continue;let r=Date.now(),i=s.filter(d=>r-d.enqueuedAt<cr),o=s.length-i.length;if(o>0&&(this.logger?.info(`queue replay: dropping ${o} stale prompt(s) for ${t.sessionId} (TTL ${cr/1e3}s)`),await Le(t.sessionId,i).catch(()=>{})),i.length===0)continue;let a=await this.loadFromDisk(t.sessionId).catch(()=>{});if(!a){this.logger?.warn(`queue replay: no meta for ${t.sessionId}; discarding ${i.length} entr${i.length===1?"y":"ies"}`),await Le(t.sessionId,[]).catch(()=>{});continue}try{let d=await this.resurrect(a);this.logger?.info(`queue replay: resurrected ${t.sessionId} and replaying ${i.length} prompt(s)`),d.replayPersistedQueue(i)}catch(d){this.logger?.warn(`queue replay: failed to resurrect ${t.sessionId}: ${d.message}`)}}}};function Ma(n,e){if(typeof n!="string"||n.length===0)return!1;let t=Ve.resolve(n),s=Ve.resolve(e);return t===s||t.startsWith(s+Ve.sep)}function Ra(n,e){let s=(n.mergedAvailableCommands().length>0?n.agentOnlyAdvertisedCommands().map(f=>f.description!==void 0?{name:f.name,description:f.description}:{name:f.name}):void 0)??e?.agentCommands,r=n.availableModes(),o=(r.length>0?r.map(f=>{let l={id:f.id};return f.name!==void 0&&(l.name=f.name),f.description!==void 0&&(l.description=f.description),l}):void 0)??e?.agentModes,a=n.availableModels(),c=(a.length>0?a.map(f=>{let l={modelId:f.modelId};return f.name!==void 0&&(l.name=f.name),f.description!==void 0&&(l.description=f.description),l}):void 0)??e?.agentModels;return yn({sessionId:n.sessionId,lineageId:e?.lineageId??Je(),upstreamSessionId:n.upstreamSessionId,importedFromSessionId:e?.importedFromSessionId,importedFromUpstreamSessionId:e?.importedFromUpstreamSessionId,importedFromMachine:e?.importedFromMachine,agentId:n.agentId,cwd:n.cwd,title:n.title,synopsis:e?.synopsis,summarizedThroughEntry:e?.summarizedThroughEntry,agentArgs:n.agentArgs,currentModel:n.currentModel??e?.currentModel,currentMode:n.currentMode??e?.currentMode,currentUsage:wr(n.currentUsage)??e?.currentUsage,agentCommands:s,agentModes:o,agentModels:c,parentSessionId:n.parentSessionId??e?.parentSessionId,forkedFromSessionId:n.forkedFromSessionId??e?.forkedFromSessionId,forkedFromMessageId:n.forkedFromMessageId??e?.forkedFromMessageId,originatingClient:n.originatingClient??e?.originatingClient,interactive:n.interactive??e?.interactive,priority:n.priority??e?.priority,createdAt:e?.createdAt??new Date(n.createdAt).toISOString()})}function wr(n){if(!n)return;let e={};return n.used!==void 0&&(e.used=n.used),n.size!==void 0&&(e.size=n.size),n.costAmount!==void 0&&(e.costAmount=n.costAmount),n.costCurrency!==void 0&&(e.costCurrency=n.costCurrency),n.cumulativeCost!==void 0&&(e.cumulativeCost=n.cumulativeCost),Object.keys(e).length>0?e:void 0}function Ea(n){return n?{...n}:void 0}function Pa(n,e){if(e&&n==="claude-acp")return{claudeCode:{options:{model:e}}}}function ur(n){let e=j(n.currentModelId)??j(n.currentModel)??j(n.modelId)??j(n.model);if(e)return e;let t=n.models;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=j(t.currentModelId)??j(t.currentModel);if(i)return i}let s=n._meta;if(s&&typeof s=="object"&&!Array.isArray(s)){for(let[i,o]of Object.entries(s))if(i!=="hydra-acp"&&o&&typeof o=="object"&&!Array.isArray(o)){let a=j(o.modelId)??j(o.model)??j(o.currentModelId);if(a)return a}}let r=At(n,"model");if(r){let i=j(r.currentValue);if(i)return i}}function j(n){if(typeof n!="string")return;let e=n.trim();return e.length>0?e:void 0}function At(n,e){let t=n.configOptions;if(Array.isArray(t))for(let s of t){if(!s||typeof s!="object"||Array.isArray(s))continue;let r=s;if(r.id===e)return r}}function lr(n){return n.length>0?n:void 0}function fr(n){let e=we(n.availableModels);if(e.length>0)return e;let t=n.models;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=we(t.availableModels);if(i.length>0)return i}let s=n._meta;if(s&&typeof s=="object"&&!Array.isArray(s)){for(let[i,o]of Object.entries(s))if(i!=="hydra-acp"&&o&&typeof o=="object"&&!Array.isArray(o)){let a=we(o.availableModels);if(a.length>0)return a}}let r=At(n,"model");if(r){let i=we(r.options);if(i.length>0)return i}return[]}function pr(n){let e=Te(n.availableModes);if(e.length>0)return e;let t=n.modes;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=Te(t.availableModes);if(i.length>0)return i}let s=n._meta;if(s&&typeof s=="object"&&!Array.isArray(s)){for(let[i,o]of Object.entries(s))if(i!=="hydra-acp"&&o&&typeof o=="object"&&!Array.isArray(o)){let a=Te(o.availableModes);if(a.length>0)return a}}let r=At(n,"mode");if(r){let i=Te(r.options);if(i.length>0)return i}return[]}function mr(n){let e=j(n.currentModeId)??j(n.currentMode)??j(n.modeId)??j(n.mode);if(e)return e;let t=n.modes;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=j(t.currentModeId)??j(t.currentMode);if(i)return i}let s=n._meta;if(s&&typeof s=="object"&&!Array.isArray(s)){for(let[i,o]of Object.entries(s))if(i!=="hydra-acp"&&o&&typeof o=="object"&&!Array.isArray(o)){let a=j(o.currentModeId)??j(o.currentMode)??j(o.modeId);if(a)return a}}let r=At(n,"mode");if(r){let i=j(r.currentValue);if(i)return i}}async function gr(n){let{agent:e,upstreamSessionId:t,persistedMode:s,agentReportedMode:r,advertisedModes:i,logger:o}=n;if(!s)return r;if(s===r)return s;if(i&&i.length>0&&!i.some(a=>a.id===s)){let a=i.map(d=>d.id).join(", ");return o?.warn(`resurrect: persisted currentMode=${JSON.stringify(s)} not in agent's availableModes ([${a}]); skipping session/set_mode, session will use ${JSON.stringify(r)}`),r}try{return o?.info(`resurrect: pushing persisted modeId=${JSON.stringify(s)} to agent (agentReported=${JSON.stringify(r)})`),await e.connection.request("session/set_mode",{sessionId:t,modeId:s}),o?.info(`resurrect: session/set_mode accepted, effectiveMode=${JSON.stringify(s)}`),s}catch(a){return o?.warn(`resurrect: session/set_mode rejected by agent for modeId=${JSON.stringify(s)} (${a.message}); session will use ${JSON.stringify(r)}`),r}}async function hr(n){let{agent:e,upstreamSessionId:t,persistedModel:s,agentReportedModel:r,logger:i}=n;if(!s)return r;if(s===r)return s;try{return i?.info(`resurrect: pushing persisted modelId=${JSON.stringify(s)} to agent (agentReported=${JSON.stringify(r)})`),await e.connection.request("session/set_model",{sessionId:t,modelId:s}),i?.info(`resurrect: session/set_model accepted, effectiveModel=${JSON.stringify(s)}`),s}catch(o){return i?.warn(`resurrect: session/set_model rejected by agent for modelId=${JSON.stringify(s)} (${o.message}); session will use ${JSON.stringify(r)}`),r}}function Ta(n){for(let e=n.length-1;e>=0;e--){let t=n[e];if(!t||t.method!=="session/update")continue;let s=t.params?.update;if(s?.sessionUpdate==="turn_complete"&&!(typeof s.messageId!="string"||s.messageId.length===0))return{index:e,messageId:s.messageId}}}async function yr(n){try{let e=await ve.readFile(w.tuiHistoryFile(n),"utf8"),t=[];for(let s of e.split(`
72
- `))if(s.length!==0)try{let r=JSON.parse(s);typeof r=="string"&&t.push(r)}catch{}return t}catch{return[]}}async function kn(n){try{let e=await ve.stat(w.historyFile(n));return{mtime:new Date(e.mtimeMs).toISOString(),hasContent:e.size>0}}catch{return{hasContent:!1}}}function St(n,e){return n.interactive!==void 0?n.interactive:n.originatingClient?.name===wt?!1:e?!0:void 0}import{spawn as _a}from"child_process";import*as Fe from"fs";import*as fe from"fs/promises";import*as vr from"path";var xt=class{windowMs;maxFailures;now;recentExits=[];tripped_;constructor(e={}){this.windowMs=e.windowMs??3e5,this.maxFailures=e.maxFailuresInWindow??10,this.now=e.now??Date.now}recordExit(e,t,s){if(e===78){let o=`exited with code 78 (unrecoverable); fix and run \`${s}s start ${t}\``;return this.tripped_=o,{tripped:o}}let r=this.now();this.recentExits.push(r);let i=r-this.windowMs;for(;this.recentExits.length>0&&this.recentExits[0]<i;)this.recentExits.shift();if(this.recentExits.length>this.maxFailures){let o=Math.round(this.windowMs/6e4),a=`${this.recentExits.length} exits in ${o}m (crash loop); fix and run \`${s}s start ${t}\``;return this.tripped_=a,{tripped:a}}return"restart"}reset(){this.recentExits=[],this.tripped_=void 0}get tripped(){return this.tripped_}};var $a=1e3,Fa=6e4,An=3e3,$e=class{entries=new Map;stopping=!1;context;tokenRegistry;breakerOptions;restartBaseMs;restartCapMs;adapter;constructor(e,t,s,r={}){this.adapter=t,this.context=s,this.tokenRegistry=r.tokenRegistry,this.breakerOptions=r.breakerOptions,this.restartBaseMs=r.restartBaseMs??$a,this.restartCapMs=r.restartCapMs??Fa;for(let i of e)this.entries.set(i.name,this.makeEntry(i))}setContext(e){this.context=e}reportVersion(e,t){let s=this.entries.get(e);s&&(s.version=t)}async start(){if(!this.context)throw new Error(`${this.managerName()}: setContext must be called before start`);await fe.mkdir(this.adapter.paths.dir(),{recursive:!0}),await this.reapOrphans();for(let e of this.entries.values())e.config.enabled&&this.spawn(e,0)}async stop(){this.stopping=!0;let e=[];for(let t of this.entries.values()){t.restartTimer&&(clearTimeout(t.restartTimer),t.restartTimer=void 0);let s=t.child;if(s){try{s.kill("SIGTERM")}catch{}e.push(new Promise(r=>{if(s.exitCode!==null||s.signalCode!==null){r();return}let i=setTimeout(()=>{try{s.kill("SIGKILL")}catch{}r()},An);s.on("exit",()=>{clearTimeout(i),r()})}))}}await Promise.allSettled(e);for(let t of this.entries.values()){try{t.logStream?.end()}catch{}t.child=void 0,t.logStream=void 0,t.pid=void 0}}list(){return[...this.entries.values()].map(e=>this.infoFor(e))}get(e){let t=this.entries.get(e);return t?this.infoFor(t):void 0}has(e){return this.entries.has(e)}async startByName(e){let t=this.entries.get(e);if(!t)throw Ye(new Error(`unknown ${this.adapter.kind}: ${e}`),"NOT_FOUND");if(t.child)throw Ye(new Error(`${this.adapter.kind} ${e} already running`),"CONFLICT");return t.restartTimer&&(clearTimeout(t.restartTimer),t.restartTimer=void 0),t.manuallyStopped=!1,t.restartCount=0,t.breaker.reset(),t.failureReason=void 0,this.spawn(t,0),this.infoFor(t)}async stopByName(e){let t=this.entries.get(e);if(!t)throw Ye(new Error(`unknown ${this.adapter.kind}: ${e}`),"NOT_FOUND");t.manuallyStopped=!0,t.restartTimer&&(clearTimeout(t.restartTimer),t.restartTimer=void 0);let s=t.child;return s?(await this.terminate(t,s),this.infoFor(t)):this.infoFor(t)}async restartByName(e){return await this.stopByName(e),this.startByName(e)}register(e){if(this.entries.has(e.name))throw Ye(new Error(`${this.adapter.kind} ${e.name} already exists`),"CONFLICT");if(!this.context)throw new Error(`${this.managerName()}: setContext must be called before register`);let t=this.makeEntry(e);return this.entries.set(e.name,t),e.enabled&&this.spawn(t,0),this.infoFor(t)}async unregister(e){let t=this.entries.get(e);if(!t)throw Ye(new Error(`unknown ${this.adapter.kind}: ${e}`),"NOT_FOUND");t.manuallyStopped=!0,t.restartTimer&&(clearTimeout(t.restartTimer),t.restartTimer=void 0);let s=t.child;s&&await this.terminate(t,s);try{t.logStream?.end()}catch{}this.entries.delete(e)}async terminate(e,t){if(t.exitCode!==null||t.signalCode!==null)return;let s=new Promise(i=>{e.exitWaiters.push(i)});try{t.kill("SIGTERM")}catch{}let r=setTimeout(()=>{try{t.kill("SIGKILL")}catch{}},An);typeof r.unref=="function"&&r.unref();try{await s}finally{clearTimeout(r)}}infoFor(e){let t;return e.failureReason!==void 0?t="failed":e.child?t="running":e.restartTimer?t="restarting":e.config.enabled?t="stopped":t="disabled",{name:e.config.name,status:t,pid:e.pid,enabled:e.config.enabled,restartCount:e.restartCount,startedAt:e.startedAt,lastExitCode:e.lastExitCode,logPath:this.adapter.paths.logFile(e.config.name),version:e.version,failureReason:e.failureReason}}makeEntry(e){return{config:e,child:void 0,logStream:void 0,restartTimer:void 0,pid:void 0,startedAt:void 0,restartCount:0,lastExitCode:void 0,manuallyStopped:!1,exitWaiters:[],version:void 0,processToken:void 0,breaker:new xt(this.breakerOptions),failureReason:void 0}}async reapOrphans(){let e;try{e=await fe.readdir(this.adapter.paths.dir())}catch(t){if(t.code==="ENOENT")return;throw t}for(let t of e){if(!t.endsWith(".pid"))continue;let s=vr.join(this.adapter.paths.dir(),t),r;try{let i=await fe.readFile(s,"utf8"),o=Number.parseInt(i.trim(),10);Number.isInteger(o)&&o>0&&(r=o)}catch{}if(typeof r=="number"&&xn(r)){try{process.kill(r,"SIGTERM")}catch{}let i=Date.now()+An;for(;Date.now()<i&&xn(r);)await new Promise(o=>setTimeout(o,50));if(xn(r))try{process.kill(r,"SIGKILL")}catch{}}await fe.unlink(s).catch(()=>{})}}spawn(e,t){if(this.stopping||e.manuallyStopped)return;let s=this.context;if(!s)throw new Error(`${this.managerName()}.spawn called before setContext`);let r=e.config,i=r.command.length>0?r.command:[r.name],o=Fe.createWriteStream(this.adapter.paths.logFile(r.name),{flags:"a"});o.write(`[hydra-acp] ${new Date().toISOString()} starting ${this.adapter.kind} ${r.name} (attempt ${t+1})
73
- `);let a=this.tokenRegistry?.mint(r.name,this.adapter.tokenRole)??s.serviceToken;e.processToken=a,e.version=void 0;let d={...process.env,HYDRA_ACP_DAEMON_URL:s.daemonUrl,HYDRA_ACP_DAEMON_HOST:s.daemonHost,HYDRA_ACP_DAEMON_PORT:String(s.daemonPort),HYDRA_ACP_TOKEN:a,HYDRA_ACP_WS_URL:s.daemonWsUrl,HYDRA_ACP_HOME:s.hydraHome,[this.adapter.nameEnvVar]:r.name,...r.env},[c,...f]=i;if(c===void 0){o.write(`[hydra-acp] ${this.adapter.kind} ${r.name} has empty command
74
- `),o.end();return}let l=[...f,...r.args],u;try{u=_a(c,l,{env:d,stdio:["ignore","pipe","pipe"],detached:!1})}catch(p){o.write(`[hydra-acp] failed to spawn ${r.name}: ${p.message}
75
- `),o.end(),this.scheduleRestart(e,t);return}if(u.stdout&&u.stdout.pipe(o,{end:!1}),u.stderr&&u.stderr.pipe(o,{end:!1}),typeof u.pid=="number")try{Fe.writeFileSync(this.adapter.paths.pidFile(r.name),`${u.pid}
76
- `,{encoding:"utf8",mode:384})}catch(p){o.write(`[hydra-acp] failed to write pid file for ${r.name}: ${p.message}
77
- `)}e.child=u,e.logStream=o,e.pid=typeof u.pid=="number"?u.pid:void 0,e.startedAt=Date.now(),e.lastExitCode=void 0,u.on("error",p=>{o.write(`[hydra-acp] ${this.adapter.kind} ${r.name} error: ${p.message}
78
- `)}),u.on("exit",(p,g)=>{try{Fe.unlinkSync(this.adapter.paths.pidFile(r.name))}catch{}o.write(`[hydra-acp] ${this.adapter.kind} ${r.name} exited code=${p??"null"} signal=${g??"null"}
79
- `),e.child=void 0,e.pid=void 0,e.lastExitCode=typeof p=="number"?p:void 0,e.processToken&&(this.tokenRegistry?.revoke(r.name),e.processToken=void 0);let m=e.exitWaiters.splice(0);for(let v of m)v();if(this.stopping||e.manuallyStopped){try{o.end()}catch{}e.logStream=void 0;return}e.restartCount+=1;let y=e.breaker.recordExit(p,r.name,this.adapter.kind);if(typeof y=="object"){e.failureReason=y.tripped,o.write(`[hydra-acp] ${this.adapter.kind} ${r.name} circuit breaker tripped: ${y.tripped}
80
- `);try{o.end()}catch{}e.logStream=void 0;return}this.scheduleRestart(e,t+1)})}scheduleRestart(e,t){if(this.stopping||e.manuallyStopped)return;let s=Math.min(this.restartBaseMs*2**Math.min(t,10),this.restartCapMs);e.restartTimer=setTimeout(()=>{e.restartTimer=void 0,this.spawn(e,t)},s),typeof e.restartTimer.unref=="function"&&e.restartTimer.unref()}managerName(){return this.adapter.kind==="extension"?"ExtensionManager":"TransformerManager"}};function xn(n){try{return process.kill(n,0),!0}catch{return!1}}function Ye(n,e){return n.code=e,n}var Oa={kind:"extension",nameEnvVar:"HYDRA_ACP_EXTENSION_NAME",tokenRole:"extension",paths:{dir:w.extensionsDir,logFile:w.extensionLogFile,pidFile:w.extensionPidFile}},Ct=class extends $e{constructor(e,t,s={}){super(e,Oa,t,s)}};var Na={kind:"transformer",nameEnvVar:"HYDRA_ACP_TRANSFORMER_NAME",tokenRole:"transformer",paths:{dir:w.transformersDir,logFile:w.transformerLogFile,pidFile:w.transformerPidFile}},Mt=class extends $e{connected=new Map;constructor(e,t,s={}){super(e,Na,t,s)}registerConnection(e,t,s){this.connected.set(e,{name:e,connection:t,intercepts:new Set(s)})}deregisterConnection(e){this.connected.delete(e)}resolveChain(e){let t=[];for(let s of e){let r=this.connected.get(s);r&&t.push(r)}return t}};var Rt=class{entries=new Map;changeHandlers=[];register(e,t,s){this.entries.set(e,{connection:t,commands:[...s]}),this.fireChanged()}clear(e){this.entries.delete(e)&&this.fireChanged()}get(e){return this.entries.get(e)}has(e){return this.entries.has(e)}list(){let e=[];for(let[t,s]of this.entries)for(let r of s.commands)e.push({name:t,command:r});return e}onChange(e){return this.changeHandlers.push(e),()=>{let t=this.changeHandlers.indexOf(e);t>=0&&this.changeHandlers.splice(t,1)}}fireChanged(){for(let e of this.changeHandlers)try{e()}catch{}}};import*as Xe from"fs/promises";import*as Et from"path";var Ke=n=>{process.stderr.write(n+`
90
+ `:"")}import{z as _}from"zod";var ka=_.object({method:_.string(),params:_.unknown(),recordedAt:_.number()}),Aa=_.object({sessionId:_.string(),lineageId:_.string(),upstreamSessionId:_.string().optional(),agentId:_.string(),cwd:_.string(),title:_.string().optional(),synopsis:qe.optional(),summarizedThroughEntry:_.number().int().nonnegative().optional(),currentModel:_.string().optional(),currentMode:_.string().optional(),currentUsage:gn.optional(),agentCommands:_.array(pn).optional(),agentModes:_.array(mn).optional(),interactive:_.boolean().optional(),originatingClient:hn.optional(),priority:_.number().int().nonnegative().optional(),createdAt:_.string(),updatedAt:_.string()}),xa=_.object({version:_.literal(1),exportedAt:_.string(),exportedFrom:_.object({hydraVersion:_.string(),machine:_.string(),hydraHost:_.string().optional()}),session:Aa,history:_.array(ka),promptHistory:_.array(_.string()).optional(),toolBlobs:_.record(_.string()).optional()});function Qe(n){let e={version:1,exportedAt:new Date().toISOString(),exportedFrom:{hydraVersion:n.hydraVersion,machine:n.machine,...n.hydraHost!==void 0&&n.hydraHost.length>0?{hydraHost:n.hydraHost}:{}},session:{sessionId:n.record.sessionId,lineageId:n.record.lineageId,...n.record.upstreamSessionId?{upstreamSessionId:n.record.upstreamSessionId}:{},agentId:n.record.agentId,cwd:n.record.cwd,...n.record.title!==void 0?{title:n.record.title}:{},...n.record.synopsis!==void 0?{synopsis:n.record.synopsis}:{},...n.record.summarizedThroughEntry!==void 0?{summarizedThroughEntry:n.record.summarizedThroughEntry}:{},...n.record.currentModel!==void 0?{currentModel:n.record.currentModel}:{},...n.record.currentMode!==void 0?{currentMode:n.record.currentMode}:{},...n.record.currentUsage!==void 0?{currentUsage:n.record.currentUsage}:{},...n.record.agentCommands!==void 0?{agentCommands:n.record.agentCommands}:{},...n.record.agentModes!==void 0?{agentModes:n.record.agentModes}:{},...n.record.interactive!==void 0?{interactive:n.record.interactive}:{},...n.record.originatingClient!==void 0?{originatingClient:n.record.originatingClient}:{},...n.record.priority!==void 0&&n.record.priority>0?{priority:n.record.priority}:{},createdAt:n.record.createdAt,updatedAt:n.record.updatedAt},history:n.history};return n.promptHistory!==void 0&&(e.promptHistory=n.promptHistory),n.toolBlobs!==void 0&&Object.keys(n.toolBlobs).length>0&&(e.toolBlobs=n.toolBlobs),e}function cr(n){return xa.parse(n)}var ur=900*1e3,Ma="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz",It=Ca(Ma,16),Ge=class n{constructor(e,t,s,r={}){this.registry=e;this.spawner=t??(i=>ae.spawn(i)),this.store=s??new gt,this.tombstones=r.tombstones??new ht,this.sessionHistoryMaxEntries=r.sessionHistoryMaxEntries??1e3,this.histories=new bt({maxEntries:this.sessionHistoryMaxEntries}),this.idleTimeoutMs=r.idleTimeoutMs??0,this.idleEventTimeoutMs=r.idleEventTimeoutMs??3e4,this.defaultModels=r.defaultModels??{},this.synopsisAgent=r.synopsisAgent,this.synopsisModel=r.synopsisModel,this.synopsisOnClose=r.synopsisOnClose??!1,this.defaultTransformers=r.defaultTransformers??[],this.logger=r.logger,this.npmRegistry=r.npmRegistry,this.extensionCommands=r.extensionCommands,this.defaultCwd=r.defaultCwd??"~",this.synopsisCoordinator=new vt({registry:this.registry,store:this.store,histories:this.histories,synopsisAgent:this.synopsisAgent,synopsisModel:this.synopsisModel,persistTitle:async(i,o)=>{let a=this.get(i);if(a){await a.retitle(o);return}await this.persistTitle(i,o)},persistSynopsis:(i,o,a)=>this.persistSynopsis(i,o,a),logger:this.logger,npmRegistry:this.npmRegistry}),this.refreshAgentCatalog()}registry;sessions=new Map;resurrectionInflight=new Map;spawner;store;tombstones;histories;idleTimeoutMs;defaultModels;synopsisAgent;synopsisModel;synopsisOnClose;defaultTransformers;idleEventTimeoutMs;sessionHistoryMaxEntries;metaWriteQueues=new Map;listCache=new Map;static LIST_CACHE_TTL_MS=500;logger;npmRegistry;extensionCommands;defaultCwd;synopsisCoordinator;agentCatalog=[];async refreshAgentCatalog(){try{let{agents:e}=await Me(this.registry);this.agentCatalog=e.map(t=>({id:t.id,name:t.name,...t.description!==void 0?{description:t.description}:{}}))}catch{}}async create(e){let t=await this.bootstrapAgent({agentId:e.agentId,cwd:e.cwd,agentArgs:e.agentArgs,mcpServers:e.mcpServers,model:e.model,onInstallProgress:e.onInstallProgress});if(e.transformChain&&e.transformChain.length>0){let r={...t.agentCapabilities??{}};for(let i of e.transformChain)if(i.intercepts.has("agent:initialize"))try{let o=await i.connection.request("hydra-acp/transformer/message",{token:`t_${It()}`,phase:"response",method:"initialize",direction:"agent\u2192daemon",sessionId:"(pre-session)",envelope:r});o.action==="stop"&&o.payload&&(r=o.payload)}catch{}t.agentCapabilities=r}let s=new we({cwd:e.cwd,agentId:e.agentId,agent:t.agent,upstreamSessionId:t.upstreamSessionId,agentMeta:t.agentMeta,agentCapabilities:t.agentCapabilities,title:e.title,agentArgs:e.agentArgs,idleTimeoutMs:this.idleTimeoutMs,idleEventTimeoutMs:this.idleEventTimeoutMs,logger:this.logger,spawnReplacementAgent:r=>this.bootstrapAgent({...r,mcpServers:[]}),listSessions:()=>this.list(),availableAgents:()=>this.agentCatalog,historyStore:this.histories,historyMaxEntries:this.sessionHistoryMaxEntries,currentModel:t.initialModel,currentMode:t.initialMode,agentModes:t.initialModes,agentModels:t.initialModels,transformChain:e.transformChain,parentSessionId:e.parentSessionId,originatingClient:e.originatingClient,interactive:e.interactive,extensionCommands:this.extensionCommands,scheduleSynopsis:()=>this.synopsisCoordinator.schedule(s.sessionId)});return await this.attachManagerHooks(s),s}async resurrect(e){let t=this.sessions.get(e.hydraSessionId);if(t){if(t.upstreamSessionId!==e.upstreamSessionId){let i=new Error(`session ${e.hydraSessionId} already exists with a different upstream id`);throw i.code=I.AlreadyAttached,i}return t}let s=this.resurrectionInflight.get(e.hydraSessionId);if(s)return s;let r=this.doResurrect(e);this.resurrectionInflight.set(e.hydraSessionId,r);try{return await r}finally{this.resurrectionInflight.delete(e.hydraSessionId)}}async doResurrect(e){let t=this.sessions.get(e.hydraSessionId);if(t)return t;let s=await this.registry.getAgent(e.agentId);if(!s){let m=new Error(`agent ${e.agentId} not found in registry; cannot resurrect`);throw m.code=I.AgentNotInstalled,m}if(e.upstreamSessionId==="")return this.doResurrectFromImport(e);if(!await this.dirExists(e.cwd))return this.doResurrectFromImport(e);let r=await oe(s,e.agentArgs??[],{npmRegistry:this.npmRegistry,onInstallProgress:e.onInstallProgress}),i=this.spawner({agentId:e.agentId,cwd:e.cwd,plan:r}),o;try{o=(await i.connection.request("initialize",{protocolVersion:le,clientCapabilities:{},clientInfo:{name:"hydra",version:Q}})).agentCapabilities}catch(m){throw await i.kill().catch(()=>{}),m}let a;try{let m=Ta(e.agentId,e.currentModel);a=await i.connection.request("session/load",{sessionId:e.upstreamSessionId,cwd:e.cwd,mcpServers:e.mcpServers??[],...m&&{_meta:m}})}catch(m){return process.stderr.write(`session/load failed for upstream ${e.upstreamSessionId} on ${e.agentId} (${m.message}); recovering via import-reseed
91
+ `),await i.kill().catch(()=>{}),this.doResurrectFromImport(e)}if(e.pendingHistorySync===!0)this.clearPendingHistorySync(e.hydraSessionId).catch(()=>{});else{let m=i.connection.drainBuffered("session/update");this.logger?.info(`resurrect: drain1 dropped ${m} buffered session/update(s) for sessionId=${e.hydraSessionId}`)}let d=gr(a??{}),c=e.agentModes??fr(mr(a??{}));this.logger?.info(`resurrect: sessionId=${e.hydraSessionId} persistedMode=${JSON.stringify(e.currentMode)} agentReportedMode=${JSON.stringify(d)} advertisedModes=${JSON.stringify(c?.map(m=>m.id))}`);let p=await hr({agent:i,upstreamSessionId:e.upstreamSessionId,persistedMode:e.currentMode,agentReportedMode:d,advertisedModes:c,logger:this.logger});this.logger?.info(`resurrect: effectiveMode=${JSON.stringify(p)} for sessionId=${e.hydraSessionId}`);let l=lr(a??{}),u=fr(pr(a??{}))??e.agentModels;if(this.logger?.info(`resurrect: sessionId=${e.hydraSessionId} persistedModel=${JSON.stringify(e.currentModel)} agentReportedModel=${JSON.stringify(l)} advertisedModels=${JSON.stringify(u?.map(m=>m.modelId))}`),e.pendingHistorySync!==!0){let m=i.connection.drainBuffered("session/update");this.logger?.info(`resurrect: drain2 (post-mode-restore) dropped ${m} buffered session/update(s) for sessionId=${e.hydraSessionId}`)}let f=await yr({agent:i,upstreamSessionId:e.upstreamSessionId,persistedModel:e.currentModel,agentReportedModel:l,logger:this.logger});if(e.pendingHistorySync!==!0){let m=i.connection.drainBuffered("session/update");this.logger?.info(`resurrect: drain3 (post-model-restore) dropped ${m} buffered session/update(s) for sessionId=${e.hydraSessionId}`)}let g=new we({sessionId:e.hydraSessionId,cwd:e.cwd,agentId:e.agentId,agent:i,upstreamSessionId:e.upstreamSessionId,agentMeta:a?._meta,agentCapabilities:o,title:e.title,agentArgs:e.agentArgs,idleTimeoutMs:this.idleTimeoutMs,logger:this.logger,spawnReplacementAgent:m=>this.bootstrapAgent({...m,mcpServers:e.mcpServers??[]}),listSessions:()=>this.list(),availableAgents:()=>this.agentCatalog,historyStore:this.histories,historyMaxEntries:this.sessionHistoryMaxEntries,currentModel:f,currentMode:p,currentUsage:e.currentUsage,agentCommands:e.agentCommands,agentModes:c,agentModels:u,firstPromptSeeded:!!e.title,createdAt:e.createdAt?new Date(e.createdAt).getTime():void 0,originatingClient:e.originatingClient,interactive:e.interactive,priority:e.priority,forkedFromSessionId:e.forkedFromSessionId,forkedFromMessageId:e.forkedFromMessageId,extensionCommands:this.extensionCommands,scheduleSynopsis:()=>this.synopsisCoordinator.schedule(g.sessionId)});return await this.attachManagerHooks(g),g}async doResurrectFromImport(e){let t=await this.resolveResurrectCwd(e.cwd),s=await this.bootstrapAgent({agentId:e.agentId,cwd:t,agentArgs:e.agentArgs,mcpServers:e.mcpServers??[],onInstallProgress:e.onInstallProgress,model:e.currentModel}),r=e.agentModes??s.initialModes,i=await hr({agent:s.agent,upstreamSessionId:s.upstreamSessionId,persistedMode:e.currentMode,agentReportedMode:s.initialMode,advertisedModes:r,logger:this.logger}),o=e.agentModels??s.initialModels,a=await yr({agent:s.agent,upstreamSessionId:s.upstreamSessionId,persistedModel:e.currentModel,agentReportedModel:s.initialModel,logger:this.logger});s.agent.connection.drainBuffered("session/update");let d=new we({sessionId:e.hydraSessionId,cwd:t,agentId:e.agentId,agent:s.agent,upstreamSessionId:s.upstreamSessionId,agentMeta:s.agentMeta,agentCapabilities:s.agentCapabilities,title:e.title,agentArgs:e.agentArgs,idleTimeoutMs:this.idleTimeoutMs,logger:this.logger,spawnReplacementAgent:c=>this.bootstrapAgent({...c,mcpServers:e.mcpServers??[]}),listSessions:()=>this.list(),availableAgents:()=>this.agentCatalog,historyStore:this.histories,historyMaxEntries:this.sessionHistoryMaxEntries,currentModel:a,currentMode:i,currentUsage:e.currentUsage,agentCommands:e.agentCommands,agentModes:r,agentModels:o,firstPromptSeeded:!!e.title,createdAt:e.createdAt?new Date(e.createdAt).getTime():void 0,originatingClient:e.originatingClient,interactive:e.interactive,priority:e.priority,forkedFromSessionId:e.forkedFromSessionId,forkedFromMessageId:e.forkedFromMessageId,extensionCommands:this.extensionCommands,scheduleSynopsis:()=>this.synopsisCoordinator.schedule(d.sessionId)});return await this.attachManagerHooks(d),d.seedFromImport().catch(()=>{}),d}async dirExists(e){try{return(await be.stat(e)).isDirectory()}catch{return!1}}async reapIfOrphanedNonInteractive(e){let t=this.sessions.get(e);!t||t.attachedCount>0||t.interactive!==!0&&(this.logger?.info(`reaping orphaned non-interactive session ${e} (agent killed, cold record kept)`),await t.close({deleteRecord:!1}).catch(()=>{}))}async resolveResurrectCwd(e){return await this.dirExists(e)?e:He(this.defaultCwd)}async syncFromAgent(e){let t=await this.registry.getAgent(e);if(!t){let f=new Error(`agent ${e} not found in registry`);throw f.code=I.AgentNotInstalled,f}let s=await oe(t,[],{npmRegistry:this.npmRegistry}),r=this.spawner({agentId:e,cwd:kt.homedir(),plan:s}),i;try{i=await r.connection.request("initialize",{protocolVersion:le,clientCapabilities:{},clientInfo:{name:"hydra",version:Q}})}catch(f){throw await r.kill().catch(()=>{}),f}if((i.agentCapabilities??{}).sessionCapabilities?.list===void 0)throw await r.kill().catch(()=>{}),new Error(`agent ${e} does not advertise sessionCapabilities.list; cannot sync`);let a;try{a=await this.collectAgentSessions(r)}catch(f){throw await r.kill().catch(()=>{}),f}await r.kill().catch(()=>{});let d=new Set;for(let f of this.sessions.values())d.add(`${f.agentId}::${f.upstreamSessionId}`);let c=await this.store.list().catch(()=>[]);for(let f of c)d.add(`${f.agentId}::${f.upstreamSessionId}`);let p=w.sessionsDir(),l=[],u=0;for(let f of a){let g=`${e}::${f.sessionId}`;if(d.has(g)){u+=1;continue}if(Ra(f.cwd,p)){u+=1;continue}let m=await this.tombstones.read(e,f.sessionId).catch(()=>{});if(m){if(!qs(m,f.updatedAt)){u+=1;continue}await this.tombstones.remove(e,f.sessionId).catch(()=>{}),this.logger?.info(`syncFromAgent: resurrecting tombstoned ${e}/${f.sessionId} (upstream updatedAt advanced past ${m.upstreamUpdatedAt??"<unset>"})`)}d.add(g);let y=`${de}${It()}`,v=new Date().toISOString(),S=f.updatedAt??v,C={sessionId:y,lineageId:Je(),upstreamSessionId:f.sessionId,agentId:e,cwd:f.cwd,pendingHistorySync:!0,interactive:!0,createdAt:S,updatedAt:S};f.title!==void 0&&(C.title=f.title);let k=yn(C);await this.store.write(k),l.push({version:1,...k})}return{synced:l,skipped:u}}async collectAgentSessions(e){let t=[],s;for(let r=0;r<100;r+=1){let i={};s!==void 0&&(i.cursor=s);let o=await e.connection.request("session/list",i),a=Array.isArray(o.sessions)?o.sessions:[];for(let d of a){if(typeof d.sessionId!="string"||typeof d.cwd!="string")continue;let c={sessionId:d.sessionId,cwd:d.cwd};typeof d.title=="string"&&(c.title=d.title),typeof d.updatedAt=="string"&&(c.updatedAt=d.updatedAt),t.push(c)}if(typeof o.nextCursor!="string"||o.nextCursor.length===0)break;s=o.nextCursor}return t}async applySeedModel(e,t,s,r){try{return await e.connection.request("session/set_model",{sessionId:t,modelId:s}),this.logger?.info(`${r}: session/set_model accepted`),!0}catch(i){return this.logger?.warn(`${r} rejected by agent (${i.message}); session will use the agent's own default`),!1}}async bootstrapAgent(e){let t=await this.registry.getAgent(e.agentId);if(!t){let i=new Error(`agent ${e.agentId} not found in registry`);throw i.code=I.AgentNotInstalled,i}let s=await oe(t,e.agentArgs??[],{npmRegistry:this.npmRegistry,onInstallProgress:e.onInstallProgress}),r=this.spawner({agentId:e.agentId,cwd:e.cwd,plan:s});try{let o=(await r.connection.request("initialize",{protocolVersion:le,clientCapabilities:{},clientInfo:{name:"hydra",version:Q}})).agentCapabilities,a=await r.connection.request("session/new",{cwd:e.cwd,mcpServers:e.mcpServers??[]}),d=a.sessionId;if(typeof d!="string")throw new Error(`agent ${e.agentId} returned a non-string sessionId from session/new`);let c=lr(a),p=pr(a),l=e.model??this.defaultModels[e.agentId];if(l&&l!==c){let g=Ee(l,p),m=e.model!==void 0?`model=${JSON.stringify(l)}`:`defaultModels[${e.agentId}]=${JSON.stringify(l)}`;if(g.kind==="exact"||g.kind==="none")await this.applySeedModel(r,d,l,m)&&(c=l);else if(g.kind==="resolved")(g.modelId===c||await this.applySeedModel(r,d,g.modelId,`${m} resolved to ${JSON.stringify(g.modelId)}`))&&(c=g.modelId);else if(g.kind==="ambiguous")this.logger?.warn(`${m} is ambiguous (trailing-segment matches [${g.candidates.join(", ")}]); skipping session/set_model, session will use ${JSON.stringify(c)}`);else{let y=p.map(v=>v.modelId).join(", ");this.logger?.warn(`${m} not in agent's availableModels ([${y}]); skipping session/set_model, session will use ${JSON.stringify(c)}`)}}let u=mr(a),f=gr(a);return{agent:r,upstreamSessionId:d,agentMeta:a._meta,agentCapabilities:o,initialModel:c,initialModels:p.length>0?p:void 0,initialModes:u.length>0?u:void 0,initialMode:f}}catch(i){throw await r.kill().catch(()=>{}),i}}async attachManagerHooks(e){e.onClose(({deleteRecord:t})=>{if(this.sessions.delete(e.sessionId),this.invalidateListCache(),t){if(e.upstreamSessionId){let s=St({interactive:e.interactive,...e.originatingClient?{originatingClient:e.originatingClient}:{}},!0);this.tombstones.add({agentId:e.agentId,upstreamSessionId:e.upstreamSessionId,deletedAt:new Date().toISOString(),upstreamUpdatedAt:new Date(e.updatedAt).toISOString(),cwd:e.cwd,title:e.title,reason:"user",...s!==void 0?{interactive:s}:{}}).catch(()=>{})}this.store.delete(e.sessionId).catch(()=>{}),this.histories.delete(e.sessionId).catch(()=>{});return}e.firstPromptSeeded&&this.synopsisOnClose&&this.synopsisCoordinator.schedule(e.sessionId)}),e.onTitleChange(t=>{this.persistTitle(e.sessionId,t).catch(()=>{})}),e.onPriorityChange(t=>{this.persistPriority(e.sessionId,t).catch(()=>{})}),e.onAgentChange(({agentId:t,upstreamSessionId:s})=>{this.persistAgentChange(e.sessionId,t,s).catch(()=>{})}),e.onModelChange(t=>{this.persistSnapshot(e.sessionId,{currentModel:t}).catch(()=>{})}),e.onModeChange(t=>{this.persistSnapshot(e.sessionId,{currentMode:t}).catch(()=>{})}),e.onInteractiveChange(t=>{this.persistSnapshot(e.sessionId,{interactive:t}).catch(()=>{})}),e.onUsageChange(t=>{this.persistSnapshot(e.sessionId,{currentUsage:vr(t)}).catch(()=>{})}),e.onAgentCommandsChange(t=>{this.persistSnapshot(e.sessionId,{agentCommands:t.map(s=>({name:s.name,...s.description!==void 0?{description:s.description}:{}}))}).catch(()=>{})}),e.onAgentModesChange(t=>{this.persistSnapshot(e.sessionId,{agentModes:t.map(s=>({id:s.id,...s.name!==void 0?{name:s.name}:{},...s.description!==void 0?{description:s.description}:{}}))}).catch(()=>{})}),e.onAgentModelsChange(t=>{this.persistSnapshot(e.sessionId,{agentModels:t.map(s=>({modelId:s.modelId,...s.name!==void 0?{name:s.name}:{},...s.description!==void 0?{description:s.description}:{}}))}).catch(()=>{})}),this.sessions.set(e.sessionId,e),this.invalidateListCache(),await this.enqueueMetaWrite(e.sessionId,async()=>{let t=await this.store.read(e.sessionId),s=Ea(e,t);await this.store.write(s)}).catch(()=>{})}async getHistory(e){if(this.sessions.has(e))return this.histories.load(e).catch(()=>[]);if(await this.store.read(e))return this.histories.load(e).catch(()=>[])}async loadHistory(e){return this.histories.load(e)}async loadToolBlob(e,t){return at(e,t)}async loadFromDisk(e){let t=await this.store.read(e);if(!t)return;let s=t.title;return s||(s=await this.deriveTitleFromHistory(e)),{hydraSessionId:t.sessionId,upstreamSessionId:t.upstreamSessionId,agentId:t.agentId,cwd:t.cwd,title:s,synopsis:t.synopsis,summarizedThroughEntry:t.summarizedThroughEntry,agentArgs:t.agentArgs,currentModel:t.currentModel,currentMode:t.currentMode,currentUsage:Pa(t.currentUsage?{...t.currentUsage,cumulativeCost:(t.currentUsage.cumulativeCost??0)+(t.currentUsage.costAmount??0),costAmount:void 0}:void 0),agentCommands:t.agentCommands,agentModes:t.agentModes,agentModels:t.agentModels,createdAt:t.createdAt,pendingHistorySync:t.pendingHistorySync,originatingClient:t.originatingClient,interactive:t.interactive,priority:t.priority,forkedFromSessionId:t.forkedFromSessionId,forkedFromMessageId:t.forkedFromMessageId}}async clearPendingHistorySync(e){await this.enqueueMetaWrite(e,async()=>{let t=await this.store.read(e);if(!t||t.pendingHistorySync!==!0)return;let s={...t};delete s.pendingHistorySync,await this.store.write(s)})}async deriveTitleFromHistory(e){let t=await this.histories.load(e).catch(()=>[]);for(let s of t){let r=s.params;if(r?.update?.sessionUpdate!=="prompt_received")continue;let i=Te(r.update.prompt),o=un(i,200);if(o)return o}}get(e){return this.sessions.get(e)}liveSessions(){return this.sessions.values()}activeAgentVersions(){let e=new Map;for(let t of this.sessions.values()){let s=t.agent.agentId,r=t.agent.version,i=e.get(s);i||(i=new Set,e.set(s,i)),i.add(r)}return e}async resolveCanonicalId(e){if(this.sessions.has(e)||await this.store.read(e))return e;if(e.startsWith(de))return;let t=de+e;if(this.sessions.has(t)||await this.store.read(t))return t}require(e){let t=this.sessions.get(e);if(!t){let s=new Error(`session ${e} not found`);throw s.code=I.SessionNotFound,s}return t}liveListEntry(e){return{sessionId:e.sessionId,upstreamSessionId:e.upstreamSessionId,cwd:e.cwd,title:e.title,agentId:e.agentId,currentModel:e.currentModel,currentUsage:e.currentUsage,parentSessionId:e.parentSessionId,forkedFromSessionId:e.forkedFromSessionId,forkedFromMessageId:e.forkedFromMessageId,originatingClient:e.originatingClient,interactive:e.interactive,updatedAt:new Date(e.updatedAt).toISOString(),attachedClients:e.attachedCount,status:"live",busy:e.turnStartedAt!==void 0,awaitingInput:e.awaitingInput}}async list(e={}){let t=`${e.cwd??""}|${e.includeNonInteractive?"1":"0"}`,s=Date.now(),r=this.listCache.get(t);if(r&&r.expiresAt>s)return r.promise;let i=this.listUncached(e);return this.listCache.set(t,{expiresAt:s+n.LIST_CACHE_TTL_MS,promise:i}),i.catch(()=>{let o=this.listCache.get(t);o&&o.promise===i&&this.listCache.delete(t)}),i}invalidateListCache(){this.listCache.clear()}async listUncached(e={}){let t=[],s=new Set,r=o=>e.includeNonInteractive?!0:o===!0;for(let o of this.sessions.values()){if(e.cwd&&o.cwd!==e.cwd)continue;s.add(o.sessionId);let a=await kn(o.sessionId),d=St({interactive:o.interactive,...o.originatingClient?{originatingClient:o.originatingClient}:{}},a.hasContent);if(!r(d))continue;let c=a.mtime??new Date(o.updatedAt).toISOString();t.push({sessionId:o.sessionId,upstreamSessionId:o.upstreamSessionId,cwd:o.cwd,title:o.title,agentId:o.agentId,currentModel:o.currentModel,currentUsage:o.currentUsage,parentSessionId:o.parentSessionId,forkedFromSessionId:o.forkedFromSessionId,forkedFromMessageId:o.forkedFromMessageId,originatingClient:o.originatingClient,interactive:d,priority:o.priority,updatedAt:c,attachedClients:o.attachedCount,status:"live",busy:o.turnStartedAt!==void 0,awaitingInput:o.awaitingInput})}let i=await this.store.list().catch(()=>[]);for(let o of i){if(s.has(o.sessionId)||e.cwd&&o.cwd!==e.cwd)continue;let a=await kn(o.sessionId),d=St(o,a.hasContent);if(!r(d))continue;let c=a.mtime??o.updatedAt;t.push({sessionId:o.sessionId,upstreamSessionId:o.upstreamSessionId,cwd:o.cwd,title:o.title,agentId:o.agentId,currentModel:o.currentModel,currentUsage:o.currentUsage?{...o.currentUsage,costAmount:(o.currentUsage.cumulativeCost??0)+(o.currentUsage.costAmount??0)||void 0}:void 0,importedFromMachine:o.importedFromMachine,importedFromUpstreamSessionId:o.importedFromUpstreamSessionId,parentSessionId:o.parentSessionId,forkedFromSessionId:o.forkedFromSessionId,forkedFromMessageId:o.forkedFromMessageId,originatingClient:o.originatingClient,interactive:d,priority:o.priority,updatedAt:c,attachedClients:0,status:"cold",busy:!1,awaitingInput:!1})}return t.sort((o,a)=>o.updatedAt<a.updatedAt?1:-1),t}async exportBundle(e,t={}){let s=await this.store.read(e);if(!s)return;let r;if(s.lineageId)r=s;else{let c=Je(),p={...s,lineageId:c};await this.enqueueMetaWrite(e,async()=>{let l=await this.store.read(e);l&&(l.lineageId||await this.store.write({...l,lineageId:c}))}).catch(()=>{}),r=p}let i=t.tools??"inline",o=await this.histories.load(e,i==="references"?{tools:"references"}:{}).catch(()=>[]),a=await wr(e);if(i!=="references")return{record:r,history:o,promptHistory:a};let d={};for(let c of sr(o)){let p=await es(e,c);p&&(d[c]=p.toString("base64"))}return{record:r,history:o,promptHistory:a,toolBlobs:d}}async importBundle(e,t={}){let s=await this.store.findByLineageId(e.session.lineageId);if(s){if(!t.replace){let o=new Error(`bundle already imported as ${s.sessionId}`);throw o.code=I.BundleAlreadyImported,o.existingSessionId=s.sessionId,o}let i=this.sessions.get(s.sessionId);return i&&await i.close({deleteRecord:!1}).catch(()=>{}),await this.writeImportedRecord({sessionId:s.sessionId,bundle:e,preservedCreatedAt:s.createdAt,cwd:t.cwd}),{sessionId:s.sessionId,importedFromSessionId:e.session.sessionId,replaced:!0}}let r=`${de}${It()}`;return await this.writeImportedRecord({sessionId:r,bundle:e,cwd:t.cwd}),{sessionId:r,importedFromSessionId:e.session.sessionId,replaced:!1}}async forkSession(e,t={}){let s=await this.store.read(e);if(!s){let g=new Error(`source session not found: ${e}`);throw g.code=I.SessionNotFound,g}let r=t.agentId??s.agentId,i=r!==s.agentId;if(i&&!await this.registry.getAgent(r)){let m=new Error(`agent ${r} not found in registry`);throw m.code=I.AgentNotInstalled,m}let o=await this.histories.load(e).catch(()=>[]),a,d;if(t.forkAt!==void 0){if(a=cn(o,t.forkAt),a<0){let g=new Error(`forkAt messageId not found in source history: ${t.forkAt}`);throw g.code=I.InvalidParams,g}d=t.forkAt}else{let g=_a(o);if(!g){let m=new Error(`source session ${e} has no completed turns to fork from`);throw m.code=I.InvalidParams,m}a=g.index,d=g.messageId}let c=o.slice(0,a+1),p=await wr(e),l={...s,lineageId:Je(),agentId:r,...i?{currentModel:void 0,currentMode:void 0,currentUsage:void 0,agentCommands:void 0,agentModes:void 0,agentModels:void 0}:{}},u=Qe({record:l,history:c,promptHistory:p.length>0?p:void 0,hydraVersion:Q,machine:kt.hostname()}),f=`${de}${It()}`;return await this.writeImportedRecord({sessionId:f,bundle:u,cwd:t.cwd,forkedFromSessionId:e,forkedFromMessageId:d}),{sessionId:f,forkedFromSessionId:e,forkedAt:d}}async writeImportedRecord(e){if(await this.histories.rewrite(e.sessionId,e.bundle.history),e.bundle.toolBlobs)for(let[i,o]of Object.entries(e.bundle.toolBlobs))await ts(e.sessionId,i,Buffer.from(o,"base64")).catch(()=>{});let t=new Date(e.bundle.session.updatedAt);Number.isNaN(t.getTime())||await be.utimes(w.historyFile(e.sessionId),t,t).catch(()=>{}),e.bundle.promptHistory&&e.bundle.promptHistory.length>0&&await dr(w.tuiHistoryFile(e.sessionId),e.bundle.promptHistory).catch(()=>{});let s=new Date().toISOString(),r=e.forkedFromSessionId!==void 0;await this.enqueueMetaWrite(e.sessionId,async()=>{await this.store.write({sessionId:e.sessionId,lineageId:e.bundle.session.lineageId,upstreamSessionId:"",...r?{forkedFromSessionId:e.forkedFromSessionId,forkedFromMessageId:e.forkedFromMessageId}:{importedFromSessionId:e.bundle.session.sessionId,importedFromUpstreamSessionId:e.bundle.session.upstreamSessionId,importedFromMachine:e.bundle.exportedFrom.machine},agentId:e.bundle.session.agentId,cwd:e.cwd??e.bundle.session.cwd,title:e.bundle.session.title,synopsis:e.bundle.session.synopsis,summarizedThroughEntry:e.bundle.session.summarizedThroughEntry,currentModel:e.bundle.session.currentModel,currentMode:e.bundle.session.currentMode,currentUsage:e.bundle.session.currentUsage,agentCommands:e.bundle.session.agentCommands,agentModes:e.bundle.session.agentModes,interactive:e.bundle.session.interactive,originatingClient:e.bundle.session.originatingClient,priority:e.bundle.session.priority,createdAt:e.preservedCreatedAt??s,updatedAt:e.bundle.session.updatedAt})})}async deleteRecord(e){let t=await this.store.read(e);if(!t)return!1;if(t.upstreamSessionId){let s=await kn(e),r=St(t,s.hasContent);await this.tombstones.add({agentId:t.agentId,upstreamSessionId:t.upstreamSessionId,deletedAt:new Date().toISOString(),upstreamUpdatedAt:t.updatedAt,cwd:t.cwd,title:t.title,reason:"user",...r!==void 0?{interactive:r}:{}}).catch(()=>{})}return await this.store.delete(e).catch(()=>{}),await this.histories.delete(e).catch(()=>{}),this.invalidateListCache(),!0}async hasRecord(e){return await this.store.read(e).catch(()=>{})!==void 0}async setPriority(e,t){let s=this.get(e);if(s)return s.setPriority(t),!0;if(!await this.hasRecord(e))return!1;let r=t===void 0||t<=0?void 0:Math.floor(t);return await this.persistPriority(e,r),!0}async persistPriority(e,t){await this.enqueueMetaWrite(e,async()=>{let s=await this.store.read(e);if(!s)return;let r={...s,updatedAt:new Date().toISOString()};t===void 0?delete r.priority:r.priority=t,await this.store.write(r)}),this.invalidateListCache()}async setTitle(e,t){let s=this.get(e);return s?(await s.retitle(t),!0):await this.hasRecord(e)?(await this.persistTitle(e,t),!0):!1}async persistTitle(e,t){await this.enqueueMetaWrite(e,async()=>{let s=await this.store.read(e);s&&await this.store.write({...s,title:t,updatedAt:new Date().toISOString()})})}async persistSynopsis(e,t,s){await this.enqueueMetaWrite(e,async()=>{let r=await this.store.read(e);r&&await this.store.write({...r,synopsis:t,summarizedThroughEntry:s,updatedAt:new Date().toISOString()})})}async persistAgentChange(e,t,s){await this.enqueueMetaWrite(e,async()=>{let r=await this.store.read(e);r&&await this.store.write({...r,agentId:t,upstreamSessionId:s,updatedAt:new Date().toISOString()})})}async persistSnapshot(e,t){await this.enqueueMetaWrite(e,async()=>{let s=await this.store.read(e);s&&await this.store.write({...s,...t.currentModel!==void 0?{currentModel:t.currentModel}:{},...t.currentMode!==void 0?{currentMode:t.currentMode}:{},...t.currentUsage!==void 0?{currentUsage:t.currentUsage}:{},...t.agentCommands!==void 0?{agentCommands:t.agentCommands}:{},...t.agentModes!==void 0?{agentModes:t.agentModes}:{},...t.agentModels!==void 0?{agentModels:t.agentModels}:{},...t.interactive!==void 0?{interactive:t.interactive}:{},...t.cwd!==void 0?{cwd:t.cwd}:{},updatedAt:new Date().toISOString()})})}enqueueMetaWrite(e,t){let r=(this.metaWriteQueues.get(e)??Promise.resolve()).then(t,t),i=r.catch(()=>{});return this.metaWriteQueues.set(e,i),i.finally(()=>{this.metaWriteQueues.get(e)===i&&this.metaWriteQueues.delete(e)}),r}async closeAll(){let e=[...this.sessions.values()];await Promise.allSettled(e.map(t=>t.close({deleteRecord:!1}))),this.sessions.clear()}async flushSynopsis(e){await this.synopsisCoordinator.flush(e)}async shutdownSynopsis(){await this.synopsisCoordinator.shutdown()}scheduleSynopsis(e){this.synopsisCoordinator.schedule(e)}async flushMetaWrites(){let e=[...this.metaWriteQueues.values()];e.length!==0&&await Promise.allSettled(e)}async flushHistoryWrites(){await this.histories.flushAll()}async resurrectPendingQueues(){let e=await this.store.list().catch(()=>[]);for(let t of e){let s=await _s(t.sessionId).catch(()=>[]);if(s.length===0)continue;let r=Date.now(),i=s.filter(d=>r-d.enqueuedAt<ur),o=s.length-i.length;if(o>0&&(this.logger?.info(`queue replay: dropping ${o} stale prompt(s) for ${t.sessionId} (TTL ${ur/1e3}s)`),await Le(t.sessionId,i).catch(()=>{})),i.length===0)continue;let a=await this.loadFromDisk(t.sessionId).catch(()=>{});if(!a){this.logger?.warn(`queue replay: no meta for ${t.sessionId}; discarding ${i.length} entr${i.length===1?"y":"ies"}`),await Le(t.sessionId,[]).catch(()=>{});continue}try{let d=await this.resurrect(a);this.logger?.info(`queue replay: resurrected ${t.sessionId} and replaying ${i.length} prompt(s)`),d.replayPersistedQueue(i)}catch(d){this.logger?.warn(`queue replay: failed to resurrect ${t.sessionId}: ${d.message}`)}}}};function Ra(n,e){if(typeof n!="string"||n.length===0)return!1;let t=Ve.resolve(n),s=Ve.resolve(e);return t===s||t.startsWith(s+Ve.sep)}function Ea(n,e){let s=(n.mergedAvailableCommands().length>0?n.agentOnlyAdvertisedCommands().map(p=>p.description!==void 0?{name:p.name,description:p.description}:{name:p.name}):void 0)??e?.agentCommands,r=n.availableModes(),o=(r.length>0?r.map(p=>{let l={id:p.id};return p.name!==void 0&&(l.name=p.name),p.description!==void 0&&(l.description=p.description),l}):void 0)??e?.agentModes,a=n.availableModels(),c=(a.length>0?a.map(p=>{let l={modelId:p.modelId};return p.name!==void 0&&(l.name=p.name),p.description!==void 0&&(l.description=p.description),l}):void 0)??e?.agentModels;return yn({sessionId:n.sessionId,lineageId:e?.lineageId??Je(),upstreamSessionId:n.upstreamSessionId,importedFromSessionId:e?.importedFromSessionId,importedFromUpstreamSessionId:e?.importedFromUpstreamSessionId,importedFromMachine:e?.importedFromMachine,agentId:n.agentId,cwd:n.cwd,title:n.title,synopsis:e?.synopsis,summarizedThroughEntry:e?.summarizedThroughEntry,agentArgs:n.agentArgs,currentModel:n.currentModel??e?.currentModel,currentMode:n.currentMode??e?.currentMode,currentUsage:vr(n.currentUsage)??e?.currentUsage,agentCommands:s,agentModes:o,agentModels:c,parentSessionId:n.parentSessionId??e?.parentSessionId,forkedFromSessionId:n.forkedFromSessionId??e?.forkedFromSessionId,forkedFromMessageId:n.forkedFromMessageId??e?.forkedFromMessageId,originatingClient:n.originatingClient??e?.originatingClient,interactive:n.interactive??e?.interactive,priority:n.priority??e?.priority,createdAt:e?.createdAt??new Date(n.createdAt).toISOString()})}function vr(n){if(!n)return;let e={};return n.used!==void 0&&(e.used=n.used),n.size!==void 0&&(e.size=n.size),n.costAmount!==void 0&&(e.costAmount=n.costAmount),n.costCurrency!==void 0&&(e.costCurrency=n.costCurrency),n.cumulativeCost!==void 0&&(e.cumulativeCost=n.cumulativeCost),Object.keys(e).length>0?e:void 0}function Pa(n){return n?{...n}:void 0}function Ta(n,e){if(e&&n==="claude-acp")return{claudeCode:{options:{model:e}}}}function lr(n){let e=j(n.currentModelId)??j(n.currentModel)??j(n.modelId)??j(n.model);if(e)return e;let t=n.models;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=j(t.currentModelId)??j(t.currentModel);if(i)return i}let s=n._meta;if(s&&typeof s=="object"&&!Array.isArray(s)){for(let[i,o]of Object.entries(s))if(i!=="hydra-acp"&&o&&typeof o=="object"&&!Array.isArray(o)){let a=j(o.modelId)??j(o.model)??j(o.currentModelId);if(a)return a}}let r=At(n,"model");if(r){let i=j(r.currentValue);if(i)return i}}function j(n){if(typeof n!="string")return;let e=n.trim();return e.length>0?e:void 0}function At(n,e){let t=n.configOptions;if(Array.isArray(t))for(let s of t){if(!s||typeof s!="object"||Array.isArray(s))continue;let r=s;if(r.id===e)return r}}function fr(n){return n.length>0?n:void 0}function pr(n){let e=fe(n.availableModels);if(e.length>0)return e;let t=n.models;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=fe(t.availableModels);if(i.length>0)return i}let s=n._meta;if(s&&typeof s=="object"&&!Array.isArray(s)){for(let[i,o]of Object.entries(s))if(i!=="hydra-acp"&&o&&typeof o=="object"&&!Array.isArray(o)){let a=fe(o.availableModels);if(a.length>0)return a}}let r=At(n,"model");if(r){let i=fe(r.options);if(i.length>0)return i}return[]}function mr(n){let e=ve(n.availableModes);if(e.length>0)return e;let t=n.modes;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=ve(t.availableModes);if(i.length>0)return i}let s=n._meta;if(s&&typeof s=="object"&&!Array.isArray(s)){for(let[i,o]of Object.entries(s))if(i!=="hydra-acp"&&o&&typeof o=="object"&&!Array.isArray(o)){let a=ve(o.availableModes);if(a.length>0)return a}}let r=At(n,"mode");if(r){let i=ve(r.options);if(i.length>0)return i}return[]}function gr(n){let e=j(n.currentModeId)??j(n.currentMode)??j(n.modeId)??j(n.mode);if(e)return e;let t=n.modes;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=j(t.currentModeId)??j(t.currentMode);if(i)return i}let s=n._meta;if(s&&typeof s=="object"&&!Array.isArray(s)){for(let[i,o]of Object.entries(s))if(i!=="hydra-acp"&&o&&typeof o=="object"&&!Array.isArray(o)){let a=j(o.currentModeId)??j(o.currentMode)??j(o.modeId);if(a)return a}}let r=At(n,"mode");if(r){let i=j(r.currentValue);if(i)return i}}async function hr(n){let{agent:e,upstreamSessionId:t,persistedMode:s,agentReportedMode:r,advertisedModes:i,logger:o}=n;if(!s)return r;if(s===r)return s;if(i&&i.length>0&&!i.some(a=>a.id===s)){let a=i.map(d=>d.id).join(", ");return o?.warn(`resurrect: persisted currentMode=${JSON.stringify(s)} not in agent's availableModes ([${a}]); skipping session/set_mode, session will use ${JSON.stringify(r)}`),r}try{return o?.info(`resurrect: pushing persisted modeId=${JSON.stringify(s)} to agent (agentReported=${JSON.stringify(r)})`),await e.connection.request("session/set_mode",{sessionId:t,modeId:s}),o?.info(`resurrect: session/set_mode accepted, effectiveMode=${JSON.stringify(s)}`),s}catch(a){return o?.warn(`resurrect: session/set_mode rejected by agent for modeId=${JSON.stringify(s)} (${a.message}); session will use ${JSON.stringify(r)}`),r}}async function yr(n){let{agent:e,upstreamSessionId:t,persistedModel:s,agentReportedModel:r,logger:i}=n;if(!s)return r;if(s===r)return s;try{return i?.info(`resurrect: pushing persisted modelId=${JSON.stringify(s)} to agent (agentReported=${JSON.stringify(r)})`),await e.connection.request("session/set_model",{sessionId:t,modelId:s}),i?.info(`resurrect: session/set_model accepted, effectiveModel=${JSON.stringify(s)}`),s}catch(o){return i?.warn(`resurrect: session/set_model rejected by agent for modelId=${JSON.stringify(s)} (${o.message}); session will use ${JSON.stringify(r)}`),r}}function _a(n){for(let e=n.length-1;e>=0;e--){let t=n[e];if(!t||t.method!=="session/update")continue;let s=t.params?.update;if(s?.sessionUpdate==="turn_complete"&&!(typeof s.messageId!="string"||s.messageId.length===0))return{index:e,messageId:s.messageId}}}async function wr(n){try{let e=await be.readFile(w.tuiHistoryFile(n),"utf8"),t=[];for(let s of e.split(`
92
+ `))if(s.length!==0)try{let r=JSON.parse(s);typeof r=="string"&&t.push(r)}catch{}return t}catch{return[]}}async function kn(n){try{let e=await be.stat(w.historyFile(n));return{mtime:new Date(e.mtimeMs).toISOString(),hasContent:e.size>0}}catch{return{hasContent:!1}}}function St(n,e){return n.interactive!==void 0?n.interactive:n.originatingClient?.name===wt?!1:e?!0:void 0}import{spawn as $a}from"child_process";import*as Oe from"fs";import*as pe from"fs/promises";import*as br from"path";var xt=class{windowMs;maxFailures;now;recentExits=[];tripped_;constructor(e={}){this.windowMs=e.windowMs??3e5,this.maxFailures=e.maxFailuresInWindow??10,this.now=e.now??Date.now}recordExit(e,t,s){if(e===78){let o=`exited with code 78 (unrecoverable); fix and run \`${s}s start ${t}\``;return this.tripped_=o,{tripped:o}}let r=this.now();this.recentExits.push(r);let i=r-this.windowMs;for(;this.recentExits.length>0&&this.recentExits[0]<i;)this.recentExits.shift();if(this.recentExits.length>this.maxFailures){let o=Math.round(this.windowMs/6e4),a=`${this.recentExits.length} exits in ${o}m (crash loop); fix and run \`${s}s start ${t}\``;return this.tripped_=a,{tripped:a}}return"restart"}reset(){this.recentExits=[],this.tripped_=void 0}get tripped(){return this.tripped_}};var Oa=1e3,Fa=6e4,An=3e3,$e=class{entries=new Map;stopping=!1;context;tokenRegistry;breakerOptions;restartBaseMs;restartCapMs;adapter;constructor(e,t,s,r={}){this.adapter=t,this.context=s,this.tokenRegistry=r.tokenRegistry,this.breakerOptions=r.breakerOptions,this.restartBaseMs=r.restartBaseMs??Oa,this.restartCapMs=r.restartCapMs??Fa;for(let i of e)this.entries.set(i.name,this.makeEntry(i))}setContext(e){this.context=e}reportVersion(e,t){let s=this.entries.get(e);s&&(s.version=t)}async start(){if(!this.context)throw new Error(`${this.managerName()}: setContext must be called before start`);await pe.mkdir(this.adapter.paths.dir(),{recursive:!0}),await this.reapOrphans();for(let e of this.entries.values())e.config.enabled&&this.spawn(e,0)}async stop(){this.stopping=!0;let e=[];for(let t of this.entries.values()){t.restartTimer&&(clearTimeout(t.restartTimer),t.restartTimer=void 0);let s=t.child;if(s){try{s.kill("SIGTERM")}catch{}e.push(new Promise(r=>{if(s.exitCode!==null||s.signalCode!==null){r();return}let i=setTimeout(()=>{try{s.kill("SIGKILL")}catch{}r()},An);s.on("exit",()=>{clearTimeout(i),r()})}))}}await Promise.allSettled(e);for(let t of this.entries.values()){try{t.logStream?.end()}catch{}t.child=void 0,t.logStream=void 0,t.pid=void 0}}list(){return[...this.entries.values()].map(e=>this.infoFor(e))}get(e){let t=this.entries.get(e);return t?this.infoFor(t):void 0}has(e){return this.entries.has(e)}async startByName(e){let t=this.entries.get(e);if(!t)throw Ye(new Error(`unknown ${this.adapter.kind}: ${e}`),"NOT_FOUND");if(t.child)throw Ye(new Error(`${this.adapter.kind} ${e} already running`),"CONFLICT");return t.restartTimer&&(clearTimeout(t.restartTimer),t.restartTimer=void 0),t.manuallyStopped=!1,t.restartCount=0,t.breaker.reset(),t.failureReason=void 0,this.spawn(t,0),this.infoFor(t)}async stopByName(e){let t=this.entries.get(e);if(!t)throw Ye(new Error(`unknown ${this.adapter.kind}: ${e}`),"NOT_FOUND");t.manuallyStopped=!0,t.restartTimer&&(clearTimeout(t.restartTimer),t.restartTimer=void 0);let s=t.child;return s?(await this.terminate(t,s),this.infoFor(t)):this.infoFor(t)}async restartByName(e){return await this.stopByName(e),this.startByName(e)}register(e){if(this.entries.has(e.name))throw Ye(new Error(`${this.adapter.kind} ${e.name} already exists`),"CONFLICT");if(!this.context)throw new Error(`${this.managerName()}: setContext must be called before register`);let t=this.makeEntry(e);return this.entries.set(e.name,t),e.enabled&&this.spawn(t,0),this.infoFor(t)}async unregister(e){let t=this.entries.get(e);if(!t)throw Ye(new Error(`unknown ${this.adapter.kind}: ${e}`),"NOT_FOUND");t.manuallyStopped=!0,t.restartTimer&&(clearTimeout(t.restartTimer),t.restartTimer=void 0);let s=t.child;s&&await this.terminate(t,s);try{t.logStream?.end()}catch{}this.entries.delete(e)}async terminate(e,t){if(t.exitCode!==null||t.signalCode!==null)return;let s=new Promise(i=>{e.exitWaiters.push(i)});try{t.kill("SIGTERM")}catch{}let r=setTimeout(()=>{try{t.kill("SIGKILL")}catch{}},An);typeof r.unref=="function"&&r.unref();try{await s}finally{clearTimeout(r)}}infoFor(e){let t;return e.failureReason!==void 0?t="failed":e.child?t="running":e.restartTimer?t="restarting":e.config.enabled?t="stopped":t="disabled",{name:e.config.name,status:t,pid:e.pid,enabled:e.config.enabled,restartCount:e.restartCount,startedAt:e.startedAt,lastExitCode:e.lastExitCode,logPath:this.adapter.paths.logFile(e.config.name),version:e.version,failureReason:e.failureReason}}makeEntry(e){return{config:e,child:void 0,logStream:void 0,restartTimer:void 0,pid:void 0,startedAt:void 0,restartCount:0,lastExitCode:void 0,manuallyStopped:!1,exitWaiters:[],version:void 0,processToken:void 0,breaker:new xt(this.breakerOptions),failureReason:void 0}}async reapOrphans(){let e;try{e=await pe.readdir(this.adapter.paths.dir())}catch(t){if(t.code==="ENOENT")return;throw t}for(let t of e){if(!t.endsWith(".pid"))continue;let s=br.join(this.adapter.paths.dir(),t),r;try{let i=await pe.readFile(s,"utf8"),o=Number.parseInt(i.trim(),10);Number.isInteger(o)&&o>0&&(r=o)}catch{}if(typeof r=="number"&&xn(r)){try{process.kill(r,"SIGTERM")}catch{}let i=Date.now()+An;for(;Date.now()<i&&xn(r);)await new Promise(o=>setTimeout(o,50));if(xn(r))try{process.kill(r,"SIGKILL")}catch{}}await pe.unlink(s).catch(()=>{})}}spawn(e,t){if(this.stopping||e.manuallyStopped)return;let s=this.context;if(!s)throw new Error(`${this.managerName()}.spawn called before setContext`);let r=e.config,i=r.command.length>0?r.command:[r.name],o=Oe.createWriteStream(this.adapter.paths.logFile(r.name),{flags:"a"});o.write(`[hydra-acp] ${new Date().toISOString()} starting ${this.adapter.kind} ${r.name} (attempt ${t+1})
93
+ `);let a=this.tokenRegistry?.mint(r.name,this.adapter.tokenRole)??s.serviceToken;e.processToken=a,e.version=void 0;let d={...process.env,HYDRA_ACP_DAEMON_URL:s.daemonUrl,HYDRA_ACP_DAEMON_HOST:s.daemonHost,HYDRA_ACP_DAEMON_PORT:String(s.daemonPort),HYDRA_ACP_TOKEN:a,HYDRA_ACP_WS_URL:s.daemonWsUrl,HYDRA_ACP_HOME:s.hydraHome,[this.adapter.nameEnvVar]:r.name,...r.env},[c,...p]=i;if(c===void 0){o.write(`[hydra-acp] ${this.adapter.kind} ${r.name} has empty command
94
+ `),o.end();return}let l=[...p,...r.args],u;try{u=$a(c,l,{env:d,stdio:["ignore","pipe","pipe"],detached:!1})}catch(f){o.write(`[hydra-acp] failed to spawn ${r.name}: ${f.message}
95
+ `),o.end(),this.scheduleRestart(e,t);return}if(u.stdout&&u.stdout.pipe(o,{end:!1}),u.stderr&&u.stderr.pipe(o,{end:!1}),typeof u.pid=="number")try{Oe.writeFileSync(this.adapter.paths.pidFile(r.name),`${u.pid}
96
+ `,{encoding:"utf8",mode:384})}catch(f){o.write(`[hydra-acp] failed to write pid file for ${r.name}: ${f.message}
97
+ `)}e.child=u,e.logStream=o,e.pid=typeof u.pid=="number"?u.pid:void 0,e.startedAt=Date.now(),e.lastExitCode=void 0,u.on("error",f=>{o.write(`[hydra-acp] ${this.adapter.kind} ${r.name} error: ${f.message}
98
+ `)}),u.on("exit",(f,g)=>{try{Oe.unlinkSync(this.adapter.paths.pidFile(r.name))}catch{}o.write(`[hydra-acp] ${this.adapter.kind} ${r.name} exited code=${f??"null"} signal=${g??"null"}
99
+ `),e.child=void 0,e.pid=void 0,e.lastExitCode=typeof f=="number"?f:void 0,e.processToken&&(this.tokenRegistry?.revoke(r.name),e.processToken=void 0);let m=e.exitWaiters.splice(0);for(let v of m)v();if(this.stopping||e.manuallyStopped){try{o.end()}catch{}e.logStream=void 0;return}e.restartCount+=1;let y=e.breaker.recordExit(f,r.name,this.adapter.kind);if(typeof y=="object"){e.failureReason=y.tripped,o.write(`[hydra-acp] ${this.adapter.kind} ${r.name} circuit breaker tripped: ${y.tripped}
100
+ `);try{o.end()}catch{}e.logStream=void 0;return}this.scheduleRestart(e,t+1)})}scheduleRestart(e,t){if(this.stopping||e.manuallyStopped)return;let s=Math.min(this.restartBaseMs*2**Math.min(t,10),this.restartCapMs);e.restartTimer=setTimeout(()=>{e.restartTimer=void 0,this.spawn(e,t)},s),typeof e.restartTimer.unref=="function"&&e.restartTimer.unref()}managerName(){return this.adapter.kind==="extension"?"ExtensionManager":"TransformerManager"}};function xn(n){try{return process.kill(n,0),!0}catch{return!1}}function Ye(n,e){return n.code=e,n}var Na={kind:"extension",nameEnvVar:"HYDRA_ACP_EXTENSION_NAME",tokenRole:"extension",paths:{dir:w.extensionsDir,logFile:w.extensionLogFile,pidFile:w.extensionPidFile}},Ct=class extends $e{constructor(e,t,s={}){super(e,Na,t,s)}};var ja={kind:"transformer",nameEnvVar:"HYDRA_ACP_TRANSFORMER_NAME",tokenRole:"transformer",paths:{dir:w.transformersDir,logFile:w.transformerLogFile,pidFile:w.transformerPidFile}},Mt=class extends $e{connected=new Map;constructor(e,t,s={}){super(e,ja,t,s)}registerConnection(e,t,s){this.connected.set(e,{name:e,connection:t,intercepts:new Set(s)})}deregisterConnection(e){this.connected.delete(e)}resolveChain(e){let t=[];for(let s of e){let r=this.connected.get(s);r&&t.push(r)}return t}};var Rt=class{entries=new Map;changeHandlers=[];register(e,t,s){this.entries.set(e,{connection:t,commands:[...s]}),this.fireChanged()}clear(e){this.entries.delete(e)&&this.fireChanged()}get(e){return this.entries.get(e)}has(e){return this.entries.has(e)}list(){let e=[];for(let[t,s]of this.entries)for(let r of s.commands)e.push({name:t,command:r});return e}onChange(e){return this.changeHandlers.push(e),()=>{let t=this.changeHandlers.indexOf(e);t>=0&&this.changeHandlers.splice(t,1)}}fireChanged(){for(let e of this.changeHandlers)try{e()}catch{}}};import*as Xe from"fs/promises";import*as Et from"path";var Ke=n=>{process.stderr.write(n+`
81
101
  `)};function Cn(n){Ke=n??(e=>process.stderr.write(e+`
82
- `))}async function br(n,e){let t=ne();if(!t)return;let s=await n.load(),r=new Map;for(let d of s.agents)r.set(d.id,d.version??"current");let i=e.activeAgentVersions(),o=Et.join(w.agentsDir(),t),a;try{a=await Xe.readdir(o,{withFileTypes:!0})}catch(d){let c=d;if(c.code==="ENOENT")return;Ke(`hydra-acp: prune: failed to read ${o}: ${c.message}`);return}for(let d of a){if(!d.isDirectory())continue;let c=d.name,f=r.get(c);if(f===void 0)continue;let l=i.get(c)??new Set,u=Et.join(o,c),p;try{p=await Xe.readdir(u,{withFileTypes:!0})}catch(g){Ke(`hydra-acp: prune: failed to read ${u}: ${g.message}`);continue}for(let g of p){if(!g.isDirectory())continue;let m=g.name;if(m===f||l.has(m)||m.includes(".partial-"))continue;let y=Et.join(u,m);try{await Xe.rm(y,{recursive:!0,force:!0}),Ke(`hydra-acp: pruned stale ${c} ${m} (${y})`)}catch(v){Ke(`hydra-acp: prune: failed to remove ${y}: ${v.message}`)}}}}function Ir(n){let e,t=!1,s=0,r=(a,d)=>{n.logger&&n.logger[a](`agent-sync: ${d}`)},i=async()=>{let a=[];try{let f=await n.registry.load();for(let l of f.agents)await sn(l)==="yes"&&a.push(l.id)}catch(f){return r("warn",`registry load failed: ${f.message}`),n.intervalMs}if(a.length===0)return n.intervalMs;let d=s%a.length;s=(s+1)%a.length;let c=a[d];try{let{synced:f,skipped:l}=await n.manager.syncFromAgent(c);r("info",`${c}: synced ${f.length}, skipped ${l}`)}catch(f){r("warn",`${c}: ${f.message}`)}return Math.max(1,Math.floor(n.intervalMs/a.length))},o=a=>{t||(e=setTimeout(()=>{i().then(d=>{o(d)}).catch(d=>{r("warn",`tick crashed: ${d.message}`),o(n.intervalMs)})},a),e.unref())};return o(n.intervalMs),()=>{t=!0,e&&(clearTimeout(e),e=void 0)}}async function Mn(n){let e=n.maxDeletions??200,t=(p,g)=>{n.logger&&n.logger[p](`session-gc: ${g}`)},s=Date.now(),r=n.maxAgeMs>0?s-n.maxAgeMs:Number.POSITIVE_INFINITY,i=await n.manager.list({includeNonInteractive:!0}).catch(p=>(t("warn",`manager.list failed: ${p.message}`),[])),o=[];for(let p of i){if(p.status!=="cold")continue;if((n.selection??"explicit")==="explicit"){if(p.interactive!==!1)continue}else if(p.interactive===!0)continue;let m=Date.parse(p.updatedAt);Number.isFinite(m)&&(n.maxAgeMs>0&&m>r||o.push({sessionId:p.sessionId,lastUsedMs:m}))}if(o.length===0)return n.verbose&&t("info","no candidates"),{considered:0,deleted:0,failed:0,deferred:0};o.sort((p,g)=>p.lastUsedMs-g.lastUsedMs);let a=o.slice(0,e),d=a[0]?.lastUsedMs,c=0,f=0;for(let{sessionId:p}of a)try{await n.manager.deleteRecord(p)&&(c+=1)}catch(g){f+=1,t("warn",`delete ${p} failed: ${g.message}`)}let l=o.length-a.length;if(n.verbose||c>0||f>0){let p=(n.selection??"explicit")==="unpromoted"?"unpromoted":"non-interactive";t("info",`swept ${c} ${p} session(s) older than ${ja(n.maxAgeMs)}`+(f>0?`; ${f} failed`:"")+(l>0?`; ${l} deferred to next sweep`:""))}let u={considered:o.length,deleted:c,failed:f,deferred:l};return d!==void 0&&(u.oldestLastUsedMs=d),u}function Sr(n){let e,t=!1,s=!1,r=i=>{t||(e=setTimeout(()=>{(async()=>{if(!s){s=!0;try{await Mn({manager:n.manager,maxAgeMs:n.maxAgeMs,selection:"unpromoted",...n.maxDeletionsPerSweep!==void 0?{maxDeletions:n.maxDeletionsPerSweep}:{},...n.logger?{logger:n.logger}:{}})}finally{s=!1}}})().catch(a=>{n.logger?.warn(`session-gc: sweep crashed: ${a.message}`)}).finally(()=>{r(n.intervalMs)})},i),e.unref())};return r(n.intervalMs),()=>{t=!0,e&&(clearTimeout(e),e=void 0)}}function ja(n){if(n<=0)return"any age";let e=n/(1440*60*1e3);if(e>=1)return`${e.toFixed(e>=10?0:1)}d`;let t=n/(3600*1e3);return`${t.toFixed(t>=10?0:1)}h`}import{createHash as Ha}from"crypto";function Rn(n){if(Array.isArray(n))return n.map(Rn);if(n!==null&&typeof n=="object"){let e={};for(let t of Object.keys(n).sort())e[t]=Rn(n[t]);return e}return n}function kr(n){let e=JSON.stringify(Rn(n));return Ha("sha256").update(e).digest("hex").slice(0,16)}import*as Cr from"path";import{createHash as Ba,randomBytes as Ua,timingSafeEqual as Da}from"crypto";var Mr="hydra_session_",La=3600*24*30,Ar=12,qa=32,Ja=50;function za(){return Cr.join(w.home(),"session-tokens.json")}function xr(n){return Ba("sha256").update(n).digest("hex")}function Rr(n){return Ua(n).toString("hex")}function Wa(){return Rr(Ar).slice(0,Ar*2)}function Qa(){return`${Mr}${Rr(qa)}`}var Pt=class n{records=new Map;writeTimer=null;writeInflight=null;filePath;constructor(e,t){this.filePath=t;for(let s of e)this.records.set(s.hash,s)}static async load(){let e=[],t=za(),s=await te(t);s&&Array.isArray(s.records)&&(e=s.records.filter(Va));let r=new n(e,t);return r.sweepExpired(new Date)>0&&await r.flush(),r}async issue(e={}){let t=Qa(),s=xr(t),r=Wa(),i=new Date,o=e.ttlSec&&e.ttlSec>0?e.ttlSec:La,a=new Date(i.getTime()+o*1e3),d={id:r,hash:s,label:e.label,createdAt:i.toISOString(),expiresAt:a.toISOString(),lastUsedAt:i.toISOString()};return this.records.set(s,d),this.scheduleWrite(),{id:r,token:t,expiresAt:d.expiresAt}}async verify(e){if(typeof e!="string"||!e.startsWith(Mr))return;let t=xr(e),s=this.records.get(t);if(!s)return;let r=Buffer.from(s.hash,"hex"),i=Buffer.from(t,"hex");if(r.length!==i.length||!Da(r,i))return;let o=new Date;if(new Date(s.expiresAt).getTime()<=o.getTime()){this.records.delete(t),this.scheduleWrite();return}return s.lastUsedAt=o.toISOString(),this.scheduleWrite(),s.id}async revoke(e){for(let[t,s]of this.records)if(s.id===e)return this.records.delete(t),this.scheduleWrite(),!0;return!1}async revokeAll(){let e=this.records.size;return this.records.clear(),this.scheduleWrite(),e}list(){return Array.from(this.records.values()).map(({id:e,label:t,createdAt:s,expiresAt:r,lastUsedAt:i})=>({id:e,label:t,createdAt:s,expiresAt:r,lastUsedAt:i})).sort((e,t)=>t.createdAt.localeCompare(e.createdAt))}sweepExpired(e=new Date){let t=0;for(let[s,r]of this.records)new Date(r.expiresAt).getTime()<=e.getTime()&&(this.records.delete(s),t+=1);return t>0&&this.scheduleWrite(),t}async flush(){this.writeTimer&&(clearTimeout(this.writeTimer),this.writeTimer=null),await this.persist()}scheduleWrite(){this.writeTimer||(this.writeTimer=setTimeout(()=>{this.writeTimer=null,this.persist().catch(()=>{})},Ja))}persist(){let e=(this.writeInflight??Promise.resolve()).catch(()=>{}).then(()=>K(this.filePath,{records:Array.from(this.records.values())},{mode:384}));return this.writeInflight=e,e.catch(()=>{}).finally(()=>{this.writeInflight===e&&(this.writeInflight=null)}),e}};function Va(n){if(!n||typeof n!="object")return!1;let e=n;return typeof e.id=="string"&&typeof e.hash=="string"&&typeof e.createdAt=="string"&&typeof e.expiresAt=="string"&&typeof e.lastUsedAt=="string"&&(e.label===void 0||typeof e.label=="string")}var Er="Bearer ",Tt=class{constructor(e){this.token=e}token;async validate(e){return Ga(e,this.token)?"service":void 0}},_t=class{constructor(e){this.store=e}store;async validate(e){return this.store.verify(e)}},$t=class{constructor(e){this.validators=e}validators;async validate(e){for(let t of this.validators){let s=await t.validate(e);if(s!==void 0)return s}}};function Pr(n){return async function(t,s){let r=t.headers.authorization;if(!r||!r.startsWith(Er)){s.code(401).send({error:"Missing bearer token"});return}let i=r.slice(Er.length).trim(),o=await n.validator.validate(i);if(!o){s.code(403).send({error:"Invalid token"});return}t.authIdentity=o}}function Tr(n){let e=n.headers["sec-websocket-protocol"],t=Array.isArray(e)?e.join(","):e;if(t)for(let s of t.split(",")){let r=s.trim(),i="hydra-acp-token.";if(r.startsWith(i))return r.slice(i.length)}if(n.url)try{let r=new URL(n.url,"http://localhost").searchParams.get("token");if(r)return r}catch{return}}var Ft=class{tokens=new Map;mint(e,t){let s=rt();return this.tokens.set(s,{name:e,kind:t}),s}revoke(e){for(let[t,s]of this.tokens)s.name===e&&this.tokens.delete(t)}resolve(e){return this.tokens.get(e)}async validate(e){let t=this.tokens.get(e);if(t)return`${t.kind}:${t.name}`}};function Ga(n,e){if(n.length!==e.length)return!1;let t=0;for(let s=0;s<n.length;s+=1)t|=n.charCodeAt(s)^e.charCodeAt(s);return t===0}var Ot=class{entries=new Map;maxFails;windowMs;constructor(e=10,t=900*1e3){this.maxFails=e,this.windowMs=t}isBlocked(e){let t=this.entries.get(e);return t?Date.now()-t.windowStart>this.windowMs?(this.entries.delete(e),!1):t.fails>=this.maxFails:!1}recordFailure(e){let t=Date.now(),s=this.entries.get(e);if(!s||t-s.windowStart>this.windowMs){this.entries.set(e,{fails:1,windowStart:t});return}s.fails+=1}recordSuccess(e){this.entries.delete(e)}};import*as tt from"os";import*as Gr from"path";import{posix as Ya}from"path";import Ka from"strip-ansi";function Ze(n){if(!n||typeof n!="object")return;let e=n;return typeof e.workerTaskId=="string"?e.workerTaskId:void 0}var Xa=/[\x00-\x08\x0b-\x1f\x7f]/g;function Z(n){return Ka(n).replace(Xa,"")}function H(n){return Z(n).replace(/[\n\t]+/g," ").replace(/ +/g," ").trim()}function Fr(n,e={}){if(!n||typeof n!="object")return null;let t=n,s=t.sessionUpdate??t.kind;if(typeof s!="string")return null;switch(s){case"agent_message_chunk":return id(t);case"agent_thought_chunk":case"agent_thought":return od(t);case"user_message_chunk":return ad(t);case"prompt_received":return dd(t);case"tool_call":return cd(t,e);case"tool_call_update":return md(t,e);case"plan":return yd(t);case"current_mode_update":return wd(t);case"current_model_update":return vd(t);case"turn_complete":return bd(t);case"usage_update":return rd(t);case"available_commands_update":return nd(t);case"available_modes_update":return sd(t);case"session_info_update":return ed(t);case"config_option_update":return Za(t);default:return{kind:"unknown",sessionUpdate:s,raw:n}}}function Za(n){let e=n.configOptions;if(!Array.isArray(e))return null;let t=[];for(let s of e){if(!s||typeof s!="object")continue;let r=s;if(typeof r.id!="string"||typeof r.currentValue!="string"||!Array.isArray(r.options))continue;let i=[];for(let o of r.options){if(!o||typeof o!="object")continue;let a=o;typeof a.value=="string"&&i.push({value:a.value,name:typeof a.name=="string"?a.name:a.value,...typeof a.description=="string"?{description:a.description}:{}})}t.push({id:r.id,name:typeof r.name=="string"?r.name:r.id,type:"select",currentValue:r.currentValue,options:i,...typeof r.category=="string"?{category:r.category}:{}})}return{kind:"config-options",options:t}}function ed(n){let e=O(n,"title"),t=e!==void 0?H(e):void 0,s=n._meta,r;if(s&&typeof s=="object"&&!Array.isArray(s)){let o=s["hydra-acp"];if(o&&typeof o=="object"&&!Array.isArray(o)){let a=o.agentId;typeof a=="string"&&(r=a)}}if(t===void 0&&r===void 0)return null;let i={kind:"session-info"};return t!==void 0&&(i.title=t),r!==void 0&&(i.agentId=r),i}function td(n){if(!Array.isArray(n))return[];let e=[];for(let t of n){if(!t||typeof t!="object")continue;let s=t;if(typeof s.name!="string"||s.name.length===0)continue;let r=s.name.startsWith("/")?s.name:`/${s.name}`,i={name:H(r)};typeof s.description=="string"&&(i.description=H(s.description)),e.push(i)}return e}function nd(n){let e=n.availableCommands??n.commands;return Array.isArray(e)?{kind:"available-commands",commands:td(e)}:null}function sd(n){let e=n.availableModes;if(!Array.isArray(e))return null;let t=[];for(let s of e){if(!s||typeof s!="object")continue;let r=s;if(typeof r.id!="string"||r.id.length===0)continue;let i={id:H(r.id)};typeof r.name=="string"&&(i.name=H(r.name)),typeof r.description=="string"&&(i.description=H(r.description)),t.push(i)}return{kind:"available-modes",modes:t}}function rd(n){let e={kind:"usage-update"};if(typeof n.used=="number"&&(e.used=n.used),typeof n.size=="number"&&(e.size=n.size),n.cost&&typeof n.cost=="object"){let t=n.cost;typeof t.amount=="number"&&(e.costAmount=t.amount),typeof t.currency=="string"&&(e.costCurrency=t.currency)}return e}function id(n){let e=et(n.content);return e===null?null:{kind:"agent-text",text:e,workerTaskId:Ze(n)}}function od(n){let e=typeof n.text=="string"?Z(n.text):et(n.content);return e===null?null:{kind:"agent-thought",text:e,workerTaskId:Ze(n)}}function ad(n){let e=n._meta;if(e&&typeof e=="object"&&!Array.isArray(e)){let s=e["hydra-acp"];if(s&&typeof s=="object"&&!Array.isArray(s)&&s.compatFor==="prompt_received")return null}let t=et(n.content);return t===null?null:{kind:"user-text",text:t}}function dd(n){let e=Id(n.prompt);return e===null?null:{kind:"user-text",text:e}}function Or(n){return n?n.toLowerCase().replace(/[_\s-]/g,"")==="exitplanmode":!1}function _r(n){if(typeof n=="string")return{text:n};if(n&&typeof n=="object"&&!Array.isArray(n)){let e=n;if(typeof e.__hydraBlob=="string")return{ref:{hash:e.__hydraBlob,bytes:typeof e.bytes=="number"?e.bytes:0}}}}function Nt(n){let e=n.content;if(Array.isArray(e))for(let s of e){if(!s||typeof s!="object")continue;let r=s;if(r.type!=="diff")continue;let i=_r(r.oldText),o=_r(r.newText);if(i===void 0&&o===void 0)continue;let a=typeof r.path=="string"?r.path:void 0;return{...a!==void 0?{path:a}:{},oldText:i?.text??"",newText:o?.text??"",...i?.ref?{oldRef:i.ref}:{},...o?.ref?{newRef:o.ref}:{}}}let t=n.rawInput;if(t&&typeof t=="object"&&!Array.isArray(t)){let s=t,r=typeof s.file_path=="string"?s.file_path:typeof s.path=="string"?s.path:void 0;if(typeof s.old_string=="string"&&typeof s.new_string=="string")return{...r!==void 0?{path:r}:{},oldText:s.old_string,newText:s.new_string};if(typeof s.content=="string")return{...r!==void 0?{path:r}:{},oldText:"",newText:s.content}}return null}function Nr(n){let e=n.rawInput;if(!e||typeof e!="object"||Array.isArray(e))return null;let t=e.plan;return typeof t!="string"||t.length===0?null:Z(t)}function cd(n,e={}){let t=O(n,"toolCallId")??O(n,"id");if(!t)return null;let s=O(n,"title")??O(n,"name")??O(n,"label")??"tool call",r=O(n,"name")??O(n,"title");if(Or(r)){let u=Nr(n);if(u!==null){let p=O(n,"status"),g={kind:"exit-plan-mode",toolCallId:t,plan:u};return p!==void 0&&(g.status=p),g}}let i=Hr(H(s),n,e),o=O(n,"status"),a=O(n,"kind"),d={kind:"tool-call",toolCallId:t,title:i};o!==void 0&&(d.status=o),a!==void 0&&(d.rawKind=a);let c=Nt(n);c!==null&&(d.editDiff=c);let f=jr(n);f!==void 0&&(d.detail=f);let l=Ze(n);return l!==void 0&&(d.workerTaskId=l),d}var $r=64;function jr(n){let e=n.rawInput;if(!e||typeof e!="object"||Array.isArray(e))return;let t=e;if(typeof t.command=="string"&&t.command.trim().length>0){let i=H(t.command).trim().replace(/^cd\s+\S+\s+&&\s+/,"");return fd(i,$r)}let s=typeof t.file_path=="string"?t.file_path:typeof t.filePath=="string"?t.filePath:typeof t.path=="string"?t.path:void 0;if(s!==void 0&&s.length>0)return pd(Ie(H(s)),$r)}var ud=["home/","Users/","root/","tmp/","var/","opt/","etc/","usr/","mnt/","private/"];function ld(n){for(let e of ud)if(n.startsWith(e))return!0;return!1}function Hr(n,e,t={}){if(n.length===0)return n;if(n.startsWith("/"))return Ie(n);if(n.startsWith("~")||!n.includes("/")||/\s/.test(n))return n;let s=e.rawInput;if(s&&typeof s=="object"&&!Array.isArray(s)){let r=s,i=[r.file_path,r.filePath,r.path];for(let o of i)if(!(typeof o!="string"||o.length===0)&&(o===`/${n}`||o.endsWith(`/${n}`)||o===n))return Ie(o)}return ld(n)?Ie(`/${n}`):t.cwd&&t.cwd.length>0?Ie(Ya.resolve(t.cwd,n)):n}function fd(n,e){return n.length>e?`${n.slice(0,e-1)}\u2026`:n}function pd(n,e){if(n.length<=e)return n;let t=n.slice(-(e-1)),s=t.indexOf("/");return s>=0&&s<e/2?`\u2026${t.slice(s)}`:`\u2026${t}`}function md(n,e={}){let t=O(n,"toolCallId")??O(n,"id");if(!t)return null;let s=O(n,"title"),r=s!==void 0?Hr(H(s),n,e):void 0,i=O(n,"status"),o=Nt(n),a=jr(n);if(!(r!==void 0||o!==null||a!==void 0||i==="completed"||i==="failed"||i==="rejected"||i==="cancelled"))return null;let c=O(n,"name")??s;if(Or(c)){let u={kind:"exit-plan-mode",toolCallId:t},p=Nr(n);return p!==null&&(u.plan=p),i!==void 0&&(u.status=i),u}let f={kind:"tool-call-update",toolCallId:t};if(r!==void 0&&(f.title=r),a!==void 0&&(f.detail=a),i!==void 0&&(f.status=i),o!==null&&(f.editDiff=o),i==="failed"){let u=gd(n);u!==null&&(f.errorText=u),hd(n,u)&&(f.upstreamInterrupted=!0)}let l=Ze(n);return l!==void 0&&(f.workerTaskId=l),f}function gd(n){let e=n.content;if(Array.isArray(e))for(let s of e){if(!s||typeof s!="object")continue;let i=et(s.content);if(i!==null&&i.length>0)return i}let t=n.rawOutput;if(t&&typeof t=="object"){let s=t.error;if(typeof s=="string"&&s.length>0)return Z(s)}return null}function hd(n,e){let t=n.rawOutput;if(t&&typeof t=="object"){let s=t.metadata;if(s&&typeof s=="object"&&s.interrupted===!0)return!0}return!!(e!==null&&e.toLowerCase().includes("tool execution aborted"))}function yd(n){let e=n.entries;if(!Array.isArray(e)||e.length===0)return null;let t=[];for(let s of e){if(!s||typeof s!="object")continue;let r=s,i=typeof r.content=="string"?H(r.content):void 0;if(!i)continue;let o={content:i};typeof r.status=="string"&&(o.status=r.status),typeof r.priority=="string"&&(o.priority=r.priority),t.push(o)}return{kind:"plan",entries:t}}function wd(n){let e=O(n,"currentModeId")??O(n,"currentMode")??O(n,"mode");return e?{kind:"mode-changed",mode:H(e)}:null}function vd(n){let e=O(n,"currentModel")??O(n,"model");if(!e)return null;let t=n.availableModels,s=Array.isArray(t)?t.map(r=>typeof r=="object"&&r!==null?r.modelId:typeof r=="string"?r:void 0).filter(r=>typeof r=="string"&&r.length>0):void 0;return{kind:"model-changed",model:H(e),...s&&s.length>0?{availableModels:s}:{}}}function bd(n){let e=O(n,"stopReason"),t=n._meta,s=t?.["hydra-acp"]?.amended!==void 0&&t["hydra-acp"].amended!==null,r={kind:"turn-complete"};return e!==void 0&&(r.stopReason=e),s&&(r.amended=!0),r}function et(n){if(typeof n=="string")return Z(n);if(!n||typeof n!="object")return null;let e=n;return e.type==="text"&&typeof e.text=="string"||typeof e.text=="string"?Z(e.text):null}function Id(n){if(!Array.isArray(n))return null;let e=[];for(let t of n){let s=et(t);s!==null&&e.push(s)}return e.length===0?null:e.join("")}function O(n,e){let t=n[e];return typeof t=="string"?t:void 0}function Br(n){let e=new Map,t=new Map;for(let r of n){let o=r.params?.update;if(!o||typeof o!="object")continue;let a=o.sessionUpdate;if(a!=="tool_call"&&a!=="tool_call_update")continue;let d=typeof o.toolCallId=="string"&&o.toolCallId.length>0?o.toolCallId:void 0,c=kd(o);if(c.length===0)continue;let f=c;if(d!==void 0){let l=e.get(d)??[],u=[];for(let p of c)l.some(g=>g.path===p.path&&g.oldText===p.oldText&&g.newText===p.newText)||u.push(p);if(u.length===0)continue;e.set(d,[...l,...u]),f=u}for(let l of f)Sd(t,l)}let s=[];for(let[r,i]of t)s.push({path:r,hunks:i.hunks,created:i.created});return s}function Ur(n){let e=[];for(let t of n){let s=-1;for(let i=e.length-1;i>=0;i--)if(e[i].newText===t.oldText){s=i;break}if(s===-1){e.push(t);continue}let r=e[s];e.splice(s,1,{oldText:r.oldText,newText:t.newText})}return e}function Sd(n,e){let t={oldText:e.oldText,newText:e.newText},s=n.get(e.path);if(s===void 0){n.set(e.path,{hunks:[t],created:e.oldText.length===0});return}s.hunks.push(t)}function kd(n){let e=[],t=n.rawInput;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=t,o=typeof i.file_path=="string"?i.file_path:typeof i.path=="string"?i.path:void 0,a=i.edits;if(o!==void 0&&Array.isArray(a)){for(let d of a){if(!d||typeof d!="object")continue;let c=d,f=typeof c.old_string=="string"?c.old_string:void 0,l=typeof c.new_string=="string"?c.new_string:void 0;f===void 0||l===void 0||e.push({path:o,oldText:f,newText:l})}if(e.length>0)return e}}let s=n.content;if(Array.isArray(s)){for(let i of s){if(!i||typeof i!="object")continue;let o=i;if(o.type!=="diff")continue;let a=typeof o.path=="string"?o.path:void 0;if(a===void 0)continue;let d=typeof o.oldText=="string"?o.oldText:"",c=typeof o.newText=="string"?o.newText:"";e.push({path:a,oldText:d,newText:c})}if(e.length>0)return e}let r=Nt(n);return r&&r.path&&e.push({path:r.path,oldText:r.oldText,newText:r.newText}),e}function Dr(n){let e=Ad(n),t=xd(e),s=[];Cd(s,n),Md(s,e,t);let r=s.join(`
102
+ `))}async function Ir(n,e){let t=ne();if(!t)return;let s=await n.load(),r=new Map;for(let d of s.agents)r.set(d.id,d.version??"current");let i=e.activeAgentVersions(),o=Et.join(w.agentsDir(),t),a;try{a=await Xe.readdir(o,{withFileTypes:!0})}catch(d){let c=d;if(c.code==="ENOENT")return;Ke(`hydra-acp: prune: failed to read ${o}: ${c.message}`);return}for(let d of a){if(!d.isDirectory())continue;let c=d.name,p=r.get(c);if(p===void 0)continue;let l=i.get(c)??new Set,u=Et.join(o,c),f;try{f=await Xe.readdir(u,{withFileTypes:!0})}catch(g){Ke(`hydra-acp: prune: failed to read ${u}: ${g.message}`);continue}for(let g of f){if(!g.isDirectory())continue;let m=g.name;if(m===p||l.has(m)||m.includes(".partial-"))continue;let y=Et.join(u,m);try{await Xe.rm(y,{recursive:!0,force:!0}),Ke(`hydra-acp: pruned stale ${c} ${m} (${y})`)}catch(v){Ke(`hydra-acp: prune: failed to remove ${y}: ${v.message}`)}}}}function Sr(n){let e,t=!1,s=0,r=(a,d)=>{n.logger&&n.logger[a](`agent-sync: ${d}`)},i=async()=>{let a=[];try{let p=await n.registry.load();for(let l of p.agents)await sn(l)==="yes"&&a.push(l.id)}catch(p){return r("warn",`registry load failed: ${p.message}`),n.intervalMs}if(a.length===0)return n.intervalMs;let d=s%a.length;s=(s+1)%a.length;let c=a[d];try{let{synced:p,skipped:l}=await n.manager.syncFromAgent(c);r("info",`${c}: synced ${p.length}, skipped ${l}`)}catch(p){r("warn",`${c}: ${p.message}`)}return Math.max(1,Math.floor(n.intervalMs/a.length))},o=a=>{t||(e=setTimeout(()=>{i().then(d=>{o(d)}).catch(d=>{r("warn",`tick crashed: ${d.message}`),o(n.intervalMs)})},a),e.unref())};return o(n.intervalMs),()=>{t=!0,e&&(clearTimeout(e),e=void 0)}}async function Mn(n){let e=n.maxDeletions??200,t=(f,g)=>{n.logger&&n.logger[f](`session-gc: ${g}`)},s=Date.now(),r=n.maxAgeMs>0?s-n.maxAgeMs:Number.POSITIVE_INFINITY,i=await n.manager.list({includeNonInteractive:!0}).catch(f=>(t("warn",`manager.list failed: ${f.message}`),[])),o=[];for(let f of i){if(f.status!=="cold")continue;if((n.selection??"explicit")==="explicit"){if(f.interactive!==!1)continue}else if(f.interactive===!0)continue;let m=Date.parse(f.updatedAt);Number.isFinite(m)&&(n.maxAgeMs>0&&m>r||o.push({sessionId:f.sessionId,lastUsedMs:m}))}if(o.length===0)return n.verbose&&t("info","no candidates"),{considered:0,deleted:0,failed:0,deferred:0};o.sort((f,g)=>f.lastUsedMs-g.lastUsedMs);let a=o.slice(0,e),d=a[0]?.lastUsedMs,c=0,p=0;for(let{sessionId:f}of a)try{await n.manager.deleteRecord(f)&&(c+=1)}catch(g){p+=1,t("warn",`delete ${f} failed: ${g.message}`)}let l=o.length-a.length;if(n.verbose||c>0||p>0){let f=(n.selection??"explicit")==="unpromoted"?"unpromoted":"non-interactive";t("info",`swept ${c} ${f} session(s) older than ${Ha(n.maxAgeMs)}`+(p>0?`; ${p} failed`:"")+(l>0?`; ${l} deferred to next sweep`:""))}let u={considered:o.length,deleted:c,failed:p,deferred:l};return d!==void 0&&(u.oldestLastUsedMs=d),u}function kr(n){let e,t=!1,s=!1,r=i=>{t||(e=setTimeout(()=>{(async()=>{if(!s){s=!0;try{await Mn({manager:n.manager,maxAgeMs:n.maxAgeMs,selection:"unpromoted",...n.maxDeletionsPerSweep!==void 0?{maxDeletions:n.maxDeletionsPerSweep}:{},...n.logger?{logger:n.logger}:{}})}finally{s=!1}}})().catch(a=>{n.logger?.warn(`session-gc: sweep crashed: ${a.message}`)}).finally(()=>{r(n.intervalMs)})},i),e.unref())};return r(n.intervalMs),()=>{t=!0,e&&(clearTimeout(e),e=void 0)}}function Ha(n){if(n<=0)return"any age";let e=n/(1440*60*1e3);if(e>=1)return`${e.toFixed(e>=10?0:1)}d`;let t=n/(3600*1e3);return`${t.toFixed(t>=10?0:1)}h`}import{createHash as Ba}from"crypto";function Rn(n){if(Array.isArray(n))return n.map(Rn);if(n!==null&&typeof n=="object"){let e={};for(let t of Object.keys(n).sort())e[t]=Rn(n[t]);return e}return n}function Ar(n){let e=JSON.stringify(Rn(n));return Ba("sha256").update(e).digest("hex").slice(0,16)}import*as Mr from"path";import{createHash as Ua,randomBytes as Da,timingSafeEqual as La}from"crypto";var Rr="hydra_session_",qa=3600*24*30,xr=12,Ja=32,za=50;function Wa(){return Mr.join(w.home(),"session-tokens.json")}function Cr(n){return Ua("sha256").update(n).digest("hex")}function Er(n){return Da(n).toString("hex")}function Qa(){return Er(xr).slice(0,xr*2)}function Va(){return`${Rr}${Er(Ja)}`}var Pt=class n{records=new Map;writeTimer=null;writeInflight=null;filePath;constructor(e,t){this.filePath=t;for(let s of e)this.records.set(s.hash,s)}static async load(){let e=[],t=Wa(),s=await te(t);s&&Array.isArray(s.records)&&(e=s.records.filter(Ga));let r=new n(e,t);return r.sweepExpired(new Date)>0&&await r.flush(),r}async issue(e={}){let t=Va(),s=Cr(t),r=Qa(),i=new Date,o=e.ttlSec&&e.ttlSec>0?e.ttlSec:qa,a=new Date(i.getTime()+o*1e3),d={id:r,hash:s,label:e.label,createdAt:i.toISOString(),expiresAt:a.toISOString(),lastUsedAt:i.toISOString()};return this.records.set(s,d),this.scheduleWrite(),{id:r,token:t,expiresAt:d.expiresAt}}async verify(e){if(typeof e!="string"||!e.startsWith(Rr))return;let t=Cr(e),s=this.records.get(t);if(!s)return;let r=Buffer.from(s.hash,"hex"),i=Buffer.from(t,"hex");if(r.length!==i.length||!La(r,i))return;let o=new Date;if(new Date(s.expiresAt).getTime()<=o.getTime()){this.records.delete(t),this.scheduleWrite();return}return s.lastUsedAt=o.toISOString(),this.scheduleWrite(),s.id}async revoke(e){for(let[t,s]of this.records)if(s.id===e)return this.records.delete(t),this.scheduleWrite(),!0;return!1}async revokeAll(){let e=this.records.size;return this.records.clear(),this.scheduleWrite(),e}list(){return Array.from(this.records.values()).map(({id:e,label:t,createdAt:s,expiresAt:r,lastUsedAt:i})=>({id:e,label:t,createdAt:s,expiresAt:r,lastUsedAt:i})).sort((e,t)=>t.createdAt.localeCompare(e.createdAt))}sweepExpired(e=new Date){let t=0;for(let[s,r]of this.records)new Date(r.expiresAt).getTime()<=e.getTime()&&(this.records.delete(s),t+=1);return t>0&&this.scheduleWrite(),t}async flush(){this.writeTimer&&(clearTimeout(this.writeTimer),this.writeTimer=null),await this.persist()}scheduleWrite(){this.writeTimer||(this.writeTimer=setTimeout(()=>{this.writeTimer=null,this.persist().catch(()=>{})},za))}persist(){let e=(this.writeInflight??Promise.resolve()).catch(()=>{}).then(()=>K(this.filePath,{records:Array.from(this.records.values())},{mode:384}));return this.writeInflight=e,e.catch(()=>{}).finally(()=>{this.writeInflight===e&&(this.writeInflight=null)}),e}};function Ga(n){if(!n||typeof n!="object")return!1;let e=n;return typeof e.id=="string"&&typeof e.hash=="string"&&typeof e.createdAt=="string"&&typeof e.expiresAt=="string"&&typeof e.lastUsedAt=="string"&&(e.label===void 0||typeof e.label=="string")}var Pr="Bearer ",Tt=class{constructor(e){this.token=e}token;async validate(e){return Ya(e,this.token)?"service":void 0}},_t=class{constructor(e){this.store=e}store;async validate(e){return this.store.verify(e)}},$t=class{constructor(e){this.validators=e}validators;async validate(e){for(let t of this.validators){let s=await t.validate(e);if(s!==void 0)return s}}};function Tr(n){return async function(t,s){let r=t.headers.authorization;if(!r||!r.startsWith(Pr)){s.code(401).send({error:"Missing bearer token"});return}let i=r.slice(Pr.length).trim(),o=await n.validator.validate(i);if(!o){s.code(403).send({error:"Invalid token"});return}t.authIdentity=o}}function _r(n){let e=n.headers["sec-websocket-protocol"],t=Array.isArray(e)?e.join(","):e;if(t)for(let s of t.split(",")){let r=s.trim(),i="hydra-acp-token.";if(r.startsWith(i))return r.slice(i.length)}if(n.url)try{let r=new URL(n.url,"http://localhost").searchParams.get("token");if(r)return r}catch{return}}var Ot=class{tokens=new Map;mint(e,t){let s=rt();return this.tokens.set(s,{name:e,kind:t}),s}revoke(e){for(let[t,s]of this.tokens)s.name===e&&this.tokens.delete(t)}resolve(e){return this.tokens.get(e)}async validate(e){let t=this.tokens.get(e);if(t)return`${t.kind}:${t.name}`}};function Ya(n,e){if(n.length!==e.length)return!1;let t=0;for(let s=0;s<n.length;s+=1)t|=n.charCodeAt(s)^e.charCodeAt(s);return t===0}var Ft=class{entries=new Map;maxFails;windowMs;constructor(e=10,t=900*1e3){this.maxFails=e,this.windowMs=t}isBlocked(e){let t=this.entries.get(e);return t?Date.now()-t.windowStart>this.windowMs?(this.entries.delete(e),!1):t.fails>=this.maxFails:!1}recordFailure(e){let t=Date.now(),s=this.entries.get(e);if(!s||t-s.windowStart>this.windowMs){this.entries.set(e,{fails:1,windowStart:t});return}s.fails+=1}recordSuccess(e){this.entries.delete(e)}};import*as tt from"os";import*as Yr from"path";import{posix as Ka}from"path";import Xa from"strip-ansi";function Ze(n){if(!n||typeof n!="object")return;let e=n;return typeof e.workerTaskId=="string"?e.workerTaskId:void 0}var Za=/[\x00-\x08\x0b-\x1f\x7f]/g;function Z(n){return Xa(n).replace(Za,"")}function H(n){return Z(n).replace(/[\n\t]+/g," ").replace(/ +/g," ").trim()}function Fr(n,e={}){if(!n||typeof n!="object")return null;let t=n,s=t.sessionUpdate??t.kind;if(typeof s!="string")return null;switch(s){case"agent_message_chunk":return od(t);case"agent_thought_chunk":case"agent_thought":return ad(t);case"user_message_chunk":return dd(t);case"prompt_received":return cd(t);case"tool_call":return ud(t,e);case"tool_call_update":return gd(t,e);case"plan":return wd(t);case"current_mode_update":return vd(t);case"current_model_update":return bd(t);case"turn_complete":return Id(t);case"usage_update":return id(t);case"available_commands_update":return sd(t);case"available_modes_update":return rd(t);case"session_info_update":return td(t);case"config_option_update":return ed(t);default:return{kind:"unknown",sessionUpdate:s,raw:n}}}function ed(n){let e=n.configOptions;if(!Array.isArray(e))return null;let t=[];for(let s of e){if(!s||typeof s!="object")continue;let r=s;if(typeof r.id!="string"||typeof r.currentValue!="string"||!Array.isArray(r.options))continue;let i=[];for(let o of r.options){if(!o||typeof o!="object")continue;let a=o;typeof a.value=="string"&&i.push({value:a.value,name:typeof a.name=="string"?a.name:a.value,...typeof a.description=="string"?{description:a.description}:{}})}t.push({id:r.id,name:typeof r.name=="string"?r.name:r.id,type:"select",currentValue:r.currentValue,options:i,...typeof r.category=="string"?{category:r.category}:{}})}return{kind:"config-options",options:t}}function td(n){let e=F(n,"title"),t=e!==void 0?H(e):void 0,s=n._meta,r;if(s&&typeof s=="object"&&!Array.isArray(s)){let o=s["hydra-acp"];if(o&&typeof o=="object"&&!Array.isArray(o)){let a=o.agentId;typeof a=="string"&&(r=a)}}if(t===void 0&&r===void 0)return null;let i={kind:"session-info"};return t!==void 0&&(i.title=t),r!==void 0&&(i.agentId=r),i}function nd(n){if(!Array.isArray(n))return[];let e=[];for(let t of n){if(!t||typeof t!="object")continue;let s=t;if(typeof s.name!="string"||s.name.length===0)continue;let r=s.name.startsWith("/")?s.name:`/${s.name}`,i={name:H(r)};typeof s.description=="string"&&(i.description=H(s.description)),e.push(i)}return e}function sd(n){let e=n.availableCommands??n.commands;return Array.isArray(e)?{kind:"available-commands",commands:nd(e)}:null}function rd(n){let e=n.availableModes;if(!Array.isArray(e))return null;let t=[];for(let s of e){if(!s||typeof s!="object")continue;let r=s;if(typeof r.id!="string"||r.id.length===0)continue;let i={id:H(r.id)};typeof r.name=="string"&&(i.name=H(r.name)),typeof r.description=="string"&&(i.description=H(r.description)),t.push(i)}return{kind:"available-modes",modes:t}}function id(n){let e={kind:"usage-update"};if(typeof n.used=="number"&&(e.used=n.used),typeof n.size=="number"&&(e.size=n.size),n.cost&&typeof n.cost=="object"){let t=n.cost;typeof t.amount=="number"&&(e.costAmount=t.amount),typeof t.currency=="string"&&(e.costCurrency=t.currency)}return e}function od(n){let e=et(n.content);return e===null?null:{kind:"agent-text",text:e,workerTaskId:Ze(n)}}function ad(n){let e=typeof n.text=="string"?Z(n.text):et(n.content);return e===null?null:{kind:"agent-thought",text:e,workerTaskId:Ze(n)}}function dd(n){let e=n._meta;if(e&&typeof e=="object"&&!Array.isArray(e)){let s=e["hydra-acp"];if(s&&typeof s=="object"&&!Array.isArray(s)&&s.compatFor==="prompt_received")return null}let t=et(n.content);return t===null?null:{kind:"user-text",text:t}}function cd(n){let e=Sd(n.prompt);return e===null?null:{kind:"user-text",text:e}}function Nr(n){return n?n.toLowerCase().replace(/[_\s-]/g,"")==="exitplanmode":!1}function $r(n){if(typeof n=="string")return{text:n};if(n&&typeof n=="object"&&!Array.isArray(n)){let e=n;if(typeof e.__hydraBlob=="string")return{ref:{hash:e.__hydraBlob,bytes:typeof e.bytes=="number"?e.bytes:0}}}}function Nt(n){let e=n.content;if(Array.isArray(e))for(let s of e){if(!s||typeof s!="object")continue;let r=s;if(r.type!=="diff")continue;let i=$r(r.oldText),o=$r(r.newText);if(i===void 0&&o===void 0)continue;let a=typeof r.path=="string"?r.path:void 0;return{...a!==void 0?{path:a}:{},oldText:i?.text??"",newText:o?.text??"",...i?.ref?{oldRef:i.ref}:{},...o?.ref?{newRef:o.ref}:{}}}let t=n.rawInput;if(t&&typeof t=="object"&&!Array.isArray(t)){let s=t,r=typeof s.file_path=="string"?s.file_path:typeof s.path=="string"?s.path:void 0;if(typeof s.old_string=="string"&&typeof s.new_string=="string")return{...r!==void 0?{path:r}:{},oldText:s.old_string,newText:s.new_string};if(typeof s.content=="string")return{...r!==void 0?{path:r}:{},oldText:"",newText:s.content}}return null}function jr(n){let e=n.rawInput;if(!e||typeof e!="object"||Array.isArray(e))return null;let t=e.plan;return typeof t!="string"||t.length===0?null:Z(t)}function ud(n,e={}){let t=F(n,"toolCallId")??F(n,"id");if(!t)return null;let s=F(n,"title")??F(n,"name")??F(n,"label")??"tool call",r=F(n,"name")??F(n,"title");if(Nr(r)){let u=jr(n);if(u!==null){let f=F(n,"status"),g={kind:"exit-plan-mode",toolCallId:t,plan:u};return f!==void 0&&(g.status=f),g}}let i=Br(H(s),n,e),o=F(n,"status"),a=F(n,"kind"),d={kind:"tool-call",toolCallId:t,title:i};o!==void 0&&(d.status=o),a!==void 0&&(d.rawKind=a);let c=Nt(n);c!==null&&(d.editDiff=c);let p=Hr(n);p!==void 0&&(d.detail=p);let l=Ze(n);return l!==void 0&&(d.workerTaskId=l),d}var Or=64;function Hr(n){let e=n.rawInput;if(!e||typeof e!="object"||Array.isArray(e))return;let t=e;if(typeof t.command=="string"&&t.command.trim().length>0){let i=H(t.command).trim().replace(/^cd\s+\S+\s+&&\s+/,"");return pd(i,Or)}let s=typeof t.file_path=="string"?t.file_path:typeof t.filePath=="string"?t.filePath:typeof t.path=="string"?t.path:void 0;if(s!==void 0&&s.length>0)return md(Se(H(s)),Or)}var ld=["home/","Users/","root/","tmp/","var/","opt/","etc/","usr/","mnt/","private/"];function fd(n){for(let e of ld)if(n.startsWith(e))return!0;return!1}function Br(n,e,t={}){if(n.length===0)return n;if(n.startsWith("/"))return Se(n);if(n.startsWith("~")||!n.includes("/")||/\s/.test(n))return n;let s=e.rawInput;if(s&&typeof s=="object"&&!Array.isArray(s)){let r=s,i=[r.file_path,r.filePath,r.path];for(let o of i)if(!(typeof o!="string"||o.length===0)&&(o===`/${n}`||o.endsWith(`/${n}`)||o===n))return Se(o)}return fd(n)?Se(`/${n}`):t.cwd&&t.cwd.length>0?Se(Ka.resolve(t.cwd,n)):n}function pd(n,e){return n.length>e?`${n.slice(0,e-1)}\u2026`:n}function md(n,e){if(n.length<=e)return n;let t=n.slice(-(e-1)),s=t.indexOf("/");return s>=0&&s<e/2?`\u2026${t.slice(s)}`:`\u2026${t}`}function gd(n,e={}){let t=F(n,"toolCallId")??F(n,"id");if(!t)return null;let s=F(n,"title"),r=s!==void 0?Br(H(s),n,e):void 0,i=F(n,"status"),o=Nt(n),a=Hr(n);if(!(r!==void 0||o!==null||a!==void 0||i==="completed"||i==="failed"||i==="rejected"||i==="cancelled"))return null;let c=F(n,"name")??s;if(Nr(c)){let u={kind:"exit-plan-mode",toolCallId:t},f=jr(n);return f!==null&&(u.plan=f),i!==void 0&&(u.status=i),u}let p={kind:"tool-call-update",toolCallId:t};if(r!==void 0&&(p.title=r),a!==void 0&&(p.detail=a),i!==void 0&&(p.status=i),o!==null&&(p.editDiff=o),i==="failed"){let u=hd(n);u!==null&&(p.errorText=u),yd(n,u)&&(p.upstreamInterrupted=!0)}let l=Ze(n);return l!==void 0&&(p.workerTaskId=l),p}function hd(n){let e=n.content;if(Array.isArray(e))for(let s of e){if(!s||typeof s!="object")continue;let i=et(s.content);if(i!==null&&i.length>0)return i}let t=n.rawOutput;if(t&&typeof t=="object"){let s=t.error;if(typeof s=="string"&&s.length>0)return Z(s)}return null}function yd(n,e){let t=n.rawOutput;if(t&&typeof t=="object"){let s=t.metadata;if(s&&typeof s=="object"&&s.interrupted===!0)return!0}return!!(e!==null&&e.toLowerCase().includes("tool execution aborted"))}function wd(n){let e=n.entries;if(!Array.isArray(e)||e.length===0)return null;let t=[];for(let s of e){if(!s||typeof s!="object")continue;let r=s,i=typeof r.content=="string"?H(r.content):void 0;if(!i)continue;let o={content:i};typeof r.status=="string"&&(o.status=r.status),typeof r.priority=="string"&&(o.priority=r.priority),t.push(o)}return{kind:"plan",entries:t}}function vd(n){let e=F(n,"currentModeId")??F(n,"currentMode")??F(n,"mode");return e?{kind:"mode-changed",mode:H(e)}:null}function bd(n){let e=F(n,"currentModel")??F(n,"model");if(!e)return null;let t=n.availableModels,s=Array.isArray(t)?t.map(r=>typeof r=="object"&&r!==null?r.modelId:typeof r=="string"?r:void 0).filter(r=>typeof r=="string"&&r.length>0):void 0;return{kind:"model-changed",model:H(e),...s&&s.length>0?{availableModels:s}:{}}}function Id(n){let e=F(n,"stopReason"),t=n._meta,s=t?.["hydra-acp"]?.amended!==void 0&&t["hydra-acp"].amended!==null,r={kind:"turn-complete"};return e!==void 0&&(r.stopReason=e),s&&(r.amended=!0),r}function et(n){if(typeof n=="string")return Z(n);if(!n||typeof n!="object")return null;let e=n;return e.type==="text"&&typeof e.text=="string"||typeof e.text=="string"?Z(e.text):null}function Sd(n){if(!Array.isArray(n))return null;let e=[];for(let t of n){let s=et(t);s!==null&&e.push(s)}return e.length===0?null:e.join("")}function F(n,e){let t=n[e];return typeof t=="string"?t:void 0}function Ur(n){let e=new Map,t=new Map;for(let r of n){let o=r.params?.update;if(!o||typeof o!="object")continue;let a=o.sessionUpdate;if(a!=="tool_call"&&a!=="tool_call_update")continue;let d=typeof o.toolCallId=="string"&&o.toolCallId.length>0?o.toolCallId:void 0,c=Ad(o);if(c.length===0)continue;let p=c;if(d!==void 0){let l=e.get(d)??[],u=[];for(let f of c)l.some(g=>g.path===f.path&&g.oldText===f.oldText&&g.newText===f.newText)||u.push(f);if(u.length===0)continue;e.set(d,[...l,...u]),p=u}for(let l of p)kd(t,l)}let s=[];for(let[r,i]of t)s.push({path:r,hunks:i.hunks,created:i.created});return s}function Dr(n){let e=[];for(let t of n){let s=-1;for(let i=e.length-1;i>=0;i--)if(e[i].newText===t.oldText){s=i;break}if(s===-1){e.push(t);continue}let r=e[s];e.splice(s,1,{oldText:r.oldText,newText:t.newText})}return e}function kd(n,e){let t={oldText:e.oldText,newText:e.newText},s=n.get(e.path);if(s===void 0){n.set(e.path,{hunks:[t],created:e.oldText.length===0});return}s.hunks.push(t)}function Ad(n){let e=[],t=n.rawInput;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=t,o=typeof i.file_path=="string"?i.file_path:typeof i.path=="string"?i.path:void 0,a=i.edits;if(o!==void 0&&Array.isArray(a)){for(let d of a){if(!d||typeof d!="object")continue;let c=d,p=typeof c.old_string=="string"?c.old_string:void 0,l=typeof c.new_string=="string"?c.new_string:void 0;p===void 0||l===void 0||e.push({path:o,oldText:p,newText:l})}if(e.length>0)return e}}let s=n.content;if(Array.isArray(s)){for(let i of s){if(!i||typeof i!="object")continue;let o=i;if(o.type!=="diff")continue;let a=typeof o.path=="string"?o.path:void 0;if(a===void 0)continue;let d=typeof o.oldText=="string"?o.oldText:"",c=typeof o.newText=="string"?o.newText:"";e.push({path:a,oldText:d,newText:c})}if(e.length>0)return e}let r=Nt(n);return r&&r.path&&e.push({path:r.path,oldText:r.oldText,newText:r.newText}),e}function Lr(n){let e=xd(n),t=Cd(e),s=[];Md(s,n),Rd(s,e,t);let r=s.join(`
83
103
  `);return r.endsWith(`
84
104
  `)||(r+=`
85
- `),r}function Ad(n){let e=[];for(let t of n.history){if(t.method!=="session/update")continue;let s=t.params;if(!s||typeof s!="object")continue;let r=Fr(s.update,{cwd:n.session.cwd});r!==null&&e.push({event:r,recordedAt:t.recordedAt})}return e}function xd(n){let e=new Map;for(let{event:t}of n){if(t.kind==="tool-call"){let s=e.get(t.toolCallId);e.set(t.toolCallId,{title:t.title,status:t.status??s?.status??"pending"});continue}if(t.kind==="tool-call-update"){let s=e.get(t.toolCallId)??{title:"tool call",status:"pending"};e.set(t.toolCallId,{title:t.title??s.title,status:t.status??s.status})}}return e}function Cd(n,e){let t=e.session,s=Hs(t.sessionId),r=t.title?.trim()||`Hydra session ${s}`;n.push(`# ${be(r)}`),n.push("");let i=[];i.push(`- **Session:** \`${s}\` (lineage \`${t.lineageId}\`)`);let o=[t.agentId];t.currentModel&&o.push(`model: ${t.currentModel}`),t.currentMode&&o.push(`mode: ${t.currentMode}`),i.push(`- **Agent:** ${o.filter(Boolean).join(" \xB7 ")}`),i.push(`- **Cwd:** ${t.cwd}`),i.push(`- **Exported:** ${e.exportedAt} from ${e.exportedFrom.machine} (hydra ${e.exportedFrom.hydraVersion})`);let a=t.currentUsage;if(a&&(a.used!==void 0||a.costAmount!==void 0)){let d=[];if(a.used!==void 0){let c=a.size!==void 0?`${En(a.size)}`:void 0;d.push(c?`${En(a.used)} / ${c} tokens`:`${En(a.used)} tokens`)}if(a.costAmount!==void 0){let c=a.costCurrency??"USD";d.push(`$${a.costAmount.toFixed(2)} ${c}`)}i.push(`- **Usage:** ${d.join(" \xB7 ")}`)}n.push(i.join(`
86
- `)),n.push("")}function Md(n,e,t){if(!e.some(c=>Rd(c.event))){n.push("_No conversation history recorded._"),n.push("");return}let s=new Set,r=0,i="",o=!1,a=()=>{i.length!==0&&(n.push(i.trimEnd()),n.push(""),i="")},d=()=>{o||(r+=1,n.push("---"),n.push(""),n.push(`## Turn ${r}`),n.push(""),o=!0)};for(let{event:c}of e)switch(c.kind){case"user-text":{a(),r+=1,n.push("---"),n.push(""),n.push(`## Turn ${r}`),n.push(""),n.push("**User:**"),n.push("");for(let f of c.text.split(`
87
- `))n.push(`> ${be(f)}`);n.push(""),n.push("**Assistant:**"),n.push(""),o=!0;break}case"agent-text":d(),i+=c.text;break;case"agent-thought":{d(),a();let f=c.text.split(`
88
- `);for(let l of f)n.push(`> _${be(l)}_`);n.push("");break}case"tool-call":{if(d(),a(),s.has(c.toolCallId))break;s.add(c.toolCallId);let f=t.get(c.toolCallId)??{title:c.title,status:c.status??"pending"};n.push(`- ${Pd(f.status)} ${Ed(f)}`),n.push("");break}case"tool-call-update":break;case"plan":{d(),a(),n.push("**Plan:**"),n.push("");for(let f of c.entries){let l=f.status==="completed"?"[x]":"[ ]";n.push(`- ${l} ${be(f.content)}`)}n.push("");break}case"mode-changed":d(),a(),n.push(`_mode: ${be(c.mode)}_`),n.push("");break;case"model-changed":d(),a(),n.push(`_model: ${be(c.model)}_`),n.push("");break;case"turn-complete":a();break;case"usage-update":case"available-commands":case"session-info":case"unknown":break}a()}function Rd(n){switch(n.kind){case"usage-update":case"available-commands":case"session-info":case"unknown":case"turn-complete":return!1;default:return!0}}function Ed(n){let e=n.status,t=e==="completed"||e===void 0?"":` _(${e})_`;return`${be(n.title)}${t}`}function Pd(n){switch(n){case"completed":return"\u2713";case"failed":return"\u2717";case"cancelled":case"rejected":return"\u2298";case"in_progress":return"\u21BB";default:return"\xB7"}}function be(n){return n.replace(/</g,"&lt;").replace(/>/g,"&gt;")}function En(n){return n.toLocaleString("en-US")}function Lr(n){return n==="127.0.0.1"||n==="::1"||n==="localhost"||n==="[::1]"}function Td(n){let e=n.trim();if(e.length===0)return{operator:"OR",terms:[]};let t=/\w+:"[^"]*"|"[^"]*"|\S+/g,s=[],r;for(;(r=t.exec(e))!==null;)s.push(r[0]);let i="OR",o=!1,a=!1,d=[];for(let f of s){let l=f.toUpperCase();l==="AND"?o=!0:l==="OR"?a=!0:d.push(f)}o?i="AND":a&&(i="OR");let c=d.map(f=>_d(f)).filter(f=>f.term.length>0);return{operator:i,terms:c}}function _d(n){let e=/^(\w+):"([^"]*)"$/.exec(n);if(e)return{scope:qr(e[1]),term:e[2]};let t=/^"([^"]*)"$/.exec(n);if(t)return{scope:"all",term:t[1]};let s=/^(prompt|response|tool):([\s\S]*)$/i.exec(n);return s?{scope:qr(s[1]),term:s[2].trim()}:{scope:"all",term:n.trim()}}function qr(n){switch(n.toLowerCase()){case"prompt":return"user";case"response":return"agent";case"tool":return"tool";default:return"all"}}function $d(n,e){return n==="all"?!0:n==="user"?e==="user":n==="agent"?e==="agent"||e==="thought":e==="tool"||e==="tool-input"}var Fd=5,Od=200,Jr=30;async function Qr(n,e,t={}){let s=Td(e);if(s.terms.length===0)return{query:e,truncated:!1,results:[]};let r=t.maxSnippetsPerSession??Fd,i=t.maxSessions??Od,o=t.sessionIds?new Set(t.sessionIds):null,a=await n.list(),d=o?a.filter(l=>o.has(l.sessionId)):a,c=[],f=!1;for(let l of d){if(c.length>=i){f=!0;break}let u=await n.loadHistory(l.sessionId).catch(()=>[]),p=Nd(u,s,r);if(p.snippets.length===0)continue;let g={sessionId:l.sessionId,cwd:l.cwd,status:l.status,updatedAt:l.updatedAt,totalMatches:p.totalMatches,snippets:p.snippets};l.title!==void 0&&(g.title=l.title),c.push(g)}return{query:e,truncated:f,results:c}}function Nd(n,e,t){if(e.terms.length===0)return{totalMatches:0,snippets:[]};let s=0,r=[];for(let{scope:i,term:o}of e.terms){let a=jd(n,o,i,t-r.length);if(e.operator==="AND"&&a.totalMatches===0)return{totalMatches:0,snippets:[]};s+=a.totalMatches,r.push(...a.snippets)}return{totalMatches:s,snippets:r}}function jd(n,e,t,s){let r=e.toLowerCase(),i=0,o=[];for(let a of n){let d=Hd(a).filter(c=>$d(t,c.kind));for(let c of d){let f=c.text.toLowerCase(),l=f.indexOf(r);if(l===-1)continue;let u=0;for(;l!==-1;)u++,l=f.indexOf(r,l+r.length);if(i+=u,o.length<s){let p=f.indexOf(r),g={kind:c.kind,text:qd(c.text,p,r.length),recordedAt:a.recordedAt};c.toolName!==void 0&&(g.toolName=c.toolName),o.push(g)}}}return{totalMatches:i,snippets:o}}function Hd(n){if(n.method!=="session/update")return[];let e=n.params;if(!e||typeof e!="object"||Array.isArray(e))return[];let t=e.update;if(!t||typeof t!="object"||Array.isArray(t))return[];let s=t,r=typeof s.sessionUpdate=="string"?s.sessionUpdate:s.kind;if(typeof r!="string")return[];switch(r){case"agent_message_chunk":{let i=jt(s.content);return i?[{kind:"agent",text:i}]:[]}case"agent_thought":case"agent_thought_chunk":{let i=typeof s.text=="string"?Z(s.text):jt(s.content);return i?[{kind:"thought",text:i}]:[]}case"user_message_chunk":{if(Dd(s))return[];let i=jt(s.content);return i?[{kind:"user",text:i}]:[]}case"prompt_received":{let i=Ld(s.prompt);return i?[{kind:"user",text:i}]:[]}case"tool_call":case"tool_call_update":return Bd(s);default:return[]}}function Bd(n){let e=zr(n,"name"),t=zr(n,"title"),s=[];if(t!==void 0){let a=H(t);if(a.length>0){let d={kind:"tool",text:a};e!==void 0&&(d.toolName=e),s.push(d)}}if(e!==void 0&&e!==t){let a=H(e);a.length>0&&s.push({kind:"tool",toolName:e,text:a})}let r=n.rawInput;if(r&&typeof r=="object"){let a=Wr(r);if(a.length>0){let d={kind:"tool-input",text:H(a)};e!==void 0&&(d.toolName=e),s.push(d)}}let i=n.locations;if(Array.isArray(i)&&i.length>0){let a=Wr(i);if(a.length>0){let d={kind:"tool-input",text:H(a)};e!==void 0&&(d.toolName=e),s.push(d)}}let o=Ud(n);if(o!==null){let a={kind:"tool",text:o};e!==void 0&&(a.toolName=e),s.push(a)}return s}function Ud(n){let e=n.content;if(Array.isArray(e))for(let s of e){if(!s||typeof s!="object")continue;let i=s.content;if(!i||typeof i!="object")continue;let o=i;if(o.type==="text"&&typeof o.text=="string"){let a=H(o.text);if(a.length>0)return a}}let t=n.rawOutput;if(t&&typeof t=="object"){let s=t.error;if(typeof s=="string"){let r=H(s);if(r.length>0)return r}}return null}function Dd(n){let e=n._meta;if(!e||typeof e!="object"||Array.isArray(e))return!1;let t=e["hydra-acp"];return!t||typeof t!="object"||Array.isArray(t)?!1:t.compatFor==="prompt_received"}function jt(n){if(typeof n=="string")return Z(n);if(!n||typeof n!="object"||Array.isArray(n))return"";let e=n;return typeof e.text=="string"?Z(e.text):""}function Ld(n){if(!Array.isArray(n))return"";let e=[];for(let t of n){let s=jt(t);s.length>0&&e.push(s)}return e.join("")}function zr(n,e){let t=n[e];return typeof t=="string"?t:void 0}function Wr(n){try{return JSON.stringify(n)}catch{return""}}function qd(n,e,t){let s=n.replace(/\s+/g," ").trim();if(s.length===0)return"";let r=s.toLowerCase(),i=n.slice(e,e+t).toLowerCase().replace(/\s+/g," ").trim(),o=i.length>0?r.indexOf(i):0;o===-1&&(o=0);let a=Math.max(0,o-Jr),d=Math.min(s.length,o+i.length+Jr),c=a>0?"\u2026":"",f=d<s.length?"\u2026":"";return`${c}${s.slice(a,d)}${f}`}function Vr(n){if(n.publicHost&&n.publicHost.length>0)return n.publicHost;if(n.host&&!Lr(n.host))return n.port!==void 0?`${n.host}:${n.port}`:n.host}function Yr(n,e,t){n.get("/v1/sessions",async s=>{let r=s.query,i=r?.includeNonInteractive==="1"||r?.includeNonInteractive==="true";return{sessions:await e.list({cwd:r?.cwd,includeNonInteractive:i})}}),n.post("/v1/sessions/search",async(s,r)=>{let i=s.body??{},o=typeof i.q=="string"?i.q:"";if(o.trim().length===0)return r.code(400).send({error:"q is required"}),r;let a=Array.isArray(i.sessionIds)?i.sessionIds.filter(c=>typeof c=="string"&&c.length>0):void 0;return await Qr(e,o,{sessionIds:a})}),n.post("/v1/sessions",async(s,r)=>{let i=s.body??{},o=He(i.cwd??t.cwd),a=i.agentId??t.agentId;try{let d=await e.create({cwd:o,agentId:a,mcpServers:i.mcpServers});r.code(201).send({sessionId:d.sessionId,agentId:d.agentId,cwd:d.cwd})}catch(d){r.code(500).send({error:d.message})}}),n.post("/v1/sessions/collect",async(s,r)=>{let i=s.body??{},o=typeof i.maxAgeDays=="number"&&i.maxAgeDays>0?i.maxAgeDays*24*60*60*1e3:0,a=typeof i.limit=="number"&&i.limit>0?i.limit:1e3,d=i.selection==="explicit"||i.selection==="unpromoted"?i.selection:"unpromoted";try{let c=await Mn({manager:e,maxAgeMs:o,maxDeletions:a,selection:d,verbose:!1});r.code(200).send(c)}catch(c){r.code(500).send({error:c.message})}}),n.post("/v1/sessions/:id/kill",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=e.get(o);if(a){a.close({deleteRecord:!1}).catch(()=>{}),r.code(202).send();return}if(!await e.hasRecord(o)){r.code(404).send({error:"session not found"});return}r.code(204).send()}),n.post("/v1/sessions/:id/stdin/open",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=e.get(o);if(!a)return r.code(404).send({error:"session not found"}),r;let d=s.body??{},c={};(d.mode==="memory"||d.mode==="file")&&(c.mode=d.mode),typeof d.capacityBytes=="number"&&(c.capacityBytes=d.capacityBytes),typeof d.fileCapBytes=="number"&&(c.fileCapBytes=d.fileCapBytes),(c.mode??"memory")==="file"&&(c.filePathFor=f=>Gr.join(tt.tmpdir(),`hydra-acp-stdin-${f}.log`));try{return a.openStream(c)}catch(f){return r.code(409).send({error:f.message}),r}}),n.post("/v1/sessions/:id/stdin",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=e.get(o);if(!a)return r.code(404).send({error:"session not found"}),r;let d=s.body??{},c=typeof d.chunk=="string"?d.chunk:"",f=d.eof===!0;try{return a.streamWrite(c,f)}catch(l){return r.code(409).send({error:l.message}),r}}),n.patch("/v1/sessions/:id",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=s.body??{};if(a.priority!==void 0){let c=a.priority,f;if(c===null||c===0)f=void 0;else if(typeof c=="number"&&Number.isInteger(c)&&c>0)f=c;else{r.code(400).send({error:"priority must be a non-negative integer (or null to clear)"});return}if(!await e.setPriority(o,f)){r.code(404).send({error:"session not found"});return}r.code(204).send();return}if(a.regen===!0){if(!(e.get(o)!==void 0||await e.hasRecord(o))){r.code(404).send({error:"session not found"});return}e.scheduleSynopsis(o),r.code(202).send();return}if(typeof a.title!="string"||a.title.trim().length===0){r.code(400).send({error:"title must be a non-empty string"});return}if(!await e.setTitle(o,a.title)){r.code(404).send({error:"session not found"});return}r.code(204).send()}),n.delete("/v1/sessions/:id",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=e.get(o);if(a){await a.close({deleteRecord:!0}),r.code(204).send();return}if(!await e.deleteRecord(o)){r.code(404).send({error:"session not found"});return}r.code(204).send()}),n.get("/v1/sessions/:id/export",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=s.query?.tools,d=a==="references"?"references":Zs(a),c=await e.exportBundle(o,d==="references"?{tools:"references"}:{});if(!c){r.code(404).send({error:"session not found"});return}let f=Qe({record:c.record,history:d==="summary"?er(c.history,"summary"):c.history,promptHistory:c.promptHistory.length>0?c.promptHistory:void 0,...c.toolBlobs!==void 0?{toolBlobs:c.toolBlobs}:{},hydraVersion:Q,machine:tt.hostname(),hydraHost:Vr(t)}),l=new Date().toISOString().replace(/[:.]/g,"-");r.header("Content-Disposition",`attachment; filename="${o}-${l}.hydra"`),r.code(200).send(f)}),n.get("/v1/sessions/:id/diff",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=await e.exportBundle(o);if(!a){r.code(404).send({error:"session not found"});return}let d=s.query??{},c=d.fold==="true"||d.fold==="1",f=typeof d.paths=="string"&&d.paths.length>0?new Set(d.paths.split(",").map(u=>u.trim()).filter(u=>u.length>0)):void 0,l=Br(a.history);f&&(l=l.filter(u=>f.has(u.path))),c&&(l=l.map(u=>({...u,hunks:Ur(u.hunks)}))),r.code(200).send(l)}),n.get("/v1/sessions/:id/tools/:hash",async(s,r)=>{let i=s.params,o=await e.resolveCanonicalId(i.id)??i.id,a=await e.loadToolBlob(o,i.hash);if(a===null){r.code(404).send({error:"tool blob not found"});return}r.header("Content-Type","text/plain; charset=utf-8"),r.code(200).send(a)}),n.get("/v1/sessions/:id/transcript",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=await e.exportBundle(o);if(!a){r.code(404).send({error:"session not found"});return}let d=Qe({record:a.record,history:a.history,promptHistory:a.promptHistory.length>0?a.promptHistory:void 0,hydraVersion:Q,machine:tt.hostname(),hydraHost:Vr(t)});r.header("Content-Type","text/markdown; charset=utf-8"),r.code(200).send(Dr(d))}),n.post("/v1/sessions/:id/fork",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=s.body??{},d={};if(a.forkAt!==void 0){if(typeof a.forkAt!="string"||a.forkAt.length===0){r.code(400).send({error:"forkAt must be a non-empty string"});return}d.forkAt=a.forkAt}if(a.cwd!==void 0){if(typeof a.cwd!="string"||a.cwd.length===0){r.code(400).send({error:"cwd must be a non-empty string"});return}d.cwd=He(a.cwd)}if(a.agentId!==void 0){if(typeof a.agentId!="string"||a.agentId.length===0){r.code(400).send({error:"agentId must be a non-empty string"});return}d.agentId=a.agentId}try{let c=await e.forkSession(o,d);r.code(201).send(c)}catch(c){let f=c;if(f.code===I.SessionNotFound){r.code(404).send({error:f.message});return}if(f.code===I.InvalidParams||f.code===I.AgentNotInstalled){r.code(400).send({error:f.message});return}r.code(500).send({error:f.message})}}),n.post("/v1/sessions/import",async(s,r)=>{let i=s.body??{};if(i.bundle===void 0){r.code(400).send({error:"missing bundle"});return}let o;if(i.cwd!==void 0){if(typeof i.cwd!="string"||i.cwd.length===0){r.code(400).send({error:"cwd must be a non-empty string"});return}o=i.cwd}let a;try{a=dr(i.bundle)}catch(d){r.code(400).send({error:"invalid bundle",details:d.message});return}try{let d=await e.importBundle(a,{replace:i.replace===!0,...o!==void 0?{cwd:o}:{}});r.code(201).send(d)}catch(d){let c=d;if(c.code===I.BundleAlreadyImported){r.code(409).send({error:"bundle already imported",existingSessionId:c.existingSessionId});return}r.code(500).send({error:c.message})}}),n.get("/v1/sessions/:id/history",async(s,r)=>{let i=s.params.id,o=s.query,a=o?.follow==="1"||o?.follow==="true",d=await e.resolveCanonicalId(i)??i,c=e.get(d),f,l,u=!1,p=[];if(c)a&&(l=c.onBroadcast(m=>{r.raw.writableEnded||(u?r.raw.write(JSON.stringify(m)+`
89
- `):p.push(m))})),f=await c.getHistorySnapshot();else{let m=await e.getHistory(d);if(m===void 0)return r.code(404).send({error:"session not found"}),r;f=m}r.raw.setHeader("Content-Type","application/x-ndjson"),r.raw.setHeader("Cache-Control","no-cache"),r.raw.statusCode=200;let g=new Set;for(let m of f??[]){r.raw.write(JSON.stringify(m)+`
90
- `);let y=m;typeof y.recordedAt=="number"&&g.add(String(y.recordedAt))}for(let m of p){let y=m,v=typeof y.recordedAt=="number"?String(y.recordedAt):"";v&&g.has(v)||r.raw.write(JSON.stringify(m)+`
91
- `)}return u=!0,l?(s.raw.on("close",()=>{l?.(),r.raw.writableEnded||r.raw.end()}),r):(r.raw.end(),r)})}function Kr(n,e,t,s={}){n.get("/v1/agents",async()=>Ce(e)),n.get("/v1/registry",async()=>e.load()),n.post("/v1/registry/refresh",async()=>{let r=await e.refresh();return{version:r.version,agentCount:r.agents.length}}),n.post("/v1/agents/:id/install",async(r,i)=>{let o=r.params.id,a=await e.getAgent(o);if(!a){i.code(404).send({error:`agent ${o} not found in registry`});return}if(a.distribution.uvx&&!a.distribution.npx&&!a.distribution.binary){i.send({agentId:a.id,version:a.version??"current",distribution:"uvx",installed:!1,message:"uvx agents resolve on first run; nothing to pre-install."});return}try{let d=await oe(a,[],{npmRegistry:s.npmRegistry}),c=a.distribution.npx?"npx":a.distribution.binary?"binary":"unknown";i.send({agentId:a.id,version:d.version,distribution:c,installed:!0,command:d.command})}catch(d){i.code(500).send({error:d.message})}}),n.post("/v1/agents/:id/sync",async(r,i)=>{let o=r.params.id;try{let{synced:a,skipped:d}=await t.syncFromAgent(o);return{synced:a.map(c=>({sessionId:c.sessionId,upstreamSessionId:c.upstreamSessionId,agentId:c.agentId,cwd:c.cwd,title:c.title,updatedAt:c.updatedAt})),skipped:d}}catch(a){let d=a;if(d.code===I.AgentNotInstalled){i.code(404).send({error:d.message});return}i.code(409).send({error:d.message})}})}function Xr(n,e,t){n.get("/v1/health",{config:{skipAuth:!0}},async()=>({status:"ok",version:e,configDigest:t}))}var Jd=/^[A-Za-z0-9._-]+$/;function Zr(n,e){n.get("/v1/extensions",async()=>({extensions:e.list()})),n.get("/v1/extensions/:name",async(t,s)=>{let r=t.params.name,i=e.get(r);if(!i){s.code(404).send({error:`unknown extension: ${r}`});return}return i}),n.post("/v1/extensions",async(t,s)=>{let r=t.body??{},i=zd(r);if("error"in i){s.code(400).send({error:i.error});return}try{let o=e.register(i.config);s.code(201).send(o)}catch(o){nt(s,o)}}),n.delete("/v1/extensions/:name",async(t,s)=>{let r=t.params.name;try{await e.unregister(r),s.code(204).send()}catch(i){nt(s,i)}}),n.post("/v1/extensions/:name/start",async(t,s)=>{let r=t.params.name;try{let i=await e.startByName(r);s.code(200).send(i)}catch(i){nt(s,i)}}),n.post("/v1/extensions/:name/stop",async(t,s)=>{let r=t.params.name;try{let i=await e.stopByName(r);s.code(200).send(i)}catch(i){nt(s,i)}}),n.post("/v1/extensions/:name/restart",async(t,s)=>{let r=t.params.name;try{let i=await e.restartByName(r);s.code(200).send(i)}catch(i){nt(s,i)}})}function nt(n,e){let t=e.code,s=e.message??"unknown error";if(t==="NOT_FOUND"){n.code(404).send({error:s});return}if(t==="CONFLICT"){n.code(409).send({error:s});return}n.code(500).send({error:s})}function zd(n){let e=n.name;if(typeof e!="string"||!Jd.test(e))return{error:"name must match [A-Za-z0-9._-]+"};let t=n.command;if(t!==void 0&&(!Array.isArray(t)||t.some(o=>typeof o!="string")))return{error:"command must be string[]"};let s=n.args;if(s!==void 0&&(!Array.isArray(s)||s.some(o=>typeof o!="string")))return{error:"args must be string[]"};let r=n.env;if(r!==void 0&&(typeof r!="object"||r===null||Array.isArray(r)))return{error:"env must be an object of string\u2192string"};if(r&&Object.values(r).some(o=>typeof o!="string"))return{error:"env values must be strings"};let i=n.enabled;return i!==void 0&&typeof i!="boolean"?{error:"enabled must be a boolean"}:{config:{name:e,command:t??[],args:s??[],env:r??{},enabled:i===void 0?!0:i}}}var Wd=/^[A-Za-z0-9._-]+$/;function ei(n,e){n.get("/v1/transformers",async()=>({transformers:e.list()})),n.get("/v1/transformers/:name",async(t,s)=>{let r=t.params.name,i=e.get(r);if(!i){s.code(404).send({error:`unknown transformer: ${r}`});return}return i}),n.post("/v1/transformers",async(t,s)=>{let r=t.body??{},i=Qd(r);if("error"in i){s.code(400).send({error:i.error});return}try{let o=e.register(i.config);s.code(201).send(o)}catch(o){st(s,o)}}),n.delete("/v1/transformers/:name",async(t,s)=>{let r=t.params.name;try{await e.unregister(r),s.code(204).send()}catch(i){st(s,i)}}),n.post("/v1/transformers/:name/start",async(t,s)=>{let r=t.params.name;try{let i=await e.startByName(r);s.code(200).send(i)}catch(i){st(s,i)}}),n.post("/v1/transformers/:name/stop",async(t,s)=>{let r=t.params.name;try{let i=await e.stopByName(r);s.code(200).send(i)}catch(i){st(s,i)}}),n.post("/v1/transformers/:name/restart",async(t,s)=>{let r=t.params.name;try{let i=await e.restartByName(r);s.code(200).send(i)}catch(i){st(s,i)}})}function st(n,e){let t=e.code,s=e.message??"unknown error";if(t==="NOT_FOUND"){n.code(404).send({error:s});return}if(t==="CONFLICT"){n.code(409).send({error:s});return}n.code(500).send({error:s})}function Qd(n){let e=n.name;if(typeof e!="string"||!Wd.test(e))return{error:"name must match [A-Za-z0-9._-]+"};let t=n.command;if(t!==void 0&&(!Array.isArray(t)||t.some(o=>typeof o!="string")))return{error:"command must be string[]"};let s=n.args;if(s!==void 0&&(!Array.isArray(s)||s.some(o=>typeof o!="string")))return{error:"args must be string[]"};let r=n.env;if(r!==void 0&&(typeof r!="object"||r===null||Array.isArray(r)))return{error:"env must be an object of string\u2192string"};if(r&&Object.values(r).some(o=>typeof o!="string"))return{error:"env values must be strings"};let i=n.enabled;return i!==void 0&&typeof i!="boolean"?{error:"enabled must be a boolean"}:{config:{name:e,command:t??[],args:s??[],env:r??{},enabled:i===void 0?!0:i}}}function ti(n,e){n.get("/v1/config",async()=>e)}import{z as Ne}from"zod";import*as Oe from"fs/promises";import*as ni from"path";import{randomBytes as tp,scrypt as Vd,timingSafeEqual as Gd}from"crypto";import{promisify as Yd}from"util";var Kd=Yd(Vd);function si(){return ni.join(w.home(),"password-hash")}var Xd=128*1024*1024;async function ri(){try{return(await Oe.readFile(si(),"utf8")).trim().length>0}catch(n){if(n.code==="ENOENT")return!1;throw n}}async function ii(n){if(typeof n!="string"||n.length===0)return!1;let e;try{e=(await Oe.readFile(si(),"utf8")).trim()}catch(c){if(c.code==="ENOENT")return!1;throw c}let t=e.split("$");if(t.length!==6||t[0]!=="scrypt")return!1;let s=parseInt(t[1],10),r=parseInt(t[2],10),i=parseInt(t[3],10);if(!Number.isFinite(s)||!Number.isFinite(r)||!Number.isFinite(i))return!1;let o=Buffer.from(t[4],"hex"),a=Buffer.from(t[5],"hex");if(o.length===0||a.length===0)return!1;let d=await Kd(n,o,a.length,{N:s,r,p:i,maxmem:Xd});return d.length!==a.length?!1:Gd(d,a)}var Zd=Ne.object({password:Ne.string().min(1),label:Ne.string().min(1).max(256).optional(),ttlSec:Ne.number().int().positive().optional()}),ec=Ne.object({id:Ne.string().optional()}).optional();function oi(n,e){n.post("/v1/auth/login",{config:{skipAuth:!0}},async(t,s)=>{let r=tc(t);if(e.rateLimiter.isBlocked(r))return s.code(429).send({error:"Too many failed attempts; try again later."});let i;try{i=Zd.parse(t.body)}catch{return s.code(400).send({error:"Invalid request body"})}if(!await ri())return s.code(403).send({error:"No password configured. Run `hydra-acp auth password` on the daemon host."});if(!await ii(i.password))return e.rateLimiter.recordFailure(r),s.code(401).send({error:"Invalid password"});e.rateLimiter.recordSuccess(r);let a=await e.store.issue({label:i.label,ttlSec:i.ttlSec});return s.code(200).send({session_token:a.token,id:a.id,expires_at:a.expiresAt})}),n.post("/v1/auth/logout",async(t,s)=>{let r;try{r=ec.parse(t.body??void 0)}catch{return s.code(400).send({error:"Invalid request body"})}let i=r?.id??t.authIdentity;if(!i||i==="service")return s.code(200).send({revoked:!1});let o=await e.store.revoke(i);return s.code(200).send({revoked:o})}),n.get("/v1/auth/verify",async(t,s)=>s.code(200).send({ok:!0})),n.get("/v1/auth/sessions",async(t,s)=>s.code(200).send({sessions:e.store.list()})),n.delete("/v1/auth/sessions/:id",async(t,s)=>{let r=t.params.id;return await e.store.revoke(r)?s.code(204).send():s.code(404).send({error:"Not found"})})}function tc(n){return n.ip||"unknown"}import{nanoid as On}from"nanoid";function Pn(n){let e=[],t=[],s=!1,r=i=>{if(!s){s=!0;for(let o of t)o(i)}};return n.on("message",(i,o)=>{if(o)return;let a=i.toString("utf8");try{let d=JSON.parse(a);for(let c of e)c(d)}catch(d){for(let c of e)c({jsonrpc:"2.0",id:0,error:{code:I.ParseError,message:`Failed to parse WS frame: ${d.message}`}})}}),n.on("close",()=>r()),n.on("error",i=>r(i)),{async send(i){if(s)throw new Error("ws is closed");let o=JSON.stringify(i);await new Promise((a,d)=>{n.send(o,c=>{if(c){d(c);return}a()})})},onMessage(i){e.push(i)},onClose(i){t.push(i)},async close(){s||(n.close(),r())}}}import{randomBytes as di}from"crypto";function ci(n,e){n.get("/acp",{websocket:!0},async(t,s)=>{let r=Tr({headers:s.headers,url:s.url});if(!r||!await e.validator.validate(r)){t.close(4401,"Unauthorized");return}let i=e.processRegistry?.resolve(r),o=Pn(t),a=new he(o),d={clientId:`hydra_client_${On(12)}`,processIdentity:i,attached:new Map};a.onClose(()=>{for(let l of d.attached.values())e.manager.get(l.sessionId)?.detach(l.clientId);d.attached.clear()});let c=(l,u)=>{if(d.attached.get(l)?.readonly){let g=new Error(`${u} not permitted on a read-only attachment`);throw g.code=I.PermissionDenied,g}};if(a.onRequest("initialize",async l=>{let u=ms.parse(l??{});u.clientInfo?.name&&(d.clientInfo={name:u.clientInfo.name,...u.clientInfo.version!==void 0?{version:u.clientInfo.version}:{}});let p=u.clientInfo?.version;return p&&i&&(i.kind==="extension"?e.onExtensionVersion?.(i.name,p):e.onTransformerVersion?.(i.name,p)),ic()}),i&&e.extensionCommands){let l=e.extensionCommands;a.onRequest("hydra-acp/commands/register",async u=>{let p=u??{},g=Array.isArray(p.commands)?p.commands.map(m=>{if(!m||typeof m!="object")return;let y=m;if(typeof y.verb!="string")return;let v={verb:y.verb};return typeof y.argsHint=="string"&&(v.argsHint=y.argsHint),typeof y.description=="string"&&(v.description=y.description),v}).filter(m=>m!==void 0):[];return l.register(i.name,a,g),{ok:!0,registered:g.length}}),a.onClose(()=>{l.clear(i.name)})}if(i&&e.extensionMcp){let l=e.extensionMcp;a.onRequest("hydra-acp/mcp_tools/register",async u=>{let p=u??{},g=typeof p.instructions=="string"?p.instructions:void 0,m=Array.isArray(p.tools)?p.tools.map(y=>{if(!y||typeof y!="object")return;let v=y;if(typeof v.name!="string"||v.name.length===0||typeof v.description!="string"||v.inputSchema===null||typeof v.inputSchema!="object")return;let S={name:v.name,description:v.description,inputSchema:v.inputSchema};return v.outputSchema!==null&&typeof v.outputSchema=="object"&&(S.outputSchema=v.outputSchema),S}).filter(y=>y!==void 0):[];if(m.length===0)throw new Error("register_mcp_tools requires at least one tool");return l.register(i.name,a,g,m),{ok:!0,registered:m.length}}),a.onClose(()=>{l.clear(i.name)})}i?.kind==="transformer"&&(a.onRequest("hydra-acp/transformer/initialize",async l=>{let u=l??{},p=Array.isArray(u.intercepts)?u.intercepts.filter(g=>typeof g=="string"):[];if(e.transformers&&(e.transformers.registerConnection(i.name,a,p),e.manager?.defaultTransformers.includes(i.name))){let g=e.transformers.resolveChain([i.name])[0];if(g)for(let m of e.manager.liveSessions())m.addTransformer(g)}return{ack:!0}}),a.onClose(()=>{e.transformers?.deregisterConnection(i.name)}),a.onRequest("hydra-acp/message/emit",async l=>{let u=l??{},p=typeof u.sessionId=="string"?u.sessionId:void 0,g=typeof u.method=="string"?u.method:void 0,m=u.envelope,y=u.route;if(!p||!g)throw Object.assign(new Error("emit_message requires sessionId and method"),{code:-32602});let v=e.manager.get(p);if(!v)throw Object.assign(new Error(`session ${p} not found`),{code:I.SessionNotFound});let S=typeof u.respondsTo=="string"?u.respondsTo:void 0;if(S)return v.dischargeClaim(S,m),{ok:!0};if(y==="chain")return{ok:!0,response:await v.emitToChain(i.name,g,m)};if(y==="daemon")return{ok:!0,response:await v.emitToChain(i.name,g,m)};throw Object.assign(new Error(`unsupported route: ${JSON.stringify(y)}`),{code:-32602})}),a.onRequest("hydra-acp/session/request_permission",async l=>{let u=l??{},p=typeof u.sessionId=="string"?u.sessionId:void 0;if(!p)throw Object.assign(new Error("hydra-acp/session/request_permission requires sessionId"),{code:-32602});let g=e.manager.get(p);if(!g)throw Object.assign(new Error(`session ${p} not found`),{code:I.SessionNotFound});return g.requestPermissionFromClients(u)}),a.onRequest("hydra-acp/transformer/attach",async l=>sc(l,i.name,e)),a.onRequest("hydra-acp/child_session/spawn",async l=>{let u=l??{},p=typeof u.agentId=="string"?u.agentId:e.defaultAgent,g=typeof u.cwd=="string"?u.cwd:void 0,m=typeof u.parentSessionId=="string"?u.parentSessionId:void 0,y=u._meta&&typeof u._meta=="object"?u._meta:void 0,v=y&&y["hydra-acp"]&&typeof y["hydra-acp"]=="object"?y["hydra-acp"]:void 0,S=v&&typeof v.title=="string"?v.title:void 0;if(!g&&m){let E=e.manager.get(m);E&&(g=E.cwd)}if(!g)throw Object.assign(new Error("child_session/spawn requires cwd (or a parentSessionId pointing at a live session whose cwd we can inherit)"),{code:-32602});let C=typeof u.interactive=="boolean"?u.interactive:!1;return{childSessionId:(await e.manager.create({agentId:p,cwd:g,parentSessionId:m,interactive:C,transformChain:[],title:S})).sessionId}}),a.onRequest("hydra-acp/session/fork",async l=>{let u=l??{};if(typeof u.sessionId!="string")throw Object.assign(new Error("fork_session requires sessionId"),{code:I.InvalidParams});let p=typeof u.forkAt=="string"?u.forkAt:void 0,g=typeof u.cwd=="string"?u.cwd:void 0,m=typeof u.agentId=="string"?u.agentId:void 0;return await e.manager.forkSession(u.sessionId,{...p!==void 0?{forkAt:p}:{},...g!==void 0?{cwd:g}:{},...m!==void 0?{agentId:m}:{}})}),a.onRequest("hydra-acp/session/delete",async l=>{let u=l??{};if(typeof u.sessionId!="string")throw Object.assign(new Error("hydra-acp/session/delete requires sessionId"),{code:I.InvalidParams});let p=await e.manager.resolveCanonicalId(u.sessionId)??u.sessionId,g=e.manager.get(p);if(g)return await g.close({deleteRecord:!0}),{deleted:!0,sessionId:p};if(!await e.manager.deleteRecord(p))throw Object.assign(new Error(`session ${p} not found`),{code:I.SessionNotFound});return{deleted:!0,sessionId:p}}),a.onRequest("hydra-acp/child_session/await",async l=>{let u=l??{},p=typeof u.childSessionId=="string"?u.childSessionId:void 0,g=u.until==="idle"?"idle":"turn_complete",m=typeof u.timeoutMs=="number"?Math.min(u.timeoutMs,30*6e4):5*6e4;if(!p)throw Object.assign(new Error("await_child requires childSessionId"),{code:-32602});let y=e.manager.get(p);if(!y)throw Object.assign(new Error(`child session ${p} not found`),{code:I.SessionNotFound});return new Promise(v=>{let S=[],C,k=()=>{clearTimeout(E),C?.(),v({entries:S})};C=y.onBroadcast(N=>{S.push(N),g==="turn_complete"&&N.params?.update?.sessionUpdate==="turn_complete"&&k()});let E=setTimeout(k,m);typeof E.unref=="function"&&E.unref(),y.onClose(()=>k())})}),a.onRequest("hydra-acp/child_session/close",async l=>{let u=l??{},p=typeof u.childSessionId=="string"?u.childSessionId:void 0;if(!p)throw Object.assign(new Error("close_child_session requires childSessionId"),{code:-32602});let g=e.manager.get(p);return g&&await g.close({deleteRecord:!1}),{ok:!0}}),a.onRequest("hydra-acp/connection/keep_alive",async l=>{let u=l??{},p=typeof u.token=="string"?u.token:void 0,g=typeof u.sessionId=="string"?u.sessionId:void 0,m=typeof u.estimatedRemainingMs=="number"?u.estimatedRemainingMs:void 0;return p&&g&&e.manager.get(g)?.keepAliveClaim(p,m),{ok:!0}})),a.onRequest("hydra-acp/session/tool_content",async l=>{let u=l??{};if(typeof u.sessionId!="string"||typeof u.hash!="string")throw Object.assign(new Error("hydra-acp/session/tool_content requires sessionId and hash"),{code:I.InvalidParams});let p=await e.manager.resolveCanonicalId(u.sessionId)??u.sessionId,g=await e.manager.loadToolBlob(p,u.hash);if(g===null)throw Object.assign(new Error("tool content not found"),{code:I.SessionNotFound});return{content:g}}),a.onRequest("session/new",async l=>{let u=gs.parse(l),p=De(l?._meta),g=Array.isArray(p.transformers)&&p.transformers.every(M=>typeof M=="string")?p.transformers:e.manager.defaultTransformers??[],m=e.transformers?.resolveChain(g)??[],y,v,S=u.mcpServers;if(p.mcpStdin===!0&&e.mcpTokenRegistry!==void 0&&e.getDaemonOrigin!==void 0){y=di(32).toString("hex"),v=e.mcpTokenRegistry.reserve(y);let P={name:"hydra-acp-stdin",type:"http",url:`${e.getDaemonOrigin()}/mcp/hydra-acp-stdin`,headers:[{name:"Authorization",value:`Bearer ${y}`}]};S=[...u.mcpServers??[],P]}let C=Bt(e);C!==void 0&&(S=[...S??[],...C.descriptors]);let k;try{k=await e.manager.create({cwd:u.cwd,agentId:p.agentId??e.defaultAgent,mcpServers:S,title:p.title,agentArgs:p.agentArgs,model:p.model,onInstallProgress:ai(a),transformChain:m,originatingClient:d.clientInfo,...p.interactive!==void 0?{interactive:p.interactive}:{}})}catch(M){throw v!==void 0&&v.abandon(M instanceof Error?M:void 0),C!==void 0&&C.abandon(M instanceof Error?M:void 0),M}if(y!==void 0&&v!==void 0&&e.mcpTokenRegistry!==void 0){let M=y,P=e.mcpTokenRegistry;v.complete(k),k.onClose(()=>{P.unbind(M)})}C!==void 0&&C.bindToSession(k);let E=Ht(a,k,d),{entries:N}=await k.attach(E,"full");d.attached.set(k.sessionId,{sessionId:k.sessionId,clientId:E.clientId,readonly:!1}),setImmediate(()=>{(async()=>{for(let M of N)await a.notify(M.method,M.params).catch(()=>{})})()});let V=Tn(k),W=_n(k);return{sessionId:k.sessionId,...V?{modes:V}:{},...W?{models:W}:{},configOptions:k.buildConfigOptions(),_meta:$n(e.manager,k,{clientId:E.clientId})}}),a.onRequest("session/attach",async l=>{let u=hs.parse(l),p=u.clientInfo?.version;p&&i&&(i.kind==="extension"?e.onExtensionVersion?.(i.name,p):e.onTransformerVersion?.(i.name,p));let g=De(u._meta),m=g.resume,y=g.readonly===!0;n.log.info(`session/attach sessionId=${u.sessionId} hasResumeHints=${!!m} readonly=${y}`);let v=m?u.sessionId:await e.manager.resolveCanonicalId(u.sessionId)??u.sessionId,S=e.manager.get(v);if(!S&&y){let M=await e.manager.loadFromDisk(v);if(!M){let F=new Error(`session ${u.sessionId} not found`);throw F.code=I.SessionNotFound,F}let P=await e.manager.loadHistory(v),B=u.clientId??`cli_${On(8)}`;d.attached.set(M.hydraSessionId,{sessionId:M.hydraSessionId,clientId:B,readonly:!0}),n.log.info(`session/attach OK (viewer) sessionId=${M.hydraSessionId} clientId=${B} attachedCount=${d.attached.size} replayed=${P.length}`);for(let F of P)await a.notify(F.method,F.params).catch(()=>{});return{sessionId:M.hydraSessionId,clientId:B,connectedClients:[B],historyPolicy:"full",replayed:P.length,_meta:rc(M)}}if(!S){let M=await e.manager.loadFromDisk(v),P=M;if(m&&(P={...M,hydraSessionId:u.sessionId,upstreamSessionId:m.upstreamSessionId,agentId:m.agentId,cwd:m.cwd,...m.title!==void 0?{title:m.title}:{},...m.agentArgs!==void 0?{agentArgs:m.agentArgs}:{}}),!P){let U=new Error(`session ${u.sessionId} not found and no resume hints provided`);throw U.code=I.SessionNotFound,U}let B=P.originatingClient?P:{...P,originatingClient:d.clientInfo},F=Bt(e);try{S=await e.manager.resurrect({...B,mcpServers:F?.descriptors,onInstallProgress:ai(a)})}catch(U){throw F!==void 0&&F.abandon(U instanceof Error?U:void 0),U}F!==void 0&&F.bindToSession(S),Fn(S,e)}let C=Ht(a,S,d,u.clientInfo,u.clientId),k=g.replayMode==="drip",{entries:E,appliedPolicy:N}=await S.attach(C,u.historyPolicy,{afterMessageId:u.afterMessageId,raw:k,...g.toolContent!==void 0?{toolContent:g.toolContent}:{}});if(d.attached.set(S.sessionId,{sessionId:S.sessionId,clientId:C.clientId,readonly:y}),n.log.info(`session/attach OK sessionId=${S.sessionId} clientId=${C.clientId} attachedCount=${d.attached.size} requestedPolicy=${u.historyPolicy} appliedPolicy=${N} replayed=${E.length} readonly=${y}${k?" replayMode=drip":""}`),k){let M=g.dripSpeed&&g.dripSpeed>0?g.dripSpeed:1,P=750;(async()=>{let B=null;for(let F of E){let U=typeof F.recordedAt=="number"?F.recordedAt:null;if(B!==null&&U!==null){let D=Math.min(P,Math.max(0,(U-B)/M));D>0&&await new Promise(je=>setTimeout(je,D))}U!==null&&(B=U);try{await a.notify(F.method,F.params)}catch{return}}})()}else for(let P=0;P<E.length;P++){let B=E[P],F=a.notify(B.method,B.params).catch(()=>{});(P+1)%200===0&&await F}S.replayPendingPermissions(C);let V=Tn(S),W=_n(S);return{sessionId:S.sessionId,clientId:C.clientId,connectedClients:S.connectedClients(C.clientId),historyPolicy:N,replayed:E.length,...V?{modes:V}:{},...W?{models:W}:{},configOptions:S.buildConfigOptions(),_meta:$n(e.manager,S)}}),a.onRequest("session/detach",async l=>{let u=ys.parse(l),p=d.attached.get(u.sessionId);if(!p){let m=new Error("client not attached to that session");throw m.code=I.SessionNotFound,m}let g=e.manager.get(u.sessionId);return g?.detach(p.clientId),d.attached.delete(u.sessionId),g&&e.manager.reapIfOrphanedNonInteractive(u.sessionId),{sessionId:u.sessionId,_meta:{[Ue]:{detachStatus:"detached"}}}}),a.onRequest("session/list",async l=>{let u=ws.parse(l??{});return{sessions:(await e.manager.list({cwd:u.cwd})).filter(y=>y.originatingClient?.name!==wt).map(bs)}}),a.onRequest("hydra-acp/agents/list",async()=>{if(!e.registry){let l=new Error("agent registry unavailable");throw l.code=I.InternalError,l}return Ce(e.registry)}),a.onRequest("session/prompt",async l=>{let u=Is.parse(l);c(u.sessionId,"session/prompt");let p=d.attached.get(u.sessionId);if(!p){n.log.warn(`session/prompt rejected: not attached sessionId=${u.sessionId} attachedKeys=[${[...d.attached.keys()].join(",")}]`);let m=new Error("not attached to session");throw m.code=I.SessionNotFound,m}let g=e.manager.get(u.sessionId);if(!g){let m=await e.manager.loadFromDisk(u.sessionId);if(!m){let S=new Error(`session ${u.sessionId} not found`);throw S.code=I.SessionNotFound,S}n.log.info(`session/prompt auto-resurrecting cold sessionId=${u.sessionId}`);let y=Bt(e);try{g=await e.manager.resurrect({...m,mcpServers:y?.descriptors})}catch(S){throw y!==void 0&&y.abandon(S instanceof Error?S:void 0),S}y!==void 0&&y.bindToSession(g),Fn(g,e);let v=Ht(a,g,d,void 0,p.clientId);await g.attach(v,"none")}return g.prompt(p.clientId,u)});let f=l=>{let u;try{u=lt.parse(l)}catch(m){n.log.warn(`session/cancel: invalid params: ${m.message}`);return}let p=d.attached.get(u.sessionId);if(!p)return;if(p.readonly){n.log.warn(`session/cancel dropped (readonly attachment) sessionId=${u.sessionId}`);return}let g=e.manager.get(u.sessionId);g&&g.cancel(p.clientId).catch(m=>{n.log.warn(`session/cancel for ${u.sessionId}: ${m.message}`)})};a.onNotification("session/cancel",f),a.onRequest("session/cancel",async l=>{let u=lt.parse(l);return c(u.sessionId,"session/cancel"),f(l),null}),a.onRequest("hydra-acp/prompt/cancel",async l=>{let u=ks.parse(l);c(u.sessionId,"hydra-acp/prompt/cancel");let p=e.manager.get(u.sessionId);if(!p){let g=new Error(`session ${u.sessionId} not found`);throw g.code=I.SessionNotFound,g}return p.cancelQueuedPrompt(u.messageId)}),a.onRequest("hydra-acp/session/force_cancel",async l=>{let u=lt.parse(l);c(u.sessionId,"hydra-acp/session/force_cancel");let p=e.manager.get(u.sessionId);if(!p){let g=new Error(`session ${u.sessionId} not found`);throw g.code=I.SessionNotFound,g}return p.forceCancel()}),a.onRequest("hydra-acp/prompt/update",async l=>{let u=As.parse(l);c(u.sessionId,"hydra-acp/prompt/update");let p=e.manager.get(u.sessionId);if(!p){let g=new Error(`session ${u.sessionId} not found`);throw g.code=I.SessionNotFound,g}return p.updateQueuedPrompt(u.messageId,u.prompt)}),a.onRequest("hydra-acp/prompt/amend",async l=>{let u=xs.parse(l);c(u.sessionId,"hydra-acp/prompt/amend");let p=d.attached.get(u.sessionId);if(!p){let m=new Error("not attached to session");throw m.code=I.SessionNotFound,m}let g=e.manager.get(u.sessionId);if(!g){let m=new Error(`session ${u.sessionId} not found`);throw m.code=I.SessionNotFound,m}return g.amendPrompt(p.clientId,u)}),a.onRequest("session/load",async l=>{let u=l??{},p=typeof u.sessionId=="string"?u.sessionId:void 0;if(!p){let k=new Error("session/load requires sessionId");throw k.code=I.InvalidParams,k}let g=await e.manager.resolveCanonicalId(p)??p,m=e.manager.get(g);if(!m){let k=await e.manager.loadFromDisk(g);if(!k){let N=new Error(`session ${p} not found in memory or on disk`);throw N.code=I.SessionNotFound,N}let E=Bt(e);try{m=await e.manager.resurrect({...k,mcpServers:E?.descriptors})}catch(N){throw E!==void 0&&E.abandon(N instanceof Error?N:void 0),N}E!==void 0&&E.bindToSession(m),Fn(m,e)}let y=Ht(a,m,d),{entries:v}=await m.attach(y,"pending_only");d.attached.set(m.sessionId,{sessionId:m.sessionId,clientId:y.clientId,readonly:!1});for(let k of v)await a.notify(k.method,k.params);m.replayPendingPermissions(y);let S=Tn(m),C=_n(m);return{sessionId:m.sessionId,...S?{modes:S}:{},...C?{models:C}:{},configOptions:m.buildConfigOptions(),_meta:$n(e.manager,m,{clientId:y.clientId})}}),a.onRequest("session/set_model",async l=>{let u=l?.sessionId;typeof u=="string"&&c(u,"session/set_model");let p=nc(l,e.manager);if(p.kind==="error"){n.log.warn(p.logMessage);let y=new Error(p.message);throw y.code=p.code,y}if(p.kind==="no_op")return n.log.warn(p.logMessage),await a.notify("session/update",{sessionId:p.sessionId,update:{sessionUpdate:"current_model_update",currentModel:p.currentModel}}).catch(()=>{}),null;n.log.info(p.logMessage);let{modelId:g}=p,m=await p.session.forwardRequest("session/set_model",{...l,modelId:g});return p.session.applyModelChange(g),m}),a.onRequest("session/set_mode",async l=>{let u=l,p=u?.sessionId;if(typeof p=="string"&&c(p,"session/set_mode"),!u||typeof u.sessionId!="string"){let y=new Error("session/set_mode requires string sessionId");throw y.code=I.InvalidParams,y}if(typeof u.modeId!="string"){let y=new Error("session/set_mode requires string modeId");throw y.code=I.InvalidParams,y}let g=e.manager.get(u.sessionId);if(!g){let y=new Error(`session ${u.sessionId} not found`);throw y.code=I.SessionNotFound,y}let m=await g.forwardRequest("session/set_mode",l);return g.applyModeChange(u.modeId),m}),a.onRequest("session/set_config_option",async l=>{let u=l,p=v=>{let S=new Error(v);return S.code=I.InvalidParams,S},g=u?.sessionId;if(typeof g=="string"&&c(g,"session/set_config_option"),!u||typeof u.sessionId!="string")throw p("session/set_config_option requires string sessionId");if(typeof u.configId!="string")throw p("session/set_config_option requires string configId");if(typeof u.value!="string")throw p("session/set_config_option requires string value");let m=e.manager.get(u.sessionId);if(!m){let v=new Error(`session ${u.sessionId} not found`);throw v.code=I.SessionNotFound,v}let y=m.buildConfigOptions().find(v=>v.id===u.configId);if(!y)throw p(`unknown configId ${JSON.stringify(u.configId)} for this session`);if(!y.options.some(v=>v.value===u.value))throw p(`value ${JSON.stringify(u.value)} is not valid for configId ${JSON.stringify(u.configId)}`);switch(u.configId){case"model":{u.value!==m.currentModel&&await m.forwardRequest("session/set_model",{sessionId:u.sessionId,modelId:u.value}),m.applyModelChange(u.value);break}case"mode":{u.value!==m.currentMode&&await m.forwardRequest("session/set_mode",{sessionId:u.sessionId,modeId:u.value}),m.applyModeChange(u.value);break}case"agent":{u.value!==m.agentId&&await m.setAgent(u.value);break}default:throw p(`configId ${JSON.stringify(u.configId)} is not settable`)}return{configOptions:m.buildConfigOptions()}}),a.setDefaultHandler(async(l,u)=>{if(!u.startsWith("session/")||l===null||typeof l!="object"){let m=new Error(`Method not found: ${u}`);throw m.code=I.MethodNotFound,m}let p=l.sessionId;if(typeof p!="string"){let m=new Error(`Method not found: ${u}`);throw m.code=I.MethodNotFound,m}c(p,u);let g=e.manager.get(p);if(!g){let m=new Error(`session ${p} not found`);throw m.code=I.SessionNotFound,m}return g.forwardRequest(u,l)})})}function ai(n){return e=>{let t={agentId:e.agentId,version:e.version,source:e.source,phase:e.phase};"receivedBytes"in e&&(t.receivedBytes=e.receivedBytes),"totalBytes"in e&&(t.totalBytes=e.totalBytes),"packageSpec"in e&&(t.packageSpec=e.packageSpec),n.notify(Cs,t).catch(()=>{})}}function Tn(n){let e=n.availableModes();if(e.length===0)return;let t=e.map(r=>{let i={id:r.id,name:r.name??r.id};return r.description!==void 0&&(i.description=r.description),i});return{currentModeId:n.currentMode??e[0].id,availableModes:t}}function _n(n){let e=n.availableModels();if(e.length===0)return;let t=e.map(r=>{let i={modelId:r.modelId};return r.name!==void 0&&(i.name=r.name),r.description!==void 0&&(i.description=r.description),i});return{currentModelId:n.currentModel??e[0].modelId,availableModels:t}}function nc(n,e){if(!n||typeof n!="object")return{kind:"error",code:I.InvalidParams,message:"session/set_model requires params",logMessage:"session/set_model rejected: params not an object"};let t=n;if(typeof t.sessionId!="string")return{kind:"error",code:I.InvalidParams,message:"session/set_model requires string sessionId",logMessage:"session/set_model rejected: missing/non-string sessionId"};if(typeof t.modelId!="string")return{kind:"error",code:I.InvalidParams,message:"session/set_model requires string modelId",logMessage:`session/set_model rejected: missing/non-string modelId sessionId=${t.sessionId}`};let s=e.get(t.sessionId);if(!s)return{kind:"error",code:I.SessionNotFound,message:`session ${t.sessionId} not found`,logMessage:`session/set_model rejected: session not found sessionId=${t.sessionId}`};let r=s.availableModels(),i=Re(t.modelId,r);if(i.kind==="none")return{kind:"ok",session:s,modelId:t.modelId,logMessage:`session/set_model passthrough (no availableModels) sessionId=${t.sessionId} modelId=${JSON.stringify(t.modelId)}`};if(i.kind==="exact")return{kind:"ok",session:s,modelId:t.modelId,logMessage:`session/set_model accepted sessionId=${t.sessionId} modelId=${JSON.stringify(t.modelId)}`};if(i.kind==="resolved")return{kind:"ok",session:s,modelId:i.modelId,logMessage:`session/set_model resolved sessionId=${t.sessionId} requested=${JSON.stringify(t.modelId)} \u2192 ${JSON.stringify(i.modelId)}`};let o=r.map(d=>d.modelId).join(", "),a=i.kind==="ambiguous"?`ambiguous (trailing-segment matches [${i.candidates.join(", ")}])`:"not in availableModels";return s.currentModel!==void 0&&s.currentModel.length>0?{kind:"no_op",session:s,sessionId:t.sessionId,currentModel:s.currentModel,logMessage:`session/set_model no_op (resyncing client) sessionId=${t.sessionId} requested=${JSON.stringify(t.modelId)} ${a} actual=${JSON.stringify(s.currentModel)} agentId=${s.agentId} known=[${o}]`}:{kind:"error",code:I.InvalidParams,message:`model "${t.modelId}" is ${a==="not in availableModels"?"not in this session's availableModels":a} (agent ${s.agentId}); known models: ${o}`,logMessage:`session/set_model rejected sessionId=${t.sessionId} modelId=${JSON.stringify(t.modelId)} ${a} agentId=${s.agentId} known=[${o}] (no current model to fall back to)`}}async function sc(n,e,t){let s=n??{},r=typeof s.sessionId=="string"?s.sessionId:void 0;if(!r)throw Object.assign(new Error("transformer/attach requires sessionId"),{code:I.InvalidParams});if(!t.transformers)throw Object.assign(new Error("transformer manager not configured"),{code:I.InternalError});let i=t.transformers.resolveChain([e])[0];if(!i)throw Object.assign(new Error(`transformer ${e} is not connected (call hydra-acp/transformer/initialize first)`),{code:I.InternalError});let o=t.manager.get(r);if(!o)throw Object.assign(new Error(`session ${r} not found`),{code:I.SessionNotFound});return o.addTransformer(i),{ok:!0}}function rc(n){let e={sessionId:n.hydraSessionId,upstreamSessionId:n.upstreamSessionId,cwd:n.cwd,title:n.title,agentId:n.agentId,currentModel:n.currentModel,currentUsage:n.currentUsage,forkedFromSessionId:n.forkedFromSessionId,forkedFromMessageId:n.forkedFromMessageId,originatingClient:n.originatingClient,interactive:n.interactive,updatedAt:n.createdAt??new Date().toISOString(),attachedClients:0,status:"cold",busy:!1,awaitingInput:!1},t={currentMode:n.currentMode,agentArgs:n.agentArgs,availableCommands:n.agentCommands,availableModes:n.agentModes,availableModels:n.agentModels};return{[Ue]:ut(e,t)}}function $n(n,e,t={}){let s=n.liveListEntry(e),r={clientId:t.clientId,currentMode:e.currentMode,agentArgs:e.agentArgs,availableCommands:e.mergedAvailableCommands(),availableModes:e.availableModes(),availableModels:e.availableModels(),turnStartedAt:e.turnStartedAt,agentCapabilities:e.agentCapabilities,queue:e.queueSnapshot()};return ct(e.agentMeta,ut(s,r))}function ic(){return{protocolVersion:le,agentInfo:{name:"hydra",version:Q},agentCapabilities:{promptCapabilities:{image:!0,audio:!0,embeddedContext:!0},mcpCapabilities:{http:!0,sse:!0},loadSession:!0,sessionCapabilities:{attach:{},list:{}}},authMethods:[{id:"bearer-token",description:"Bearer token presented at WS upgrade"}],_meta:ct(void 0,{prompt:{queueing:!0,cancelling:!0,updating:!0,amending:!0,pipelining:!1},agents:{list:!0,installProgress:!0}})}}function Fn(n,e){if(!(!e.transformers||!e.manager))for(let t of e.manager.defaultTransformers){let s=e.transformers.resolveChain([t])[0];s&&n.addTransformer(s)}}function Ht(n,e,t,s,r){return{clientId:r??`cli_${On(8)}`,connection:n,clientInfo:s}}function Bt(n){if(n.extensionMcp===void 0||n.mcpTokenRegistry===void 0||n.getDaemonOrigin===void 0)return;let e=n.extensionMcp.list();if(e.length===0)return;let t=di(32).toString("hex"),s=n.mcpTokenRegistry.reserve(t),r=n.getDaemonOrigin(),i=e.map(a=>({name:a,type:"http",url:`${r}/mcp/${a}`,headers:[{name:"Authorization",value:`Bearer ${t}`}]})),o=n.mcpTokenRegistry;return{descriptors:i,bindToSession:a=>{s.complete(a),a.onClose(()=>{o.unbind(t)})},abandon:a=>s.abandon(a)}}var Ut=class{byToken=new Map;reserve(e){if(this.byToken.has(e))throw new Error("mcp token already bound");let t,s,r=new Promise((o,a)=>{t=o,s=a});r.catch(()=>{});let i={session:void 0,sessionReady:r,disposers:[]};return this.byToken.set(e,i),{complete:o=>{i.session=o,t(o)},abandon:o=>{this.byToken.delete(e),s(o??new Error("mcp token reservation abandoned"))}}}bind(e,t){let{complete:s}=this.reserve(e);s(t)}lookup(e){return this.byToken.get(e)}addDisposer(e,t){let s=this.byToken.get(e);s!==void 0&&s.disposers.push(t)}async unbind(e){let t=this.byToken.get(e);if(t!==void 0){this.byToken.delete(e);for(let s of t.disposers)try{await s()}catch{}}}size(){return this.byToken.size}};import{randomUUID as oc}from"crypto";import{McpServer as ac}from"@modelcontextprotocol/sdk/server/mcp.js";import{StreamableHTTPServerTransport as dc}from"@modelcontextprotocol/sdk/server/streamableHttp.js";import{z as q}from"zod";var ui="Bearer ";function Dt(n){let e=n.headers.authorization;if(typeof e!="string"||!e.startsWith(ui))return;let t=e.slice(ui.length).trim();return t.length>0?t:void 0}function cc(n){let e=new ac({name:"hydra-acp-stdin",version:"1.0.0"},{instructions:"Piped input from `hydra cat --stream` is exposed here as a byte stream. Use `tail` for the latest N bytes (good for finding the end of a log), `head` for the first N bytes (good for headers/preamble), `read` for windowed reads against an absolute byte cursor, `wait_for_more` to block until new bytes arrive past a cursor, and `info` for the current cursors/capacity/closed status. Byte payloads come back base64-encoded."});return e.registerTool("tail",{description:"Return the most recent `bytes` bytes of piped stdin (capped server-side, default 64 KiB max). `truncated:true` means older bytes existed but have been evicted from the ring.",inputSchema:{bytes:q.number().int().min(1).describe("How many trailing bytes to return.")}},async({bytes:t})=>{let s=n.streamTail(t);return{content:[{type:"text",text:JSON.stringify(s)}],structuredContent:s}}),e.registerTool("head",{description:"Return the first `bytes` bytes of piped stdin (capped server-side, default 64 KiB max). `truncated:true` means the head has already been evicted from the ring and the returned bytes start at the oldest still-resident cursor.",inputSchema:{bytes:q.number().int().min(1).describe("How many leading bytes to return.")}},async({bytes:t})=>{let s=n.streamHead(t);return{content:[{type:"text",text:JSON.stringify(s)}],structuredContent:s}}),e.registerTool("read",{description:"Read up to `max_bytes` bytes starting at absolute byte `cursor`. Returns `{bytes, nextCursor, gap?, eof?}` \u2014 `gap` is the number of bytes silently skipped because the ring had evicted them; `eof:true` means the producer closed and there is nothing left to read.",inputSchema:{cursor:q.number().int().min(0).describe("Absolute byte offset to start reading from. Use 0 to read from the very beginning (may produce a gap if old bytes have been evicted)."),max_bytes:q.number().int().min(1).optional().describe("Optional cap on how many bytes to return. Server caps at 64 KiB regardless."),wait_ms:q.number().int().min(0).optional().describe("If no bytes are available, block up to this many ms for more (capped server-side at 60_000).")}},async({cursor:t,max_bytes:s,wait_ms:r})=>{let i=await n.streamRead(t,s,r);return{content:[{type:"text",text:JSON.stringify(i)}],structuredContent:i}}),e.registerTool("wait_for_more",{description:"Block until bytes are available past `cursor`, the stream closes, or `timeout_ms` elapses. Returns one of {data, eof, timeout} plus the current `writeCursor`. Use this when you've consumed everything up to a cursor and want to wait for more without busy-polling.",inputSchema:{cursor:q.number().int().min(0).describe("The cursor you've already consumed up to."),timeout_ms:q.number().int().min(0).describe("Maximum ms to block (server caps at 60_000).")}},async({cursor:t,timeout_ms:s})=>{let r=await n.streamWaitFor(t,s),i=n.streamInfo(),o={outcome:r,writeCursor:i.writeCursor,closed:i.closed};return{content:[{type:"text",text:JSON.stringify(o)}],structuredContent:o}}),e.registerTool("grep",{description:"Scan piped stdin line-by-line and return lines matching `pattern`. Prefer this over `read` when the question is 'find lines that mention X' \u2014 it filters server-side so you don't pull and decode 64 KiB base64 windows. Returns `{matches: [{cursor, line, before?, after?}], truncated, nextCursor, gap?, scannedBytes, eof?}`. Lines come back as decoded UTF-8 strings (not base64). When `truncated:true`, re-call with `cursor: nextCursor` to resume.",inputSchema:{pattern:q.string().min(1).describe("Search pattern. Treated as a JavaScript regular expression by default (set `regex:false` for a literal substring match)."),regex:q.boolean().optional().describe("Default true. Pass false to treat `pattern` as a literal substring."),case_insensitive:q.boolean().optional().describe("Default false. Pass true for case-insensitive matching."),invert:q.boolean().optional().describe("Default false. Pass true to return lines that do NOT match the pattern."),max_matches:q.number().int().min(1).optional().describe("Default 100. Capped server-side at 1000."),max_bytes:q.number().int().min(1).optional().describe("Default 64 KiB output. Capped server-side at 256 KiB."),context_before:q.number().int().min(0).optional().describe("Default 0. Number of lines before each match to include (capped at 20)."),context_after:q.number().int().min(0).optional().describe("Default 0. Number of lines after each match to include (capped at 20)."),cursor:q.number().int().min(0).optional().describe("Optional absolute byte offset to start scanning from. Omit to scan from the oldest still-resident byte. Pass the `nextCursor` from a previous truncated call to resume.")}},async t=>{let s={pattern:t.pattern};t.regex!==void 0&&(s.regex=t.regex),t.case_insensitive!==void 0&&(s.caseInsensitive=t.case_insensitive),t.invert!==void 0&&(s.invert=t.invert),t.max_matches!==void 0&&(s.maxMatches=t.max_matches),t.max_bytes!==void 0&&(s.maxBytes=t.max_bytes),t.context_before!==void 0&&(s.contextBefore=t.context_before),t.context_after!==void 0&&(s.contextAfter=t.context_after),t.cursor!==void 0&&(s.cursor=t.cursor);let r=n.streamGrep(s),i=r;return{content:[{type:"text",text:JSON.stringify(r)}],structuredContent:i}}),e.registerTool("info",{description:"Report cursor / capacity / closed state of the stdin ring. Cheap; safe to call repeatedly.",inputSchema:{}},async()=>{let t=n.streamInfo();return{content:[{type:"text",text:JSON.stringify(t)}],structuredContent:t}}),e}var uc=1e4;function li(n,e){let t=new Map;async function s(o,a){let d=t.get(o);if(d!==void 0)return d.transport;let c=cc(a),f=new dc({sessionIdGenerator:()=>oc()});await c.connect(f);let l={server:c,transport:f};return t.set(o,l),e.addDisposer(o,async()=>{t.delete(o);try{await f.close()}catch{}try{await c.close()}catch{}}),f}async function r(o,a){let d=Dt(o);if(d===void 0){a.code(401).send({error:"missing bearer token"});return}let c=e.lookup(d);if(c===void 0){a.code(404).send({error:"unknown stdin token"});return}let f;if(c.session!==void 0)f=c.session;else{let u,p=new Promise(m=>{u=setTimeout(()=>m(void 0),uc)}),g=await Promise.race([c.sessionReady.catch(()=>{}),p]);if(u!==void 0&&clearTimeout(u),g===void 0){a.code(503).send({error:"session not ready"});return}f=g}let l=await s(d,f);a.hijack(),await l.handleRequest(o.raw,a.raw,o.body)}let i={config:{skipAuth:!0}};n.post("/mcp/hydra-acp-stdin",i,async(o,a)=>{await r(o,a)}),n.get("/mcp/hydra-acp-stdin",i,async(o,a)=>{await r(o,a)}),n.delete("/mcp/hydra-acp-stdin",i,async(o,a)=>{await r(o,a)})}var Lt=class{byName=new Map;changeHandlers=[];register(e,t,s,r){this.byName.set(e,{connection:t,instructions:s,tools:[...r]}),this.fireChanged(e,"register")}clear(e){this.byName.delete(e)&&this.fireChanged(e,"clear")}lookup(e){return this.byName.get(e)}list(){return Array.from(this.byName.keys())}onChange(e){return this.changeHandlers.push(e),()=>{let t=this.changeHandlers.indexOf(e);t>=0&&this.changeHandlers.splice(t,1)}}fireChanged(e,t){for(let s of this.changeHandlers)try{s(e,t)}catch{}}};import{StreamableHTTPServerTransport as yc}from"@modelcontextprotocol/sdk/server/streamableHttp.js";import{randomUUID as wc}from"crypto";import{Server as lc}from"@modelcontextprotocol/sdk/server/index.js";import{CallToolRequestSchema as fc,ListToolsRequestSchema as pc}from"@modelcontextprotocol/sdk/types.js";var mc=6e4;function fi(n,e,t,s={}){let r=s.invokeTimeoutMs??mc,i=typeof t=="function"?async()=>t():async()=>t,o=new lc({name:n,version:"1.0.0"},{capabilities:{tools:{listChanged:!1}},...e.instructions!==void 0?{instructions:e.instructions}:{}}),a=new Map(e.tools.map(d=>[d.name,d]));return o.setRequestHandler(pc,async()=>({tools:e.tools.map(d=>({name:d.name,description:d.description,inputSchema:d.inputSchema,...d.outputSchema!==void 0?{outputSchema:d.outputSchema}:{}}))})),o.setRequestHandler(fc,async d=>{let c=d.params.name;if(!a.has(c))return qt(`unknown tool: ${c}`);try{let f=await i(),l=await gc(e.connection,n,c,d.params.arguments??{},f,r);return hc(l,c)}catch(f){return qt(f instanceof Error?f.message:String(f))}}),o}async function gc(n,e,t,s,r,i){let o,a=new Promise((d,c)=>{o=setTimeout(()=>c(new Error(`extension timeout after ${i}ms`)),i)});try{return await Promise.race([n.request("hydra-acp/mcp_tools/invoke",{server:e,tool:t,args:s,sessionId:r}),a])}finally{o!==void 0&&clearTimeout(o)}}function hc(n,e){if(n===null||typeof n!="object")return qt(`extension ${e} returned non-object`);let t=n;return Array.isArray(t.content)?t:qt(`extension ${e} omitted content array`)}function qt(n){return{content:[{type:"text",text:n}],isError:!0}}var vc=1e4;function pi(n,e,t,s={}){let r=new Map;async function i(f){try{await f.transport.close()}catch{}try{await f.server.close()}catch{}}function o(f){for(let l of r.values()){let u=l.get(f);u!==void 0&&(l.delete(f),i(u))}}t.onChange(f=>{o(f)});async function a(f,l){let u=r.get(f);u===void 0&&(u=new Map,r.set(f,u),e.addDisposer(f,async()=>{let S=r.get(f);if(S!==void 0){r.delete(f);for(let C of S.values())await i(C)}}));let p=u.get(l);if(p!==void 0)return p.transport;let g=t.lookup(l);if(g===void 0)return;let y=fi(l,g,async()=>{let S=e.lookup(f);if(S===void 0)throw new Error("mcp token no longer bound");if(S.session!==void 0)return S.session.sessionId;let C,k=new Promise(N=>{C=setTimeout(()=>N(void 0),vc)}),E=await Promise.race([S.sessionReady.catch(()=>{}),k]);if(C!==void 0&&clearTimeout(C),E===void 0)throw new Error("session not ready");return E.sessionId},s.buildOptions),v=new yc({sessionIdGenerator:()=>wc()});return await y.connect(v),u.set(l,{server:y,transport:v}),v}async function d(f,l){let u=Dt(f);if(u===void 0){l.code(401).send({error:"missing bearer token"});return}if(e.lookup(u)===void 0){l.code(404).send({error:"unknown mcp token"});return}let g=f.params.name,m=await a(u,g);if(m===void 0){l.code(404).send({error:`unknown mcp server: ${g}`});return}l.hijack(),await m.handleRequest(f.raw,l.raw,f.body)}let c={config:{skipAuth:!0}};n.post("/mcp/:name",c,async(f,l)=>{await d(f,l)}),n.get("/mcp/:name",c,async(f,l)=>{await d(f,l)}),n.delete("/mcp/:name",c,async(f,l)=>{await d(f,l)})}async function kc(n,e){xc(n);let t=n.daemon.tls?{key:await pe.readFile(n.daemon.tls.key),cert:await pe.readFile(n.daemon.tls.cert)}:void 0;await pe.mkdir(w.home(),{recursive:!0});let{stream:s,fileStream:r}=await Ac(n.daemon.logLevel),i=bc({logger:{level:n.daemon.logLevel,stream:s},https:t??null,bodyLimit:256*1024*1024});await i.register(Ic,{options:{handleProtocols:Hn}}),Xt($=>{i.log.info($)}),tn($=>{i.log.info($)});let o=await Pt.load(),a=new Ot,d=new Ft,c=new $t([new Tt(e),new _t(o),d]),f=Pr({validator:c});i.addHook("onRequest",async($,me)=>{$.routeOptions.config?.skipAuth||$.url==="/acp"||$.url?.startsWith("/acp?")||await f($,me)});let l=setInterval(()=>{o.sweepExpired()},300*1e3);l.unref();let u=new Be(n,{onFetched:()=>{br(u,y)}});Cn($=>i.log.info($));let p={info:$=>i.log.info($),warn:$=>i.log.warn($)},g=$=>ae.spawn({...$,stderrTailBytes:n.daemon.agentStderrTailBytes,logger:p});Zn(n.compressToolContent);let m=new Rt,y=new Ge(u,g,void 0,{idleTimeoutMs:n.daemon.sessionIdleTimeoutSeconds*1e3,defaultModels:n.defaultModels,synopsisAgent:n.synopsisAgent,synopsisModel:n.synopsisModel,synopsisOnClose:n.synopsisOnClose,defaultTransformers:n.defaultTransformers,sessionHistoryMaxEntries:n.daemon.sessionHistoryMaxEntries,logger:p,npmRegistry:n.npmRegistry,extensionCommands:m,defaultCwd:n.defaultCwd}),v=new Ct(Wn(n),void 0,{tokenRegistry:d}),S=new Mt(Qn(n),void 0,{tokenRegistry:d});Xr(i,Q,kr(n)),Yr(i,y,{agentId:n.defaultAgent,cwd:n.defaultCwd,publicHost:n.daemon.publicHost,host:n.daemon.host,port:n.daemon.port}),Kr(i,u,y,{npmRegistry:n.npmRegistry}),Zr(i,v),ei(i,S),ti(i,{defaultAgent:n.defaultAgent,defaultCwd:n.defaultCwd,defaultModels:{...n.defaultModels},...n.synopsisAgent!==void 0?{synopsisAgent:n.synopsisAgent}:{},...n.synopsisModel!==void 0?{synopsisModel:n.synopsisModel}:{},synopsisOnClose:n.synopsisOnClose,defaultTransformers:[...n.defaultTransformers]}),oi(i,{store:o,rateLimiter:a});let C=new Ut,k=new Lt;li(i,C),pi(i,C,k);let E,N=()=>{if(E!==void 0)return E;let $=i.server.address(),me=$&&typeof $=="object"?$.port:n.daemon.port;return E=`${n.daemon.tls?"https":"http"}://${n.daemon.host}:${me}`,E};ci(i,{validator:c,manager:y,defaultAgent:n.defaultAgent,processRegistry:d,onExtensionVersion:($,me)=>v.reportVersion($,me),onTransformerVersion:($,me)=>S.reportVersion($,me),transformers:S,extensionCommands:m,mcpTokenRegistry:C,extensionMcp:k,getDaemonOrigin:N,registry:u}),await i.listen({host:n.daemon.host,port:n.daemon.port});let V=i.server.address(),W=V&&typeof V=="object"?V.port:n.daemon.port;await pe.mkdir(w.home(),{recursive:!0}),await pe.writeFile(w.pidFile(),JSON.stringify({pid:process.pid,host:n.daemon.host,port:W,startedAt:new Date().toISOString()})+`
92
- `,{encoding:"utf8",mode:384});let M=n.daemon.tls?"https":"http",P=n.daemon.tls?"wss":"ws",B={daemonUrl:`${M}://${n.daemon.host}:${W}`,daemonHost:n.daemon.host,daemonPort:W,serviceToken:e,daemonWsUrl:`${P}://${n.daemon.host}:${W}/acp`,hydraHome:w.home()};v.setContext(B),S.setContext(B),await v.start(),await S.start(),y.resurrectPendingQueues().catch($=>{i.log.warn(`queue replay scan failed: ${$.message}`)});let F=n.daemon.agentSyncIntervalMinutes*60*1e3,U=F>0?Ir({registry:u,manager:y,intervalMs:F,logger:p}):void 0,D=n.daemon.sessionGcIntervalMinutes*60*1e3,je=n.daemon.sessionGcMaxAgeDays*24*60*60*1e3,Nn=D>0?Sr({manager:y,intervalMs:D,maxAgeMs:je,logger:p}):void 0;return{app:i,manager:y,registry:u,extensions:v,transformers:S,mcpTokenRegistry:C,extensionMcp:k,processRegistry:d,shutdown:async()=>{Nn&&Nn(),U&&U(),clearInterval(l),await o.flush(),await v.stop(),await S.stop(),await y.closeAll(),await y.flushSynopsis(3e4),await y.shutdownSynopsis(),await y.flushMetaWrites(),await y.flushHistoryWrites(),Xt(null),tn(null),Cn(null),await i.close();try{gi.unlinkSync(w.pidFile())}catch{}try{r.flushSync()}catch{}}}}async function Ac(n){let e=await Sc({file:w.logFile(),size:"10m",frequency:"daily",mkdir:!0,symlink:!0}),t=mi.destination(2);return{stream:mi.multistream([{stream:e,level:n},{stream:t,level:n}]),fileStream:e}}function xc(n){let e=n.daemon.host;if(!(e==="127.0.0.1"||e==="::1"||e==="localhost"||e==="[::1]")&&!n.daemon.tls)throw new Error(`Refusing to bind to non-loopback host ${e} without TLS configured.`)}export{ae as AgentInstance,he as JsonRpcConnection,Be as Registry,ye as Session,Ge as SessionManager,$i as defaultConfig,wi as ensureServiceToken,rt as generateServiceToken,Gn as loadConfig,yi as loadServiceToken,rn as ndjsonStreamFromStdio,w as paths,oe as planSpawn,kc as startDaemon,_i as writeConfig,Pn as wsToMessageStream};
105
+ `),r}function xd(n){let e=[];for(let t of n.history){if(t.method!=="session/update")continue;let s=t.params;if(!s||typeof s!="object")continue;let r=Fr(s.update,{cwd:n.session.cwd});r!==null&&e.push({event:r,recordedAt:t.recordedAt})}return e}function Cd(n){let e=new Map;for(let{event:t}of n){if(t.kind==="tool-call"){let s=e.get(t.toolCallId);e.set(t.toolCallId,{title:t.title,status:t.status??s?.status??"pending"});continue}if(t.kind==="tool-call-update"){let s=e.get(t.toolCallId)??{title:"tool call",status:"pending"};e.set(t.toolCallId,{title:t.title??s.title,status:t.status??s.status})}}return e}function Md(n,e){let t=e.session,s=Bs(t.sessionId),r=t.title?.trim()||`Hydra session ${s}`;n.push(`# ${Ie(r)}`),n.push("");let i=[];i.push(`- **Session:** \`${s}\` (lineage \`${t.lineageId}\`)`);let o=[t.agentId];t.currentModel&&o.push(`model: ${t.currentModel}`),t.currentMode&&o.push(`mode: ${t.currentMode}`),i.push(`- **Agent:** ${o.filter(Boolean).join(" \xB7 ")}`),i.push(`- **Cwd:** ${t.cwd}`),i.push(`- **Exported:** ${e.exportedAt} from ${e.exportedFrom.machine} (hydra ${e.exportedFrom.hydraVersion})`);let a=t.currentUsage;if(a&&(a.used!==void 0||a.costAmount!==void 0)){let d=[];if(a.used!==void 0){let c=a.size!==void 0?`${En(a.size)}`:void 0;d.push(c?`${En(a.used)} / ${c} tokens`:`${En(a.used)} tokens`)}if(a.costAmount!==void 0){let c=a.costCurrency??"USD";d.push(`$${a.costAmount.toFixed(2)} ${c}`)}i.push(`- **Usage:** ${d.join(" \xB7 ")}`)}n.push(i.join(`
106
+ `)),n.push("")}function Rd(n,e,t){if(!e.some(c=>Ed(c.event))){n.push("_No conversation history recorded._"),n.push("");return}let s=new Set,r=0,i="",o=!1,a=()=>{i.length!==0&&(n.push(i.trimEnd()),n.push(""),i="")},d=()=>{o||(r+=1,n.push("---"),n.push(""),n.push(`## Turn ${r}`),n.push(""),o=!0)};for(let{event:c}of e)switch(c.kind){case"user-text":{a(),r+=1,n.push("---"),n.push(""),n.push(`## Turn ${r}`),n.push(""),n.push("**User:**"),n.push("");for(let p of c.text.split(`
107
+ `))n.push(`> ${Ie(p)}`);n.push(""),n.push("**Assistant:**"),n.push(""),o=!0;break}case"agent-text":d(),i+=c.text;break;case"agent-thought":{d(),a();let p=c.text.split(`
108
+ `);for(let l of p)n.push(`> _${Ie(l)}_`);n.push("");break}case"tool-call":{if(d(),a(),s.has(c.toolCallId))break;s.add(c.toolCallId);let p=t.get(c.toolCallId)??{title:c.title,status:c.status??"pending"};n.push(`- ${Td(p.status)} ${Pd(p)}`),n.push("");break}case"tool-call-update":break;case"plan":{d(),a(),n.push("**Plan:**"),n.push("");for(let p of c.entries){let l=p.status==="completed"?"[x]":"[ ]";n.push(`- ${l} ${Ie(p.content)}`)}n.push("");break}case"mode-changed":d(),a(),n.push(`_mode: ${Ie(c.mode)}_`),n.push("");break;case"model-changed":d(),a(),n.push(`_model: ${Ie(c.model)}_`),n.push("");break;case"turn-complete":a();break;case"usage-update":case"available-commands":case"session-info":case"unknown":break}a()}function Ed(n){switch(n.kind){case"usage-update":case"available-commands":case"session-info":case"unknown":case"turn-complete":return!1;default:return!0}}function Pd(n){let e=n.status,t=e==="completed"||e===void 0?"":` _(${e})_`;return`${Ie(n.title)}${t}`}function Td(n){switch(n){case"completed":return"\u2713";case"failed":return"\u2717";case"cancelled":case"rejected":return"\u2298";case"in_progress":return"\u21BB";default:return"\xB7"}}function Ie(n){return n.replace(/</g,"&lt;").replace(/>/g,"&gt;")}function En(n){return n.toLocaleString("en-US")}function qr(n){return n==="127.0.0.1"||n==="::1"||n==="localhost"||n==="[::1]"}function _d(n){let e=n.trim();if(e.length===0)return{operator:"OR",terms:[]};let t=/\w+:"[^"]*"|"[^"]*"|\S+/g,s=[],r;for(;(r=t.exec(e))!==null;)s.push(r[0]);let i="OR",o=!1,a=!1,d=[];for(let p of s){let l=p.toUpperCase();l==="AND"?o=!0:l==="OR"?a=!0:d.push(p)}o?i="AND":a&&(i="OR");let c=d.map(p=>$d(p)).filter(p=>p.term.length>0);return{operator:i,terms:c}}function $d(n){let e=/^(\w+):"([^"]*)"$/.exec(n);if(e)return{scope:Jr(e[1]),term:e[2]};let t=/^"([^"]*)"$/.exec(n);if(t)return{scope:"all",term:t[1]};let s=/^(prompt|response|tool):([\s\S]*)$/i.exec(n);return s?{scope:Jr(s[1]),term:s[2].trim()}:{scope:"all",term:n.trim()}}function Jr(n){switch(n.toLowerCase()){case"prompt":return"user";case"response":return"agent";case"tool":return"tool";default:return"all"}}function Od(n,e){return n==="all"?!0:n==="user"?e==="user":n==="agent"?e==="agent"||e==="thought":e==="tool"||e==="tool-input"}var Fd=5,Nd=200,zr=30;async function Vr(n,e,t={}){let s=_d(e);if(s.terms.length===0)return{query:e,truncated:!1,results:[]};let r=t.maxSnippetsPerSession??Fd,i=t.maxSessions??Nd,o=t.sessionIds?new Set(t.sessionIds):null,a=await n.list(),d=o?a.filter(l=>o.has(l.sessionId)):a,c=[],p=!1;for(let l of d){if(c.length>=i){p=!0;break}let u=await n.loadHistory(l.sessionId).catch(()=>[]),f=jd(u,s,r);if(f.snippets.length===0)continue;let g={sessionId:l.sessionId,cwd:l.cwd,status:l.status,updatedAt:l.updatedAt,totalMatches:f.totalMatches,snippets:f.snippets};l.title!==void 0&&(g.title=l.title),c.push(g)}return{query:e,truncated:p,results:c}}function jd(n,e,t){if(e.terms.length===0)return{totalMatches:0,snippets:[]};let s=0,r=[];for(let{scope:i,term:o}of e.terms){let a=Hd(n,o,i,t-r.length);if(e.operator==="AND"&&a.totalMatches===0)return{totalMatches:0,snippets:[]};s+=a.totalMatches,r.push(...a.snippets)}return{totalMatches:s,snippets:r}}function Hd(n,e,t,s){let r=e.toLowerCase(),i=0,o=[];for(let a of n){let d=Bd(a).filter(c=>Od(t,c.kind));for(let c of d){let p=c.text.toLowerCase(),l=p.indexOf(r);if(l===-1)continue;let u=0;for(;l!==-1;)u++,l=p.indexOf(r,l+r.length);if(i+=u,o.length<s){let f=p.indexOf(r),g={kind:c.kind,text:Jd(c.text,f,r.length),recordedAt:a.recordedAt};c.toolName!==void 0&&(g.toolName=c.toolName),o.push(g)}}}return{totalMatches:i,snippets:o}}function Bd(n){if(n.method!=="session/update")return[];let e=n.params;if(!e||typeof e!="object"||Array.isArray(e))return[];let t=e.update;if(!t||typeof t!="object"||Array.isArray(t))return[];let s=t,r=typeof s.sessionUpdate=="string"?s.sessionUpdate:s.kind;if(typeof r!="string")return[];switch(r){case"agent_message_chunk":{let i=jt(s.content);return i?[{kind:"agent",text:i}]:[]}case"agent_thought":case"agent_thought_chunk":{let i=typeof s.text=="string"?Z(s.text):jt(s.content);return i?[{kind:"thought",text:i}]:[]}case"user_message_chunk":{if(Ld(s))return[];let i=jt(s.content);return i?[{kind:"user",text:i}]:[]}case"prompt_received":{let i=qd(s.prompt);return i?[{kind:"user",text:i}]:[]}case"tool_call":case"tool_call_update":return Ud(s);default:return[]}}function Ud(n){let e=Wr(n,"name"),t=Wr(n,"title"),s=[];if(t!==void 0){let a=H(t);if(a.length>0){let d={kind:"tool",text:a};e!==void 0&&(d.toolName=e),s.push(d)}}if(e!==void 0&&e!==t){let a=H(e);a.length>0&&s.push({kind:"tool",toolName:e,text:a})}let r=n.rawInput;if(r&&typeof r=="object"){let a=Qr(r);if(a.length>0){let d={kind:"tool-input",text:H(a)};e!==void 0&&(d.toolName=e),s.push(d)}}let i=n.locations;if(Array.isArray(i)&&i.length>0){let a=Qr(i);if(a.length>0){let d={kind:"tool-input",text:H(a)};e!==void 0&&(d.toolName=e),s.push(d)}}let o=Dd(n);if(o!==null){let a={kind:"tool",text:o};e!==void 0&&(a.toolName=e),s.push(a)}return s}function Dd(n){let e=n.content;if(Array.isArray(e))for(let s of e){if(!s||typeof s!="object")continue;let i=s.content;if(!i||typeof i!="object")continue;let o=i;if(o.type==="text"&&typeof o.text=="string"){let a=H(o.text);if(a.length>0)return a}}let t=n.rawOutput;if(t&&typeof t=="object"){let s=t.error;if(typeof s=="string"){let r=H(s);if(r.length>0)return r}}return null}function Ld(n){let e=n._meta;if(!e||typeof e!="object"||Array.isArray(e))return!1;let t=e["hydra-acp"];return!t||typeof t!="object"||Array.isArray(t)?!1:t.compatFor==="prompt_received"}function jt(n){if(typeof n=="string")return Z(n);if(!n||typeof n!="object"||Array.isArray(n))return"";let e=n;return typeof e.text=="string"?Z(e.text):""}function qd(n){if(!Array.isArray(n))return"";let e=[];for(let t of n){let s=jt(t);s.length>0&&e.push(s)}return e.join("")}function Wr(n,e){let t=n[e];return typeof t=="string"?t:void 0}function Qr(n){try{return JSON.stringify(n)}catch{return""}}function Jd(n,e,t){let s=n.replace(/\s+/g," ").trim();if(s.length===0)return"";let r=s.toLowerCase(),i=n.slice(e,e+t).toLowerCase().replace(/\s+/g," ").trim(),o=i.length>0?r.indexOf(i):0;o===-1&&(o=0);let a=Math.max(0,o-zr),d=Math.min(s.length,o+i.length+zr),c=a>0?"\u2026":"",p=d<s.length?"\u2026":"";return`${c}${s.slice(a,d)}${p}`}function Gr(n){if(n.publicHost&&n.publicHost.length>0)return n.publicHost;if(n.host&&!qr(n.host))return n.port!==void 0?`${n.host}:${n.port}`:n.host}function Kr(n,e,t){n.get("/v1/sessions",async s=>{let r=s.query,i=r?.includeNonInteractive==="1"||r?.includeNonInteractive==="true";return{sessions:await e.list({cwd:r?.cwd,includeNonInteractive:i})}}),n.post("/v1/sessions/search",async(s,r)=>{let i=s.body??{},o=typeof i.q=="string"?i.q:"";if(o.trim().length===0)return r.code(400).send({error:"q is required"}),r;let a=Array.isArray(i.sessionIds)?i.sessionIds.filter(c=>typeof c=="string"&&c.length>0):void 0;return await Vr(e,o,{sessionIds:a})}),n.post("/v1/sessions",async(s,r)=>{let i=s.body??{},o=He(i.cwd??t.cwd),a=i.agentId??t.agentId;try{let d=await e.create({cwd:o,agentId:a,mcpServers:i.mcpServers});r.code(201).send({sessionId:d.sessionId,agentId:d.agentId,cwd:d.cwd})}catch(d){r.code(500).send({error:d.message})}}),n.post("/v1/sessions/collect",async(s,r)=>{let i=s.body??{},o=typeof i.maxAgeDays=="number"&&i.maxAgeDays>0?i.maxAgeDays*24*60*60*1e3:0,a=typeof i.limit=="number"&&i.limit>0?i.limit:1e3,d=i.selection==="explicit"||i.selection==="unpromoted"?i.selection:"unpromoted";try{let c=await Mn({manager:e,maxAgeMs:o,maxDeletions:a,selection:d,verbose:!1});r.code(200).send(c)}catch(c){r.code(500).send({error:c.message})}}),n.post("/v1/sessions/:id/kill",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=e.get(o);if(a){a.close({deleteRecord:!1}).catch(()=>{}),r.code(202).send();return}if(!await e.hasRecord(o)){r.code(404).send({error:"session not found"});return}r.code(204).send()}),n.post("/v1/sessions/:id/stdin/open",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=e.get(o);if(!a)return r.code(404).send({error:"session not found"}),r;let d=s.body??{},c={};(d.mode==="memory"||d.mode==="file")&&(c.mode=d.mode),typeof d.capacityBytes=="number"&&(c.capacityBytes=d.capacityBytes),typeof d.fileCapBytes=="number"&&(c.fileCapBytes=d.fileCapBytes),(c.mode??"memory")==="file"&&(c.filePathFor=p=>Yr.join(tt.tmpdir(),`hydra-acp-stdin-${p}.log`));try{return a.openStream(c)}catch(p){return r.code(409).send({error:p.message}),r}}),n.post("/v1/sessions/:id/stdin",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=e.get(o);if(!a)return r.code(404).send({error:"session not found"}),r;let d=s.body??{},c=typeof d.chunk=="string"?d.chunk:"",p=d.eof===!0;try{return a.streamWrite(c,p)}catch(l){return r.code(409).send({error:l.message}),r}}),n.patch("/v1/sessions/:id",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=s.body??{};if(a.priority!==void 0){let c=a.priority,p;if(c===null||c===0)p=void 0;else if(typeof c=="number"&&Number.isInteger(c)&&c>0)p=c;else{r.code(400).send({error:"priority must be a non-negative integer (or null to clear)"});return}if(!await e.setPriority(o,p)){r.code(404).send({error:"session not found"});return}r.code(204).send();return}if(a.regen===!0){if(!(e.get(o)!==void 0||await e.hasRecord(o))){r.code(404).send({error:"session not found"});return}e.scheduleSynopsis(o),r.code(202).send();return}if(typeof a.title!="string"||a.title.trim().length===0){r.code(400).send({error:"title must be a non-empty string"});return}if(!await e.setTitle(o,a.title)){r.code(404).send({error:"session not found"});return}r.code(204).send()}),n.delete("/v1/sessions/:id",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=e.get(o);if(a){await a.close({deleteRecord:!0}),r.code(204).send();return}if(!await e.deleteRecord(o)){r.code(404).send({error:"session not found"});return}r.code(204).send()}),n.get("/v1/sessions/:id/export",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=s.query?.tools,d=a==="references"?"references":er(a),c=await e.exportBundle(o,d==="references"?{tools:"references"}:{});if(!c){r.code(404).send({error:"session not found"});return}let p=Qe({record:c.record,history:d==="summary"?tr(c.history,"summary"):c.history,promptHistory:c.promptHistory.length>0?c.promptHistory:void 0,...c.toolBlobs!==void 0?{toolBlobs:c.toolBlobs}:{},hydraVersion:Q,machine:tt.hostname(),hydraHost:Gr(t)}),l=new Date().toISOString().replace(/[:.]/g,"-");r.header("Content-Disposition",`attachment; filename="${o}-${l}.hydra"`),r.code(200).send(p)}),n.get("/v1/sessions/:id/diff",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=await e.exportBundle(o);if(!a){r.code(404).send({error:"session not found"});return}let d=s.query??{},c=d.fold==="true"||d.fold==="1",p=typeof d.paths=="string"&&d.paths.length>0?new Set(d.paths.split(",").map(u=>u.trim()).filter(u=>u.length>0)):void 0,l=Ur(a.history);p&&(l=l.filter(u=>p.has(u.path))),c&&(l=l.map(u=>({...u,hunks:Dr(u.hunks)}))),r.code(200).send(l)}),n.get("/v1/sessions/:id/tools/:hash",async(s,r)=>{let i=s.params,o=await e.resolveCanonicalId(i.id)??i.id,a=await e.loadToolBlob(o,i.hash);if(a===null){r.code(404).send({error:"tool blob not found"});return}r.header("Content-Type","text/plain; charset=utf-8"),r.code(200).send(a)}),n.get("/v1/sessions/:id/transcript",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=await e.exportBundle(o);if(!a){r.code(404).send({error:"session not found"});return}let d=Qe({record:a.record,history:a.history,promptHistory:a.promptHistory.length>0?a.promptHistory:void 0,hydraVersion:Q,machine:tt.hostname(),hydraHost:Gr(t)});r.header("Content-Type","text/markdown; charset=utf-8"),r.code(200).send(Lr(d))}),n.post("/v1/sessions/:id/fork",async(s,r)=>{let i=s.params.id,o=await e.resolveCanonicalId(i)??i,a=s.body??{},d={};if(a.forkAt!==void 0){if(typeof a.forkAt!="string"||a.forkAt.length===0){r.code(400).send({error:"forkAt must be a non-empty string"});return}d.forkAt=a.forkAt}if(a.cwd!==void 0){if(typeof a.cwd!="string"||a.cwd.length===0){r.code(400).send({error:"cwd must be a non-empty string"});return}d.cwd=He(a.cwd)}if(a.agentId!==void 0){if(typeof a.agentId!="string"||a.agentId.length===0){r.code(400).send({error:"agentId must be a non-empty string"});return}d.agentId=a.agentId}try{let c=await e.forkSession(o,d);r.code(201).send(c)}catch(c){let p=c;if(p.code===I.SessionNotFound){r.code(404).send({error:p.message});return}if(p.code===I.InvalidParams||p.code===I.AgentNotInstalled){r.code(400).send({error:p.message});return}r.code(500).send({error:p.message})}}),n.post("/v1/sessions/import",async(s,r)=>{let i=s.body??{};if(i.bundle===void 0){r.code(400).send({error:"missing bundle"});return}let o;if(i.cwd!==void 0){if(typeof i.cwd!="string"||i.cwd.length===0){r.code(400).send({error:"cwd must be a non-empty string"});return}o=i.cwd}let a;try{a=cr(i.bundle)}catch(d){r.code(400).send({error:"invalid bundle",details:d.message});return}try{let d=await e.importBundle(a,{replace:i.replace===!0,...o!==void 0?{cwd:o}:{}});r.code(201).send(d)}catch(d){let c=d;if(c.code===I.BundleAlreadyImported){r.code(409).send({error:"bundle already imported",existingSessionId:c.existingSessionId});return}r.code(500).send({error:c.message})}}),n.get("/v1/sessions/:id/history",async(s,r)=>{let i=s.params.id,o=s.query,a=o?.follow==="1"||o?.follow==="true",d=await e.resolveCanonicalId(i)??i,c=e.get(d),p,l,u=!1,f=[];if(c)a&&(l=c.onBroadcast(m=>{r.raw.writableEnded||(u?r.raw.write(JSON.stringify(m)+`
109
+ `):f.push(m))})),p=await c.getHistorySnapshot();else{let m=await e.getHistory(d);if(m===void 0)return r.code(404).send({error:"session not found"}),r;p=m}r.raw.setHeader("Content-Type","application/x-ndjson"),r.raw.setHeader("Cache-Control","no-cache"),r.raw.statusCode=200;let g=new Set;for(let m of p??[]){r.raw.write(JSON.stringify(m)+`
110
+ `);let y=m;typeof y.recordedAt=="number"&&g.add(String(y.recordedAt))}for(let m of f){let y=m,v=typeof y.recordedAt=="number"?String(y.recordedAt):"";v&&g.has(v)||r.raw.write(JSON.stringify(m)+`
111
+ `)}return u=!0,l?(s.raw.on("close",()=>{l?.(),r.raw.writableEnded||r.raw.end()}),r):(r.raw.end(),r)})}function Xr(n,e,t,s={}){n.get("/v1/agents",async()=>Me(e)),n.get("/v1/registry",async()=>e.load()),n.post("/v1/registry/refresh",async()=>{let r=await e.refresh();return{version:r.version,agentCount:r.agents.length}}),n.post("/v1/agents/:id/install",async(r,i)=>{let o=r.params.id,a=await e.getAgent(o);if(!a){i.code(404).send({error:`agent ${o} not found in registry`});return}if(a.distribution.uvx&&!a.distribution.npx&&!a.distribution.binary){i.send({agentId:a.id,version:a.version??"current",distribution:"uvx",installed:!1,message:"uvx agents resolve on first run; nothing to pre-install."});return}try{let d=await oe(a,[],{npmRegistry:s.npmRegistry}),c=a.distribution.npx?"npx":a.distribution.binary?"binary":"unknown";i.send({agentId:a.id,version:d.version,distribution:c,installed:!0,command:d.command})}catch(d){i.code(500).send({error:d.message})}}),n.post("/v1/agents/:id/sync",async(r,i)=>{let o=r.params.id;try{let{synced:a,skipped:d}=await t.syncFromAgent(o);return{synced:a.map(c=>({sessionId:c.sessionId,upstreamSessionId:c.upstreamSessionId,agentId:c.agentId,cwd:c.cwd,title:c.title,updatedAt:c.updatedAt})),skipped:d}}catch(a){let d=a;if(d.code===I.AgentNotInstalled){i.code(404).send({error:d.message});return}i.code(409).send({error:d.message})}})}function Zr(n,e,t){n.get("/v1/health",{config:{skipAuth:!0}},async()=>({status:"ok",version:e,configDigest:t}))}var zd=/^[A-Za-z0-9._-]+$/;function ei(n,e){n.get("/v1/extensions",async()=>({extensions:e.list()})),n.get("/v1/extensions/:name",async(t,s)=>{let r=t.params.name,i=e.get(r);if(!i){s.code(404).send({error:`unknown extension: ${r}`});return}return i}),n.post("/v1/extensions",async(t,s)=>{let r=t.body??{},i=Wd(r);if("error"in i){s.code(400).send({error:i.error});return}try{let o=e.register(i.config);s.code(201).send(o)}catch(o){nt(s,o)}}),n.delete("/v1/extensions/:name",async(t,s)=>{let r=t.params.name;try{await e.unregister(r),s.code(204).send()}catch(i){nt(s,i)}}),n.post("/v1/extensions/:name/start",async(t,s)=>{let r=t.params.name;try{let i=await e.startByName(r);s.code(200).send(i)}catch(i){nt(s,i)}}),n.post("/v1/extensions/:name/stop",async(t,s)=>{let r=t.params.name;try{let i=await e.stopByName(r);s.code(200).send(i)}catch(i){nt(s,i)}}),n.post("/v1/extensions/:name/restart",async(t,s)=>{let r=t.params.name;try{let i=await e.restartByName(r);s.code(200).send(i)}catch(i){nt(s,i)}})}function nt(n,e){let t=e.code,s=e.message??"unknown error";if(t==="NOT_FOUND"){n.code(404).send({error:s});return}if(t==="CONFLICT"){n.code(409).send({error:s});return}n.code(500).send({error:s})}function Wd(n){let e=n.name;if(typeof e!="string"||!zd.test(e))return{error:"name must match [A-Za-z0-9._-]+"};let t=n.command;if(t!==void 0&&(!Array.isArray(t)||t.some(o=>typeof o!="string")))return{error:"command must be string[]"};let s=n.args;if(s!==void 0&&(!Array.isArray(s)||s.some(o=>typeof o!="string")))return{error:"args must be string[]"};let r=n.env;if(r!==void 0&&(typeof r!="object"||r===null||Array.isArray(r)))return{error:"env must be an object of string\u2192string"};if(r&&Object.values(r).some(o=>typeof o!="string"))return{error:"env values must be strings"};let i=n.enabled;return i!==void 0&&typeof i!="boolean"?{error:"enabled must be a boolean"}:{config:{name:e,command:t??[],args:s??[],env:r??{},enabled:i===void 0?!0:i}}}var Qd=/^[A-Za-z0-9._-]+$/;function ti(n,e){n.get("/v1/transformers",async()=>({transformers:e.list()})),n.get("/v1/transformers/:name",async(t,s)=>{let r=t.params.name,i=e.get(r);if(!i){s.code(404).send({error:`unknown transformer: ${r}`});return}return i}),n.post("/v1/transformers",async(t,s)=>{let r=t.body??{},i=Vd(r);if("error"in i){s.code(400).send({error:i.error});return}try{let o=e.register(i.config);s.code(201).send(o)}catch(o){st(s,o)}}),n.delete("/v1/transformers/:name",async(t,s)=>{let r=t.params.name;try{await e.unregister(r),s.code(204).send()}catch(i){st(s,i)}}),n.post("/v1/transformers/:name/start",async(t,s)=>{let r=t.params.name;try{let i=await e.startByName(r);s.code(200).send(i)}catch(i){st(s,i)}}),n.post("/v1/transformers/:name/stop",async(t,s)=>{let r=t.params.name;try{let i=await e.stopByName(r);s.code(200).send(i)}catch(i){st(s,i)}}),n.post("/v1/transformers/:name/restart",async(t,s)=>{let r=t.params.name;try{let i=await e.restartByName(r);s.code(200).send(i)}catch(i){st(s,i)}})}function st(n,e){let t=e.code,s=e.message??"unknown error";if(t==="NOT_FOUND"){n.code(404).send({error:s});return}if(t==="CONFLICT"){n.code(409).send({error:s});return}n.code(500).send({error:s})}function Vd(n){let e=n.name;if(typeof e!="string"||!Qd.test(e))return{error:"name must match [A-Za-z0-9._-]+"};let t=n.command;if(t!==void 0&&(!Array.isArray(t)||t.some(o=>typeof o!="string")))return{error:"command must be string[]"};let s=n.args;if(s!==void 0&&(!Array.isArray(s)||s.some(o=>typeof o!="string")))return{error:"args must be string[]"};let r=n.env;if(r!==void 0&&(typeof r!="object"||r===null||Array.isArray(r)))return{error:"env must be an object of string\u2192string"};if(r&&Object.values(r).some(o=>typeof o!="string"))return{error:"env values must be strings"};let i=n.enabled;return i!==void 0&&typeof i!="boolean"?{error:"enabled must be a boolean"}:{config:{name:e,command:t??[],args:s??[],env:r??{},enabled:i===void 0?!0:i}}}function ni(n,e){n.get("/v1/config",async()=>e)}import{z as Ne}from"zod";import*as Fe from"fs/promises";import*as si from"path";import{randomBytes as np,scrypt as Gd,timingSafeEqual as Yd}from"crypto";import{promisify as Kd}from"util";var Xd=Kd(Gd);function ri(){return si.join(w.home(),"password-hash")}var Zd=128*1024*1024;async function ii(){try{return(await Fe.readFile(ri(),"utf8")).trim().length>0}catch(n){if(n.code==="ENOENT")return!1;throw n}}async function oi(n){if(typeof n!="string"||n.length===0)return!1;let e;try{e=(await Fe.readFile(ri(),"utf8")).trim()}catch(c){if(c.code==="ENOENT")return!1;throw c}let t=e.split("$");if(t.length!==6||t[0]!=="scrypt")return!1;let s=parseInt(t[1],10),r=parseInt(t[2],10),i=parseInt(t[3],10);if(!Number.isFinite(s)||!Number.isFinite(r)||!Number.isFinite(i))return!1;let o=Buffer.from(t[4],"hex"),a=Buffer.from(t[5],"hex");if(o.length===0||a.length===0)return!1;let d=await Xd(n,o,a.length,{N:s,r,p:i,maxmem:Zd});return d.length!==a.length?!1:Yd(d,a)}var ec=Ne.object({password:Ne.string().min(1),label:Ne.string().min(1).max(256).optional(),ttlSec:Ne.number().int().positive().optional()}),tc=Ne.object({id:Ne.string().optional()}).optional();function ai(n,e){n.post("/v1/auth/login",{config:{skipAuth:!0}},async(t,s)=>{let r=nc(t);if(e.rateLimiter.isBlocked(r))return s.code(429).send({error:"Too many failed attempts; try again later."});let i;try{i=ec.parse(t.body)}catch{return s.code(400).send({error:"Invalid request body"})}if(!await ii())return s.code(403).send({error:"No password configured. Run `hydra-acp auth password` on the daemon host."});if(!await oi(i.password))return e.rateLimiter.recordFailure(r),s.code(401).send({error:"Invalid password"});e.rateLimiter.recordSuccess(r);let a=await e.store.issue({label:i.label,ttlSec:i.ttlSec});return s.code(200).send({session_token:a.token,id:a.id,expires_at:a.expiresAt})}),n.post("/v1/auth/logout",async(t,s)=>{let r;try{r=tc.parse(t.body??void 0)}catch{return s.code(400).send({error:"Invalid request body"})}let i=r?.id??t.authIdentity;if(!i||i==="service")return s.code(200).send({revoked:!1});let o=await e.store.revoke(i);return s.code(200).send({revoked:o})}),n.get("/v1/auth/verify",async(t,s)=>s.code(200).send({ok:!0})),n.get("/v1/auth/sessions",async(t,s)=>s.code(200).send({sessions:e.store.list()})),n.delete("/v1/auth/sessions/:id",async(t,s)=>{let r=t.params.id;return await e.store.revoke(r)?s.code(204).send():s.code(404).send({error:"Not found"})})}function nc(n){return n.ip||"unknown"}import{nanoid as Fn}from"nanoid";function Pn(n){let e=[],t=[],s=!1,r=i=>{if(!s){s=!0;for(let o of t)o(i)}};return n.on("message",(i,o)=>{if(o)return;let a=i.toString("utf8");try{let d=JSON.parse(a);for(let c of e)c(d)}catch(d){for(let c of e)c({jsonrpc:"2.0",id:0,error:{code:I.ParseError,message:`Failed to parse WS frame: ${d.message}`}})}}),n.on("close",()=>r()),n.on("error",i=>r(i)),{async send(i){if(s)throw new Error("ws is closed");let o=JSON.stringify(i);await new Promise((a,d)=>{n.send(o,c=>{if(c){d(c);return}a()})})},onMessage(i){e.push(i)},onClose(i){t.push(i)},async close(){s||(n.close(),r())}}}import{randomBytes as ci}from"crypto";function ui(n,e){n.get("/acp",{websocket:!0},async(t,s)=>{let r=_r({headers:s.headers,url:s.url});if(!r||!await e.validator.validate(r)){t.close(4401,"Unauthorized");return}let i=e.processRegistry?.resolve(r),o=Pn(t),a=new ye(o),d={clientId:`hydra_client_${Fn(12)}`,processIdentity:i,attached:new Map};a.onClose(()=>{for(let l of d.attached.values())e.manager.get(l.sessionId)?.detach(l.clientId);d.attached.clear()});let c=(l,u)=>{if(d.attached.get(l)?.readonly){let g=new Error(`${u} not permitted on a read-only attachment`);throw g.code=I.PermissionDenied,g}};if(a.onRequest("initialize",async l=>{let u=ms.parse(l??{});u.clientInfo?.name&&(d.clientInfo={name:u.clientInfo.name,...u.clientInfo.version!==void 0?{version:u.clientInfo.version}:{}});let f=u.clientInfo?.version;return f&&i&&(i.kind==="extension"?e.onExtensionVersion?.(i.name,f):e.onTransformerVersion?.(i.name,f)),oc()}),i&&e.extensionCommands){let l=e.extensionCommands;a.onRequest("hydra-acp/commands/register",async u=>{let f=u??{},g=Array.isArray(f.commands)?f.commands.map(m=>{if(!m||typeof m!="object")return;let y=m;if(typeof y.verb!="string")return;let v={verb:y.verb};return typeof y.argsHint=="string"&&(v.argsHint=y.argsHint),typeof y.description=="string"&&(v.description=y.description),v}).filter(m=>m!==void 0):[];return l.register(i.name,a,g),{ok:!0,registered:g.length}}),a.onClose(()=>{l.clear(i.name)})}if(i&&e.extensionMcp){let l=e.extensionMcp;a.onRequest("hydra-acp/mcp_tools/register",async u=>{let f=u??{},g=typeof f.instructions=="string"?f.instructions:void 0,m=Array.isArray(f.tools)?f.tools.map(y=>{if(!y||typeof y!="object")return;let v=y;if(typeof v.name!="string"||v.name.length===0||typeof v.description!="string"||v.inputSchema===null||typeof v.inputSchema!="object")return;let S={name:v.name,description:v.description,inputSchema:v.inputSchema};return v.outputSchema!==null&&typeof v.outputSchema=="object"&&(S.outputSchema=v.outputSchema),S}).filter(y=>y!==void 0):[];if(m.length===0)throw new Error("register_mcp_tools requires at least one tool");return l.register(i.name,a,g,m),{ok:!0,registered:m.length}}),a.onClose(()=>{l.clear(i.name)})}i?.kind==="transformer"&&(a.onRequest("hydra-acp/transformer/initialize",async l=>{let u=l??{},f=Array.isArray(u.intercepts)?u.intercepts.filter(g=>typeof g=="string"):[];if(e.transformers&&(e.transformers.registerConnection(i.name,a,f),e.manager?.defaultTransformers.includes(i.name))){let g=e.transformers.resolveChain([i.name])[0];if(g)for(let m of e.manager.liveSessions())m.addTransformer(g)}return{ack:!0}}),a.onClose(()=>{e.transformers?.deregisterConnection(i.name)}),a.onRequest("hydra-acp/message/emit",async l=>{let u=l??{},f=typeof u.sessionId=="string"?u.sessionId:void 0,g=typeof u.method=="string"?u.method:void 0,m=u.envelope,y=u.route;if(!f||!g)throw Object.assign(new Error("emit_message requires sessionId and method"),{code:-32602});let v=e.manager.get(f);if(!v)throw Object.assign(new Error(`session ${f} not found`),{code:I.SessionNotFound});let S=typeof u.respondsTo=="string"?u.respondsTo:void 0;if(S)return v.dischargeClaim(S,m),{ok:!0};if(y==="chain")return{ok:!0,response:await v.emitToChain(i.name,g,m)};if(y==="daemon")return{ok:!0,response:await v.emitToChain(i.name,g,m)};throw Object.assign(new Error(`unsupported route: ${JSON.stringify(y)}`),{code:-32602})}),a.onRequest("hydra-acp/session/request_permission",async l=>{let u=l??{},f=typeof u.sessionId=="string"?u.sessionId:void 0;if(!f)throw Object.assign(new Error("hydra-acp/session/request_permission requires sessionId"),{code:-32602});let g=e.manager.get(f);if(!g)throw Object.assign(new Error(`session ${f} not found`),{code:I.SessionNotFound});return g.requestPermissionFromClients(u)}),a.onRequest("hydra-acp/transformer/attach",async l=>rc(l,i.name,e)),a.onRequest("hydra-acp/child_session/spawn",async l=>{let u=l??{},f=typeof u.agentId=="string"?u.agentId:e.defaultAgent,g=typeof u.cwd=="string"?u.cwd:void 0,m=typeof u.parentSessionId=="string"?u.parentSessionId:void 0,y=u._meta&&typeof u._meta=="object"?u._meta:void 0,v=y&&y["hydra-acp"]&&typeof y["hydra-acp"]=="object"?y["hydra-acp"]:void 0,S=v&&typeof v.title=="string"?v.title:void 0;if(!g&&m){let E=e.manager.get(m);E&&(g=E.cwd)}if(!g)throw Object.assign(new Error("child_session/spawn requires cwd (or a parentSessionId pointing at a live session whose cwd we can inherit)"),{code:-32602});let C=typeof u.interactive=="boolean"?u.interactive:!1;return{childSessionId:(await e.manager.create({agentId:f,cwd:g,parentSessionId:m,interactive:C,transformChain:[],title:S})).sessionId}}),a.onRequest("hydra-acp/session/fork",async l=>{let u=l??{};if(typeof u.sessionId!="string")throw Object.assign(new Error("fork_session requires sessionId"),{code:I.InvalidParams});let f=typeof u.forkAt=="string"?u.forkAt:void 0,g=typeof u.cwd=="string"?u.cwd:void 0,m=typeof u.agentId=="string"?u.agentId:void 0;return await e.manager.forkSession(u.sessionId,{...f!==void 0?{forkAt:f}:{},...g!==void 0?{cwd:g}:{},...m!==void 0?{agentId:m}:{}})}),a.onRequest("hydra-acp/session/delete",async l=>{let u=l??{};if(typeof u.sessionId!="string")throw Object.assign(new Error("hydra-acp/session/delete requires sessionId"),{code:I.InvalidParams});let f=await e.manager.resolveCanonicalId(u.sessionId)??u.sessionId,g=e.manager.get(f);if(g)return await g.close({deleteRecord:!0}),{deleted:!0,sessionId:f};if(!await e.manager.deleteRecord(f))throw Object.assign(new Error(`session ${f} not found`),{code:I.SessionNotFound});return{deleted:!0,sessionId:f}}),a.onRequest("hydra-acp/child_session/await",async l=>{let u=l??{},f=typeof u.childSessionId=="string"?u.childSessionId:void 0,g=u.until==="idle"?"idle":"turn_complete",m=typeof u.timeoutMs=="number"?Math.min(u.timeoutMs,30*6e4):5*6e4;if(!f)throw Object.assign(new Error("await_child requires childSessionId"),{code:-32602});let y=e.manager.get(f);if(!y)throw Object.assign(new Error(`child session ${f} not found`),{code:I.SessionNotFound});return new Promise(v=>{let S=[],C,k=()=>{clearTimeout(E),C?.(),v({entries:S})};C=y.onBroadcast(N=>{S.push(N),g==="turn_complete"&&N.params?.update?.sessionUpdate==="turn_complete"&&k()});let E=setTimeout(k,m);typeof E.unref=="function"&&E.unref(),y.onClose(()=>k())})}),a.onRequest("hydra-acp/child_session/close",async l=>{let u=l??{},f=typeof u.childSessionId=="string"?u.childSessionId:void 0;if(!f)throw Object.assign(new Error("close_child_session requires childSessionId"),{code:-32602});let g=e.manager.get(f);return g&&await g.close({deleteRecord:!1}),{ok:!0}}),a.onRequest("hydra-acp/connection/keep_alive",async l=>{let u=l??{},f=typeof u.token=="string"?u.token:void 0,g=typeof u.sessionId=="string"?u.sessionId:void 0,m=typeof u.estimatedRemainingMs=="number"?u.estimatedRemainingMs:void 0;return f&&g&&e.manager.get(g)?.keepAliveClaim(f,m),{ok:!0}})),a.onRequest("hydra-acp/session/tool_content",async l=>{let u=l??{};if(typeof u.sessionId!="string"||typeof u.hash!="string")throw Object.assign(new Error("hydra-acp/session/tool_content requires sessionId and hash"),{code:I.InvalidParams});let f=await e.manager.resolveCanonicalId(u.sessionId)??u.sessionId,g=await e.manager.loadToolBlob(f,u.hash);if(g===null)throw Object.assign(new Error("tool content not found"),{code:I.SessionNotFound});return{content:g}}),a.onRequest("session/new",async l=>{let u=gs.parse(l),f=De(l?._meta),g=Array.isArray(f.transformers)&&f.transformers.every(M=>typeof M=="string")?f.transformers:e.manager.defaultTransformers??[],m=e.transformers?.resolveChain(g)??[],y,v,S=u.mcpServers;if(f.mcpStdin===!0&&e.mcpTokenRegistry!==void 0&&e.getDaemonOrigin!==void 0){y=ci(32).toString("hex"),v=e.mcpTokenRegistry.reserve(y);let P={name:"hydra-acp-stdin",type:"http",url:`${e.getDaemonOrigin()}/mcp/hydra-acp-stdin`,headers:[{name:"Authorization",value:`Bearer ${y}`}]};S=[...u.mcpServers??[],P]}let C=Bt(e);C!==void 0&&(S=[...S??[],...C.descriptors]);let k;try{k=await e.manager.create({cwd:u.cwd,agentId:f.agentId??e.defaultAgent,mcpServers:S,title:f.title,agentArgs:f.agentArgs,model:f.model,onInstallProgress:di(a),transformChain:m,originatingClient:d.clientInfo,...f.interactive!==void 0?{interactive:f.interactive}:{}})}catch(M){throw v!==void 0&&v.abandon(M instanceof Error?M:void 0),C!==void 0&&C.abandon(M instanceof Error?M:void 0),M}if(y!==void 0&&v!==void 0&&e.mcpTokenRegistry!==void 0){let M=y,P=e.mcpTokenRegistry;v.complete(k),k.onClose(()=>{P.unbind(M)})}C!==void 0&&C.bindToSession(k);let E=Ht(a,k,d),{entries:N}=await k.attach(E,"full");d.attached.set(k.sessionId,{sessionId:k.sessionId,clientId:E.clientId,readonly:!1}),setImmediate(()=>{(async()=>{for(let M of N)await a.notify(M.method,M.params).catch(()=>{})})()});let V=Tn(k),W=_n(k);return{sessionId:k.sessionId,...V?{modes:V}:{},...W?{models:W}:{},configOptions:k.buildConfigOptions(),_meta:$n(e.manager,k,{clientId:E.clientId})}}),a.onRequest("session/attach",async l=>{let u=hs.parse(l),f=u.clientInfo?.version;f&&i&&(i.kind==="extension"?e.onExtensionVersion?.(i.name,f):e.onTransformerVersion?.(i.name,f));let g=De(u._meta),m=g.resume,y=g.readonly===!0;n.log.info(`session/attach sessionId=${u.sessionId} hasResumeHints=${!!m} readonly=${y}`);let v=m?u.sessionId:await e.manager.resolveCanonicalId(u.sessionId)??u.sessionId,S=e.manager.get(v);if(!S&&y){let M=await e.manager.loadFromDisk(v);if(!M){let O=new Error(`session ${u.sessionId} not found`);throw O.code=I.SessionNotFound,O}let P=await e.manager.loadHistory(v),B=u.clientId??`cli_${Fn(8)}`;d.attached.set(M.hydraSessionId,{sessionId:M.hydraSessionId,clientId:B,readonly:!0}),n.log.info(`session/attach OK (viewer) sessionId=${M.hydraSessionId} clientId=${B} attachedCount=${d.attached.size} replayed=${P.length}`);for(let O of P)await a.notify(O.method,O.params).catch(()=>{});return{sessionId:M.hydraSessionId,clientId:B,connectedClients:[B],historyPolicy:"full",replayed:P.length,_meta:ic(M)}}if(!S){let M=await e.manager.loadFromDisk(v),P=M;if(m&&(P={...M,hydraSessionId:u.sessionId,upstreamSessionId:m.upstreamSessionId,agentId:m.agentId,cwd:m.cwd,...m.title!==void 0?{title:m.title}:{},...m.agentArgs!==void 0?{agentArgs:m.agentArgs}:{}}),!P){let U=new Error(`session ${u.sessionId} not found and no resume hints provided`);throw U.code=I.SessionNotFound,U}let B=P.originatingClient?P:{...P,originatingClient:d.clientInfo},O=Bt(e);try{S=await e.manager.resurrect({...B,mcpServers:O?.descriptors,onInstallProgress:di(a)})}catch(U){throw O!==void 0&&O.abandon(U instanceof Error?U:void 0),U}O!==void 0&&O.bindToSession(S),On(S,e)}let C=Ht(a,S,d,u.clientInfo,u.clientId),k=g.replayMode==="drip",{entries:E,appliedPolicy:N}=await S.attach(C,u.historyPolicy,{afterMessageId:u.afterMessageId,raw:k,...g.toolContent!==void 0?{toolContent:g.toolContent}:{}});if(d.attached.set(S.sessionId,{sessionId:S.sessionId,clientId:C.clientId,readonly:y}),n.log.info(`session/attach OK sessionId=${S.sessionId} clientId=${C.clientId} attachedCount=${d.attached.size} requestedPolicy=${u.historyPolicy} appliedPolicy=${N} replayed=${E.length} readonly=${y}${k?" replayMode=drip":""}`),k){let M=g.dripSpeed&&g.dripSpeed>0?g.dripSpeed:1,P=750;(async()=>{let B=null;for(let O of E){let U=typeof O.recordedAt=="number"?O.recordedAt:null;if(B!==null&&U!==null){let D=Math.min(P,Math.max(0,(U-B)/M));D>0&&await new Promise(je=>setTimeout(je,D))}U!==null&&(B=U);try{await a.notify(O.method,O.params)}catch{return}}})()}else for(let P=0;P<E.length;P++){let B=E[P],O=a.notify(B.method,B.params).catch(()=>{});(P+1)%200===0&&await O}S.replayPendingPermissions(C);let V=Tn(S),W=_n(S);return{sessionId:S.sessionId,clientId:C.clientId,connectedClients:S.connectedClients(C.clientId),historyPolicy:N,replayed:E.length,...V?{modes:V}:{},...W?{models:W}:{},configOptions:S.buildConfigOptions(),_meta:$n(e.manager,S)}}),a.onRequest("session/detach",async l=>{let u=ys.parse(l),f=d.attached.get(u.sessionId);if(!f){let m=new Error("client not attached to that session");throw m.code=I.SessionNotFound,m}let g=e.manager.get(u.sessionId);return g?.detach(f.clientId),d.attached.delete(u.sessionId),g&&e.manager.reapIfOrphanedNonInteractive(u.sessionId),{sessionId:u.sessionId,_meta:{[Ue]:{detachStatus:"detached"}}}}),a.onRequest("session/list",async l=>{let u=ws.parse(l??{});return{sessions:(await e.manager.list({cwd:u.cwd})).filter(y=>y.originatingClient?.name!==wt).map(bs)}}),a.onRequest("hydra-acp/agents/list",async()=>{if(!e.registry){let l=new Error("agent registry unavailable");throw l.code=I.InternalError,l}return Me(e.registry)}),a.onRequest("session/prompt",async l=>{let u=Is.parse(l);c(u.sessionId,"session/prompt");let f=d.attached.get(u.sessionId);if(!f){n.log.warn(`session/prompt rejected: not attached sessionId=${u.sessionId} attachedKeys=[${[...d.attached.keys()].join(",")}]`);let m=new Error("not attached to session");throw m.code=I.SessionNotFound,m}let g=e.manager.get(u.sessionId);if(!g){let m=await e.manager.loadFromDisk(u.sessionId);if(!m){let S=new Error(`session ${u.sessionId} not found`);throw S.code=I.SessionNotFound,S}n.log.info(`session/prompt auto-resurrecting cold sessionId=${u.sessionId}`);let y=Bt(e);try{g=await e.manager.resurrect({...m,mcpServers:y?.descriptors})}catch(S){throw y!==void 0&&y.abandon(S instanceof Error?S:void 0),S}y!==void 0&&y.bindToSession(g),On(g,e);let v=Ht(a,g,d,void 0,f.clientId);await g.attach(v,"none")}return g.prompt(f.clientId,u)});let p=l=>{let u;try{u=lt.parse(l)}catch(m){n.log.warn(`session/cancel: invalid params: ${m.message}`);return}let f=d.attached.get(u.sessionId);if(!f)return;if(f.readonly){n.log.warn(`session/cancel dropped (readonly attachment) sessionId=${u.sessionId}`);return}let g=e.manager.get(u.sessionId);g&&g.cancel(f.clientId).catch(m=>{n.log.warn(`session/cancel for ${u.sessionId}: ${m.message}`)})};a.onNotification("session/cancel",p),a.onRequest("session/cancel",async l=>{let u=lt.parse(l);return c(u.sessionId,"session/cancel"),p(l),null}),a.onRequest("hydra-acp/prompt/cancel",async l=>{let u=ks.parse(l);c(u.sessionId,"hydra-acp/prompt/cancel");let f=e.manager.get(u.sessionId);if(!f){let g=new Error(`session ${u.sessionId} not found`);throw g.code=I.SessionNotFound,g}return f.cancelQueuedPrompt(u.messageId)}),a.onRequest("hydra-acp/session/force_cancel",async l=>{let u=lt.parse(l);c(u.sessionId,"hydra-acp/session/force_cancel");let f=e.manager.get(u.sessionId);if(!f){let g=new Error(`session ${u.sessionId} not found`);throw g.code=I.SessionNotFound,g}return f.forceCancel()}),a.onRequest("hydra-acp/prompt/update",async l=>{let u=As.parse(l);c(u.sessionId,"hydra-acp/prompt/update");let f=e.manager.get(u.sessionId);if(!f){let g=new Error(`session ${u.sessionId} not found`);throw g.code=I.SessionNotFound,g}return f.updateQueuedPrompt(u.messageId,u.prompt)}),a.onRequest("hydra-acp/prompt/amend",async l=>{let u=xs.parse(l);c(u.sessionId,"hydra-acp/prompt/amend");let f=d.attached.get(u.sessionId);if(!f){let m=new Error("not attached to session");throw m.code=I.SessionNotFound,m}let g=e.manager.get(u.sessionId);if(!g){let m=new Error(`session ${u.sessionId} not found`);throw m.code=I.SessionNotFound,m}return g.amendPrompt(f.clientId,u)}),a.onRequest("session/load",async l=>{let u=l??{},f=typeof u.sessionId=="string"?u.sessionId:void 0;if(!f){let k=new Error("session/load requires sessionId");throw k.code=I.InvalidParams,k}let g=await e.manager.resolveCanonicalId(f)??f,m=e.manager.get(g);if(!m){let k=await e.manager.loadFromDisk(g);if(!k){let N=new Error(`session ${f} not found in memory or on disk`);throw N.code=I.SessionNotFound,N}let E=Bt(e);try{m=await e.manager.resurrect({...k,mcpServers:E?.descriptors})}catch(N){throw E!==void 0&&E.abandon(N instanceof Error?N:void 0),N}E!==void 0&&E.bindToSession(m),On(m,e)}let y=Ht(a,m,d),{entries:v}=await m.attach(y,"pending_only");d.attached.set(m.sessionId,{sessionId:m.sessionId,clientId:y.clientId,readonly:!1});for(let k of v)await a.notify(k.method,k.params);m.replayPendingPermissions(y);let S=Tn(m),C=_n(m);return{sessionId:m.sessionId,...S?{modes:S}:{},...C?{models:C}:{},configOptions:m.buildConfigOptions(),_meta:$n(e.manager,m,{clientId:y.clientId})}}),a.onRequest("session/set_model",async l=>{let u=l?.sessionId;typeof u=="string"&&c(u,"session/set_model");let f=sc(l,e.manager);if(f.kind==="error"){n.log.warn(f.logMessage);let y=new Error(f.message);throw y.code=f.code,y}if(f.kind==="no_op")return n.log.warn(f.logMessage),await a.notify("session/update",{sessionId:f.sessionId,update:{sessionUpdate:"current_model_update",currentModel:f.currentModel}}).catch(()=>{}),null;n.log.info(f.logMessage);let{modelId:g}=f,m=await f.session.forwardRequest("session/set_model",{...l,modelId:g});return f.session.applyModelChange(g),m}),a.onRequest("session/set_mode",async l=>{let u=l,f=u?.sessionId;if(typeof f=="string"&&c(f,"session/set_mode"),!u||typeof u.sessionId!="string"){let y=new Error("session/set_mode requires string sessionId");throw y.code=I.InvalidParams,y}if(typeof u.modeId!="string"){let y=new Error("session/set_mode requires string modeId");throw y.code=I.InvalidParams,y}let g=e.manager.get(u.sessionId);if(!g){let y=new Error(`session ${u.sessionId} not found`);throw y.code=I.SessionNotFound,y}let m=await g.forwardRequest("session/set_mode",l);return g.applyModeChange(u.modeId),m}),a.onRequest("session/set_config_option",async l=>{let u=l,f=v=>{let S=new Error(v);return S.code=I.InvalidParams,S},g=u?.sessionId;if(typeof g=="string"&&c(g,"session/set_config_option"),!u||typeof u.sessionId!="string")throw f("session/set_config_option requires string sessionId");if(typeof u.configId!="string")throw f("session/set_config_option requires string configId");if(typeof u.value!="string")throw f("session/set_config_option requires string value");let m=e.manager.get(u.sessionId);if(!m){let v=new Error(`session ${u.sessionId} not found`);throw v.code=I.SessionNotFound,v}let y=m.buildConfigOptions().find(v=>v.id===u.configId);if(!y)throw f(`unknown configId ${JSON.stringify(u.configId)} for this session`);if(!y.options.some(v=>v.value===u.value))throw f(`value ${JSON.stringify(u.value)} is not valid for configId ${JSON.stringify(u.configId)}`);switch(u.configId){case"model":{u.value!==m.currentModel&&await m.forwardRequest("session/set_model",{sessionId:u.sessionId,modelId:u.value}),m.applyModelChange(u.value);break}case"mode":{u.value!==m.currentMode&&await m.forwardRequest("session/set_mode",{sessionId:u.sessionId,modeId:u.value}),m.applyModeChange(u.value);break}case"agent":{u.value!==m.agentId&&await m.setAgent(u.value);break}default:throw f(`configId ${JSON.stringify(u.configId)} is not settable`)}return{configOptions:m.buildConfigOptions()}}),a.setDefaultHandler(async(l,u)=>{if(!u.startsWith("session/")||l===null||typeof l!="object"){let m=new Error(`Method not found: ${u}`);throw m.code=I.MethodNotFound,m}let f=l.sessionId;if(typeof f!="string"){let m=new Error(`Method not found: ${u}`);throw m.code=I.MethodNotFound,m}c(f,u);let g=e.manager.get(f);if(!g){let m=new Error(`session ${f} not found`);throw m.code=I.SessionNotFound,m}return g.forwardRequest(u,l)})})}function di(n){return e=>{let t={agentId:e.agentId,version:e.version,source:e.source,phase:e.phase};"receivedBytes"in e&&(t.receivedBytes=e.receivedBytes),"totalBytes"in e&&(t.totalBytes=e.totalBytes),"packageSpec"in e&&(t.packageSpec=e.packageSpec),n.notify(Cs,t).catch(()=>{})}}function Tn(n){let e=n.availableModes();if(e.length===0)return;let t=e.map(r=>{let i={id:r.id,name:r.name??r.id};return r.description!==void 0&&(i.description=r.description),i});return{currentModeId:n.currentMode??e[0].id,availableModes:t}}function _n(n){let e=n.availableModels();if(e.length===0)return;let t=e.map(r=>{let i={modelId:r.modelId};return r.name!==void 0&&(i.name=r.name),r.description!==void 0&&(i.description=r.description),i});return{currentModelId:n.currentModel??e[0].modelId,availableModels:t}}function sc(n,e){if(!n||typeof n!="object")return{kind:"error",code:I.InvalidParams,message:"session/set_model requires params",logMessage:"session/set_model rejected: params not an object"};let t=n;if(typeof t.sessionId!="string")return{kind:"error",code:I.InvalidParams,message:"session/set_model requires string sessionId",logMessage:"session/set_model rejected: missing/non-string sessionId"};if(typeof t.modelId!="string")return{kind:"error",code:I.InvalidParams,message:"session/set_model requires string modelId",logMessage:`session/set_model rejected: missing/non-string modelId sessionId=${t.sessionId}`};let s=e.get(t.sessionId);if(!s)return{kind:"error",code:I.SessionNotFound,message:`session ${t.sessionId} not found`,logMessage:`session/set_model rejected: session not found sessionId=${t.sessionId}`};let r=s.availableModels(),i=Ee(t.modelId,r);if(i.kind==="none")return{kind:"ok",session:s,modelId:t.modelId,logMessage:`session/set_model passthrough (no availableModels) sessionId=${t.sessionId} modelId=${JSON.stringify(t.modelId)}`};if(i.kind==="exact")return{kind:"ok",session:s,modelId:t.modelId,logMessage:`session/set_model accepted sessionId=${t.sessionId} modelId=${JSON.stringify(t.modelId)}`};if(i.kind==="resolved")return{kind:"ok",session:s,modelId:i.modelId,logMessage:`session/set_model resolved sessionId=${t.sessionId} requested=${JSON.stringify(t.modelId)} \u2192 ${JSON.stringify(i.modelId)}`};let o=r.map(d=>d.modelId).join(", "),a=i.kind==="ambiguous"?`ambiguous (trailing-segment matches [${i.candidates.join(", ")}])`:"not in availableModels";return s.currentModel!==void 0&&s.currentModel.length>0?{kind:"no_op",session:s,sessionId:t.sessionId,currentModel:s.currentModel,logMessage:`session/set_model no_op (resyncing client) sessionId=${t.sessionId} requested=${JSON.stringify(t.modelId)} ${a} actual=${JSON.stringify(s.currentModel)} agentId=${s.agentId} known=[${o}]`}:{kind:"error",code:I.InvalidParams,message:`model "${t.modelId}" is ${a==="not in availableModels"?"not in this session's availableModels":a} (agent ${s.agentId}); known models: ${o}`,logMessage:`session/set_model rejected sessionId=${t.sessionId} modelId=${JSON.stringify(t.modelId)} ${a} agentId=${s.agentId} known=[${o}] (no current model to fall back to)`}}async function rc(n,e,t){let s=n??{},r=typeof s.sessionId=="string"?s.sessionId:void 0;if(!r)throw Object.assign(new Error("transformer/attach requires sessionId"),{code:I.InvalidParams});if(!t.transformers)throw Object.assign(new Error("transformer manager not configured"),{code:I.InternalError});let i=t.transformers.resolveChain([e])[0];if(!i)throw Object.assign(new Error(`transformer ${e} is not connected (call hydra-acp/transformer/initialize first)`),{code:I.InternalError});let o=t.manager.get(r);if(!o)throw Object.assign(new Error(`session ${r} not found`),{code:I.SessionNotFound});return o.addTransformer(i),{ok:!0}}function ic(n){let e={sessionId:n.hydraSessionId,upstreamSessionId:n.upstreamSessionId,cwd:n.cwd,title:n.title,agentId:n.agentId,currentModel:n.currentModel,currentUsage:n.currentUsage,forkedFromSessionId:n.forkedFromSessionId,forkedFromMessageId:n.forkedFromMessageId,originatingClient:n.originatingClient,interactive:n.interactive,updatedAt:n.createdAt??new Date().toISOString(),attachedClients:0,status:"cold",busy:!1,awaitingInput:!1},t={currentMode:n.currentMode,agentArgs:n.agentArgs,availableCommands:n.agentCommands,availableModes:n.agentModes,availableModels:n.agentModels};return{[Ue]:ut(e,t)}}function $n(n,e,t={}){let s=n.liveListEntry(e),r={clientId:t.clientId,currentMode:e.currentMode,agentArgs:e.agentArgs,availableCommands:e.mergedAvailableCommands(),availableModes:e.availableModes(),availableModels:e.availableModels(),turnStartedAt:e.turnStartedAt,agentCapabilities:e.agentCapabilities,queue:e.queueSnapshot()};return ct(e.agentMeta,ut(s,r))}function oc(){return{protocolVersion:le,agentInfo:{name:"hydra",version:Q},agentCapabilities:{promptCapabilities:{image:!0,audio:!0,embeddedContext:!0},mcpCapabilities:{http:!0,sse:!0},loadSession:!0,sessionCapabilities:{attach:{},list:{}}},authMethods:[{id:"bearer-token",description:"Bearer token presented at WS upgrade"}],_meta:ct(void 0,{prompt:{queueing:!0,cancelling:!0,updating:!0,amending:!0,pipelining:!1},agents:{list:!0,installProgress:!0}})}}function On(n,e){if(!(!e.transformers||!e.manager))for(let t of e.manager.defaultTransformers){let s=e.transformers.resolveChain([t])[0];s&&n.addTransformer(s)}}function Ht(n,e,t,s,r){return{clientId:r??`cli_${Fn(8)}`,connection:n,clientInfo:s}}function Bt(n){if(n.extensionMcp===void 0||n.mcpTokenRegistry===void 0||n.getDaemonOrigin===void 0)return;let e=n.extensionMcp.list();if(e.length===0)return;let t=ci(32).toString("hex"),s=n.mcpTokenRegistry.reserve(t),r=n.getDaemonOrigin(),i=e.map(a=>({name:a,type:"http",url:`${r}/mcp/${a}`,headers:[{name:"Authorization",value:`Bearer ${t}`}]})),o=n.mcpTokenRegistry;return{descriptors:i,bindToSession:a=>{s.complete(a),a.onClose(()=>{o.unbind(t)})},abandon:a=>s.abandon(a)}}var Ut=class{byToken=new Map;reserve(e){if(this.byToken.has(e))throw new Error("mcp token already bound");let t,s,r=new Promise((o,a)=>{t=o,s=a});r.catch(()=>{});let i={session:void 0,sessionReady:r,disposers:[]};return this.byToken.set(e,i),{complete:o=>{i.session=o,t(o)},abandon:o=>{this.byToken.delete(e),s(o??new Error("mcp token reservation abandoned"))}}}bind(e,t){let{complete:s}=this.reserve(e);s(t)}lookup(e){return this.byToken.get(e)}addDisposer(e,t){let s=this.byToken.get(e);s!==void 0&&s.disposers.push(t)}async unbind(e){let t=this.byToken.get(e);if(t!==void 0){this.byToken.delete(e);for(let s of t.disposers)try{await s()}catch{}}}size(){return this.byToken.size}};import{randomUUID as ac}from"crypto";import{McpServer as dc}from"@modelcontextprotocol/sdk/server/mcp.js";import{StreamableHTTPServerTransport as cc}from"@modelcontextprotocol/sdk/server/streamableHttp.js";import{z as q}from"zod";var li="Bearer ";function Dt(n){let e=n.headers.authorization;if(typeof e!="string"||!e.startsWith(li))return;let t=e.slice(li.length).trim();return t.length>0?t:void 0}function uc(n){let e=new dc({name:"hydra-acp-stdin",version:"1.0.0"},{instructions:"Piped input from `hydra cat --stream` is exposed here as a byte stream. Use `tail` for the latest N bytes (good for finding the end of a log), `head` for the first N bytes (good for headers/preamble), `read` for windowed reads against an absolute byte cursor, `wait_for_more` to block until new bytes arrive past a cursor, and `info` for the current cursors/capacity/closed status. Byte payloads come back base64-encoded."});return e.registerTool("tail",{description:"Return the most recent `bytes` bytes of piped stdin (capped server-side, default 64 KiB max). `truncated:true` means older bytes existed but have been evicted from the ring.",inputSchema:{bytes:q.number().int().min(1).describe("How many trailing bytes to return.")}},async({bytes:t})=>{let s=n.streamTail(t);return{content:[{type:"text",text:JSON.stringify(s)}],structuredContent:s}}),e.registerTool("head",{description:"Return the first `bytes` bytes of piped stdin (capped server-side, default 64 KiB max). `truncated:true` means the head has already been evicted from the ring and the returned bytes start at the oldest still-resident cursor.",inputSchema:{bytes:q.number().int().min(1).describe("How many leading bytes to return.")}},async({bytes:t})=>{let s=n.streamHead(t);return{content:[{type:"text",text:JSON.stringify(s)}],structuredContent:s}}),e.registerTool("read",{description:"Read up to `max_bytes` bytes starting at absolute byte `cursor`. Returns `{bytes, nextCursor, gap?, eof?}` \u2014 `gap` is the number of bytes silently skipped because the ring had evicted them; `eof:true` means the producer closed and there is nothing left to read.",inputSchema:{cursor:q.number().int().min(0).describe("Absolute byte offset to start reading from. Use 0 to read from the very beginning (may produce a gap if old bytes have been evicted)."),max_bytes:q.number().int().min(1).optional().describe("Optional cap on how many bytes to return. Server caps at 64 KiB regardless."),wait_ms:q.number().int().min(0).optional().describe("If no bytes are available, block up to this many ms for more (capped server-side at 60_000).")}},async({cursor:t,max_bytes:s,wait_ms:r})=>{let i=await n.streamRead(t,s,r);return{content:[{type:"text",text:JSON.stringify(i)}],structuredContent:i}}),e.registerTool("wait_for_more",{description:"Block until bytes are available past `cursor`, the stream closes, or `timeout_ms` elapses. Returns one of {data, eof, timeout} plus the current `writeCursor`. Use this when you've consumed everything up to a cursor and want to wait for more without busy-polling.",inputSchema:{cursor:q.number().int().min(0).describe("The cursor you've already consumed up to."),timeout_ms:q.number().int().min(0).describe("Maximum ms to block (server caps at 60_000).")}},async({cursor:t,timeout_ms:s})=>{let r=await n.streamWaitFor(t,s),i=n.streamInfo(),o={outcome:r,writeCursor:i.writeCursor,closed:i.closed};return{content:[{type:"text",text:JSON.stringify(o)}],structuredContent:o}}),e.registerTool("grep",{description:"Scan piped stdin line-by-line and return lines matching `pattern`. Prefer this over `read` when the question is 'find lines that mention X' \u2014 it filters server-side so you don't pull and decode 64 KiB base64 windows. Returns `{matches: [{cursor, line, before?, after?}], truncated, nextCursor, gap?, scannedBytes, eof?}`. Lines come back as decoded UTF-8 strings (not base64). When `truncated:true`, re-call with `cursor: nextCursor` to resume.",inputSchema:{pattern:q.string().min(1).describe("Search pattern. Treated as a JavaScript regular expression by default (set `regex:false` for a literal substring match)."),regex:q.boolean().optional().describe("Default true. Pass false to treat `pattern` as a literal substring."),case_insensitive:q.boolean().optional().describe("Default false. Pass true for case-insensitive matching."),invert:q.boolean().optional().describe("Default false. Pass true to return lines that do NOT match the pattern."),max_matches:q.number().int().min(1).optional().describe("Default 100. Capped server-side at 1000."),max_bytes:q.number().int().min(1).optional().describe("Default 64 KiB output. Capped server-side at 256 KiB."),context_before:q.number().int().min(0).optional().describe("Default 0. Number of lines before each match to include (capped at 20)."),context_after:q.number().int().min(0).optional().describe("Default 0. Number of lines after each match to include (capped at 20)."),cursor:q.number().int().min(0).optional().describe("Optional absolute byte offset to start scanning from. Omit to scan from the oldest still-resident byte. Pass the `nextCursor` from a previous truncated call to resume.")}},async t=>{let s={pattern:t.pattern};t.regex!==void 0&&(s.regex=t.regex),t.case_insensitive!==void 0&&(s.caseInsensitive=t.case_insensitive),t.invert!==void 0&&(s.invert=t.invert),t.max_matches!==void 0&&(s.maxMatches=t.max_matches),t.max_bytes!==void 0&&(s.maxBytes=t.max_bytes),t.context_before!==void 0&&(s.contextBefore=t.context_before),t.context_after!==void 0&&(s.contextAfter=t.context_after),t.cursor!==void 0&&(s.cursor=t.cursor);let r=n.streamGrep(s),i=r;return{content:[{type:"text",text:JSON.stringify(r)}],structuredContent:i}}),e.registerTool("info",{description:"Report cursor / capacity / closed state of the stdin ring. Cheap; safe to call repeatedly.",inputSchema:{}},async()=>{let t=n.streamInfo();return{content:[{type:"text",text:JSON.stringify(t)}],structuredContent:t}}),e}var lc=1e4;function fi(n,e){let t=new Map;async function s(o,a){let d=t.get(o);if(d!==void 0)return d.transport;let c=uc(a),p=new cc({sessionIdGenerator:()=>ac()});await c.connect(p);let l={server:c,transport:p};return t.set(o,l),e.addDisposer(o,async()=>{t.delete(o);try{await p.close()}catch{}try{await c.close()}catch{}}),p}async function r(o,a){let d=Dt(o);if(d===void 0){a.code(401).send({error:"missing bearer token"});return}let c=e.lookup(d);if(c===void 0){a.code(404).send({error:"unknown stdin token"});return}let p;if(c.session!==void 0)p=c.session;else{let u,f=new Promise(m=>{u=setTimeout(()=>m(void 0),lc)}),g=await Promise.race([c.sessionReady.catch(()=>{}),f]);if(u!==void 0&&clearTimeout(u),g===void 0){a.code(503).send({error:"session not ready"});return}p=g}let l=await s(d,p);a.hijack(),await l.handleRequest(o.raw,a.raw,o.body)}let i={config:{skipAuth:!0}};n.post("/mcp/hydra-acp-stdin",i,async(o,a)=>{await r(o,a)}),n.get("/mcp/hydra-acp-stdin",i,async(o,a)=>{await r(o,a)}),n.delete("/mcp/hydra-acp-stdin",i,async(o,a)=>{await r(o,a)})}var Lt=class{byName=new Map;changeHandlers=[];register(e,t,s,r){this.byName.set(e,{connection:t,instructions:s,tools:[...r]}),this.fireChanged(e,"register")}clear(e){this.byName.delete(e)&&this.fireChanged(e,"clear")}lookup(e){return this.byName.get(e)}list(){return Array.from(this.byName.keys())}onChange(e){return this.changeHandlers.push(e),()=>{let t=this.changeHandlers.indexOf(e);t>=0&&this.changeHandlers.splice(t,1)}}fireChanged(e,t){for(let s of this.changeHandlers)try{s(e,t)}catch{}}};import{StreamableHTTPServerTransport as wc}from"@modelcontextprotocol/sdk/server/streamableHttp.js";import{randomUUID as vc}from"crypto";import{Server as fc}from"@modelcontextprotocol/sdk/server/index.js";import{CallToolRequestSchema as pc,ListToolsRequestSchema as mc}from"@modelcontextprotocol/sdk/types.js";var gc=6e4;function pi(n,e,t,s={}){let r=s.invokeTimeoutMs??gc,i=typeof t=="function"?async()=>t():async()=>t,o=new fc({name:n,version:"1.0.0"},{capabilities:{tools:{listChanged:!1}},...e.instructions!==void 0?{instructions:e.instructions}:{}}),a=new Map(e.tools.map(d=>[d.name,d]));return o.setRequestHandler(mc,async()=>({tools:e.tools.map(d=>({name:d.name,description:d.description,inputSchema:d.inputSchema,...d.outputSchema!==void 0?{outputSchema:d.outputSchema}:{}}))})),o.setRequestHandler(pc,async d=>{let c=d.params.name;if(!a.has(c))return qt(`unknown tool: ${c}`);try{let p=await i(),l=await hc(e.connection,n,c,d.params.arguments??{},p,r);return yc(l,c)}catch(p){return qt(p instanceof Error?p.message:String(p))}}),o}async function hc(n,e,t,s,r,i){let o,a=new Promise((d,c)=>{o=setTimeout(()=>c(new Error(`extension timeout after ${i}ms`)),i)});try{return await Promise.race([n.request("hydra-acp/mcp_tools/invoke",{server:e,tool:t,args:s,sessionId:r}),a])}finally{o!==void 0&&clearTimeout(o)}}function yc(n,e){if(n===null||typeof n!="object")return qt(`extension ${e} returned non-object`);let t=n;return Array.isArray(t.content)?t:qt(`extension ${e} omitted content array`)}function qt(n){return{content:[{type:"text",text:n}],isError:!0}}var bc=1e4;function mi(n,e,t,s={}){let r=new Map;async function i(p){try{await p.transport.close()}catch{}try{await p.server.close()}catch{}}function o(p){for(let l of r.values()){let u=l.get(p);u!==void 0&&(l.delete(p),i(u))}}t.onChange(p=>{o(p)});async function a(p,l){let u=r.get(p);u===void 0&&(u=new Map,r.set(p,u),e.addDisposer(p,async()=>{let S=r.get(p);if(S!==void 0){r.delete(p);for(let C of S.values())await i(C)}}));let f=u.get(l);if(f!==void 0)return f.transport;let g=t.lookup(l);if(g===void 0)return;let y=pi(l,g,async()=>{let S=e.lookup(p);if(S===void 0)throw new Error("mcp token no longer bound");if(S.session!==void 0)return S.session.sessionId;let C,k=new Promise(N=>{C=setTimeout(()=>N(void 0),bc)}),E=await Promise.race([S.sessionReady.catch(()=>{}),k]);if(C!==void 0&&clearTimeout(C),E===void 0)throw new Error("session not ready");return E.sessionId},s.buildOptions),v=new wc({sessionIdGenerator:()=>vc()});return await y.connect(v),u.set(l,{server:y,transport:v}),v}async function d(p,l){let u=Dt(p);if(u===void 0){l.code(401).send({error:"missing bearer token"});return}if(e.lookup(u)===void 0){l.code(404).send({error:"unknown mcp token"});return}let g=p.params.name,m=await a(u,g);if(m===void 0){l.code(404).send({error:`unknown mcp server: ${g}`});return}l.hijack(),await m.handleRequest(p.raw,l.raw,p.body)}let c={config:{skipAuth:!0}};n.post("/mcp/:name",c,async(p,l)=>{await d(p,l)}),n.get("/mcp/:name",c,async(p,l)=>{await d(p,l)}),n.delete("/mcp/:name",c,async(p,l)=>{await d(p,l)})}async function Ac(n,e){Cc(n);let t=n.daemon.tls?{key:await me.readFile(n.daemon.tls.key),cert:await me.readFile(n.daemon.tls.cert)}:void 0;await me.mkdir(w.home(),{recursive:!0});let{stream:s,fileStream:r}=await xc(n.daemon.logLevel),i=Ic({logger:{level:n.daemon.logLevel,stream:s},https:t??null,bodyLimit:256*1024*1024});await i.register(Sc,{options:{handleProtocols:Hn}}),Xt($=>{i.log.info($)}),tn($=>{i.log.info($)});let o=await Pt.load(),a=new Ft,d=new Ot,c=new $t([new Tt(e),new _t(o),d]),p=Tr({validator:c});i.addHook("onRequest",async($,ge)=>{$.routeOptions.config?.skipAuth||$.url==="/acp"||$.url?.startsWith("/acp?")||await p($,ge)});let l=setInterval(()=>{o.sweepExpired()},300*1e3);l.unref();let u=new Be(n,{onFetched:()=>{Ir(u,y)}});Cn($=>i.log.info($));let f={info:$=>i.log.info($),warn:$=>i.log.warn($)},g=$=>ae.spawn({...$,stderrTailBytes:n.daemon.agentStderrTailBytes,logger:f});Zn(n.compressToolContent);let m=new Rt,y=new Ge(u,g,void 0,{idleTimeoutMs:n.daemon.sessionIdleTimeoutSeconds*1e3,defaultModels:n.defaultModels,synopsisAgent:n.synopsisAgent,synopsisModel:n.synopsisModel,synopsisOnClose:n.synopsisOnClose,defaultTransformers:n.defaultTransformers,sessionHistoryMaxEntries:n.daemon.sessionHistoryMaxEntries,logger:f,npmRegistry:n.npmRegistry,extensionCommands:m,defaultCwd:n.defaultCwd}),v=new Ct(Wn(n),void 0,{tokenRegistry:d}),S=new Mt(Qn(n),void 0,{tokenRegistry:d});Zr(i,Q,Ar(n)),Kr(i,y,{agentId:n.defaultAgent,cwd:n.defaultCwd,publicHost:n.daemon.publicHost,host:n.daemon.host,port:n.daemon.port}),Xr(i,u,y,{npmRegistry:n.npmRegistry}),ei(i,v),ti(i,S),ni(i,{defaultAgent:n.defaultAgent,defaultCwd:n.defaultCwd,defaultModels:{...n.defaultModels},...n.synopsisAgent!==void 0?{synopsisAgent:n.synopsisAgent}:{},...n.synopsisModel!==void 0?{synopsisModel:n.synopsisModel}:{},synopsisOnClose:n.synopsisOnClose,defaultTransformers:[...n.defaultTransformers]}),ai(i,{store:o,rateLimiter:a});let C=new Ut,k=new Lt;fi(i,C),mi(i,C,k);let E,N=()=>{if(E!==void 0)return E;let $=i.server.address(),ge=$&&typeof $=="object"?$.port:n.daemon.port;return E=`${n.daemon.tls?"https":"http"}://${n.daemon.host}:${ge}`,E};ui(i,{validator:c,manager:y,defaultAgent:n.defaultAgent,processRegistry:d,onExtensionVersion:($,ge)=>v.reportVersion($,ge),onTransformerVersion:($,ge)=>S.reportVersion($,ge),transformers:S,extensionCommands:m,mcpTokenRegistry:C,extensionMcp:k,getDaemonOrigin:N,registry:u}),await i.listen({host:n.daemon.host,port:n.daemon.port});let V=i.server.address(),W=V&&typeof V=="object"?V.port:n.daemon.port;await me.mkdir(w.home(),{recursive:!0}),await me.writeFile(w.pidFile(),JSON.stringify({pid:process.pid,host:n.daemon.host,port:W,startedAt:new Date().toISOString()})+`
112
+ `,{encoding:"utf8",mode:384});let M=n.daemon.tls?"https":"http",P=n.daemon.tls?"wss":"ws",B={daemonUrl:`${M}://${n.daemon.host}:${W}`,daemonHost:n.daemon.host,daemonPort:W,serviceToken:e,daemonWsUrl:`${P}://${n.daemon.host}:${W}/acp`,hydraHome:w.home()};v.setContext(B),S.setContext(B),await v.start(),await S.start(),y.resurrectPendingQueues().catch($=>{i.log.warn(`queue replay scan failed: ${$.message}`)});let O=n.daemon.agentSyncIntervalMinutes*60*1e3,U=O>0?Sr({registry:u,manager:y,intervalMs:O,logger:f}):void 0,D=n.daemon.sessionGcIntervalMinutes*60*1e3,je=n.daemon.sessionGcMaxAgeDays*24*60*60*1e3,Nn=D>0?kr({manager:y,intervalMs:D,maxAgeMs:je,logger:f}):void 0;return{app:i,manager:y,registry:u,extensions:v,transformers:S,mcpTokenRegistry:C,extensionMcp:k,processRegistry:d,shutdown:async()=>{Nn&&Nn(),U&&U(),clearInterval(l),await o.flush(),await v.stop(),await S.stop(),await y.closeAll(),await y.flushSynopsis(3e4),await y.shutdownSynopsis(),await y.flushMetaWrites(),await y.flushHistoryWrites(),Xt(null),tn(null),Cn(null),await i.close();try{hi.unlinkSync(w.pidFile())}catch{}try{r.flushSync()}catch{}}}}async function xc(n){let e=await kc({file:w.logFile(),size:"10m",frequency:"daily",mkdir:!0,symlink:!0}),t=gi.destination(2);return{stream:gi.multistream([{stream:e,level:n},{stream:t,level:n}]),fileStream:e}}function Cc(n){let e=n.daemon.host;if(!(e==="127.0.0.1"||e==="::1"||e==="localhost"||e==="[::1]")&&!n.daemon.tls)throw new Error(`Refusing to bind to non-loopback host ${e} without TLS configured.`)}export{ae as AgentInstance,ye as JsonRpcConnection,Be as Registry,we as Session,Ge as SessionManager,Oi as defaultConfig,vi as ensureServiceToken,rt as generateServiceToken,Gn as loadConfig,wi as loadServiceToken,rn as ndjsonStreamFromStdio,w as paths,oe as planSpawn,Ac as startDaemon,$i as writeConfig,Pn as wsToMessageStream};