@hydra-acp/cli 0.1.64 → 0.1.66
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli.js +313 -303
- package/dist/index.d.ts +1 -1
- package/dist/index.js +15 -15
- package/package.json +1 -1
package/dist/index.js
CHANGED
|
@@ -1,13 +1,13 @@
|
|
|
1
|
-
import*as li from"fs";import*as pe from"fs/promises";import
|
|
1
|
+
import*as li from"fs";import*as pe from"fs/promises";import pc from"fastify";import mc from"@fastify/websocket";var On="acp.v1";function Fn(n,e){return n.has(On)?On:!1}import ui from"pino";import gc from"pino-roll";import*as Bn from"fs/promises";import{homedir as qt}from"os";import{z as b}from"zod";import*as R from"path";import*as Lt from"os";var fi="HYDRA_ACP_HOME";function Se(n){let e=Lt.homedir();return e?n===e?"~":n.startsWith(e+"/")?"~"+n.slice(e.length):n:n}function P(){let n=process.env[fi];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(Lt.homedir(),".hydra-acp")}var w={home:P,config:()=>R.join(P(),"config.json"),authToken:()=>R.join(P(),"auth-token"),remotes:()=>R.join(P(),"remotes.json"),pidFile:()=>R.join(P(),"daemon.pid"),logFile:()=>R.join(P(),"daemon.log"),currentLogFile:()=>R.join(P(),"current.log"),registryCache:()=>R.join(P(),"registry.json"),agentsDir:()=>R.join(P(),"agents"),agentLogFile:n=>R.join(P(),"agents","logs",`${n}.log`),agentInstallDir:(n,e,t)=>R.join(P(),"agents",e,n,t),agentNpmInstallDir:(n,e,t)=>R.join(P(),"agents",e,n,t,`node${process.versions.modules}`),sessionsDir:()=>R.join(P(),"sessions"),sessionDir:n=>R.join(P(),"sessions",n),sessionFile:n=>R.join(P(),"sessions",n,"meta.json"),historyFile:n=>R.join(P(),"sessions",n,"history.jsonl"),toolsDir:n=>R.join(P(),"sessions",n,"tools"),toolBlobFile:(n,e)=>R.join(P(),"sessions",n,"tools",e),queueFile:n=>R.join(P(),"sessions",n,"queue.ndjson"),tombstonesDir:()=>R.join(P(),"sessions",".tombstones"),tombstoneAgentDir:n=>R.join(P(),"sessions",".tombstones",encodeURIComponent(n)),tombstoneFile:(n,e)=>R.join(P(),"sessions",".tombstones",encodeURIComponent(n),encodeURIComponent(e)),extensionsDir:()=>R.join(P(),"extensions"),extensionLogFile:n=>R.join(P(),"extensions",`${n}.log`),extensionPidFile:n=>R.join(P(),"extensions",`${n}.pid`),transformersDir:()=>R.join(P(),"transformers"),transformerLogFile:n=>R.join(P(),"transformers",`${n}.log`),transformerPidFile:n=>R.join(P(),"transformers",`${n}.pid`),transformerState:(n,e)=>R.join(P(),"sessions",n,"transformer-state",e),tuiHistoryFile:n=>R.join(P(),"sessions",n,"prompt-history"),globalTuiHistoryFile:()=>R.join(P(),"prompt-history"),tuiLogFile:()=>R.join(P(),"tui.log"),shimWireLogFile:()=>R.join(P(),"shim-wire.log")};import*as Ae from"fs/promises";function st(){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 Nn(){try{let e=(await Ae.readFile(w.authToken(),"utf8")).trim();return e.length>0?e:void 0}catch(n){if(n.code==="ENOENT")return;throw n}}async function pi(){let n=await Nn();if(!n)throw new Error(`No service token found at ${w.authToken()}. Run \`hydra-acp init\` to create one.`);return n}async function Dt(n){await Ae.mkdir(w.home(),{recursive:!0}),await Ae.writeFile(w.authToken(),n+`
|
|
2
2
|
`,{encoding:"utf8",mode:384})}async function mi(){let n=await Nn();if(n)return n;let e=st();return await Dt(e),process.stderr.write(`hydra-acp: initialized ${w.authToken()} with a fresh service token.
|
|
3
3
|
`),e}import*as ee from"fs/promises";import*as jn from"fs";import{randomBytes as gi}from"crypto";async function K(n,e,t={}){let r=(t.pretty??!0?JSON.stringify(e,null,2):JSON.stringify(e))+`
|
|
4
|
-
`;await hi(n,r,t)}async function hi(n,e,t={}){let s=yi(n);await ee.mkdir(s,{recursive:!0});let r=`${n}.tmp-${process.pid}-${wi()}`;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{jn.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 yi(n){let e=n.lastIndexOf("/");return e<=0?".":n.slice(0,e)}function wi(){return gi(4).toString("hex")}var Un="https://cdn.agentclientprotocol.com/registry/v1/latest/registry.json",vi=b.object({cert:b.string(),key:b.string()}),Ln=55514,bi=b.object({host:b.string().default("127.0.0.1"),port:b.number().int().positive().default(Ln),logLevel:b.enum(["debug","info","warn","error"]).default("info"),tls:vi.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)}),Ii=b.object({url:b.string().url().default(Un),ttlHours:b.number().positive().default(24),pinned:b.boolean().default(!1)}),Si=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()}),Ai=b.object({packageSpec:b.string().optional()}),ki=b.object({repaintThrottleMs:b.number().int().nonnegative().default(1e3),maxScrollbackLines:b.number().int().positive().default(1e4),mouse:b.boolean().default(!1),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()}),
|
|
5
|
-
`)}async function Wn(){return await Mi(),Dn.parse(await zn())}async function Ri(n){await K(w.config(),n,{mode:384})}function Ei(){return Dn.parse({})}function
|
|
4
|
+
`;await hi(n,r,t)}async function hi(n,e,t={}){let s=yi(n);await ee.mkdir(s,{recursive:!0});let r=`${n}.tmp-${process.pid}-${wi()}`;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{jn.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 yi(n){let e=n.lastIndexOf("/");return e<=0?".":n.slice(0,e)}function wi(){return gi(4).toString("hex")}var Un="https://cdn.agentclientprotocol.com/registry/v1/latest/registry.json",vi=b.object({cert:b.string(),key:b.string()}),Ln=55514,bi=b.object({host:b.string().default("127.0.0.1"),port:b.number().int().positive().default(Ln),logLevel:b.enum(["debug","info","warn","error"]).default("info"),tls:vi.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)}),Ii=b.object({url:b.string().url().default(Un),ttlHours:b.number().positive().default(24),pinned:b.boolean().default(!1)}),Si=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()}),Ai=b.object({packageSpec:b.string().optional()}),ki=b.object({repaintThrottleMs:b.number().int().nonnegative().default(1e3),maxScrollbackLines:b.number().int().positive().default(1e4),mouse:b.boolean().default(!1),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()}),Hn=b.string().min(1).regex(/^[A-Za-z0-9._-]+$/,"extension name must be filename-safe"),Ci=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)}),xi=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)}),Dn=b.object({daemon:bi.default({}),registry:Ii.default({url:Un,ttlHours:24,pinned:!1}),agents:b.record(b.string(),Si).default({}),agentOverrides:b.record(b.string(),Ai).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(Hn,Ci).default({}),transformers:b.record(Hn,xi).default({}),defaultTransformers:b.array(b.string()).default([]),npmRegistry:b.string().url().optional(),tui:ki.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"})});function qn(n){return Object.entries(n.extensions).map(([e,t])=>({name:e,...t}))}function Jn(n){return Object.entries(n.transformers).map(([e,t])=>({name:e,...t}))}async function zn(){return await te(w.config())??{}}async function Mi(){let n=await zn(),e=n.daemon,t=e&&typeof e.authToken=="string"?e.authToken:void 0;if(!t)return;let s=!1;try{await Bn.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 Dt(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 Wn(){return await Mi(),Dn.parse(await zn())}async function Ri(n){await K(w.config(),n,{mode:384})}function Ei(){return Dn.parse({})}function He(n){return n==="~"||n==="$HOME"?qt():n.startsWith("~/")?qt()+n.slice(1):n.startsWith("$HOME/")?qt()+n.slice(5):n}import*as U from"fs/promises";import{createHash as Pi}from"crypto";import{gzip as Ti,gunzip as _i}from"zlib";import{promisify as Qn}from"util";var Vn=Qn(Ti),$i=Qn(_i),Gn=/^[A-Za-z0-9_-]+$/,Oi=/^[a-f0-9]{64}$/,Jt=!0;function Yn(n){Jt=n}function rt(n,e){return Gn.test(n)?e===void 0||Oi.test(e):!1}function it(n,e){return`${w.toolBlobFile(n,e)}.gz`}async function zt(n,e){if(!rt(n))return null;let t=Pi("sha256").update(e,"utf8").digest("hex"),s=it(n,t),r=w.toolBlobFile(n,t);for(let a of[s,r])try{return await U.access(a),t}catch{}await U.mkdir(w.toolsDir(n),{recursive:!0});let i=Jt?s:r,o=Jt?await Vn(Buffer.from(e,"utf8")):Buffer.from(e,"utf8");return await U.writeFile(i,o,{mode:384,flag:"wx"}).catch(async a=>{if(a.code!=="EEXIST")throw a}),t}async function ot(n,e){if(!rt(n,e))return null;try{let t=await U.readFile(it(n,e));return(await $i(t)).toString("utf8")}catch{}try{return await U.readFile(w.toolBlobFile(n,e),"utf8")}catch{return null}}async function Kn(n,e){if(!rt(n,e))return null;try{return await U.readFile(it(n,e))}catch{}try{let t=await U.readFile(w.toolBlobFile(n,e));return await Vn(t)}catch{return null}}async function Xn(n,e,t){if(!rt(n,e))return;let s=it(n,e);try{await U.access(s);return}catch{}await U.mkdir(w.toolsDir(n),{recursive:!0}),await U.writeFile(s,t,{mode:384,flag:"wx"}).catch(r=>{if(r.code!=="EEXIST")throw r})}async function Zn(n){Gn.test(n)&&await U.rm(w.toolsDir(n),{recursive:!0,force:!0}).catch(()=>{})}import*as us from"fs/promises";import*as at from"path";import{z as C}from"zod";import*as ts from"fs";import*as D from"fs/promises";import*as ue from"path";import{spawn as ns}from"child_process";import{Readable as Fi}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 Vt(n,e=ne()){if(e)return n[e]}var ke=n=>{process.stderr.write(n+`
|
|
6
6
|
`)};function Gt(n){ke=n??(e=>process.stderr.write(e+`
|
|
7
|
-
`))}async function ss(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 Qt(s))return s;if(await Ni({agentId:n.agentId,version:n.version,archiveUrl:n.target.archive,installDir:t,onProgress:n.onProgress}),!await Qt(s))throw new Error(`Agent ${n.agentId}: extracted archive did not contain ${n.target.cmd} (looked in ${t})`);return process.platform!=="win32"&&await D.chmod(s,493).catch(()=>{}),s}async function Ni(n){await D.mkdir(ue.dirname(n.installDir),{recursive:!0});let e=await D.mkdtemp(`${n.installDir}.partial-`);try{ke(`hydra-acp: downloading ${n.agentId} from ${n.archiveUrl}`);let t=await ji({url:n.archiveUrl,dir:e,agentId:n.agentId,version:n.version,onProgress:n.onProgress});ke(`hydra-acp: extracting ${n.agentId}`),Ce(n.onProgress,{phase:"extract",agentId:n.agentId,version:n.version}),await
|
|
7
|
+
`))}async function ss(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 Qt(s))return s;if(await Ni({agentId:n.agentId,version:n.version,archiveUrl:n.target.archive,installDir:t,onProgress:n.onProgress}),!await Qt(s))throw new Error(`Agent ${n.agentId}: extracted archive did not contain ${n.target.cmd} (looked in ${t})`);return process.platform!=="win32"&&await D.chmod(s,493).catch(()=>{}),s}async function Ni(n){await D.mkdir(ue.dirname(n.installDir),{recursive:!0});let e=await D.mkdtemp(`${n.installDir}.partial-`);try{ke(`hydra-acp: downloading ${n.agentId} from ${n.archiveUrl}`);let t=await ji({url:n.archiveUrl,dir:e,agentId:n.agentId,version:n.version,onProgress:n.onProgress});ke(`hydra-acp: extracting ${n.agentId}`),Ce(n.onProgress,{phase:"extract",agentId:n.agentId,version:n.version}),await Bi(t,e),await D.unlink(t).catch(()=>{});try{await D.rename(e,n.installDir)}catch(s){let r=s;if((r.code==="EEXIST"||r.code==="ENOTEMPTY")&&await Qt(n.installDir)){await D.rm(e,{recursive:!0,force:!0}).catch(()=>{}),Ce(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version});return}throw s}ke(`hydra-acp: installed ${n.agentId} to ${n.installDir}`),Ce(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version})}catch(t){throw await D.rm(e,{recursive:!0,force:!0}).catch(()=>{}),t}}function Ce(n,e){if(n)try{n(e)}catch{}}async function ji(n){let e=Hi(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=ts.createWriteStream(t),o=Fi.fromWeb(s.body);Ce(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,Ce(n.onProgress,{phase:"download_progress",agentId:n.agentId,version:n.version,receivedBytes:a,totalBytes:r})),f-d>=p&&(d=f,ke(es(n.agentId,a,r)))}),await new Promise((u,f)=>{o.on("error",f),i.on("error",f),i.on("finish",()=>u()),o.pipe(i)}),ke(es(n.agentId,a,r,!0)),Ce(n.onProgress,{phase:"download_done",agentId:n.agentId,version:n.version,receivedBytes:a,totalBytes:r}),t}function es(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 Hi(n){let e=new URL(n);return ue.posix.basename(e.pathname)||"archive"}async function Bi(n,e){let t=n.toLowerCase();if(t.endsWith(".tar.gz")||t.endsWith(".tgz")||t.endsWith(".tar")){await Wt("tar",["-xf",n,"-C",e]);return}if(t.endsWith(".zip")){if(await Ui("unzip")){await Wt("unzip",["-q",n,"-d",e]);return}await Wt("tar",["-xf",n,"-C",e]);return}throw new Error(`Unsupported archive format: ${n}`)}function Wt(n,e){return new Promise((t,s)=>{let r=ns(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 Ui(n){return new Promise(e=>{let t=process.platform==="win32"?"where":"which",s=ns(t,[n],{stdio:"ignore"});s.on("error",()=>e(!1)),s.on("exit",r=>e(r===0))})}async function Qt(n){try{return await D.access(n),!0}catch{return!1}}import*as Q from"fs/promises";import*as se from"path";import{spawn as Li}from"child_process";var Kt=n=>{process.stderr.write(n+`
|
|
8
8
|
`)};function Xt(n){Kt=n??(e=>process.stderr.write(e+`
|
|
9
9
|
`))}async function is(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=Wi(n.packageSpec),r=Qi(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 rs(i);if(l)return l.binPath}await Di({agentId:n.agentId,version:n.version,packageSpec:n.packageSpec,installDir:t,registry:n.registry,onProgress:n.onProgress});let a=await rs(i);if(a)return a.binPath;let d=await as(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 Di(n){await Q.mkdir(se.dirname(n.installDir),{recursive:!0});let e=await Q.mkdtemp(`${n.installDir}.partial-`);try{Kt(`hydra-acp: installing ${n.packageSpec} for ${n.agentId} into ${e}`),Yt(n.onProgress,{phase:"install_start",agentId:n.agentId,version:n.version,packageSpec:n.packageSpec}),await zi({packageSpec:n.packageSpec,cwd:e,registry:n.registry});try{await Q.rename(e,n.installDir)}catch(t){let s=t;if((s.code==="EEXIST"||s.code==="ENOTEMPTY")&&await ge(n.installDir)){await Q.rm(e,{recursive:!0,force:!0}).catch(()=>{}),Yt(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version});return}throw t}Kt(`hydra-acp: installed ${n.agentId} to ${n.installDir}`),Yt(n.onProgress,{phase:"installed",agentId:n.agentId,version:n.version})}catch(t){throw await Q.rm(e,{recursive:!0,force:!0}).catch(()=>{}),t}}function Yt(n,e){if(n)try{n(e)}catch{}}var qi=5,Ji=25;function zi(n){return os(n,0)}async function os(n,e){try{await new Promise((t,s)=>{let r=n.registry?["--registry",n.registry]:[],i;try{i=Li("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<qi)return await new Promise(r=>setTimeout(r,Ji*(e+1))),os(n,e+1);throw t}}function Wi(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 Qi(n){let e=n.lastIndexOf("/");return e===-1?n:n.slice(e+1)}async function as(n,e){let t=se.join(n,"node_modules",e,"package.json");try{let s=await Q.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 rs(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 as(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 Q.access(n),!0}catch{return!1}}var Vi=C.object({package:C.string(),bin:C.string().optional(),args:C.array(C.string()).optional(),env:C.record(C.string()).optional()}),xe=C.object({archive:C.string().url().optional(),cmd:C.string().optional(),args:C.array(C.string()).optional(),env:C.record(C.string()).optional()}),Gi=C.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()}),Yi=C.object({package:C.string(),args:C.array(C.string()).optional(),env:C.record(C.string()).optional()}),Ki=C.object({command:C.string(),args:C.array(C.string()).optional(),env:C.record(C.string()).optional()}),Xi=C.object({npx:Vi.optional(),binary:Gi.optional(),uvx:Yi.optional(),exec:Ki.optional()}),Zi=C.object({id:C.string(),name:C.string(),version:C.string().optional(),description:C.string().optional(),authors:C.array(C.string()).optional(),license:C.string().optional(),icon:C.string().optional(),repository:C.string().optional(),website:C.string().optional(),distribution:Xi}),ds=C.object({version:C.string(),agents:C.array(Zi),extensions:C.array(C.unknown()).optional()}),He=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=>Zt(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:eo(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=ds.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=ds.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 eo(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 Zt(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 en(o),source:s.has(o.id)?"local":"registry"})));return{version:t.version,fetchedAt:n.lastFetchedAt(),agents:i}}async function en(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=Vt(n.distribution.binary,e);if(s?.cmd){let r=at.resolve(w.agentInstallDir(n.id,e,t),s.cmd);if(await cs(r))return"yes"}}if(n.distribution.npx){let s=n.distribution.npx,r=s.bin??Zt(n)??s.package,i=w.agentNpmInstallDir(n.id,e,t),o=at.join(i,"node_modules",".bin",r);if(await cs(o))return"yes"}return!n.distribution.npx&&!n.distribution.binary&&n.distribution.uvx?"lazy":"no"}async function cs(n){try{return await us.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??Zt(n)??r.package,a=t.onInstallProgress;return{command:await is({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=Vt(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 ss({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 io}from"child_process";import*as lt from"fs";import*as ks 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},ls=h.object({protocolVersion:h.number().optional(),clientCapabilities:h.record(h.unknown()).optional(),clientInfo:h.object({name:h.string(),version:h.string().optional()}).optional()}),to=h.enum(["full","pending_only","none","after_message"]),fs=h.object({cwd:h.string(),mcpServers:h.array(h.unknown()).optional(),_meta:h.record(h.unknown()).optional()}),no=h.object({upstreamSessionId:h.string(),agentId:h.string(),cwd:h.string(),title:h.string().optional(),agentArgs:h.array(h.string()).optional()}),ps=h.object({sessionId:h.string(),historyPolicy:to.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 Le(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=no.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=hs.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 dt(n,e){return{...n??{},[Ue]:e}}var ms=h.object({sessionId:h.string()}),gs=h.object({cwd:h.string().optional(),cursor:h.string().optional()}),hs=h.object({used:h.number().optional(),size:h.number().optional(),costAmount:h.number().optional(),costCurrency:h.string().optional()}),Xc=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:hs.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(),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()}),so=h.object({sessionId:h.string(),cwd:h.string(),title:h.string().optional(),updatedAt:h.string().optional(),_meta:h.record(h.unknown()).optional()}),Zc=h.object({sessions:h.array(so),nextCursor:h.string().optional()});function ct(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),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 ys(n){let e={sessionId:n.sessionId,cwd:n.cwd,updatedAt:n.updatedAt,_meta:dt(n._meta,ct(n))};return n.title!==void 0&&(e.title=n.title),e}var ws=h.object({sessionId:h.string(),prompt:h.array(h.unknown()),_meta:h.record(h.unknown()).optional()}),ut=h.object({sessionId:h.string()}),vs=h.object({clientId:h.string(),name:h.string().optional(),version:h.string().optional()}),eu=h.object({sessionId:h.string(),messageId:h.string(),originator:vs,prompt:h.array(h.unknown()),position:h.number().int().nonnegative(),queueDepth:h.number().int().positive(),enqueuedAt:h.number()}),tu=h.object({sessionId:h.string(),messageId:h.string(),prompt:h.array(h.unknown())}),nu=h.object({sessionId:h.string(),messageId:h.string(),reason:h.enum(["started","cancelled","abandoned"])}),bs=h.object({sessionId:h.string(),messageId:h.string()}),su=h.object({cancelled:h.boolean(),reason:h.enum(["ok","not_found","already_running"])}),Is=h.object({sessionId:h.string(),messageId:h.string(),prompt:h.array(h.unknown())}),ru=h.object({updated:h.boolean(),reason:h.enum(["ok","not_found","already_running"])}),Ss=h.object({sessionId:h.string(),targetMessageId:h.string(),prompt:h.array(h.unknown()),replaceQueue:h.boolean().optional(),onTargetCompleted:h.enum(["reject","send_anyway"]).optional()}),iu=h.object({amended:h.boolean(),reason:h.enum(["ok","target_completed","target_cancelled","target_not_found"]),messageId:h.string().optional()}),ou=h.object({sessionId:h.string(),cancelledMessageId:h.string(),newMessageId:h.string(),prompt:h.array(h.unknown()),originator:vs,amendedAt:h.number()}),au=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()}),As="hydra-acp/agents/install_progress";function tn(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(`
|
|
10
|
+
stderr: ${p}`:`npm install ${n.packageSpec} failed (${c})`))})})}catch(t){if(t.code==="ETXTBSY"&&e<qi)return await new Promise(r=>setTimeout(r,Ji*(e+1))),os(n,e+1);throw t}}function Wi(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 Qi(n){let e=n.lastIndexOf("/");return e===-1?n:n.slice(e+1)}async function as(n,e){let t=se.join(n,"node_modules",e,"package.json");try{let s=await Q.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 rs(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 as(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 Q.access(n),!0}catch{return!1}}var Vi=C.object({package:C.string(),bin:C.string().optional(),args:C.array(C.string()).optional(),env:C.record(C.string()).optional()}),xe=C.object({archive:C.string().url().optional(),cmd:C.string().optional(),args:C.array(C.string()).optional(),env:C.record(C.string()).optional()}),Gi=C.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()}),Yi=C.object({package:C.string(),args:C.array(C.string()).optional(),env:C.record(C.string()).optional()}),Ki=C.object({command:C.string(),args:C.array(C.string()).optional(),env:C.record(C.string()).optional()}),Xi=C.object({npx:Vi.optional(),binary:Gi.optional(),uvx:Yi.optional(),exec:Ki.optional()}),Zi=C.object({id:C.string(),name:C.string(),version:C.string().optional(),description:C.string().optional(),authors:C.array(C.string()).optional(),license:C.string().optional(),icon:C.string().optional(),repository:C.string().optional(),website:C.string().optional(),distribution:Xi}),ds=C.object({version:C.string(),agents:C.array(Zi),extensions:C.array(C.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=>Zt(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:eo(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=ds.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=ds.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 eo(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 Zt(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 en(o),source:s.has(o.id)?"local":"registry"})));return{version:t.version,fetchedAt:n.lastFetchedAt(),agents:i}}async function en(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=Vt(n.distribution.binary,e);if(s?.cmd){let r=at.resolve(w.agentInstallDir(n.id,e,t),s.cmd);if(await cs(r))return"yes"}}if(n.distribution.npx){let s=n.distribution.npx,r=s.bin??Zt(n)??s.package,i=w.agentNpmInstallDir(n.id,e,t),o=at.join(i,"node_modules",".bin",r);if(await cs(o))return"yes"}return!n.distribution.npx&&!n.distribution.binary&&n.distribution.uvx?"lazy":"no"}async function cs(n){try{return await us.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??Zt(n)??r.package,a=t.onInstallProgress;return{command:await is({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=Vt(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 ss({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 io}from"child_process";import*as lt from"fs";import*as ks 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},ls=h.object({protocolVersion:h.number().optional(),clientCapabilities:h.record(h.unknown()).optional(),clientInfo:h.object({name:h.string(),version:h.string().optional()}).optional()}),to=h.enum(["full","pending_only","none","after_message"]),fs=h.object({cwd:h.string(),mcpServers:h.array(h.unknown()).optional(),_meta:h.record(h.unknown()).optional()}),no=h.object({upstreamSessionId:h.string(),agentId:h.string(),cwd:h.string(),title:h.string().optional(),agentArgs:h.array(h.string()).optional()}),ps=h.object({sessionId:h.string(),historyPolicy:to.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 Le(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=no.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=hs.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 dt(n,e){return{...n??{},[Ue]:e}}var ms=h.object({sessionId:h.string()}),gs=h.object({cwd:h.string().optional(),cursor:h.string().optional()}),hs=h.object({used:h.number().optional(),size:h.number().optional(),costAmount:h.number().optional(),costCurrency:h.string().optional()}),Zc=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:hs.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(),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()}),so=h.object({sessionId:h.string(),cwd:h.string(),title:h.string().optional(),updatedAt:h.string().optional(),_meta:h.record(h.unknown()).optional()}),eu=h.object({sessions:h.array(so),nextCursor:h.string().optional()});function ct(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),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 ys(n){let e={sessionId:n.sessionId,cwd:n.cwd,updatedAt:n.updatedAt,_meta:dt(n._meta,ct(n))};return n.title!==void 0&&(e.title=n.title),e}var ws=h.object({sessionId:h.string(),prompt:h.array(h.unknown()),_meta:h.record(h.unknown()).optional()}),ut=h.object({sessionId:h.string()}),vs=h.object({clientId:h.string(),name:h.string().optional(),version:h.string().optional()}),tu=h.object({sessionId:h.string(),messageId:h.string(),originator:vs,prompt:h.array(h.unknown()),position:h.number().int().nonnegative(),queueDepth:h.number().int().positive(),enqueuedAt:h.number()}),nu=h.object({sessionId:h.string(),messageId:h.string(),prompt:h.array(h.unknown())}),su=h.object({sessionId:h.string(),messageId:h.string(),reason:h.enum(["started","cancelled","abandoned"])}),bs=h.object({sessionId:h.string(),messageId:h.string()}),ru=h.object({cancelled:h.boolean(),reason:h.enum(["ok","not_found","already_running"])}),Is=h.object({sessionId:h.string(),messageId:h.string(),prompt:h.array(h.unknown())}),iu=h.object({updated:h.boolean(),reason:h.enum(["ok","not_found","already_running"])}),Ss=h.object({sessionId:h.string(),targetMessageId:h.string(),prompt:h.array(h.unknown()),replaceQueue:h.boolean().optional(),onTargetCompleted:h.enum(["reject","send_anyway"]).optional()}),ou=h.object({amended:h.boolean(),reason:h.enum(["ok","target_completed","target_cancelled","target_not_found"]),messageId:h.string().optional()}),au=h.object({sessionId:h.string(),cancelledMessageId:h.string(),newMessageId:h.string(),prompt:h.array(h.unknown()),originator:vs,amendedAt:h.number()}),du=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()}),As="hydra-acp/agents/install_progress";function tn(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
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
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 ro}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=ro(),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 oo=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??oo,this.killEscalationMs=e.killEscalationMs??ao,this.logger=e.logger,this.fileLog=co(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()} ---
|
|
@@ -16,10 +16,10 @@ stderr: ${p}`:`npm install ${n.packageSpec} failed (${c})`))})})}catch(t){if(t.c
|
|
|
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
17
|
stderr: ${t}`:e}static spawn(e){let t={...process.env,...e.plan.env,...e.extraEnv??{}},s=io(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)})}},ao=2e3;function co(n){try{let e=w.agentLogFile(n);lt.mkdirSync(ks.dirname(e),{recursive:!0});let t=lt.createWriteStream(e,{flags:"a"});return t.on("error",()=>{}),t}catch{return}}import*as be from"fs/promises";import*as St from"os";import*as Ve from"path";import{customAlphabet as Ia}from"nanoid";import{customAlphabet as _s}from"nanoid";import*as xs from"fs/promises";var uo=64*1024*1024,lo=1*1024*1024,Re=64*1024,fo=6e4,po=100,mo=1e3,go=64*1024,ho=256*1024,Cs=20;function yo(n){return n.replace(/[.*+?^${}()|[\]\\]/g,"\\$&")}var ft=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??uo,this.maxCapacityBytes<=0)throw new Error("capacityBytes must be > 0");this.currentCapacityBytes=Math.min(lo,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,fo));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(yo(e.pattern),d),p=e.invert??!1,l=Math.max(1,Math.min(e.maxMatches??po,mo)),u=Math.max(1,Math.min(e.maxBytes??go,ho)),f=Math.max(0,Math.min(e.contextBefore??0,Cs)),g=Math.max(0,Math.min(e.contextAfter??0,Cs)),m=[],y=[],v=[],S=0,E=!1,x=0,M=0,J=(_,A)=>{for(let
|
|
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)})}},ao=2e3;function co(n){try{let e=w.agentLogFile(n);lt.mkdirSync(ks.dirname(e),{recursive:!0});let t=lt.createWriteStream(e,{flags:"a"});return t.on("error",()=>{}),t}catch{return}}import*as be from"fs/promises";import*as St from"os";import*as Ve from"path";import{customAlphabet as Ia}from"nanoid";import{customAlphabet as _s}from"nanoid";import*as xs from"fs/promises";var uo=64*1024*1024,lo=1*1024*1024,Re=64*1024,fo=6e4,po=100,mo=1e3,go=64*1024,ho=256*1024,Cs=20;function yo(n){return n.replace(/[.*+?^${}()|[\]\\]/g,"\\$&")}var ft=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??uo,this.maxCapacityBytes<=0)throw new Error("capacityBytes must be > 0");this.currentCapacityBytes=Math.min(lo,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,fo));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(yo(e.pattern),d),p=e.invert??!1,l=Math.max(1,Math.min(e.maxMatches??po,mo)),u=Math.max(1,Math.min(e.maxBytes??go,ho)),f=Math.max(0,Math.min(e.contextBefore??0,Cs)),g=Math.max(0,Math.min(e.contextAfter??0,Cs)),m=[],y=[],v=[],S=0,E=!1,x=0,M=0,J=(_,A)=>{for(let B of v)B.remaining>0&&(B.match.after===void 0&&(B.match.after=[]),B.match.after.push({cursor:_,line:A}),B.remaining--,S+=A.length);for(;v.length>0&&v[0].remaining===0;)v.shift();if(c.test(A)!==p&&m.length<l){let B={cursor:_,line:A};if(f>0&&y.length>0){B.before=y.slice();for(let je of B.before)S+=je.line.length}S+=A.length,m.push(B),g>0&&v.push({match:B,remaining:g})}if(f>0)for(y.push({cursor:_,line:A});y.length>f;)y.shift();let F=m.length>=l&&v.length===0,Y=S>=u;return F||Y};for(let _=0;_<o.length;_++){if(o[_]!==10)continue;let A=o.subarray(x,_).toString("utf8"),$=r+x;if(x=_+1,M=x,J($,A)){E=!0;break}}if(!E&&x<o.length&&this.closed){let _=o.subarray(x).toString("utf8"),A=r+x;J(A,_)&&(E=!0),M=o.length}let G=Math.min(r+M,this.writeCursor),z={matches:m,truncated:E,nextCursor:G,scannedBytes:M};return i>0&&(z.gap=i),this.closed&&G>=this.writeCursor&&(z.eof=!0),z}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(()=>xs.appendFile(t,r)).catch(o=>{this.logWriteError?.(o)}),i&&!this.fileCapReached&&(this.fileCapReached=!0,this.onFileCapReached?.())}async drainFileWrites(){await this.fileWriteChain.catch(()=>{})}};var pt=[{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)"}],Iu=new Map(pt.map(n=>[n.verb,n]));function Ms(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=Ms(n),s=e.map(r=>r.modelId).filter(r=>Ms(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 Rs(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=nn(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=nn(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?vo(r,i.outIndex,wo(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)?bo(d,t.get(l)??[]):d;l!==void 0&&s.has(l)&&!Io(u)&&(u=So(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 nn(n){if(n.method!=="session/update")return;let t=n.params?.update;if(!(!t||typeof t!="object"||Array.isArray(t)))return t}function wo(n){if(!n||typeof n!="object")return"";let e=n;return typeof e.text=="string"?e.text:""}function vo(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 bo(n,e){let t=n.params??{},s=t.update??{};return{...n,params:{...t,update:{...s,content:e}}}}function Io(n){let t=nn(n)?.rawInput;return!!t&&typeof t=="object"&&!Array.isArray(t)&&Object.keys(t).length>0}function So(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 De(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 Es(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 Ps(n){let e=w.queueFile(n);await re.unlink(e).catch(()=>{})}import*as Ns from"fs/promises";var $s="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz",Os=_s($s,16),Ts=_s($s,16),de="hydra_session_";function Pe(){return`m_${Os()}`}function Fs(n){return n.startsWith(de)?n.slice(de.length):n}var Ao=1e3,sn=300*1e3,ko=64,we=class n{sessionId;cwd;agentId;agent;upstreamSessionId;agentMeta;agentCapabilities;agentArgs;parentSessionId;forkedFromSessionId;forkedFromMessageId;originatingClient;_interactive;get interactive(){return this._interactive}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=[];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;recentlyTerminal=new Map;streamBuffer;streamFilePath;constructor(e){this.sessionId=e.sessionId??`${de}${Os()}`,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.historyStore=e.historyStore,this.historyMaxEntries=e.historyMaxEntries??Ao,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){To(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_${Ts()}`,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))},sn);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=_o(i);if(o!==null){this.setAgentAdvertisedCommands(o);return}let a=Po(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 V(new Error("session is closed"),I.SessionNotFound);if(this.clients.has(e.clientId))throw V(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:Rs(o),r=await this.getHistorySnapshot(t.toolContent??"inline"),i=this.buildStateSnapshotReplay();if(e==="after_message"){let o=t.afterMessageId?on(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 V(new Error("client not attached"),I.SessionNotFound);if(this.closing)throw V(new Error("session is closing; new prompts cannot be accepted"),I.SessionClosing);let r=ye((t??{}).prompt).trim();if(r.startsWith("/hydra"))return this.handleSlashCommand(r);let i=Pe();if(this.maybeSeedTitleFromPrompt(t),this._firstPromptSeeded=!0,!(Le((t??{})._meta).ancillary===!0)&&this._interactive===void 0){this._interactive=!0;for(let a of this.interactiveHandlers)try{a(!0)}catch{}}return this.enqueueUserPrompt(s,t,i)}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=ye(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>ko;){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 V(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.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 V(new Error("client not attached"),I.SessionNotFound);this.lastCancelAt=Date.now(),await this.agent.connection.notify("session/cancel",{sessionId:this.upstreamSessionId})}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){let i=this.rewriteForAgent(t);for(let o=r;o<this.transformChain.length;o++){let a=this.transformChain[o];if(s.has(a.name))continue;let d=`request:${e}`;if(!a.intercepts.has(d))continue;let c=`t_${Ts()}`,p;try{p=await a.connection.request("hydra-acp/transformer/message",{token:c,phase:"request",method:e,direction:"client\u2192agent",sessionId:this.sessionId,envelope:i})}catch(u){this.logger?.warn(`transformer ${a.name} error on ${d}: ${u.message}`);continue}let l=p?.action??"continue";if(l==="stop")return p?.payload??rn(e);if(l==="processing"){let u=o,f=i,g=new Set(s);return new Promise(m=>{let y=setTimeout(()=>{this.pendingClaims.delete(c)&&(this.broadcastQueueNotification("hydra-acp/transformer/abandoned_request",{sessionId:this.sessionId,token:c,transformerName:a.name}),this.forwardRequest(e,f,new Set([...g,a.name]),u+1).then(m).catch(()=>m(rn(e))))},sn);typeof y.unref=="function"&&y.unref(),this.pendingClaims.set(c,{resolve:m,timer:y,transformerName:a.name,method:e,envelope:f,chainIdx:u,originatedBy:g,side:"request"})})}s.add(a.name)}return this.agent.connection.request(e,i)}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):sn,i=setTimeout(()=>{this.pendingClaims.delete(e)&&(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)):this.forwardRequest(s.method,s.envelope,new Set([...s.originatedBy,s.transformerName]),s.chainIdx+1).then(s.resolve).catch(()=>s.resolve(rn(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}await 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=ye((e??{}).prompt),r=an(s,200);r&&(this._firstPromptSeeded=!0,this.setTitle(r))}maybeApplyAgentModel(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="current_model_update")return!1;let r=ve(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=ve(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(Mo(this.agentAdvertisedCommands,e)){this.broadcastMergedCommands();return}this.agentAdvertisedCommands=e;for(let t of this.agentCommandsHandlers)try{t(e)}catch{}this.broadcastMergedCommands()}setAgentAdvertisedModes(e){if(Ro(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(",")}]`),Eo(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)}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=`hydra ${t} ${s.verb}`,o={name:s.argsHint?`${r} ${s.argsHint}`:r};s.description&&(o.description=s.description),e.push(o)}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){let s=e.slice(6).trim().match(/^(\S+)(?:\s+([\s\S]*))?$/),r=s?.[1]??"",i=(s?.[2]??"").trim();if(r==="")return{stopReason:"end_turn"};if(pt.some(d=>d.verb===r))switch(r){case"title":return this.runTitleCommand(i);case"agent":return this.runAgentCommand(i);case"kill":return this.runKillCommand();case"restart":return this.runRestartCommand();default:{let d=new Error(`no dispatcher for /hydra verb ${r}`);throw d.code=I.InternalError,d}}if(this.extensionCommands?.has(r))return this.runExtensionCommand(r,i);let o=pt.map(d=>d.verb);if(this.extensionCommands){let d=new Set;for(let{name:c}of this.extensionCommands.list())d.has(c)||(o.push(c),d.add(c))}let a=new Error(`unknown /hydra verb: ${r} (known: ${o.join(", ")})`);throw a.code=I.InvalidParams,a}runExtensionCommand(e,t){return this.enqueuePrompt(async()=>{let s=this.extensionCommands?.get(e);if(!s)return this.emitExtensionReply(`extension "${e}" is no longer connected`);let r=t.match(/^(\S+)(?:\s+([\s\S]*))?$/),i=r?.[1]??"",o=(r?.[2]??"").trim();if(i===""){let c=s.commands.map(p=>p.verb).join(", ");return this.emitExtensionReply(`/hydra ${e} requires a verb (known: ${c||"(none)"})`)}if(!s.commands.some(c=>c.verb===i)){let c=s.commands.map(p=>p.verb).join(", ");return this.emitExtensionReply(`unknown verb "${i}" for ${e} (known: ${c||"(none)"})`)}let a;try{a=await s.connection.request("hydra-acp/commands/invoke",{sessionId:this.sessionId,verb:i,args:o})}catch(c){return this.emitExtensionReply(`${e} ${i}: ${c.message}`)}let d=a&&typeof a=="object"&&typeof a.text=="string"?a.text:"";return d.length>0?this.emitExtensionReply(d):{stopReason:"end_turn"}})}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 Ps(n){let e=w.queueFile(n);await re.unlink(e).catch(()=>{})}import*as Ns from"fs/promises";var $s="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz",Os=_s($s,16),Ts=_s($s,16),de="hydra_session_";function Pe(){return`m_${Os()}`}function Fs(n){return n.startsWith(de)?n.slice(de.length):n}var Ao=1e3,sn=300*1e3,ko=64,we=class n{sessionId;cwd;agentId;agent;upstreamSessionId;agentMeta;agentCapabilities;agentArgs;parentSessionId;forkedFromSessionId;forkedFromMessageId;originatingClient;_interactive;get interactive(){return this._interactive}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=[];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;recentlyTerminal=new Map;streamBuffer;streamFilePath;constructor(e){this.sessionId=e.sessionId??`${de}${Os()}`,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.historyStore=e.historyStore,this.historyMaxEntries=e.historyMaxEntries??Ao,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){To(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_${Ts()}`,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))},sn);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=_o(i);if(o!==null){this.setAgentAdvertisedCommands(o);return}let a=Po(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 V(new Error("session is closed"),I.SessionNotFound);if(this.clients.has(e.clientId))throw V(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:Rs(o),r=await this.getHistorySnapshot(t.toolContent??"inline"),i=this.buildStateSnapshotReplay();if(e==="after_message"){let o=t.afterMessageId?on(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 V(new Error("client not attached"),I.SessionNotFound);if(this.closing)throw V(new Error("session is closing; new prompts cannot be accepted"),I.SessionClosing);let r=ye((t??{}).prompt).trim();if(r.startsWith("/hydra"))return this.handleSlashCommand(r);let i=Pe();if(this.maybeSeedTitleFromPrompt(t),this._firstPromptSeeded=!0,!(Le((t??{})._meta).ancillary===!0)&&this._interactive!==!0){this._interactive=!0;for(let a of this.interactiveHandlers)try{a(!0)}catch{}}return this.enqueueUserPrompt(s,t,i)}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=ye(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>ko;){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 V(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.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 V(new Error("client not attached"),I.SessionNotFound);this.lastCancelAt=Date.now(),await this.agent.connection.notify("session/cancel",{sessionId:this.upstreamSessionId})}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){let i=this.rewriteForAgent(t);for(let o=r;o<this.transformChain.length;o++){let a=this.transformChain[o];if(s.has(a.name))continue;let d=`request:${e}`;if(!a.intercepts.has(d))continue;let c=`t_${Ts()}`,p;try{p=await a.connection.request("hydra-acp/transformer/message",{token:c,phase:"request",method:e,direction:"client\u2192agent",sessionId:this.sessionId,envelope:i})}catch(u){this.logger?.warn(`transformer ${a.name} error on ${d}: ${u.message}`);continue}let l=p?.action??"continue";if(l==="stop")return p?.payload??rn(e);if(l==="processing"){let u=o,f=i,g=new Set(s);return new Promise(m=>{let y=setTimeout(()=>{this.pendingClaims.delete(c)&&(this.broadcastQueueNotification("hydra-acp/transformer/abandoned_request",{sessionId:this.sessionId,token:c,transformerName:a.name}),this.forwardRequest(e,f,new Set([...g,a.name]),u+1).then(m).catch(()=>m(rn(e))))},sn);typeof y.unref=="function"&&y.unref(),this.pendingClaims.set(c,{resolve:m,timer:y,transformerName:a.name,method:e,envelope:f,chainIdx:u,originatedBy:g,side:"request"})})}s.add(a.name)}return this.agent.connection.request(e,i)}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):sn,i=setTimeout(()=>{this.pendingClaims.delete(e)&&(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)):this.forwardRequest(s.method,s.envelope,new Set([...s.originatedBy,s.transformerName]),s.chainIdx+1).then(s.resolve).catch(()=>s.resolve(rn(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=ye((e??{}).prompt),r=an(s,200);r&&(this._firstPromptSeeded=!0,this.setTitle(r))}maybeApplyAgentModel(e){let s=(e??{}).update??{};if(s.sessionUpdate!=="current_model_update")return!1;let r=ve(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=ve(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(Mo(this.agentAdvertisedCommands,e)){this.broadcastMergedCommands();return}this.agentAdvertisedCommands=e;for(let t of this.agentCommandsHandlers)try{t(e)}catch{}this.broadcastMergedCommands()}setAgentAdvertisedModes(e){if(Ro(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(",")}]`),Eo(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)}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}`,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){let s=e.slice(6).trim().match(/^(\S+)(?:\s+([\s\S]*))?$/),r=s?.[1]??"",i=(s?.[2]??"").trim();if(r==="")return{stopReason:"end_turn"};if(pt.some(c=>c.verb===r))switch(r){case"title":return this.runTitleCommand(i);case"agent":return this.runAgentCommand(i);case"kill":return this.runKillCommand();case"restart":return this.runRestartCommand();default:{let c=new Error(`no dispatcher for /hydra verb ${r}`);throw c.code=I.InternalError,c}}if(this.extensionCommands?.has(r))return this.runExtensionCommand(r,i);let o=`hydra-acp-${r}`;if(this.extensionCommands?.has(o))return this.runExtensionCommand(o,i);let a=pt.map(c=>c.verb);if(this.extensionCommands){let c=new Set;for(let{name:p}of this.extensionCommands.list())c.has(p)||(a.push(p),c.add(p))}let d=new Error(`unknown /hydra verb: ${r} (known: ${a.join(", ")})`);throw d.code=I.InvalidParams,d}runExtensionCommand(e,t){return this.enqueuePrompt(async()=>{let s=this.extensionCommands?.get(e);if(!s)return this.emitExtensionReply(`extension "${e}" is no longer connected`);let r=t.match(/^(\S+)(?:\s+([\s\S]*))?$/),i=r?.[1]??"",o=(r?.[2]??"").trim();if(!s.commands.some(c=>c.verb===i)){let c=s.commands.map(p=>p.verb).join(", ");return this.emitExtensionReply(`/hydra ${e}${i?` ${i}`:""}: unknown verb (known: ${c||"(none)"})`)}let a;try{a=await s.connection.request("hydra-acp/commands/invoke",{sessionId:this.sessionId,verb:i,args:o})}catch(c){return this.emitExtensionReply(`${e} ${i}: ${c.message}`)}let d=a&&typeof a=="object"&&typeof a.text=="string"?a.text:"";return d.length>0?this.emitExtensionReply(d):{stopReason:"end_turn"}})}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(`
|
|
@@ -42,7 +42,7 @@ Available models:
|
|
|
42
42
|
${o}
|
|
43
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 V(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(async()=>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){if(!e)throw V(new Error("/hydra agent requires an agent id"),I.InvalidParams);if(e===this.agentId)throw V(new Error(`already on agent ${e}`),I.InvalidParams);if(!this.spawnReplacementAgent)throw V(new Error("agent switching not configured for this session"),I.InternalError);let t=this.spawnReplacementAgent;return this.enqueuePrompt(async()=>{let 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(){if(!this.spawnReplacementAgent)throw V(new Error("agent restart not configured for this session"),I.InternalError);return this.enqueuePrompt(async()=>(await this.respawnAgent(),{stopReason:"end_turn"}))}async forceCancel(){if(this.closed||this.closing)throw V(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=ye(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(`
|
|
44
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 ft(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),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(()=>Ps(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(()=>Ns.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=!xo(e,r),o=i?$o(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 handlePermissionRequest(e){let t=[...this.clients.values()];if(t.length===0)throw V(new Error("no clients attached to handle permission request"),I.PermissionDenied);let s=this.rewriteForClient(e),r=Oo(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=Fo({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};return e.clientInfo?.name&&(i.name=e.clientInfo.name),e.clientInfo?.version&&(i.version=e.clientInfo.version),new Promise((o,a)=>{let d={kind:"user",messageId:s,originator:i,clientId:e.clientId,prompt:r,enqueuedAt:Date.now(),cancelled:!1,resolve:o,reject:a};this.promptQueue.push(d),this.persistRewrite(),this.broadcastQueueAdded(d),this.drainQueue()})}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(()=>De(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=ye(e.prompt).replace(/\s+$/,"");if(!t.includes(`
|
|
45
|
-
`)&&(t==="/model"||t.startsWith("/model ")||t==="/mode"||t.startsWith("/mode ")||t==="/sessions"||t==="/help")){let r;return t==="/sessions"?r=await this.handleSessionsCommand():t==="/help"?r=await this.handleHelpCommand():t==="/mode"||t.startsWith("/mode ")?r=await this.handleModeCommand(t):r=await this.handleModelCommand(t),this.closed||this.broadcastTurnComplete(e.clientId,r,e.messageId,e.wasAmend),this.clearAmendIfMatches(e.messageId),r}let s;try{s=await this.agent.connection.request("session/prompt",{sessionId:this.upstreamSessionId,prompt:e.prompt})}catch(r){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),r}return this.closed||this.broadcastTurnComplete(e.clientId,s,e.messageId,e.wasAmend),this.clearAmendIfMatches(e.messageId),s}clearAmendIfMatches(e){this.amendInProgress?.cancelledMessageId===e&&(this.amendInProgress=void 0)}};function V(n,e){return n.code=e,n}var Co=new Set(["session_info_update","current_model_update","current_mode_update","available_commands_update","available_modes_update","usage_update","config_option_update"]);function xo(n,e){if(n!=="session/update")return!1;let s=(e??{}).update?.sessionUpdate;return typeof s=="string"&&Co.has(s)}function Mo(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 Ro(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 Eo(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 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.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 Po(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 To(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 _o(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 $o(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 on(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 Oo(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 Fo(n){let e=No(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=jo(n.resolver),t}function No(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 jo(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 ye(n){return typeof n=="string"?n:Array.isArray(n)?n.map(e=>e&&typeof e=="object"&&typeof e.text=="string"?e.text:"").join(""):""}function rn(n){return n==="session/prompt"?{stopReason:"stopped"}:{}}function an(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
|
|
45
|
+
`)&&(t==="/model"||t.startsWith("/model ")||t==="/mode"||t.startsWith("/mode ")||t==="/sessions"||t==="/help")){let r;return t==="/sessions"?r=await this.handleSessionsCommand():t==="/help"?r=await this.handleHelpCommand():t==="/mode"||t.startsWith("/mode ")?r=await this.handleModeCommand(t):r=await this.handleModelCommand(t),this.closed||this.broadcastTurnComplete(e.clientId,r,e.messageId,e.wasAmend),this.clearAmendIfMatches(e.messageId),r}let s;try{s=await this.agent.connection.request("session/prompt",{sessionId:this.upstreamSessionId,prompt:e.prompt})}catch(r){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),r}return this.closed||this.broadcastTurnComplete(e.clientId,s,e.messageId,e.wasAmend),this.clearAmendIfMatches(e.messageId),s}clearAmendIfMatches(e){this.amendInProgress?.cancelledMessageId===e&&(this.amendInProgress=void 0)}};function V(n,e){return n.code=e,n}var Co=new Set(["session_info_update","current_model_update","current_mode_update","available_commands_update","available_modes_update","usage_update","config_option_update"]);function xo(n,e){if(n!=="session/update")return!1;let s=(e??{}).update?.sessionUpdate;return typeof s=="string"&&Co.has(s)}function Mo(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 Ro(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 Eo(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 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.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 Po(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 To(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 _o(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 $o(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 on(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 Oo(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 Fo(n){let e=No(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=jo(n.resolver),t}function No(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 jo(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 ye(n){return typeof n=="string"?n:Array.isArray(n)?n.map(e=>e&&typeof e=="object"&&typeof e.text=="string"?e.text:"").join(""):""}function rn(n){return n==="session/prompt"?{stopReason:"stopped"}:{}}function an(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 Uo}from"nanoid";import{z as k}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()}),Ho=200,dn=`Reply with ONLY a JSON object with exactly these keys, no prose, no markdown, no code fences:
|
|
46
46
|
{
|
|
47
47
|
"title": "short summary, max 80 chars",
|
|
48
48
|
"synopsis": {
|
|
@@ -52,13 +52,13 @@ ${o}
|
|
|
52
52
|
"open_threads": ["work started but not finished"]
|
|
53
53
|
}
|
|
54
54
|
}
|
|
55
|
-
Use empty arrays/strings where a field doesn't apply.`;function
|
|
55
|
+
Use empty arrays/strings where a field doesn't apply.`;function Hs(n){let e=n.trim();if(e.length===0)return;let t=js(e);if(t===void 0){let i=e.indexOf("{"),o=e.lastIndexOf("}");if(i<0||o<=i||(t=js(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,Ho))}if(s.synopsis!==void 0&&s.synopsis!==null){let i=qe.safeParse(s.synopsis);i.success&&Bo(i.data)&&(r.synopsis=i.data)}if(!(r.title===void 0&&r.synopsis===void 0))return r}function js(n){try{return JSON.parse(n)}catch{return}}function Bo(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 Lo="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz",Do=Uo(Lo,16),qo="hydra_lineage_";function Je(){return`${qo}${Do()}`}var un=k.object({name:k.string(),description:k.string().optional()}),ln=k.object({id:k.string(),name:k.string().optional(),description:k.string().optional()}),Jo=k.object({modelId:k.string(),name:k.string().optional(),description:k.string().optional()}),fn=k.object({used:k.number().optional(),size:k.number().optional(),costAmount:k.number().optional(),costCurrency:k.string().optional(),cumulativeCost:k.number().optional()}),pn=k.object({name:k.string(),version:k.string().optional()}),zo=k.object({version:k.literal(1),sessionId:k.string(),lineageId:k.string().optional(),upstreamSessionId:k.string(),importedFromSessionId:k.string().optional(),importedFromUpstreamSessionId:k.string().optional(),importedFromMachine:k.string().optional(),agentId:k.string(),cwd:k.string(),title:k.string().optional(),synopsis:qe.optional(),summarizedThroughEntry:k.number().int().nonnegative().optional(),agentArgs:k.array(k.string()).optional(),currentModel:k.string().optional(),currentMode:k.string().optional(),currentUsage:fn.optional(),agentCommands:k.array(un).optional(),agentModes:k.array(ln).optional(),agentModels:k.array(Jo).optional(),pendingHistorySync:k.boolean().optional(),parentSessionId:k.string().optional(),forkedFromSessionId:k.string().optional(),forkedFromMessageId:k.string().optional(),originatingClient:pn.optional(),interactive:k.boolean().optional(),createdAt:k.string(),updatedAt:k.string()}),cn=/^[A-Za-z0-9_-]+$/;function Wo(n){if(!cn.test(n))throw new Error(`unsafe session id: ${n}`)}var mt=class{async write(e){Wo(e.sessionId);let t={version:1,...e};await K(w.sessionFile(e.sessionId),t,{mode:384})}async read(e){if(!cn.test(e))return;let t=await te(w.sessionFile(e));if(t!==void 0)try{return zo.parse(t)}catch{return}}async delete(e){if(cn.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 mn(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,createdAt:n.createdAt??e,updatedAt:n.updatedAt??e}}import*as ce from"fs/promises";import{z as ie}from"zod";var Qo=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()}),gt=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 Qo.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 Bs(n,e){return e===void 0?!1:n.upstreamUpdatedAt===void 0?!0:e>n.upstreamUpdatedAt}import*as Gs from"fs/promises";import{fileURLToPath as Vo}from"url";import*as ze from"path";import*as ht from"fs";function Go(){try{let n=ze.dirname(Vo(import.meta.url));for(let e=0;e<8;e+=1){let t=ze.join(n,"package.json");if(ht.existsSync(t)){let r=JSON.parse(ht.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 W=Go(),yt="hydra-acp-cat";var Us=`[older history truncated]
|
|
56
56
|
`,Yo=["file_path","path","command","pattern","query"];function Ls(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=ea(d.prompt).trim();p.length>0&&s.push(`User: ${p}`)}else if(c==="agent_message_chunk"){let p=ta(d.content);p.length>0&&(r+=p)}else c==="tool_call"?(i(),s.push(Ko(d))):c==="turn_complete"&&i()}return i(),sa(s.join(`
|
|
57
57
|
`),t)}function Ko(n){let e=Xo(n),t=Zo(n.rawInput);return t.length===0?`Tool: ${e}`:`Tool: ${e}(${t.join(", ")})`}function Xo(n){return typeof n.name=="string"&&n.name.length>0?n.name:typeof n.title=="string"&&n.title.length>0?n.title:"(unnamed)"}function Zo(n){if(!n||typeof n!="object"||Array.isArray(n))return[];let e=n,t=[];for(let s of Yo){let r=e[s];typeof r=="string"&&r.length>0&&t.push(`${s}=${na(r,200)}`)}return t}function ea(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 ta(n){if(!n||typeof n!="object")return"";let e=n.text;return typeof e=="string"?e:""}function na(n,e){return n.length<=e?n:n.slice(0,e-1)+"\u2026"}function sa(n,e){if(n.length<=e)return n;let t=n.split(`
|
|
58
58
|
`);for(;t.length>0;){let s=Us+t.join(`
|
|
59
59
|
`);if(s.length<=e)return s;t.shift()}return Us}var ra=12e4;async function qs(n){let e=n.timeoutMs??ra,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:W}}),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=oa(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=ia(g);m.length>0&&c.push(m)});let p=Ls(n.history,{maxChars:n.maxTranscriptChars}),l=p.length>0?`${p}
|
|
60
60
|
|
|
61
|
-
${dn}`:dn;await t.connection.request("session/prompt",{sessionId:d,prompt:[{type:"text",text:l}]});let u=c.join(""),f=
|
|
61
|
+
${dn}`:dn;await t.connection.request("session/prompt",{sessionId:d,prompt:[{type:"text",text:l}]});let u=c.join(""),f=Hs(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 ia(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 oa(n){let e=new Set;Ds(e,n.availableModels);let t=n.models;return t&&typeof t=="object"&&!Array.isArray(t)&&Ds(e,t.availableModels),e}function Ds(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 aa(n){let e=new Map;for(let t of Vs(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 Ws(n){return aa(n).map(e=>e.name)}function da(n){let e=new Map;for(let t of Vs(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 Qs(n){return da(n).map(e=>e.path)}function Vs(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:Js(i),paths:new Set},e.set(d,c)):c.toolName=Js(i);for(let p of zs(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 zs(i.rawInput,i.locations))a.paths.add(d)}return e}function Js(n){return typeof n.name=="string"&&n.name.length>0?n.name:typeof n.title=="string"&&n.title.length>0?n.title:"(unnamed)"}function zs(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 ca=2,wt=class{constructor(e){this.opts=e;this.maxConcurrent=e.maxConcurrent??ca}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 Gs.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=ua(p.synopsis,s);l&&la(l)?(await this.opts.persistSynopsis(e,l,s.length),this.opts.logger?.info(`synopsis: persisted sessionId=${e} title=${JSON.stringify(!!p.title)} fields=${fa(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 ua(n,e){let t=Qs(e),s=Ws(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 la(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 fa(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 q from"fs/promises";function Ys(n){return n==="summary"?"summary":"inline"}var gn=256;function Ks(n,e){return e!=="summary"?n:n.map(pa)}function pa(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(Xs));let o=r.rawOutput;if(o&&typeof o=="object"&&!Array.isArray(o)){let a=o,d={};a.error!==void 0&&(d.error=hn(a.error)),a.metadata!==void 0&&(d.metadata=a.metadata),i.rawOutput=d}return{...n,params:{...t,update:i}}}function ma(n){return!!n&&typeof n=="object"&&!Array.isArray(n)&&n.type==="diff"}function Xs(n){if(ma(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=hn(e.text)),typeof e.content=="string"?t.content=hn(e.content):e.content&&typeof e.content=="object"&&(t.content=Xs(e.content)),t}function hn(n){if(typeof n=="string"&&n.length>gn){let e=n.length-gn;return`${n.slice(0,gn)}\u2026[+${e} chars omitted from summary export]`}return n}var ga=2048;function Zs(n){let e=new Set,t=s=>{if(er(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 er(n){return!!n&&typeof n=="object"&&!Array.isArray(n)&&typeof n.__hydraBlob=="string"}function ha(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 vn(n,e){if(!ha(n))return n;let t=n.params,s=t.update,r=await yn(s,e);return{...n,params:{...t,update:r}}}async function yn(n,e){if(typeof n=="string"){if(n.length<=ga)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 yn(s,e));return t}if(n&&typeof n=="object"){let t={};for(let[s,r]of Object.entries(n))t[s]=await yn(r,e);return t}return n}async function tr(n,e){let t=n.params;if(!t||typeof t!="object"||Array.isArray(t))return n;let s=await wn(t,e);return s===t?n:{...n,params:s}}async function wn(n,e){if(er(n))return await e(n.__hydraBlob)??"";if(Array.isArray(n)){let t=!1,s=[];for(let r of n){let i=await wn(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 wn(i,e);o!==i&&(t=!0),s[r]=o}return t?s:n}return n}var We=/^[A-Za-z0-9_-]+$/,ya=1e3,vt=class{writeQueues=new Map;maxEntries;constructor(e={}){this.maxEntries=e.maxEntries??ya}async append(e,t){if(We.test(e))return this.enqueue(e,async()=>{await q.mkdir(w.sessionDir(e),{recursive:!0});let s=await vn(t,i=>zt(e,i)),r=JSON.stringify(s)+`
|
|
62
62
|
`;await q.appendFile(w.historyFile(e),r,{encoding:"utf8",mode:384})})}async rewrite(e,t){if(We.test(e))return this.enqueue(e,async()=>{await q.mkdir(w.sessionDir(e),{recursive:!0});let s=[];for(let i of t)s.push(await vn(i,o=>zt(e,o)));let r=s.length===0?"":s.map(i=>JSON.stringify(i)).join(`
|
|
63
63
|
`)+`
|
|
64
64
|
`;await q.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 q.readFile(w.historyFile(e),"utf8")}catch(o){if(o.code==="ENOENT")return;throw o}let r=s.split(`
|
|
@@ -68,7 +68,7 @@ ${dn}`:dn;await t.connection.request("session/prompt",{sessionId:d,prompt:[{type
|
|
|
68
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 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 ot(e,l);return d.set(l,f),f},p=[];for(let l of a)p.push(await tr(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 q.unlink(w.historyFile(e))}catch(t){if(t.code!=="ENOENT")throw t}await Zn(e);try{await q.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 nr}from"fs";import*as sr from"path";async function rr(n,e){await nr.mkdir(sr.dirname(n),{recursive:!0});let t=e.map(s=>JSON.stringify(s));await nr.writeFile(n,t.length>0?t.join(`
|
|
69
69
|
`)+`
|
|
70
70
|
`:"")}import{z as T}from"zod";var wa=T.object({method:T.string(),params:T.unknown(),recordedAt:T.number()}),va=T.object({sessionId:T.string(),lineageId:T.string(),upstreamSessionId:T.string().optional(),agentId:T.string(),cwd:T.string(),title:T.string().optional(),synopsis:qe.optional(),summarizedThroughEntry:T.number().int().nonnegative().optional(),currentModel:T.string().optional(),currentMode:T.string().optional(),currentUsage:fn.optional(),agentCommands:T.array(un).optional(),agentModes:T.array(ln).optional(),interactive:T.boolean().optional(),originatingClient:pn.optional(),createdAt:T.string(),updatedAt:T.string()}),ba=T.object({version:T.literal(1),exportedAt:T.string(),exportedFrom:T.object({hydraVersion:T.string(),machine:T.string(),hydraHost:T.string().optional()}),session:va,history:T.array(wa),promptHistory:T.array(T.string()).optional(),toolBlobs:T.record(T.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}:{},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 ir(n){return ba.parse(n)}var or=900*1e3,Sa="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz",bt=Ia(Sa,16),Ge=class n{constructor(e,t,s,r={}){this.registry=e;this.spawner=t??(i=>ae.spawn(i)),this.store=s??new mt,this.tombstones=r.tombstones??new gt,this.sessionHistoryMaxEntries=r.sessionHistoryMaxEntries??1e3,this.histories=new vt({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 wt({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_${bt()}`,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:W}})).agentCapabilities}catch(m){throw await i.kill().catch(()=>{}),m}let a;try{let m=xa(e.agentId,e.currentModel);a=await i.connection.request("session/load",{sessionId:e.upstreamSessionId,cwd:e.cwd,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=lr(a??{}),c=e.agentModes??dr(ur(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 fr({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=ar(a??{}),u=dr(cr(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 pr({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:[]}),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,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:[],onInstallProgress:e.onInstallProgress,model:e.currentModel}),r=e.agentModes??s.initialModes,i=await fr({agent:s.agent,upstreamSessionId:s.upstreamSessionId,persistedMode:e.currentMode,agentReportedMode:s.initialMode,advertisedModes:r,logger:this.logger}),o=e.agentModels??s.initialModels,a=await pr({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:[]}),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,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:Be(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:St.homedir(),plan:s}),i;try{i=await r.connection.request("initialize",{protocolVersion:le,clientCapabilities:{},clientInfo:{name:"hydra",version:W}})}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(Aa(f.cwd,p)){u+=1;continue}let m=await this.tombstones.read(e,f.sessionId).catch(()=>{});if(m){if(!Hs(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}${bt()}`,v=new Date().toISOString(),S=f.updatedAt??v,E={sessionId:y,lineageId:Je(),upstreamSessionId:f.sessionId,agentId:e,cwd:f.cwd,pendingHistorySync:!0,interactive:!0,createdAt:S,updatedAt:S};f.title!==void 0&&(E.title=f.title);let x=mn(E);await this.store.write(x),l.push({version:1,...x})}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:W}})).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=ar(a),p=cr(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=ur(a),f=lr(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=It({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.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:gr(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=ka(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 ot(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:Ca(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,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=ye(r.update.prompt),o=an(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 bn(o.sessionId),d=It({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,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 bn(o.sessionId),d=It(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,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 mr(e);if(i!=="references")return{record:r,history:o,promptHistory:a};let d={};for(let c of Zs(o)){let p=await Kn(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}${bt()}`;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=on(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=Ma(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 mr(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:W,machine:St.hostname()}),f=`${de}${bt()}`;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 Xn(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 rr(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,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 bn(e),r=It(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 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 Es(t.sessionId).catch(()=>[]);if(s.length===0)continue;let r=Date.now(),i=s.filter(d=>r-d.enqueuedAt<or),o=s.length-i.length;if(o>0&&(this.logger?.info(`queue replay: dropping ${o} stale prompt(s) for ${t.sessionId} (TTL ${or/1e3}s)`),await De(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 De(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 Aa(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 ka(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 mn({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:gr(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,createdAt:e?.createdAt??new Date(n.createdAt).toISOString()})}function gr(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 Ca(n){return n?{...n}:void 0}function xa(n,e){if(e&&n==="claude-acp")return{claudeCode:{options:{model:e}}}}function ar(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 dr(n){return n.length>0?n:void 0}function cr(n){let e=ve(n.availableModels);if(e.length>0)return e;let t=n.models;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=ve(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=ve(o.availableModels);if(a.length>0)return a}}let r=At(n,"model");if(r){let i=ve(r.options);if(i.length>0)return i}return[]}function ur(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 lr(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 fr(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 pr(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 Ma(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 mr(n){try{let e=await be.readFile(w.tuiHistoryFile(n),"utf8"),t=[];for(let s of e.split(`
|
|
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=lr(a??{}),c=e.agentModes??dr(ur(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 fr({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=ar(a??{}),u=dr(cr(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 pr({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:[]}),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,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:[],onInstallProgress:e.onInstallProgress,model:e.currentModel}),r=e.agentModes??s.initialModes,i=await fr({agent:s.agent,upstreamSessionId:s.upstreamSessionId,persistedMode:e.currentMode,agentReportedMode:s.initialMode,advertisedModes:r,logger:this.logger}),o=e.agentModels??s.initialModels,a=await pr({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:[]}),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,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:St.homedir(),plan:s}),i;try{i=await r.connection.request("initialize",{protocolVersion:le,clientCapabilities:{},clientInfo:{name:"hydra",version:W}})}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(Aa(f.cwd,p)){u+=1;continue}let m=await this.tombstones.read(e,f.sessionId).catch(()=>{});if(m){if(!Bs(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}${bt()}`,v=new Date().toISOString(),S=f.updatedAt??v,E={sessionId:y,lineageId:Je(),upstreamSessionId:f.sessionId,agentId:e,cwd:f.cwd,pendingHistorySync:!0,interactive:!0,createdAt:S,updatedAt:S};f.title!==void 0&&(E.title=f.title);let x=mn(E);await this.store.write(x),l.push({version:1,...x})}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:W}})).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=ar(a),p=cr(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=ur(a),f=lr(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=It({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.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:gr(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=ka(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 ot(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:Ca(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,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=ye(r.update.prompt),o=an(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 bn(o.sessionId),d=It({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,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 bn(o.sessionId),d=It(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,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 mr(e);if(i!=="references")return{record:r,history:o,promptHistory:a};let d={};for(let c of Zs(o)){let p=await Kn(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}${bt()}`;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=on(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=Ma(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 mr(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:W,machine:St.hostname()}),f=`${de}${bt()}`;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 Xn(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 rr(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,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 bn(e),r=It(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 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 Es(t.sessionId).catch(()=>[]);if(s.length===0)continue;let r=Date.now(),i=s.filter(d=>r-d.enqueuedAt<or),o=s.length-i.length;if(o>0&&(this.logger?.info(`queue replay: dropping ${o} stale prompt(s) for ${t.sessionId} (TTL ${or/1e3}s)`),await De(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 De(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 Aa(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 ka(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 mn({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:gr(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,createdAt:e?.createdAt??new Date(n.createdAt).toISOString()})}function gr(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 Ca(n){return n?{...n}:void 0}function xa(n,e){if(e&&n==="claude-acp")return{claudeCode:{options:{model:e}}}}function ar(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 dr(n){return n.length>0?n:void 0}function cr(n){let e=ve(n.availableModels);if(e.length>0)return e;let t=n.models;if(t&&typeof t=="object"&&!Array.isArray(t)){let i=ve(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=ve(o.availableModels);if(a.length>0)return a}}let r=At(n,"model");if(r){let i=ve(r.options);if(i.length>0)return i}return[]}function ur(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 lr(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 fr(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 pr(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 Ma(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 mr(n){try{let e=await be.readFile(w.tuiHistoryFile(n),"utf8"),t=[];for(let s of e.split(`
|
|
72
72
|
`))if(s.length!==0)try{let r=JSON.parse(s);typeof r=="string"&&t.push(r)}catch{}return t}catch{return[]}}async function bn(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 It(n,e){return n.interactive!==void 0?n.interactive:n.originatingClient?.name===yt?!1:e?!0:void 0}import{spawn as Ra}from"child_process";import*as Oe from"fs";import*as fe from"fs/promises";import*as hr from"path";var kt=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 Ea=1e3,Pa=6e4,In=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??Ea,this.restartCapMs=r.restartCapMs??Pa;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()},In);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{}},In);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 kt(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=hr.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"&&Sn(r)){try{process.kill(r,"SIGTERM")}catch{}let i=Date.now()+In;for(;Date.now()<i&&Sn(r);)await new Promise(o=>setTimeout(o,50));if(Sn(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=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})
|
|
73
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,...p]=i;if(c===void 0){o.write(`[hydra-acp] ${this.adapter.kind} ${r.name} has empty command
|
|
74
74
|
`),o.end();return}let l=[...p,...r.args],u;try{u=Ra(c,l,{env:d,stdio:["ignore","pipe","pipe"],detached:!1})}catch(f){o.write(`[hydra-acp] failed to spawn ${r.name}: ${f.message}
|
|
@@ -79,14 +79,14 @@ ${dn}`:dn;await t.connection.request("session/prompt",{sessionId:d,prompt:[{type
|
|
|
79
79
|
`),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}
|
|
80
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 Sn(n){try{return process.kill(n,0),!0}catch{return!1}}function Ye(n,e){return n.code=e,n}var Ta={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,Ta,t,s)}};var _a={kind:"transformer",nameEnvVar:"HYDRA_ACP_TRANSFORMER_NAME",tokenRole:"transformer",paths:{dir:w.transformersDir,logFile:w.transformerLogFile,pidFile:w.transformerPidFile}},xt=class extends $e{connected=new Map;constructor(e,t,s={}){super(e,_a,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 Mt=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 Rt from"path";var Ke=n=>{process.stderr.write(n+`
|
|
81
81
|
`)};function An(n){Ke=n??(e=>process.stderr.write(e+`
|
|
82
|
-
`))}async function yr(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=Rt.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=Rt.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=Rt.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 wr(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 en(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 kn(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 ${$a(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 vr(n){let e,t=!1,s=!1,r=i=>{t||(e=setTimeout(()=>{(async()=>{if(!s){s=!0;try{await kn({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 $a(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 Oa}from"crypto";function Cn(n){if(Array.isArray(n))return n.map(Cn);if(n!==null&&typeof n=="object"){let e={};for(let t of Object.keys(n).sort())e[t]=Cn(n[t]);return e}return n}function br(n){let e=JSON.stringify(Cn(n));return Oa("sha256").update(e).digest("hex").slice(0,16)}import*as Ar from"path";import{createHash as Fa,randomBytes as Na,timingSafeEqual as ja}from"crypto";var kr="hydra_session_",Ba=3600*24*30,Ir=12,Ha=32,Ua=50;function La(){return Ar.join(w.home(),"session-tokens.json")}function Sr(n){return Fa("sha256").update(n).digest("hex")}function Cr(n){return Na(n).toString("hex")}function Da(){return Cr(Ir).slice(0,Ir*2)}function qa(){return`${kr}${Cr(Ha)}`}var Et=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=La(),s=await te(t);s&&Array.isArray(s.records)&&(e=s.records.filter(Ja));let r=new n(e,t);return r.sweepExpired(new Date)>0&&await r.flush(),r}async issue(e={}){let t=qa(),s=Sr(t),r=Da(),i=new Date,o=e.ttlSec&&e.ttlSec>0?e.ttlSec:Ba,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(kr))return;let t=Sr(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||!ja(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(()=>{})},Ua))}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 Ja(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 xr="Bearer ",Pt=class{constructor(e){this.token=e}token;async validate(e){return za(e,this.token)?"service":void 0}},Tt=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 Mr(n){return async function(t,s){let r=t.headers.authorization;if(!r||!r.startsWith(xr)){s.code(401).send({error:"Missing bearer token"});return}let i=r.slice(xr.length).trim(),o=await n.validator.validate(i);if(!o){s.code(403).send({error:"Invalid token"});return}t.authIdentity=o}}function Rr(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 $t=class{tokens=new Map;mint(e,t){let s=st();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 za(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 et from"os";import*as zr from"path";import Wa from"strip-ansi";var Qa=/[\x00-\x08\x0b-\x1f\x7f]/g;function Z(n){return Wa(n).replace(Qa,"")}function B(n){return Z(n).replace(/[\n\t]+/g," ").replace(/ +/g," ").trim()}function Tr(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 ed(t);case"agent_thought_chunk":case"agent_thought":return td(t);case"user_message_chunk":return nd(t);case"prompt_received":return sd(t);case"tool_call":return rd(t,e);case"tool_call_update":return cd(t,e);case"plan":return fd(t);case"current_mode_update":return pd(t);case"current_model_update":return md(t);case"turn_complete":return gd(t);case"usage_update":return Za(t);case"available_commands_update":return Ka(t);case"available_modes_update":return Xa(t);case"session_info_update":return Ga(t);case"config_option_update":return Va(t);default:return{kind:"unknown",sessionUpdate:s,raw:n}}}function Va(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 Ga(n){let e=N(n,"title"),t=e!==void 0?B(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 Ya(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:B(r)};typeof s.description=="string"&&(i.description=B(s.description)),e.push(i)}return e}function Ka(n){let e=n.availableCommands??n.commands;return Array.isArray(e)?{kind:"available-commands",commands:Ya(e)}:null}function Xa(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:B(r.id)};typeof r.name=="string"&&(i.name=B(r.name)),typeof r.description=="string"&&(i.description=B(r.description)),t.push(i)}return{kind:"available-modes",modes:t}}function Za(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 ed(n){let e=Ze(n.content);return e===null?null:{kind:"agent-text",text:e}}function td(n){let e=typeof n.text=="string"?Z(n.text):Ze(n.content);return e===null?null:{kind:"agent-thought",text:e}}function nd(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=Ze(n.content);return t===null?null:{kind:"user-text",text:t}}function sd(n){let e=hd(n.prompt);return e===null?null:{kind:"user-text",text:e}}function _r(n){return n?n.toLowerCase().replace(/[_\s-]/g,"")==="exitplanmode":!1}function Er(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 $r(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=Er(r.oldText),o=Er(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 Or(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 rd(n,e={}){let t=N(n,"toolCallId")??N(n,"id");if(!t)return null;let s=N(n,"title")??N(n,"name")??N(n,"label")??"tool call",r=N(n,"name")??N(n,"title");if(_r(r)){let l=Or(n);if(l!==null){let u=N(n,"status"),f={kind:"exit-plan-mode",toolCallId:t,plan:l};return u!==void 0&&(f.status=u),f}}let i=Nr(B(s),n,e),o=N(n,"status"),a=N(n,"kind"),d={kind:"tool-call",toolCallId:t,title:i};o!==void 0&&(d.status=o),a!==void 0&&(d.rawKind=a);let c=$r(n);c!==null&&(d.editDiff=c);let p=Fr(n);return p!==void 0&&(d.detail=p),d}var Pr=64;function Fr(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=B(t.command).trim().replace(/^cd\s+\S+\s+&&\s+/,"");return ad(i,Pr)}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 dd(Se(B(s)),Pr)}var id=["home/","Users/","root/","tmp/","var/","opt/","etc/","usr/","mnt/","private/"];function od(n){for(let e of id)if(n.startsWith(e))return!0;return!1}function Nr(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)}if(od(n))return Se(`/${n}`);if(t.cwd&&t.cwd.length>0){let r=t.cwd.endsWith("/")?t.cwd.slice(0,-1):t.cwd;return Se(`${r}/${n}`)}return n}function ad(n,e){return n.length>e?`${n.slice(0,e-1)}\u2026`:n}function dd(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 cd(n,e={}){let t=N(n,"toolCallId")??N(n,"id");if(!t)return null;let s=N(n,"title"),r=s!==void 0?Nr(B(s),n,e):void 0,i=N(n,"status"),o=$r(n),a=Fr(n);if(!(r!==void 0||o!==null||a!==void 0||i==="completed"||i==="failed"||i==="rejected"||i==="cancelled"))return null;let c=N(n,"name")??s;if(_r(c)){let l={kind:"exit-plan-mode",toolCallId:t},u=Or(n);return u!==null&&(l.plan=u),i!==void 0&&(l.status=i),l}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 l=ud(n);l!==null&&(p.errorText=l),ld(n,l)&&(p.upstreamInterrupted=!0)}return p}function ud(n){let e=n.content;if(Array.isArray(e))for(let s of e){if(!s||typeof s!="object")continue;let i=Ze(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 ld(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 fd(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"?B(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 pd(n){let e=N(n,"currentModeId")??N(n,"currentMode")??N(n,"mode");return e?{kind:"mode-changed",mode:B(e)}:null}function md(n){let e=N(n,"currentModel")??N(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:B(e),...s&&s.length>0?{availableModels:s}:{}}}function gd(n){let e=N(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 Ze(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 hd(n){if(!Array.isArray(n))return null;let e=[];for(let t of n){let s=Ze(t);s!==null&&e.push(s)}return e.length===0?null:e.join("")}function N(n,e){let t=n[e];return typeof t=="string"?t:void 0}function jr(n){let e=yd(n),t=wd(e),s=[];vd(s,n),bd(s,e,t);let r=s.join(`
|
|
82
|
+
`))}async function yr(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=Rt.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=Rt.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=Rt.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 wr(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 en(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 kn(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 ${$a(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 vr(n){let e,t=!1,s=!1,r=i=>{t||(e=setTimeout(()=>{(async()=>{if(!s){s=!0;try{await kn({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 $a(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 Oa}from"crypto";function Cn(n){if(Array.isArray(n))return n.map(Cn);if(n!==null&&typeof n=="object"){let e={};for(let t of Object.keys(n).sort())e[t]=Cn(n[t]);return e}return n}function br(n){let e=JSON.stringify(Cn(n));return Oa("sha256").update(e).digest("hex").slice(0,16)}import*as Ar from"path";import{createHash as Fa,randomBytes as Na,timingSafeEqual as ja}from"crypto";var kr="hydra_session_",Ha=3600*24*30,Ir=12,Ba=32,Ua=50;function La(){return Ar.join(w.home(),"session-tokens.json")}function Sr(n){return Fa("sha256").update(n).digest("hex")}function Cr(n){return Na(n).toString("hex")}function Da(){return Cr(Ir).slice(0,Ir*2)}function qa(){return`${kr}${Cr(Ba)}`}var Et=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=La(),s=await te(t);s&&Array.isArray(s.records)&&(e=s.records.filter(Ja));let r=new n(e,t);return r.sweepExpired(new Date)>0&&await r.flush(),r}async issue(e={}){let t=qa(),s=Sr(t),r=Da(),i=new Date,o=e.ttlSec&&e.ttlSec>0?e.ttlSec:Ha,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(kr))return;let t=Sr(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||!ja(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(()=>{})},Ua))}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 Ja(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 xr="Bearer ",Pt=class{constructor(e){this.token=e}token;async validate(e){return za(e,this.token)?"service":void 0}},Tt=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 Mr(n){return async function(t,s){let r=t.headers.authorization;if(!r||!r.startsWith(xr)){s.code(401).send({error:"Missing bearer token"});return}let i=r.slice(xr.length).trim(),o=await n.validator.validate(i);if(!o){s.code(403).send({error:"Invalid token"});return}t.authIdentity=o}}function Rr(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 $t=class{tokens=new Map;mint(e,t){let s=st();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 za(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 et from"os";import*as zr from"path";import Wa from"strip-ansi";var Qa=/[\x00-\x08\x0b-\x1f\x7f]/g;function Z(n){return Wa(n).replace(Qa,"")}function H(n){return Z(n).replace(/[\n\t]+/g," ").replace(/ +/g," ").trim()}function Tr(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 ed(t);case"agent_thought_chunk":case"agent_thought":return td(t);case"user_message_chunk":return nd(t);case"prompt_received":return sd(t);case"tool_call":return rd(t,e);case"tool_call_update":return cd(t,e);case"plan":return fd(t);case"current_mode_update":return pd(t);case"current_model_update":return md(t);case"turn_complete":return gd(t);case"usage_update":return Za(t);case"available_commands_update":return Ka(t);case"available_modes_update":return Xa(t);case"session_info_update":return Ga(t);case"config_option_update":return Va(t);default:return{kind:"unknown",sessionUpdate:s,raw:n}}}function Va(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 Ga(n){let e=N(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 Ya(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 Ka(n){let e=n.availableCommands??n.commands;return Array.isArray(e)?{kind:"available-commands",commands:Ya(e)}:null}function Xa(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 Za(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 ed(n){let e=Ze(n.content);return e===null?null:{kind:"agent-text",text:e}}function td(n){let e=typeof n.text=="string"?Z(n.text):Ze(n.content);return e===null?null:{kind:"agent-thought",text:e}}function nd(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=Ze(n.content);return t===null?null:{kind:"user-text",text:t}}function sd(n){let e=hd(n.prompt);return e===null?null:{kind:"user-text",text:e}}function _r(n){return n?n.toLowerCase().replace(/[_\s-]/g,"")==="exitplanmode":!1}function Er(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 $r(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=Er(r.oldText),o=Er(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 Or(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 rd(n,e={}){let t=N(n,"toolCallId")??N(n,"id");if(!t)return null;let s=N(n,"title")??N(n,"name")??N(n,"label")??"tool call",r=N(n,"name")??N(n,"title");if(_r(r)){let l=Or(n);if(l!==null){let u=N(n,"status"),f={kind:"exit-plan-mode",toolCallId:t,plan:l};return u!==void 0&&(f.status=u),f}}let i=Nr(H(s),n,e),o=N(n,"status"),a=N(n,"kind"),d={kind:"tool-call",toolCallId:t,title:i};o!==void 0&&(d.status=o),a!==void 0&&(d.rawKind=a);let c=$r(n);c!==null&&(d.editDiff=c);let p=Fr(n);return p!==void 0&&(d.detail=p),d}var Pr=64;function Fr(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 ad(i,Pr)}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 dd(Se(H(s)),Pr)}var id=["home/","Users/","root/","tmp/","var/","opt/","etc/","usr/","mnt/","private/"];function od(n){for(let e of id)if(n.startsWith(e))return!0;return!1}function Nr(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)}if(od(n))return Se(`/${n}`);if(t.cwd&&t.cwd.length>0){let r=t.cwd.endsWith("/")?t.cwd.slice(0,-1):t.cwd;return Se(`${r}/${n}`)}return n}function ad(n,e){return n.length>e?`${n.slice(0,e-1)}\u2026`:n}function dd(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 cd(n,e={}){let t=N(n,"toolCallId")??N(n,"id");if(!t)return null;let s=N(n,"title"),r=s!==void 0?Nr(H(s),n,e):void 0,i=N(n,"status"),o=$r(n),a=Fr(n);if(!(r!==void 0||o!==null||a!==void 0||i==="completed"||i==="failed"||i==="rejected"||i==="cancelled"))return null;let c=N(n,"name")??s;if(_r(c)){let l={kind:"exit-plan-mode",toolCallId:t},u=Or(n);return u!==null&&(l.plan=u),i!==void 0&&(l.status=i),l}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 l=ud(n);l!==null&&(p.errorText=l),ld(n,l)&&(p.upstreamInterrupted=!0)}return p}function ud(n){let e=n.content;if(Array.isArray(e))for(let s of e){if(!s||typeof s!="object")continue;let i=Ze(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 ld(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 fd(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 pd(n){let e=N(n,"currentModeId")??N(n,"currentMode")??N(n,"mode");return e?{kind:"mode-changed",mode:H(e)}:null}function md(n){let e=N(n,"currentModel")??N(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 gd(n){let e=N(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 Ze(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 hd(n){if(!Array.isArray(n))return null;let e=[];for(let t of n){let s=Ze(t);s!==null&&e.push(s)}return e.length===0?null:e.join("")}function N(n,e){let t=n[e];return typeof t=="string"?t:void 0}function jr(n){let e=yd(n),t=wd(e),s=[];vd(s,n),bd(s,e,t);let r=s.join(`
|
|
83
83
|
`);return r.endsWith(`
|
|
84
84
|
`)||(r+=`
|
|
85
85
|
`),r}function yd(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=Tr(s.update,{cwd:n.session.cwd});r!==null&&e.push({event:r,recordedAt:t.recordedAt})}return e}function wd(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 vd(n,e){let t=e.session,s=Fs(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?`${xn(a.size)}`:void 0;d.push(c?`${xn(a.used)} / ${c} tokens`:`${xn(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
86
|
`)),n.push("")}function bd(n,e,t){if(!e.some(c=>Id(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(`
|
|
87
87
|
`))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(`
|
|
88
|
-
`);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(`- ${Ad(p.status)} ${Sd(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 Id(n){switch(n.kind){case"usage-update":case"available-commands":case"session-info":case"unknown":case"turn-complete":return!1;default:return!0}}function Sd(n){let e=n.status,t=e==="completed"||e===void 0?"":` _(${e})_`;return`${Ie(n.title)}${t}`}function Ad(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,"<").replace(/>/g,">")}function xn(n){return n.toLocaleString("en-US")}function Br(n){return n==="127.0.0.1"||n==="::1"||n==="localhost"||n==="[::1]"}function kd(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=>Cd(p)).filter(p=>p.term.length>0);return{operator:i,terms:c}}function Cd(n){let e=/^(\w+):"([^"]*)"$/.exec(n);if(e)return{scope:Hr(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:Hr(s[1]),term:s[2].trim()}:{scope:"all",term:n.trim()}}function Hr(n){switch(n.toLowerCase()){case"prompt":return"user";case"response":return"agent";case"tool":return"tool";default:return"all"}}function xd(n,e){return n==="all"?!0:n==="user"?e==="user":n==="agent"?e==="agent"||e==="thought":e==="tool"||e==="tool-input"}var Md=5,Rd=200,Ur=30;async function qr(n,e,t={}){let s=kd(e);if(s.terms.length===0)return{query:e,truncated:!1,results:[]};let r=t.maxSnippetsPerSession??Md,i=t.maxSessions??Rd,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=Ed(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 Ed(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=Pd(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 Pd(n,e,t,s){let r=e.toLowerCase(),i=0,o=[];for(let a of n){let d=Td(a).filter(c=>xd(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:Nd(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 Td(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=Ft(s.content);return i?[{kind:"agent",text:i}]:[]}case"agent_thought":case"agent_thought_chunk":{let i=typeof s.text=="string"?Z(s.text):Ft(s.content);return i?[{kind:"thought",text:i}]:[]}case"user_message_chunk":{if(Od(s))return[];let i=Ft(s.content);return i?[{kind:"user",text:i}]:[]}case"prompt_received":{let i=Fd(s.prompt);return i?[{kind:"user",text:i}]:[]}case"tool_call":case"tool_call_update":return _d(s);default:return[]}}function _d(n){let e=Lr(n,"name"),t=Lr(n,"title"),s=[];if(t!==void 0){let a=B(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=B(e);a.length>0&&s.push({kind:"tool",toolName:e,text:a})}let r=n.rawInput;if(r&&typeof r=="object"){let a=Dr(r);if(a.length>0){let d={kind:"tool-input",text:B(a)};e!==void 0&&(d.toolName=e),s.push(d)}}let i=n.locations;if(Array.isArray(i)&&i.length>0){let a=Dr(i);if(a.length>0){let d={kind:"tool-input",text:B(a)};e!==void 0&&(d.toolName=e),s.push(d)}}let o=$d(n);if(o!==null){let a={kind:"tool",text:o};e!==void 0&&(a.toolName=e),s.push(a)}return s}function $d(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=B(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=B(s);if(r.length>0)return r}}return null}function Od(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 Ft(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 Fd(n){if(!Array.isArray(n))return"";let e=[];for(let t of n){let s=Ft(t);s.length>0&&e.push(s)}return e.join("")}function Lr(n,e){let t=n[e];return typeof t=="string"?t:void 0}function Dr(n){try{return JSON.stringify(n)}catch{return""}}function Nd(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-Ur),d=Math.min(s.length,o+i.length+Ur),c=a>0?"\u2026":"",p=d<s.length?"\u2026":"";return`${c}${s.slice(a,d)}${p}`}function Jr(n){if(n.publicHost&&n.publicHost.length>0)return n.publicHost;if(n.host&&!Br(n.host))return n.port!==void 0?`${n.host}:${n.port}`:n.host}function Wr(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=Be(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 kn({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=>zr.join(et.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.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":Ys(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"?Ks(c.history,"summary"):c.history,promptHistory:c.promptHistory.length>0?c.promptHistory:void 0,...c.toolBlobs!==void 0?{toolBlobs:c.toolBlobs}:{},hydraVersion:W,machine:et.hostname(),hydraHost:Jr(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/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:W,machine:et.hostname(),hydraHost:Jr(t)});r.header("Content-Type","text/markdown; charset=utf-8"),r.code(200).send(jr(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=Be(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=ir(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)+`
|
|
88
|
+
`);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(`- ${Ad(p.status)} ${Sd(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 Id(n){switch(n.kind){case"usage-update":case"available-commands":case"session-info":case"unknown":case"turn-complete":return!1;default:return!0}}function Sd(n){let e=n.status,t=e==="completed"||e===void 0?"":` _(${e})_`;return`${Ie(n.title)}${t}`}function Ad(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,"<").replace(/>/g,">")}function xn(n){return n.toLocaleString("en-US")}function Hr(n){return n==="127.0.0.1"||n==="::1"||n==="localhost"||n==="[::1]"}function kd(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=>Cd(p)).filter(p=>p.term.length>0);return{operator:i,terms:c}}function Cd(n){let e=/^(\w+):"([^"]*)"$/.exec(n);if(e)return{scope:Br(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:Br(s[1]),term:s[2].trim()}:{scope:"all",term:n.trim()}}function Br(n){switch(n.toLowerCase()){case"prompt":return"user";case"response":return"agent";case"tool":return"tool";default:return"all"}}function xd(n,e){return n==="all"?!0:n==="user"?e==="user":n==="agent"?e==="agent"||e==="thought":e==="tool"||e==="tool-input"}var Md=5,Rd=200,Ur=30;async function qr(n,e,t={}){let s=kd(e);if(s.terms.length===0)return{query:e,truncated:!1,results:[]};let r=t.maxSnippetsPerSession??Md,i=t.maxSessions??Rd,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=Ed(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 Ed(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=Pd(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 Pd(n,e,t,s){let r=e.toLowerCase(),i=0,o=[];for(let a of n){let d=Td(a).filter(c=>xd(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:Nd(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 Td(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=Ft(s.content);return i?[{kind:"agent",text:i}]:[]}case"agent_thought":case"agent_thought_chunk":{let i=typeof s.text=="string"?Z(s.text):Ft(s.content);return i?[{kind:"thought",text:i}]:[]}case"user_message_chunk":{if(Od(s))return[];let i=Ft(s.content);return i?[{kind:"user",text:i}]:[]}case"prompt_received":{let i=Fd(s.prompt);return i?[{kind:"user",text:i}]:[]}case"tool_call":case"tool_call_update":return _d(s);default:return[]}}function _d(n){let e=Lr(n,"name"),t=Lr(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=Dr(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=Dr(i);if(a.length>0){let d={kind:"tool-input",text:H(a)};e!==void 0&&(d.toolName=e),s.push(d)}}let o=$d(n);if(o!==null){let a={kind:"tool",text:o};e!==void 0&&(a.toolName=e),s.push(a)}return s}function $d(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 Od(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 Ft(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 Fd(n){if(!Array.isArray(n))return"";let e=[];for(let t of n){let s=Ft(t);s.length>0&&e.push(s)}return e.join("")}function Lr(n,e){let t=n[e];return typeof t=="string"?t:void 0}function Dr(n){try{return JSON.stringify(n)}catch{return""}}function Nd(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-Ur),d=Math.min(s.length,o+i.length+Ur),c=a>0?"\u2026":"",p=d<s.length?"\u2026":"";return`${c}${s.slice(a,d)}${p}`}function Jr(n){if(n.publicHost&&n.publicHost.length>0)return n.publicHost;if(n.host&&!Hr(n.host))return n.port!==void 0?`${n.host}:${n.port}`:n.host}function Wr(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 kn({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=>zr.join(et.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.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":Ys(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"?Ks(c.history,"summary"):c.history,promptHistory:c.promptHistory.length>0?c.promptHistory:void 0,...c.toolBlobs!==void 0?{toolBlobs:c.toolBlobs}:{},hydraVersion:W,machine:et.hostname(),hydraHost:Jr(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/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:W,machine:et.hostname(),hydraHost:Jr(t)});r.header("Content-Type","text/markdown; charset=utf-8"),r.code(200).send(jr(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=ir(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)+`
|
|
89
89
|
`):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)+`
|
|
90
90
|
`);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)+`
|
|
91
|
-
`)}return u=!0,l?(s.raw.on("close",()=>{l?.(),r.raw.writableEnded||r.raw.end()}),r):(r.raw.end(),r)})}function Qr(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 Vr(n,e,t){n.get("/v1/health",{config:{skipAuth:!0}},async()=>({status:"ok",version:e,configDigest:t}))}var jd=/^[A-Za-z0-9._-]+$/;function Gr(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=Bd(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){tt(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){tt(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){tt(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){tt(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){tt(s,i)}})}function tt(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 Bd(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 Hd=/^[A-Za-z0-9._-]+$/;function Yr(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=Ud(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/transformers/: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/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){nt(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){nt(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){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 Ud(n){let e=n.name;if(typeof e!="string"||!Hd.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 Kr(n,e){n.get("/v1/config",async()=>e)}import{z as Ne}from"zod";import*as Fe from"fs/promises";import*as Xr from"path";import{randomBytes as Lf,scrypt as Ld,timingSafeEqual as Dd}from"crypto";import{promisify as qd}from"util";var Jd=qd(Ld);function Zr(){return Xr.join(w.home(),"password-hash")}var zd=128*1024*1024;async function ei(){try{return(await Fe.readFile(Zr(),"utf8")).trim().length>0}catch(n){if(n.code==="ENOENT")return!1;throw n}}async function ti(n){if(typeof n!="string"||n.length===0)return!1;let e;try{e=(await Fe.readFile(Zr(),"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 Jd(n,o,a.length,{N:s,r,p:i,maxmem:zd});return d.length!==a.length?!1:Dd(d,a)}var Wd=Ne.object({password:Ne.string().min(1),label:Ne.string().min(1).max(256).optional(),ttlSec:Ne.number().int().positive().optional()}),Qd=Ne.object({id:Ne.string().optional()}).optional();function ni(n,e){n.post("/v1/auth/login",{config:{skipAuth:!0}},async(t,s)=>{let r=Vd(t);if(e.rateLimiter.isBlocked(r))return s.code(429).send({error:"Too many failed attempts; try again later."});let i;try{i=Wd.parse(t.body)}catch{return s.code(400).send({error:"Invalid request body"})}if(!await ei())return s.code(403).send({error:"No password configured. Run `hydra-acp auth password` on the daemon host."});if(!await ti(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=Qd.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 Vd(n){return n.ip||"unknown"}import{nanoid as _n}from"nanoid";function Mn(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 si}from"crypto";function ii(n,e){n.get("/acp",{websocket:!0},async(t,s)=>{let r=Rr({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=Mn(t),a=new he(o),d={clientId:`hydra_client_${_n(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=ls.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)),Kd()}),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"||y.verb.length===0)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 await v.emitToChain(i.name,g,m),{ok:!0};if(y==="daemon")return await v.emitToChain(i.name,g,m),{ok:!0};throw Object.assign(new Error(`unsupported route: ${JSON.stringify(y)}`),{code:-32602})}),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;if(!g)throw Object.assign(new Error("spawn_child_session requires cwd"),{code:-32602});return{childSessionId:(await e.manager.create({agentId:f,cwd:g,parentSessionId:m,transformChain:[]})).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=[],E,x=()=>{clearTimeout(M),E?.(),v({entries:S})};E=y.onBroadcast(J=>{S.push(J),g==="turn_complete"&&J.params?.update?.sessionUpdate==="turn_complete"&&x()});let M=setTimeout(x,m);typeof M.unref=="function"&&M.unref(),y.onClose(()=>x())})}),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=fs.parse(l),f=Le(l?._meta),g=Array.isArray(f.transformers)&&f.transformers.every(A=>typeof A=="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=si(32).toString("hex"),v=e.mcpTokenRegistry.reserve(y);let $={name:"hydra-acp-stdin",type:"http",url:`${e.getDaemonOrigin()}/mcp/hydra-acp-stdin`,headers:[{name:"Authorization",value:`Bearer ${y}`}]};S=[...u.mcpServers??[],$]}let E,x;if(e.extensionMcp!==void 0&&e.mcpTokenRegistry!==void 0&&e.getDaemonOrigin!==void 0){let A=e.extensionMcp.list();if(A.length>0){E=si(32).toString("hex"),x=e.mcpTokenRegistry.reserve(E);let $=e.getDaemonOrigin(),F=A.map(Y=>({name:Y,type:"http",url:`${$}/mcp/${Y}`,headers:[{name:"Authorization",value:`Bearer ${E}`}]}));S=[...S??[],...F]}}let M;try{M=await e.manager.create({cwd:u.cwd,agentId:f.agentId??e.defaultAgent,mcpServers:S,title:f.title,agentArgs:f.agentArgs,model:f.model,onInstallProgress:ri(a),transformChain:m,originatingClient:d.clientInfo,...f.interactive!==void 0?{interactive:f.interactive}:{}})}catch(A){throw v!==void 0&&v.abandon(A instanceof Error?A:void 0),x!==void 0&&x.abandon(A instanceof Error?A:void 0),A}if(y!==void 0&&v!==void 0&&e.mcpTokenRegistry!==void 0){let A=y,$=e.mcpTokenRegistry;v.complete(M),M.onClose(()=>{$.unbind(A)})}if(E!==void 0&&x!==void 0&&e.mcpTokenRegistry!==void 0){let A=E,$=e.mcpTokenRegistry;x.complete(M),M.onClose(()=>{$.unbind(A)})}let J=Nt(a,M,d),{entries:G}=await M.attach(J,"full");d.attached.set(M.sessionId,{sessionId:M.sessionId,clientId:J.clientId,readonly:!1}),setImmediate(()=>{(async()=>{for(let A of G)await a.notify(A.method,A.params).catch(()=>{})})()});let z=Rn(M),_=En(M);return{sessionId:M.sessionId,...z?{modes:z}:{},..._?{models:_}:{},configOptions:M.buildConfigOptions(),_meta:Pn(e.manager,M,{clientId:J.clientId})}}),a.onRequest("session/attach",async l=>{let u=ps.parse(l),f=u.clientInfo?.version;f&&i&&(i.kind==="extension"?e.onExtensionVersion?.(i.name,f):e.onTransformerVersion?.(i.name,f));let g=Le(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 _=await e.manager.loadFromDisk(v);if(!_){let F=new Error(`session ${u.sessionId} not found`);throw F.code=I.SessionNotFound,F}let A=await e.manager.loadHistory(v),$=u.clientId??`cli_${_n(8)}`;d.attached.set(_.hydraSessionId,{sessionId:_.hydraSessionId,clientId:$,readonly:!0}),n.log.info(`session/attach OK (viewer) sessionId=${_.hydraSessionId} clientId=${$} attachedCount=${d.attached.size} replayed=${A.length}`);for(let F of A)await a.notify(F.method,F.params).catch(()=>{});return{sessionId:_.hydraSessionId,clientId:$,connectedClients:[$],historyPolicy:"full",replayed:A.length,_meta:Yd(_)}}if(!S){let _=await e.manager.loadFromDisk(v),A=_;if(m&&(A={..._,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}:{}}),!A){let F=new Error(`session ${u.sessionId} not found and no resume hints provided`);throw F.code=I.SessionNotFound,F}let $=A.originatingClient?A:{...A,originatingClient:d.clientInfo};S=await e.manager.resurrect({...$,onInstallProgress:ri(a)}),Tn(S,e)}let E=Nt(a,S,d,u.clientInfo,u.clientId),x=g.replayMode==="drip",{entries:M,appliedPolicy:J}=await S.attach(E,u.historyPolicy,{afterMessageId:u.afterMessageId,raw:x,...g.toolContent!==void 0?{toolContent:g.toolContent}:{}});if(d.attached.set(S.sessionId,{sessionId:S.sessionId,clientId:E.clientId,readonly:y}),n.log.info(`session/attach OK sessionId=${S.sessionId} clientId=${E.clientId} attachedCount=${d.attached.size} requestedPolicy=${u.historyPolicy} appliedPolicy=${J} replayed=${M.length} readonly=${y}${x?" replayMode=drip":""}`),x){let _=g.dripSpeed&&g.dripSpeed>0?g.dripSpeed:1,A=750;(async()=>{let $=null;for(let F of M){let Y=typeof F.recordedAt=="number"?F.recordedAt:null;if($!==null&&Y!==null){let H=Math.min(A,Math.max(0,(Y-$)/_));H>0&&await new Promise(je=>setTimeout(je,H))}Y!==null&&($=Y);try{await a.notify(F.method,F.params)}catch{return}}})()}else for(let A=0;A<M.length;A++){let $=M[A],F=a.notify($.method,$.params).catch(()=>{});(A+1)%200===0&&await F}S.replayPendingPermissions(E);let G=Rn(S),z=En(S);return{sessionId:S.sessionId,clientId:E.clientId,connectedClients:S.connectedClients(E.clientId),historyPolicy:J,replayed:M.length,...G?{modes:G}:{},...z?{models:z}:{},configOptions:S.buildConfigOptions(),_meta:Pn(e.manager,S)}}),a.onRequest("session/detach",async l=>{let u=ms.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=gs.parse(l??{});return{sessions:(await e.manager.list({cwd:u.cwd})).filter(y=>y.originatingClient?.name!==yt).map(ys)}}),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=ws.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 v=new Error(`session ${u.sessionId} not found`);throw v.code=I.SessionNotFound,v}n.log.info(`session/prompt auto-resurrecting cold sessionId=${u.sessionId}`),g=await e.manager.resurrect(m),Tn(g,e);let y=Nt(a,g,d,void 0,f.clientId);await g.attach(y,"none")}return g.prompt(f.clientId,u)});let p=l=>{let u;try{u=ut.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=ut.parse(l);return c(u.sessionId,"session/cancel"),p(l),null}),a.onRequest("hydra-acp/prompt/cancel",async l=>{let u=bs.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=ut.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=Is.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=Ss.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 x=new Error("session/load requires sessionId");throw x.code=I.InvalidParams,x}let g=await e.manager.resolveCanonicalId(f)??f,m=e.manager.get(g);if(!m){let x=await e.manager.loadFromDisk(g);if(!x){let M=new Error(`session ${f} not found in memory or on disk`);throw M.code=I.SessionNotFound,M}m=await e.manager.resurrect(x),Tn(m,e)}let y=Nt(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 x of v)await a.notify(x.method,x.params);m.replayPendingPermissions(y);let S=Rn(m),E=En(m);return{sessionId:m.sessionId,...S?{modes:S}:{},...E?{models:E}:{},configOptions:m.buildConfigOptions(),_meta:Pn(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=Gd(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 ri(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(As,t).catch(()=>{})}}function Rn(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 En(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 Gd(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)`}}function Yd(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]:ct(e,t)}}function Pn(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 dt(e.agentMeta,ct(s,r))}function Kd(){return{protocolVersion:le,agentInfo:{name:"hydra",version:W},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:dt(void 0,{prompt:{queueing:!0,cancelling:!0,updating:!0,amending:!0,pipelining:!1},agents:{list:!0,installProgress:!0}})}}function Tn(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 Nt(n,e,t,s,r){return{clientId:r??`cli_${_n(8)}`,connection:n,clientInfo:s}}var jt=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 Xd}from"crypto";import{McpServer as Zd}from"@modelcontextprotocol/sdk/server/mcp.js";import{StreamableHTTPServerTransport as ec}from"@modelcontextprotocol/sdk/server/streamableHttp.js";import{z as L}from"zod";var oi="Bearer ";function Bt(n){let e=n.headers.authorization;if(typeof e!="string"||!e.startsWith(oi))return;let t=e.slice(oi.length).trim();return t.length>0?t:void 0}function tc(n){let e=new Zd({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:L.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:L.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:L.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:L.number().int().min(1).optional().describe("Optional cap on how many bytes to return. Server caps at 64 KiB regardless."),wait_ms:L.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:L.number().int().min(0).describe("The cursor you've already consumed up to."),timeout_ms:L.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:L.string().min(1).describe("Search pattern. Treated as a JavaScript regular expression by default (set `regex:false` for a literal substring match)."),regex:L.boolean().optional().describe("Default true. Pass false to treat `pattern` as a literal substring."),case_insensitive:L.boolean().optional().describe("Default false. Pass true for case-insensitive matching."),invert:L.boolean().optional().describe("Default false. Pass true to return lines that do NOT match the pattern."),max_matches:L.number().int().min(1).optional().describe("Default 100. Capped server-side at 1000."),max_bytes:L.number().int().min(1).optional().describe("Default 64 KiB output. Capped server-side at 256 KiB."),context_before:L.number().int().min(0).optional().describe("Default 0. Number of lines before each match to include (capped at 20)."),context_after:L.number().int().min(0).optional().describe("Default 0. Number of lines after each match to include (capped at 20)."),cursor:L.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 nc=1e4;function ai(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=tc(a),p=new ec({sessionIdGenerator:()=>Xd()});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=Bt(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),nc)}),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 Ht=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 cc}from"@modelcontextprotocol/sdk/server/streamableHttp.js";import{randomUUID as uc}from"crypto";import{Server as sc}from"@modelcontextprotocol/sdk/server/index.js";import{CallToolRequestSchema as rc,ListToolsRequestSchema as ic}from"@modelcontextprotocol/sdk/types.js";var oc=6e4;function di(n,e,t={}){let s=t.invokeTimeoutMs??oc,r=new sc({name:n,version:"1.0.0"},{capabilities:{tools:{listChanged:!1}},...e.instructions!==void 0?{instructions:e.instructions}:{}}),i=new Map(e.tools.map(o=>[o.name,o]));return r.setRequestHandler(ic,async()=>({tools:e.tools.map(o=>({name:o.name,description:o.description,inputSchema:o.inputSchema,...o.outputSchema!==void 0?{outputSchema:o.outputSchema}:{}}))})),r.setRequestHandler(rc,async o=>{let a=o.params.name;if(!i.has(a))return Ut(`unknown tool: ${a}`);try{let d=await ac(e.connection,n,a,o.params.arguments??{},s);return dc(d,a)}catch(d){return Ut(d instanceof Error?d.message:String(d))}}),r}async function ac(n,e,t,s,r){let i,o=new Promise((a,d)=>{i=setTimeout(()=>d(new Error(`extension timeout after ${r}ms`)),r)});try{return await Promise.race([n.request("hydra-acp/mcp_tools/invoke",{server:e,tool:t,args:s}),o])}finally{i!==void 0&&clearTimeout(i)}}function dc(n,e){if(n===null||typeof n!="object")return Ut(`extension ${e} returned non-object`);let t=n;return Array.isArray(t.content)?t:Ut(`extension ${e} omitted content array`)}function Ut(n){return{content:[{type:"text",text:n}],isError:!0}}var lc=1e4;function ci(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 v=r.get(p);if(v!==void 0){r.delete(p);for(let S of v.values())await i(S)}}));let f=u.get(l);if(f!==void 0)return f.transport;let g=t.lookup(l);if(g===void 0)return;let m=di(l,g,s.buildOptions),y=new cc({sessionIdGenerator:()=>uc()});return await m.connect(y),u.set(l,{server:m,transport:y}),y}async function d(p,l){let u=Bt(p);if(u===void 0){l.code(401).send({error:"missing bearer token"});return}let f=e.lookup(u);if(f===void 0){l.code(404).send({error:"unknown mcp token"});return}if(f.session===void 0){let y,v=new Promise(E=>{y=setTimeout(()=>E(void 0),lc)}),S=await Promise.race([f.sessionReady.catch(()=>{}),v]);if(y!==void 0&&clearTimeout(y),S===void 0){l.code(503).send({error:"session not ready"});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 gc(n,e){yc(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 hc(n.daemon.logLevel),i=fc({logger:{level:n.daemon.logLevel,stream:s},https:t??null,bodyLimit:256*1024*1024});await i.register(pc,{options:{handleProtocols:Fn}}),Gt(O=>{i.log.info(O)}),Xt(O=>{i.log.info(O)});let o=await Et.load(),a=new Ot,d=new $t,c=new _t([new Pt(e),new Tt(o),d]),p=Mr({validator:c});i.addHook("onRequest",async(O,me)=>{O.routeOptions.config?.skipAuth||O.url==="/acp"||O.url?.startsWith("/acp?")||await p(O,me)});let l=setInterval(()=>{o.sweepExpired()},300*1e3);l.unref();let u=new He(n,{onFetched:()=>{yr(u,y)}});An(O=>i.log.info(O));let f={info:O=>i.log.info(O),warn:O=>i.log.warn(O)},g=O=>ae.spawn({...O,stderrTailBytes:n.daemon.agentStderrTailBytes,logger:f});Yn(n.compressToolContent);let m=new Mt,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(qn(n),void 0,{tokenRegistry:d}),S=new xt(Jn(n),void 0,{tokenRegistry:d});Vr(i,W,br(n)),Wr(i,y,{agentId:n.defaultAgent,cwd:n.defaultCwd,publicHost:n.daemon.publicHost,host:n.daemon.host,port:n.daemon.port}),Qr(i,u,y,{npmRegistry:n.npmRegistry}),Gr(i,v),Yr(i,S),Kr(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]}),ni(i,{store:o,rateLimiter:a});let E=new jt,x=new Ht;ai(i,E),ci(i,E,x);let M,J=()=>{if(M!==void 0)return M;let O=i.server.address(),me=O&&typeof O=="object"?O.port:n.daemon.port;return M=`${n.daemon.tls?"https":"http"}://${n.daemon.host}:${me}`,M};ii(i,{validator:c,manager:y,defaultAgent:n.defaultAgent,processRegistry:d,onExtensionVersion:(O,me)=>v.reportVersion(O,me),onTransformerVersion:(O,me)=>S.reportVersion(O,me),transformers:S,extensionCommands:m,mcpTokenRegistry:E,extensionMcp:x,getDaemonOrigin:J,registry:u}),await i.listen({host:n.daemon.host,port:n.daemon.port});let G=i.server.address(),z=G&&typeof G=="object"?G.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:z,startedAt:new Date().toISOString()})+`
|
|
92
|
-
`,{encoding:"utf8",mode:384});let _=n.daemon.tls?"https":"http",A=n.daemon.tls?"wss":"ws",$={daemonUrl:`${_}://${n.daemon.host}:${z}`,daemonHost:n.daemon.host,daemonPort:z,serviceToken:e,daemonWsUrl:`${A}://${n.daemon.host}:${z}/acp`,hydraHome:w.home()};v.setContext($),S.setContext($),await v.start(),await S.start(),y.resurrectPendingQueues().catch(O=>{i.log.warn(`queue replay scan failed: ${O.message}`)});let F=n.daemon.agentSyncIntervalMinutes*60*1e3,Y=F>0?wr({registry:u,manager:y,intervalMs:F,logger:f}):void 0,
|
|
91
|
+
`)}return u=!0,l?(s.raw.on("close",()=>{l?.(),r.raw.writableEnded||r.raw.end()}),r):(r.raw.end(),r)})}function Qr(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 Vr(n,e,t){n.get("/v1/health",{config:{skipAuth:!0}},async()=>({status:"ok",version:e,configDigest:t}))}var jd=/^[A-Za-z0-9._-]+$/;function Gr(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=Hd(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){tt(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){tt(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){tt(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){tt(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){tt(s,i)}})}function tt(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 Hd(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 Bd=/^[A-Za-z0-9._-]+$/;function Yr(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=Ud(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/transformers/: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/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){nt(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){nt(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){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 Ud(n){let e=n.name;if(typeof e!="string"||!Bd.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 Kr(n,e){n.get("/v1/config",async()=>e)}import{z as Ne}from"zod";import*as Fe from"fs/promises";import*as Xr from"path";import{randomBytes as Df,scrypt as Ld,timingSafeEqual as Dd}from"crypto";import{promisify as qd}from"util";var Jd=qd(Ld);function Zr(){return Xr.join(w.home(),"password-hash")}var zd=128*1024*1024;async function ei(){try{return(await Fe.readFile(Zr(),"utf8")).trim().length>0}catch(n){if(n.code==="ENOENT")return!1;throw n}}async function ti(n){if(typeof n!="string"||n.length===0)return!1;let e;try{e=(await Fe.readFile(Zr(),"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 Jd(n,o,a.length,{N:s,r,p:i,maxmem:zd});return d.length!==a.length?!1:Dd(d,a)}var Wd=Ne.object({password:Ne.string().min(1),label:Ne.string().min(1).max(256).optional(),ttlSec:Ne.number().int().positive().optional()}),Qd=Ne.object({id:Ne.string().optional()}).optional();function ni(n,e){n.post("/v1/auth/login",{config:{skipAuth:!0}},async(t,s)=>{let r=Vd(t);if(e.rateLimiter.isBlocked(r))return s.code(429).send({error:"Too many failed attempts; try again later."});let i;try{i=Wd.parse(t.body)}catch{return s.code(400).send({error:"Invalid request body"})}if(!await ei())return s.code(403).send({error:"No password configured. Run `hydra-acp auth password` on the daemon host."});if(!await ti(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=Qd.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 Vd(n){return n.ip||"unknown"}import{nanoid as _n}from"nanoid";function Mn(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 si}from"crypto";function ii(n,e){n.get("/acp",{websocket:!0},async(t,s)=>{let r=Rr({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=Mn(t),a=new he(o),d={clientId:`hydra_client_${_n(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=ls.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)),Xd()}),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/transformer/attach",async l=>Yd(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;if(!g&&m){let S=e.manager.get(m);S&&(g=S.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 y=typeof u.interactive=="boolean"?u.interactive:!1;return{childSessionId:(await e.manager.create({agentId:f,cwd:g,parentSessionId:m,interactive:y,transformChain:[]})).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=[],E,x=()=>{clearTimeout(M),E?.(),v({entries:S})};E=y.onBroadcast(J=>{S.push(J),g==="turn_complete"&&J.params?.update?.sessionUpdate==="turn_complete"&&x()});let M=setTimeout(x,m);typeof M.unref=="function"&&M.unref(),y.onClose(()=>x())})}),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=fs.parse(l),f=Le(l?._meta),g=Array.isArray(f.transformers)&&f.transformers.every(A=>typeof A=="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=si(32).toString("hex"),v=e.mcpTokenRegistry.reserve(y);let $={name:"hydra-acp-stdin",type:"http",url:`${e.getDaemonOrigin()}/mcp/hydra-acp-stdin`,headers:[{name:"Authorization",value:`Bearer ${y}`}]};S=[...u.mcpServers??[],$]}let E,x;if(e.extensionMcp!==void 0&&e.mcpTokenRegistry!==void 0&&e.getDaemonOrigin!==void 0){let A=e.extensionMcp.list();if(A.length>0){E=si(32).toString("hex"),x=e.mcpTokenRegistry.reserve(E);let $=e.getDaemonOrigin(),F=A.map(Y=>({name:Y,type:"http",url:`${$}/mcp/${Y}`,headers:[{name:"Authorization",value:`Bearer ${E}`}]}));S=[...S??[],...F]}}let M;try{M=await e.manager.create({cwd:u.cwd,agentId:f.agentId??e.defaultAgent,mcpServers:S,title:f.title,agentArgs:f.agentArgs,model:f.model,onInstallProgress:ri(a),transformChain:m,originatingClient:d.clientInfo,...f.interactive!==void 0?{interactive:f.interactive}:{}})}catch(A){throw v!==void 0&&v.abandon(A instanceof Error?A:void 0),x!==void 0&&x.abandon(A instanceof Error?A:void 0),A}if(y!==void 0&&v!==void 0&&e.mcpTokenRegistry!==void 0){let A=y,$=e.mcpTokenRegistry;v.complete(M),M.onClose(()=>{$.unbind(A)})}if(E!==void 0&&x!==void 0&&e.mcpTokenRegistry!==void 0){let A=E,$=e.mcpTokenRegistry;x.complete(M),M.onClose(()=>{$.unbind(A)})}let J=Nt(a,M,d),{entries:G}=await M.attach(J,"full");d.attached.set(M.sessionId,{sessionId:M.sessionId,clientId:J.clientId,readonly:!1}),setImmediate(()=>{(async()=>{for(let A of G)await a.notify(A.method,A.params).catch(()=>{})})()});let z=Rn(M),_=En(M);return{sessionId:M.sessionId,...z?{modes:z}:{},..._?{models:_}:{},configOptions:M.buildConfigOptions(),_meta:Pn(e.manager,M,{clientId:J.clientId})}}),a.onRequest("session/attach",async l=>{let u=ps.parse(l),f=u.clientInfo?.version;f&&i&&(i.kind==="extension"?e.onExtensionVersion?.(i.name,f):e.onTransformerVersion?.(i.name,f));let g=Le(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 _=await e.manager.loadFromDisk(v);if(!_){let F=new Error(`session ${u.sessionId} not found`);throw F.code=I.SessionNotFound,F}let A=await e.manager.loadHistory(v),$=u.clientId??`cli_${_n(8)}`;d.attached.set(_.hydraSessionId,{sessionId:_.hydraSessionId,clientId:$,readonly:!0}),n.log.info(`session/attach OK (viewer) sessionId=${_.hydraSessionId} clientId=${$} attachedCount=${d.attached.size} replayed=${A.length}`);for(let F of A)await a.notify(F.method,F.params).catch(()=>{});return{sessionId:_.hydraSessionId,clientId:$,connectedClients:[$],historyPolicy:"full",replayed:A.length,_meta:Kd(_)}}if(!S){let _=await e.manager.loadFromDisk(v),A=_;if(m&&(A={..._,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}:{}}),!A){let F=new Error(`session ${u.sessionId} not found and no resume hints provided`);throw F.code=I.SessionNotFound,F}let $=A.originatingClient?A:{...A,originatingClient:d.clientInfo};S=await e.manager.resurrect({...$,onInstallProgress:ri(a)}),Tn(S,e)}let E=Nt(a,S,d,u.clientInfo,u.clientId),x=g.replayMode==="drip",{entries:M,appliedPolicy:J}=await S.attach(E,u.historyPolicy,{afterMessageId:u.afterMessageId,raw:x,...g.toolContent!==void 0?{toolContent:g.toolContent}:{}});if(d.attached.set(S.sessionId,{sessionId:S.sessionId,clientId:E.clientId,readonly:y}),n.log.info(`session/attach OK sessionId=${S.sessionId} clientId=${E.clientId} attachedCount=${d.attached.size} requestedPolicy=${u.historyPolicy} appliedPolicy=${J} replayed=${M.length} readonly=${y}${x?" replayMode=drip":""}`),x){let _=g.dripSpeed&&g.dripSpeed>0?g.dripSpeed:1,A=750;(async()=>{let $=null;for(let F of M){let Y=typeof F.recordedAt=="number"?F.recordedAt:null;if($!==null&&Y!==null){let B=Math.min(A,Math.max(0,(Y-$)/_));B>0&&await new Promise(je=>setTimeout(je,B))}Y!==null&&($=Y);try{await a.notify(F.method,F.params)}catch{return}}})()}else for(let A=0;A<M.length;A++){let $=M[A],F=a.notify($.method,$.params).catch(()=>{});(A+1)%200===0&&await F}S.replayPendingPermissions(E);let G=Rn(S),z=En(S);return{sessionId:S.sessionId,clientId:E.clientId,connectedClients:S.connectedClients(E.clientId),historyPolicy:J,replayed:M.length,...G?{modes:G}:{},...z?{models:z}:{},configOptions:S.buildConfigOptions(),_meta:Pn(e.manager,S)}}),a.onRequest("session/detach",async l=>{let u=ms.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=gs.parse(l??{});return{sessions:(await e.manager.list({cwd:u.cwd})).filter(y=>y.originatingClient?.name!==yt).map(ys)}}),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=ws.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 v=new Error(`session ${u.sessionId} not found`);throw v.code=I.SessionNotFound,v}n.log.info(`session/prompt auto-resurrecting cold sessionId=${u.sessionId}`),g=await e.manager.resurrect(m),Tn(g,e);let y=Nt(a,g,d,void 0,f.clientId);await g.attach(y,"none")}return g.prompt(f.clientId,u)});let p=l=>{let u;try{u=ut.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=ut.parse(l);return c(u.sessionId,"session/cancel"),p(l),null}),a.onRequest("hydra-acp/prompt/cancel",async l=>{let u=bs.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=ut.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=Is.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=Ss.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 x=new Error("session/load requires sessionId");throw x.code=I.InvalidParams,x}let g=await e.manager.resolveCanonicalId(f)??f,m=e.manager.get(g);if(!m){let x=await e.manager.loadFromDisk(g);if(!x){let M=new Error(`session ${f} not found in memory or on disk`);throw M.code=I.SessionNotFound,M}m=await e.manager.resurrect(x),Tn(m,e)}let y=Nt(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 x of v)await a.notify(x.method,x.params);m.replayPendingPermissions(y);let S=Rn(m),E=En(m);return{sessionId:m.sessionId,...S?{modes:S}:{},...E?{models:E}:{},configOptions:m.buildConfigOptions(),_meta:Pn(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=Gd(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 ri(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(As,t).catch(()=>{})}}function Rn(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 En(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 Gd(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 Yd(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 Kd(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]:ct(e,t)}}function Pn(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 dt(e.agentMeta,ct(s,r))}function Xd(){return{protocolVersion:le,agentInfo:{name:"hydra",version:W},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:dt(void 0,{prompt:{queueing:!0,cancelling:!0,updating:!0,amending:!0,pipelining:!1},agents:{list:!0,installProgress:!0}})}}function Tn(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 Nt(n,e,t,s,r){return{clientId:r??`cli_${_n(8)}`,connection:n,clientInfo:s}}var jt=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 Zd}from"crypto";import{McpServer as ec}from"@modelcontextprotocol/sdk/server/mcp.js";import{StreamableHTTPServerTransport as tc}from"@modelcontextprotocol/sdk/server/streamableHttp.js";import{z as L}from"zod";var oi="Bearer ";function Ht(n){let e=n.headers.authorization;if(typeof e!="string"||!e.startsWith(oi))return;let t=e.slice(oi.length).trim();return t.length>0?t:void 0}function nc(n){let e=new ec({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:L.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:L.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:L.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:L.number().int().min(1).optional().describe("Optional cap on how many bytes to return. Server caps at 64 KiB regardless."),wait_ms:L.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:L.number().int().min(0).describe("The cursor you've already consumed up to."),timeout_ms:L.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:L.string().min(1).describe("Search pattern. Treated as a JavaScript regular expression by default (set `regex:false` for a literal substring match)."),regex:L.boolean().optional().describe("Default true. Pass false to treat `pattern` as a literal substring."),case_insensitive:L.boolean().optional().describe("Default false. Pass true for case-insensitive matching."),invert:L.boolean().optional().describe("Default false. Pass true to return lines that do NOT match the pattern."),max_matches:L.number().int().min(1).optional().describe("Default 100. Capped server-side at 1000."),max_bytes:L.number().int().min(1).optional().describe("Default 64 KiB output. Capped server-side at 256 KiB."),context_before:L.number().int().min(0).optional().describe("Default 0. Number of lines before each match to include (capped at 20)."),context_after:L.number().int().min(0).optional().describe("Default 0. Number of lines after each match to include (capped at 20)."),cursor:L.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 sc=1e4;function ai(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=nc(a),p=new tc({sessionIdGenerator:()=>Zd()});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=Ht(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),sc)}),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 Bt=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 uc}from"@modelcontextprotocol/sdk/server/streamableHttp.js";import{randomUUID as lc}from"crypto";import{Server as rc}from"@modelcontextprotocol/sdk/server/index.js";import{CallToolRequestSchema as ic,ListToolsRequestSchema as oc}from"@modelcontextprotocol/sdk/types.js";var ac=6e4;function di(n,e,t={}){let s=t.invokeTimeoutMs??ac,r=new rc({name:n,version:"1.0.0"},{capabilities:{tools:{listChanged:!1}},...e.instructions!==void 0?{instructions:e.instructions}:{}}),i=new Map(e.tools.map(o=>[o.name,o]));return r.setRequestHandler(oc,async()=>({tools:e.tools.map(o=>({name:o.name,description:o.description,inputSchema:o.inputSchema,...o.outputSchema!==void 0?{outputSchema:o.outputSchema}:{}}))})),r.setRequestHandler(ic,async o=>{let a=o.params.name;if(!i.has(a))return Ut(`unknown tool: ${a}`);try{let d=await dc(e.connection,n,a,o.params.arguments??{},s);return cc(d,a)}catch(d){return Ut(d instanceof Error?d.message:String(d))}}),r}async function dc(n,e,t,s,r){let i,o=new Promise((a,d)=>{i=setTimeout(()=>d(new Error(`extension timeout after ${r}ms`)),r)});try{return await Promise.race([n.request("hydra-acp/mcp_tools/invoke",{server:e,tool:t,args:s}),o])}finally{i!==void 0&&clearTimeout(i)}}function cc(n,e){if(n===null||typeof n!="object")return Ut(`extension ${e} returned non-object`);let t=n;return Array.isArray(t.content)?t:Ut(`extension ${e} omitted content array`)}function Ut(n){return{content:[{type:"text",text:n}],isError:!0}}var fc=1e4;function ci(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 v=r.get(p);if(v!==void 0){r.delete(p);for(let S of v.values())await i(S)}}));let f=u.get(l);if(f!==void 0)return f.transport;let g=t.lookup(l);if(g===void 0)return;let m=di(l,g,s.buildOptions),y=new uc({sessionIdGenerator:()=>lc()});return await m.connect(y),u.set(l,{server:m,transport:y}),y}async function d(p,l){let u=Ht(p);if(u===void 0){l.code(401).send({error:"missing bearer token"});return}let f=e.lookup(u);if(f===void 0){l.code(404).send({error:"unknown mcp token"});return}if(f.session===void 0){let y,v=new Promise(E=>{y=setTimeout(()=>E(void 0),fc)}),S=await Promise.race([f.sessionReady.catch(()=>{}),v]);if(y!==void 0&&clearTimeout(y),S===void 0){l.code(503).send({error:"session not ready"});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 hc(n,e){wc(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 yc(n.daemon.logLevel),i=pc({logger:{level:n.daemon.logLevel,stream:s},https:t??null,bodyLimit:256*1024*1024});await i.register(mc,{options:{handleProtocols:Fn}}),Gt(O=>{i.log.info(O)}),Xt(O=>{i.log.info(O)});let o=await Et.load(),a=new Ot,d=new $t,c=new _t([new Pt(e),new Tt(o),d]),p=Mr({validator:c});i.addHook("onRequest",async(O,me)=>{O.routeOptions.config?.skipAuth||O.url==="/acp"||O.url?.startsWith("/acp?")||await p(O,me)});let l=setInterval(()=>{o.sweepExpired()},300*1e3);l.unref();let u=new Be(n,{onFetched:()=>{yr(u,y)}});An(O=>i.log.info(O));let f={info:O=>i.log.info(O),warn:O=>i.log.warn(O)},g=O=>ae.spawn({...O,stderrTailBytes:n.daemon.agentStderrTailBytes,logger:f});Yn(n.compressToolContent);let m=new Mt,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(qn(n),void 0,{tokenRegistry:d}),S=new xt(Jn(n),void 0,{tokenRegistry:d});Vr(i,W,br(n)),Wr(i,y,{agentId:n.defaultAgent,cwd:n.defaultCwd,publicHost:n.daemon.publicHost,host:n.daemon.host,port:n.daemon.port}),Qr(i,u,y,{npmRegistry:n.npmRegistry}),Gr(i,v),Yr(i,S),Kr(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]}),ni(i,{store:o,rateLimiter:a});let E=new jt,x=new Bt;ai(i,E),ci(i,E,x);let M,J=()=>{if(M!==void 0)return M;let O=i.server.address(),me=O&&typeof O=="object"?O.port:n.daemon.port;return M=`${n.daemon.tls?"https":"http"}://${n.daemon.host}:${me}`,M};ii(i,{validator:c,manager:y,defaultAgent:n.defaultAgent,processRegistry:d,onExtensionVersion:(O,me)=>v.reportVersion(O,me),onTransformerVersion:(O,me)=>S.reportVersion(O,me),transformers:S,extensionCommands:m,mcpTokenRegistry:E,extensionMcp:x,getDaemonOrigin:J,registry:u}),await i.listen({host:n.daemon.host,port:n.daemon.port});let G=i.server.address(),z=G&&typeof G=="object"?G.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:z,startedAt:new Date().toISOString()})+`
|
|
92
|
+
`,{encoding:"utf8",mode:384});let _=n.daemon.tls?"https":"http",A=n.daemon.tls?"wss":"ws",$={daemonUrl:`${_}://${n.daemon.host}:${z}`,daemonHost:n.daemon.host,daemonPort:z,serviceToken:e,daemonWsUrl:`${A}://${n.daemon.host}:${z}/acp`,hydraHome:w.home()};v.setContext($),S.setContext($),await v.start(),await S.start(),y.resurrectPendingQueues().catch(O=>{i.log.warn(`queue replay scan failed: ${O.message}`)});let F=n.daemon.agentSyncIntervalMinutes*60*1e3,Y=F>0?wr({registry:u,manager:y,intervalMs:F,logger:f}):void 0,B=n.daemon.sessionGcIntervalMinutes*60*1e3,je=n.daemon.sessionGcMaxAgeDays*24*60*60*1e3,$n=B>0?vr({manager:y,intervalMs:B,maxAgeMs:je,logger:f}):void 0;return{app:i,manager:y,registry:u,extensions:v,transformers:S,mcpTokenRegistry:E,extensionMcp:x,processRegistry:d,shutdown:async()=>{$n&&$n(),Y&&Y(),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(),Gt(null),Xt(null),An(null),await i.close();try{li.unlinkSync(w.pidFile())}catch{}try{r.flushSync()}catch{}}}}async function yc(n){let e=await gc({file:w.logFile(),size:"10m",frequency:"daily",mkdir:!0,symlink:!0}),t=ui.destination(2);return{stream:ui.multistream([{stream:e,level:n},{stream:t,level:n}]),fileStream:e}}function wc(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,we as Session,Ge as SessionManager,Ei as defaultConfig,mi as ensureServiceToken,st as generateServiceToken,Wn as loadConfig,pi as loadServiceToken,tn as ndjsonStreamFromStdio,w as paths,oe as planSpawn,hc as startDaemon,Ri as writeConfig,Mn as wsToMessageStream};
|