ideawave 0.0.34 → 0.0.35
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.mjs +2 -2
- package/package.json +1 -1
package/dist/index.mjs
CHANGED
|
@@ -15,7 +15,7 @@ ${Q}`:null].filter(Boolean).join(`
|
|
|
15
15
|
|
|
16
16
|
`));return u($)}function s$($,Z){if(!Z)return $;if(!$)return Z;if($===Z||$.endsWith(Z))return $;if(Z.startsWith($))return Z;return r(`${$}
|
|
17
17
|
|
|
18
|
-
${Z}`)}function Q$($){if(!Array.isArray($))return null;let Z=$.flatMap((X)=>{if(!X||typeof X!=="object")return[];let J=X;if(typeof J.path!=="string"||!J.path.trim())return[];let Q=typeof J.line==="number"?`:${J.line}`:"";return[`${J.path}${Q}`]});return Z.length>0?Z.slice(0,5):null}function o$($){let{ctx:Z,appendText:X,appendToolActivity:J}=$,Q=$.flushIntervalMs??r2,j="",W="",G=0,V=null,z={},D=new Map,U=(F)=>{if(D.set(F.toolCallId,F),!J)return;J({boardId:Z.boardId,cardId:Z.cardId,runId:Z.runId,entry:F}).catch((K)=>{console.warn("[streamAdapter] appendToolActivity failed",K)})},H=(F)=>{let K=`${Z.runId}:thought`,C=D.get(K);if(!C||C.status!=="running")return;U({...C,status:"completed",completedAt:C.completedAt??F})},B=(F)=>{for(let K of D.values()){if(K.status!=="running"||K.rawTitle.toLowerCase()!=="toolsearch")continue;U({...K,status:"completed",completedAt:K.completedAt??F})}},k=(F)=>{for(let K of Array.from(D.values())){if(K.status!=="running")continue;if(K.kind==="thought"||K.rawTitle==="agent_thought")continue;U({...K,status:"completed",completedAt:K.completedAt??F})}},q="",M=0,Y=null,P=null,O=(F)=>{if(F===q)return;q=F,M=Date.now(),X({boardId:Z.boardId,cardId:Z.cardId,runId:Z.runId,text:F}).catch((K)=>{console.warn("[streamAdapter] appendText failed",K)})},A=()=>{if(Y!==null)clearTimeout(Y),Y=null;if(P!==null){let F=P;P=null,O(F)}},f=(F)=>{let K=Date.now()-M;if(K>=Q&&Y===null){O(F);return}if(P=F,Y===null){let C=Math.max(0,Q-K);Y=setTimeout(()=>{Y=null,A()},C)}};return{onSessionUpdate:async(F)=>{let K=F.update;switch(z[K.sessionUpdate]=(z[K.sessionUpdate]??0)+1,K.sessionUpdate){case"agent_message_chunk":{let C=new Date().toISOString();if(H(C),K.content.type==="text")j+=K.content.text,f(j);else{let w=`agent_message_chunk:${K.content.type}`;z[w]=(z[w]??0)+1}return}case"agent_thought_chunk":{if(K.content.type!=="text")return;W+=K.content.text;let C=new Date().toISOString(),w=`${Z.runId}:thought`,x=D.get(w);U({toolCallId:w,rawTitle:"agent_thought",title:"Thought",kind:"thought",status:"running",startedAt:x?.startedAt??C,completedAt:null,contentPreview:r(W.trim())});return}case"user_message_chunk":return;case"tool_call":{G+=1;let C=i$(K.title),w=$$(K.status),x=new Date().toISOString();H(x),B(x),U({toolCallId:K.toolCallId,rawTitle:C,title:X$(C,K.rawInput),kind:Z$(K.kind),status:w,startedAt:x,completedAt:w==="running"?null:x,inputPreview:u(K.rawInput),outputPreview:J$(K.rawOutput),contentPreview:P0(K.content),locations:Q$(K.locations)});return}case"tool_call_update":{let C=D.get(K.toolCallId);if(!C){let Z0=i$(K.title),m=$$(K.status),l=new Date().toISOString();U({toolCallId:K.toolCallId,rawTitle:Z0,title:X$(Z0,K.rawInput),kind:Z$(K.kind),status:m,startedAt:l,completedAt:m==="running"?null:l,inputPreview:u(K.rawInput),outputPreview:J$(K.rawOutput),contentPreview:P0(K.content),locations:Q$(K.locations)});return}let w=K.status===void 0||K.status===null?C.status:$$(K.status),x=typeof K.title==="string"&&K.title.trim().length>0?K.title:C.rawTitle,R=typeof K.title==="string"&&K.title.trim().length>0,b=w==="running"?C.completedAt:C.completedAt??new Date().toISOString(),S=K.content===void 0?C.contentPreview:s$(C.contentPreview,P0(K.content)),n=K.rawOutput===void 0?C.outputPreview:s$(C.outputPreview,J$(K.rawOutput));U({...C,rawTitle:x,title:R?X$(x,K.rawInput):C.title,kind:K.kind===void 0?C.kind:Z$(K.kind),status:w,completedAt:b,inputPreview:K.rawInput===void 0?C.inputPreview:u(K.rawInput),outputPreview:n,contentPreview:S,locations:K.locations===void 0?C.locations:Q$(K.locations)});return}case"plan":V=K.entries;return;case"available_commands_update":return;case"current_mode_update":return;default:return}},getFinalText:()=>j,getToolCallCount:()=>G,getLatestPlan:()=>V,getUpdateTypeCounts:()=>({...z}),flush:(F)=>{let K=new Date().toISOString();if(H(K),F?.completeOpenToolCalls)k(K);A()}}}var Y0=1,t$=-32000;function r$($){if($.awaitingInput)return!1;return $.nowMs-$.lastUpdateMs>$.windowMs}function X1($){if(!$||typeof $!=="object")return!1;return $.code===t$}var X3=300000,J3=5000;function J1(){return{name:"ideawave-cli",title:"IdeaWave CLI",version:"0.0.34".trim()?"0.0.34":"dev"}}function Q1($){if($===Y0)return;throw new L(`Unsupported ACP protocol version ${$}; expected ${Y0}`)}function j1($){return $3.toWeb($)}function W1($){return Z3.toWeb($)}function j$($){return $!==null&&typeof $==="object"}function Q3($){if(!Array.isArray($))return[];let Z=[];for(let X of $){if(!j$(X))continue;if(typeof X.group==="string"&&Array.isArray(X.options)){let J=typeof X.name==="string"?X.name:X.group;for(let Q of X.options){if(!j$(Q))continue;if(typeof Q.value!=="string"||typeof Q.name!=="string")continue;Z.push({value:Q.value,name:Q.name,description:typeof Q.description==="string"?Q.description:null,group:X.group,groupName:J})}continue}if(typeof X.value!=="string"||typeof X.name!=="string")continue;Z.push({value:X.value,name:X.name,description:typeof X.description==="string"?X.description:null,group:null,groupName:null})}return Z}function j3($){if(!j$($))return null;if(typeof $.id!=="string"||typeof $.name!=="string")return null;let Z=typeof $.category==="string"?$.category:null,X=typeof $.description==="string"?$.description:null;if($.type==="select"){let J=typeof $.currentValue==="string"?$.currentValue:null;return{id:$.id,name:$.name,category:Z,description:X,type:"select",currentValue:J,options:Q3($.options)}}if($.type==="boolean"){let J=typeof $.currentValue==="boolean"?$.currentValue:typeof $.value==="boolean"?$.value:null;return{id:$.id,name:$.name,category:Z,description:X,type:"boolean",currentValue:J,options:[]}}return null}function W$($){let Z=($.configOptions??[]).map(j3).filter((G)=>G!==null),X=Z.find((G)=>G.type==="select"&&G.category==="model")??Z.find((G)=>G.type==="select"&&G.id==="model")??null,J=X?.options.map((G)=>({id:G.value,name:G.name,description:G.description??null})),Q=$.models?.availableModels?.map((G)=>({id:G.modelId,name:G.name??null,description:G.description??null})),j=J&&J.length>0?J:Q&&Q.length>0?Q:null,W=(typeof X?.currentValue==="string"?X.currentValue:null)??$.models?.currentModelId??null;return{models:j,currentModelId:W,configOptions:Z,modelConfigId:X?.id??null}}function W3($,Z){let X=[];if($.trim())X.push({type:"text",text:$});for(let J of Z)if(J.content.trim())X.push({type:"text",text:J.content});return X}function K3($){if(typeof $.value==="boolean")return{sessionId:$.sessionId,configId:$.configId,type:"boolean",value:$.value};return{sessionId:$.sessionId,configId:$.configId,value:$.value}}function V3($,Z){let X=W$({configOptions:Z.configOptions});return{models:X.models??$.models,currentModelId:X.currentModelId??$.currentModelId,configOptions:X.configOptions,modelConfigId:X.modelConfigId??$.modelConfigId}}async function e$($){try{let Z=await $.conn.setSessionConfigOption(K3({sessionId:$.sessionId,configId:$.configId,value:$.value}));return V3($.metadata,Z)}catch(Z){return console.warn(`[acp] setSessionConfigOption(${$.label}) failed; using agent default.`,Z instanceof Error?Z.message:Z),$.metadata}}async function K$($){let Z=r0($.spawn),X=[],J=0,Q=4096;Z.stderr.on("data",(x)=>{X.push(x),J+=x.length;while(J>Q*2&&X.length>1){let R=X.shift();if(R)J-=R.length}});let j=()=>{let x=Buffer.concat(X).toString("utf8");return x.length>Q?x.slice(-Q):x},W=d$($.spawn.cwd),G={current:null},V={at:Date.now()},z=W1(Z.stdin),D=j1(Z.stdout),U=Z1(z,D),B=new $1((x)=>({requestPermission:async(R)=>{let b=R.options,S=b?.find((n)=>n.kind==="allow_once")??b?.find((n)=>n.kind==="allow_always");if(S)return{outcome:{outcome:"selected",optionId:S.optionId}};return{outcome:{outcome:"cancelled"}}},sessionUpdate:async(R)=>{V.at=Date.now();let b=G.current;if(b)await b.onSessionUpdate(R)},readTextFile:W.readTextFile}),U),k=()=>Z.kill();if($.signal)$.signal.addEventListener("abort",k);let q=()=>{if($.signal)$.signal.removeEventListener("abort",k)},M=async(x,R)=>{if(q(),Z.kill(),X1(x))throw new X0(x.message??"Authentication required",R);if(x instanceof Error&&x.message==="ACP connection closed"){if($.signal?.aborted)throw new L("ACP session cancelled: abort signal received");let b=await Promise.race([Z.processCrashed,Z.waitForExit,new Promise((S)=>setTimeout(()=>S({stderr:""}),500))]);throw new L(`Runtime process exited before completion: ${b.stderr?.trim()??""}`)}throw x},Y;try{Y=await B.initialize({protocolVersion:Y0,clientCapabilities:t0,clientInfo:J1()})}catch(x){throw await M(x,[]),Error("unreachable")}try{Q1(Y.protocolVersion)}catch(x){throw q(),Z.kill(),x}let P=Y.authMethods?.map((x)=>x.id)??[],O;try{O=await B.newSession({cwd:$.spawn.cwd,mcpServers:$.mcpServers??[]})}catch(x){throw await M(x,P),Error("unreachable")}let A=O.sessionId,f=W$(O),F=$.modelId??null;if(F&&F!==f.currentModelId)if(f.modelConfigId){let x=f;if(f=await e$({conn:B,sessionId:A,configId:f.modelConfigId,value:F,metadata:f,label:`${f.modelConfigId}=${F}`}),f===x)try{await B.unstable_setSessionModel({sessionId:A,modelId:F})}catch(R){console.warn(`[acp] setSessionModel(${F}) failed; using agent default.`,R instanceof Error?R.message:R)}}else try{await B.unstable_setSessionModel({sessionId:A,modelId:F})}catch(x){console.warn(`[acp] setSessionModel(${F}) failed; using agent default.`,x instanceof Error?x.message:x)}for(let[x,R]of Object.entries($.configValues??{}).sort(([b],[S])=>b.localeCompare(S))){if(x===f.modelConfigId)continue;if(!f.configOptions.some((b)=>b.id===x))continue;f=await e$({conn:B,sessionId:A,configId:x,value:R,metadata:f,label:`${x}=${String(R)}`})}q();let K=!1,C=async()=>{if(K)return;K=!0,Z.kill(),await Promise.race([Z.waitForExit,new Promise((x)=>setTimeout(x,2000))])},w=!1;return Z.processCrashed.then(()=>{w=!0}),{sessionId:A,availableModels:f.models,currentModelId:f.currentModelId,availableConfigOptions:f.configOptions,isAlive:()=>!K&&!w,dispose:C,prompt:(x)=>Y3({proc:Z,conn:B,sessionId:A,activeAdapter:G,lastUpdate:V,collectStderrTail:j},x)}}async function Y3($,Z){let{proc:X,conn:J,sessionId:Q,activeAdapter:j,lastUpdate:W}=$,G={boardId:Z.boardId,cardId:Z.cardId,runId:Z.runId},V=o$({ctx:G,appendText:Z.appendText,appendToolActivity:Z.appendToolActivity});j.current=V,W.at=Date.now();let z=!1,D=null,U=()=>X.kill();Z.signal.addEventListener("abort",U);try{let H=async()=>{let M=Z.heartbeatMs??X3,Y=Math.min(J3,Math.max(50,Math.floor(M/2)));D=setInterval(()=>{let O=Z.isAwaitingInput?.()??!1;if(O){W.at=Date.now();return}if(r$({nowMs:Date.now(),lastUpdateMs:W.at,windowMs:M,awaitingInput:O}))z=!0,J.cancel({sessionId:Q}).catch(()=>{return}),X.kill()},Y);let P=W3(Z.prompt.systemPrompt,Z.prompt.messages);await J.prompt({sessionId:Q,prompt:P}),V.flush({completeOpenToolCalls:!0})},B=X.waitForExit.then((M)=>({kind:"exited",exit:M})),k=H().then(()=>({kind:"ok"})).catch(async(M)=>{if(M instanceof Error&&M.message==="ACP connection closed")return{kind:"crashed",exit:await Promise.race([X.processCrashed,X.waitForExit])};throw M}),q=await Promise.race([k,X.processCrashed.then((M)=>({kind:"crashed",exit:M})),B]);if(z)throw new o;if(q.kind==="crashed")throw new L(`Runtime process exited before completion: ${q.exit.stderr.trim()}`);if(q.kind==="exited"){if(Z.signal.aborted)throw new L("ACP session cancelled: abort signal received");if(z)throw new o;throw new L(`Runtime process exited unexpectedly: ${q.exit.stderr.trim()}`)}return{modelReported:"unknown",finalText:V.getFinalText(),toolCallsExecuted:V.getToolCallCount(),stderrTail:$.collectStderrTail(),updateTypeCounts:V.getUpdateTypeCounts()}}finally{if(D!==null)clearInterval(D);Z.signal.removeEventListener("abort",U),V.flush(),j.current=null}}var M3=30000;async function K1($){let Z=r0($.spawn),X=W1(Z.stdin),J=j1(Z.stdout),Q=Z1(X,J),W=new $1((z)=>({requestPermission:async()=>({outcome:{outcome:"cancelled"}}),sessionUpdate:async()=>{return},readTextFile:async()=>({content:""})}),Q),G=()=>Z.kill();if($.signal)$.signal.addEventListener("abort",G);let V=null;try{let z=(async()=>{let U=await W.initialize({protocolVersion:Y0,clientCapabilities:t0,clientInfo:J1()});Q1(U.protocolVersion);let H=U.authMethods?.map((Y)=>Y.id)??[],B;try{B=await W.newSession({cwd:$.spawn.cwd,mcpServers:[]})}catch(Y){if(X1(Y))throw new X0(Y.message??"Authentication required",H);throw Y}let k=W$(B),q=k.models??[],M=k.currentModelId;return W.cancel({sessionId:B.sessionId}).catch(()=>{return}),{models:q,configOptions:k.configOptions,currentModelId:M}})(),D=new Promise((U,H)=>{V=setTimeout(()=>{H(new o)},M3)});return await Promise.race([z,D])}catch(z){if(z instanceof Error&&z.message==="ACP connection closed"){if($.signal?.aborted)throw new L("ACP model discovery cancelled: abort signal received");throw new L("Runtime process exited during model discovery.")}throw z}finally{if(V)clearTimeout(V);if($.signal)$.signal.removeEventListener("abort",G);Z.kill()}}var E=new Map,p=null;function A0($){$.then((Z)=>Z.dispose()).catch(()=>{return})}function G3(){if(p!==null)return;p=setInterval(()=>{let $=Date.now();for(let[Z,X]of E){if(X.inUse)continue;if($-X.lastUsed<600000)continue;E.delete(Z),A0(X.sessionPromise)}if(E.size===0&&p!==null)clearInterval(p),p=null},60000),p.unref?.()}function z3(){if(E.size<200)return;let $=null,Z=1/0;for(let[J,Q]of E){if(Q.inUse)continue;if(Q.lastUsed>=Z)continue;$=J,Z=Q.lastUsed}if(!$)return;let X=E.get($);if(!X)return;E.delete($),A0(X.sessionPromise)}function V$($,Z,X){return`${$}:${Z}:${X}`}async function Y$($){let Z=E.get($.key);if(Z)if(Z.fingerprint===$.fingerprint){Z.inUse=!0,Z.lastUsed=Date.now();try{let J=await Z.sessionPromise;if(J.isAlive())return J}catch{}E.delete($.key)}else E.delete($.key),A0(Z.sessionPromise);let X=$.open();z3(),E.set($.key,{key:$.key,fingerprint:$.fingerprint,sessionPromise:X,lastUsed:Date.now(),inUse:!0}),G3();try{return await X}catch(J){throw E.delete($.key),J}}function M$($,Z=!1){let X=E.get($);if(!X)return;if(Z){E.delete($),A0(X.sessionPromise);return}X.inUse=!1,X.lastUsed=Date.now()}async function D3(){let $=Array.from(E.values());if(E.clear(),p!==null)clearInterval(p),p=null;await Promise.all($.map((Z)=>Z.sessionPromise.then((X)=>X.dispose()).catch(()=>{return})))}function G$($){let Z=Object.entries($.configValues??{}),X=Z.length>0?JSON.stringify(Object.fromEntries(Z.sort(([J],[Q])=>J.localeCompare(Q)))):"";return[$.adapterId,$.configId??"",$.modelId??"",X,$.mcpUrl??"",$.runFolder??"",$.toolSurface??"full"].join(" ")}var V1=!1;function H3(){if(V1)return;V1=!0;let $=()=>{D3()};process.once("beforeExit",$),process.once("SIGTERM",$),process.once("SIGINT",$)}H3();import M0 from"node:fs/promises";import q3 from"node:path";import{z as N}from"zod";var Y1=N.object({id:N.string(),name:N.string().nullable().default(null),description:N.string().nullable().default(null)}),z1=N.union([N.string(),N.boolean()]),B3=N.object({value:N.string(),name:N.string(),description:N.string().nullable().default(null),group:N.string().nullable().default(null),groupName:N.string().nullable().default(null)}),M1=N.object({id:N.string(),name:N.string(),category:N.string().nullable().default(null),description:N.string().nullable().default(null),type:N.enum(["select","boolean"]),currentValue:z1.nullable().default(null),options:N.array(B3).default([])}),z$=N.record(N.string(),z1).default({}),U3=N.discriminatedUnion("kind",[N.object({id:N.string(),kind:N.literal("registered"),adapterId:N.enum(["claude-code","opencode","codex","copilot"]),displayName:N.string(),runFolder:N.string().nullable(),defaultModelId:N.string().nullable().default(null),defaultConfigValues:z$,discoveredModels:N.array(Y1).default([]),discoveredConfigOptions:N.array(M1).default([]),discoveredAt:N.string().nullable().default(null)}),N.object({id:N.string(),kind:N.literal("custom"),displayName:N.string(),command:N.string(),args:N.array(N.string()),env:N.record(N.string(),N.string()),runFolder:N.string().nullable(),defaultModelId:N.string().nullable().default(null),defaultConfigValues:z$,discoveredModels:N.array(Y1).default([]),discoveredConfigOptions:N.array(M1).default([]),discoveredAt:N.string().nullable().default(null)})]),G1=N.object({runtimeConfigId:N.string(),runFolder:N.string().nullable(),modelId:N.string().nullable().default(null),configValues:z$}),f0=N.object({version:N.literal(1),globalDefaultConfigId:N.string().nullable(),runtimes:N.array(U3),boardOverrides:N.record(N.string(),G1).default({}),cardOverrides:N.record(N.string(),G1).default({})});import N3 from"node:os";import _0 from"node:path";function x3(){let $=N3.homedir();switch(process.platform){case"win32":return process.env.APPDATA??_0.join($,"AppData","Roaming");case"darwin":return _0.join($,"Library","Application Support");default:return process.env.XDG_CONFIG_HOME??_0.join($,".config")}}function O0(){let $=process.env.IDEAWAVE_RUNTIME_CONFIG_PATH;if($)return $;return _0.join(x3(),"ideawave","runtimes.json")}var F3={version:1,globalDefaultConfigId:null,runtimes:[],boardOverrides:{},cardOverrides:{}},D1=new Map;function k3($){return D1.get($)??Promise.resolve()}function C3($,Z){D1.set($,Z)}async function v($){let Z=$??O0(),X;try{X=await M0.readFile(Z,"utf-8")}catch(Q){if(Q.code==="ENOENT")return structuredClone(F3);throw Q}return f0.parse(JSON.parse(X))}async function h($,Z){let X=Z??O0(),J=k3(X).then(async()=>{let Q=await v(X),j=structuredClone(Q),W=await $(j);return await P3(j,X),W});return C3(X,J.catch(()=>{return})),J}async function P3($,Z){let X=Z??O0(),J=q3.dirname(X);await M0.mkdir(J,{recursive:!0});let Q=`${X}.tmp.${process.pid}.${Math.random().toString(36).slice(2)}`,j=JSON.stringify($,null,2);await M0.writeFile(Q,j,"utf-8");try{await M0.rename(Q,X)}catch(W){throw await M0.unlink(Q).catch(()=>{return}),W}}import f3 from"node:fs/promises";import w0 from"node:os";var H1="claude",D$="@agentclientprotocol/claude-agent-acp",B1=["claude-agent-acp","claude-code-acp"],U1="Install Claude Code from https://github.com/anthropics/claude-code",N1={id:"claude-code",displayName:"Claude Code",kind:"registered",resolveSpawnConfig:async()=>{if(!await _(H1))throw new L(`Claude Code CLI not found. ${U1}`);let Z=x0(D$);if(Z)return{command:Z.command,args:Z.args,env:y(Z.env)};let X=J0(D$),J=X?.status==="installing";for(let j of B1){let W=await _(j);if(W&&!J&&X?.status!=="failed")return{command:W,args:[],env:y()}}let Q=X?.status==="failed"?`Claude Code ACP bridge install failed. ${X.error??"Restart IdeaWave CLI so it can warm the pinned bridge again."}`:"Claude Code ACP bridge is not ready. Keep IdeaWave CLI running and retry in a moment.";throw new L(Q)},checkReadiness:async()=>{if(!await _(H1))return{status:"not_installed",detail:U1};let Z=J0(D$);if(Z?.status==="ready")return{status:"ready"};if(Z?.status==="installing")return{status:"not_running",detail:"Installing Claude Code ACP bridge..."};if(Z?.status==="failed")return{status:"not_running",detail:Z.error??"Claude Code ACP bridge install failed. Restart IdeaWave CLI so it can warm the pinned bridge again."};for(let X of B1)if(await _(X))return{status:"ready"};return{status:"not_running",detail:"Installing Claude Code ACP bridge..."}},authHint:"Run `claude login` (or set ANTHROPIC_API_KEY)."};var x1="codex",H$="@zed-industries/codex-acp",q1="codex-acp",F1="Install Codex CLI from https://developers.openai.com/codex/cli",k1={id:"codex",displayName:"Codex",kind:"registered",resolveSpawnConfig:async()=>{if(!await _(x1))throw new L(`Codex CLI not found. ${F1}`);let Z=x0(H$);if(Z)return{command:Z.command,args:Z.args,env:y(Z.env)};let X=J0(H$),J=X?.status==="installing",Q=await _(q1);if(Q&&!J&&X?.status!=="failed")return{command:Q,args:[],env:y()};let j=X?.status==="failed"?`Codex ACP bridge install failed. ${X.error??"Restart IdeaWave CLI so it can warm the pinned bridge again."}`:"Codex ACP bridge is not ready. Keep IdeaWave CLI running and retry in a moment.";throw new L(j)},checkReadiness:async()=>{if(!await _(x1))return{status:"not_installed",detail:F1};let Z=J0(H$);if(Z?.status==="ready")return{status:"ready"};if(Z?.status==="installing")return{status:"not_running",detail:"Installing Codex ACP bridge..."};if(Z?.status==="failed")return{status:"not_running",detail:Z.error??"Codex ACP bridge install failed. Restart IdeaWave CLI so it can warm the pinned bridge again."};if(await _(q1))return{status:"ready"};return{status:"not_running",detail:"Installing Codex ACP bridge..."}},authHint:"Run `codex login` (or set OPENAI_API_KEY)."};var C1="copilot",P1="copilot-acp",L1="npm i -g @github/copilot",A1={id:"copilot",displayName:"GitHub Copilot",kind:"registered",resolveSpawnConfig:async()=>{let $=await _(C1);if($)return{command:$,args:["--acp"],env:y()};let Z=await _(P1);if(Z)return{command:Z,args:[],env:y()};throw new L(`Copilot not installed. Run: ${L1}`)},checkReadiness:async()=>{if(await _(C1))return{status:"ready"};if(await _(P1))return{status:"ready"};return{status:"not_installed",detail:L1}},authHint:"Run `gh auth login` (or `copilot login`)."};import L3 from"node:fs/promises";function A3($){return $.startsWith("/")||/^[a-zA-Z]:[\\/]/.test($)}async function f1($){if(A3($))try{return await L3.access($),!0}catch{return!1}return await _($)!==null}var _1={id:"custom",displayName:"Custom ACP",kind:"custom",resolveSpawnConfig:async($)=>{if($.kind!=="custom")throw new L("customAdapter received non-custom entry");if(!await f1($.command))throw new L(`custom: command "${$.command}" not found`);return{command:$.command,args:$.args,env:y($.env)}},checkReadiness:async($)=>{if($.kind!=="custom")return{status:"unknown"};return await f1($.command)?{status:"ready"}:{status:"not_installed"}}};var O1="opencode",w1="Install OpenCode from https://opencode.ai",R1={id:"opencode",displayName:"OpenCode",kind:"registered",resolveSpawnConfig:async()=>{let $=await _(O1);if(!$)throw new L(`OpenCode not installed. ${w1}`);return{command:$,args:["acp"],env:y()}},checkReadiness:async()=>{if(!await _(O1))return{status:"not_installed",detail:w1};return{status:"ready"}},authHint:"Run `opencode auth login <provider>` (or set the provider's API-key env var)."};var b1={"claude-code":N1,opencode:R1,codex:k1,copilot:A1};function c($){if($.kind==="custom")return _1;let Z=b1[$.adapterId];if(!Z)throw Error(`Unknown registered adapter: ${$.adapterId}`);return Z}function Q0(){return Object.values(b1)}async function E1($){let Z={version:1,globalDefaultConfigId:null,runtimes:[],boardOverrides:{},cardOverrides:{}},X;try{X=await f3.readFile($,"utf-8")}catch(J){if(J.code==="ENOENT")return structuredClone(Z);throw J}return f0.parse(JSON.parse(X))}function B$($,Z,X){return{...$.defaultConfigValues??{},...Z?.configValues??{},...X?.configValues??{}}}async function e($){let Z=$.configPath?await E1($.configPath):await v(),X=$.cardId?Z.cardOverrides[$.cardId]:void 0;if(X?.runtimeConfigId){let Q=Z.runtimes.find((j)=>j.id===X.runtimeConfigId);if(Q)return{entry:Q,adapter:c(Q),runFolder:X.runFolder??Q.runFolder??w0.homedir(),modelId:X.modelId??Q.defaultModelId??null,configValues:B$(Q,void 0,X)}}let J=Z.boardOverrides[$.boardId];if(J?.runtimeConfigId){let Q=Z.runtimes.find((j)=>j.id===J.runtimeConfigId);if(Q)return{entry:Q,adapter:c(Q),runFolder:J.runFolder??Q.runFolder??w0.homedir(),modelId:J.modelId??Q.defaultModelId??null,configValues:B$(Q,J)}}if(Z.globalDefaultConfigId){let Q=Z.runtimes.find((j)=>j.id===Z.globalDefaultConfigId);if(Q)return{entry:Q,adapter:c(Q),runFolder:X?.runFolder??J?.runFolder??Q.runFolder??w0.homedir(),modelId:X?.modelId??J?.modelId??Q.defaultModelId??null,configValues:B$(Q,J,X)}}throw new s}async function T1($,Z={}){let J=(Z.configPath?await E1(Z.configPath):await v()).runtimes.find((Q)=>Q.id===$);if(!J)return null;return{entry:J,adapter:c(J),runFolder:J.runFolder??w0.homedir(),modelId:Z.modelId??J.defaultModelId??null,configValues:{...J.defaultConfigValues??{},...Z.configValues??{}}}}var $0=new Map,U$=new Set;function N$($){$0.get($)?.controller.abort()}function y1($){for(let[Z,X]of $0){if(X.frontendSessionId!==$)continue;X.controller.abort(),$0.delete(Z)}}function I1(){for(let $ of $0.values())$.controller.abort();$0.clear()}function v1($,Z,X,J){if((!Z.availableModels||Z.availableModels.length===0)&&Z.availableConfigOptions.length===0)return;h((Q)=>{let j=Q.runtimes.find((W)=>W.id===$);if(j){if(Z.availableModels&&Z.availableModels.length>0)j.discoveredModels=q0({adapterId:X,models:Z.availableModels,currentModelId:Z.currentModelId??J});if(Z.availableConfigOptions.length>0)j.discoveredConfigOptions=Z.availableConfigOptions}}).catch(()=>{return})}function _3($,Z){return`${$}
|
|
18
|
+
${Z}`)}function Q$($){if(!Array.isArray($))return null;let Z=$.flatMap((X)=>{if(!X||typeof X!=="object")return[];let J=X;if(typeof J.path!=="string"||!J.path.trim())return[];let Q=typeof J.line==="number"?`:${J.line}`:"";return[`${J.path}${Q}`]});return Z.length>0?Z.slice(0,5):null}function o$($){let{ctx:Z,appendText:X,appendToolActivity:J}=$,Q=$.flushIntervalMs??r2,j="",W="",G=0,V=null,z={},D=new Map,U=(F)=>{if(D.set(F.toolCallId,F),!J)return;J({boardId:Z.boardId,cardId:Z.cardId,runId:Z.runId,entry:F}).catch((K)=>{console.warn("[streamAdapter] appendToolActivity failed",K)})},H=(F)=>{let K=`${Z.runId}:thought`,C=D.get(K);if(!C||C.status!=="running")return;U({...C,status:"completed",completedAt:C.completedAt??F})},B=(F)=>{for(let K of D.values()){if(K.status!=="running"||K.rawTitle.toLowerCase()!=="toolsearch")continue;U({...K,status:"completed",completedAt:K.completedAt??F})}},k=(F)=>{for(let K of Array.from(D.values())){if(K.status!=="running")continue;if(K.kind==="thought"||K.rawTitle==="agent_thought")continue;U({...K,status:"completed",completedAt:K.completedAt??F})}},q="",M=0,Y=null,P=null,O=(F)=>{if(F===q)return;q=F,M=Date.now(),X({boardId:Z.boardId,cardId:Z.cardId,runId:Z.runId,text:F}).catch((K)=>{console.warn("[streamAdapter] appendText failed",K)})},A=()=>{if(Y!==null)clearTimeout(Y),Y=null;if(P!==null){let F=P;P=null,O(F)}},f=(F)=>{let K=Date.now()-M;if(K>=Q&&Y===null){O(F);return}if(P=F,Y===null){let C=Math.max(0,Q-K);Y=setTimeout(()=>{Y=null,A()},C)}};return{onSessionUpdate:async(F)=>{let K=F.update;switch(z[K.sessionUpdate]=(z[K.sessionUpdate]??0)+1,K.sessionUpdate){case"agent_message_chunk":{let C=new Date().toISOString();if(H(C),K.content.type==="text")j+=K.content.text,f(j);else{let w=`agent_message_chunk:${K.content.type}`;z[w]=(z[w]??0)+1}return}case"agent_thought_chunk":{if(K.content.type!=="text")return;W+=K.content.text;let C=new Date().toISOString(),w=`${Z.runId}:thought`,x=D.get(w);U({toolCallId:w,rawTitle:"agent_thought",title:"Thought",kind:"thought",status:"running",startedAt:x?.startedAt??C,completedAt:null,contentPreview:r(W.trim())});return}case"user_message_chunk":return;case"tool_call":{G+=1;let C=i$(K.title),w=$$(K.status),x=new Date().toISOString();H(x),B(x),U({toolCallId:K.toolCallId,rawTitle:C,title:X$(C,K.rawInput),kind:Z$(K.kind),status:w,startedAt:x,completedAt:w==="running"?null:x,inputPreview:u(K.rawInput),outputPreview:J$(K.rawOutput),contentPreview:P0(K.content),locations:Q$(K.locations)});return}case"tool_call_update":{let C=D.get(K.toolCallId);if(!C){let Z0=i$(K.title),m=$$(K.status),l=new Date().toISOString();U({toolCallId:K.toolCallId,rawTitle:Z0,title:X$(Z0,K.rawInput),kind:Z$(K.kind),status:m,startedAt:l,completedAt:m==="running"?null:l,inputPreview:u(K.rawInput),outputPreview:J$(K.rawOutput),contentPreview:P0(K.content),locations:Q$(K.locations)});return}let w=K.status===void 0||K.status===null?C.status:$$(K.status),x=typeof K.title==="string"&&K.title.trim().length>0?K.title:C.rawTitle,R=typeof K.title==="string"&&K.title.trim().length>0,b=w==="running"?C.completedAt:C.completedAt??new Date().toISOString(),S=K.content===void 0?C.contentPreview:s$(C.contentPreview,P0(K.content)),n=K.rawOutput===void 0?C.outputPreview:s$(C.outputPreview,J$(K.rawOutput));U({...C,rawTitle:x,title:R?X$(x,K.rawInput):C.title,kind:K.kind===void 0?C.kind:Z$(K.kind),status:w,completedAt:b,inputPreview:K.rawInput===void 0?C.inputPreview:u(K.rawInput),outputPreview:n,contentPreview:S,locations:K.locations===void 0?C.locations:Q$(K.locations)});return}case"plan":V=K.entries;return;case"available_commands_update":return;case"current_mode_update":return;default:return}},getFinalText:()=>j,getToolCallCount:()=>G,getLatestPlan:()=>V,getUpdateTypeCounts:()=>({...z}),flush:(F)=>{let K=new Date().toISOString();if(H(K),F?.completeOpenToolCalls)k(K);A()}}}var Y0=1,t$=-32000;function r$($){if($.awaitingInput)return!1;return $.nowMs-$.lastUpdateMs>$.windowMs}function X1($){if(!$||typeof $!=="object")return!1;return $.code===t$}var X3=300000,J3=5000;function J1(){return{name:"ideawave-cli",title:"IdeaWave CLI",version:"0.0.35".trim()?"0.0.35":"dev"}}function Q1($){if($===Y0)return;throw new L(`Unsupported ACP protocol version ${$}; expected ${Y0}`)}function j1($){return $3.toWeb($)}function W1($){return Z3.toWeb($)}function j$($){return $!==null&&typeof $==="object"}function Q3($){if(!Array.isArray($))return[];let Z=[];for(let X of $){if(!j$(X))continue;if(typeof X.group==="string"&&Array.isArray(X.options)){let J=typeof X.name==="string"?X.name:X.group;for(let Q of X.options){if(!j$(Q))continue;if(typeof Q.value!=="string"||typeof Q.name!=="string")continue;Z.push({value:Q.value,name:Q.name,description:typeof Q.description==="string"?Q.description:null,group:X.group,groupName:J})}continue}if(typeof X.value!=="string"||typeof X.name!=="string")continue;Z.push({value:X.value,name:X.name,description:typeof X.description==="string"?X.description:null,group:null,groupName:null})}return Z}function j3($){if(!j$($))return null;if(typeof $.id!=="string"||typeof $.name!=="string")return null;let Z=typeof $.category==="string"?$.category:null,X=typeof $.description==="string"?$.description:null;if($.type==="select"){let J=typeof $.currentValue==="string"?$.currentValue:null;return{id:$.id,name:$.name,category:Z,description:X,type:"select",currentValue:J,options:Q3($.options)}}if($.type==="boolean"){let J=typeof $.currentValue==="boolean"?$.currentValue:typeof $.value==="boolean"?$.value:null;return{id:$.id,name:$.name,category:Z,description:X,type:"boolean",currentValue:J,options:[]}}return null}function W$($){let Z=($.configOptions??[]).map(j3).filter((G)=>G!==null),X=Z.find((G)=>G.type==="select"&&G.category==="model")??Z.find((G)=>G.type==="select"&&G.id==="model")??null,J=X?.options.map((G)=>({id:G.value,name:G.name,description:G.description??null})),Q=$.models?.availableModels?.map((G)=>({id:G.modelId,name:G.name??null,description:G.description??null})),j=J&&J.length>0?J:Q&&Q.length>0?Q:null,W=(typeof X?.currentValue==="string"?X.currentValue:null)??$.models?.currentModelId??null;return{models:j,currentModelId:W,configOptions:Z,modelConfigId:X?.id??null}}function W3($,Z){let X=[];if($.trim())X.push({type:"text",text:$});for(let J of Z)if(J.content.trim())X.push({type:"text",text:J.content});return X}function K3($){if(typeof $.value==="boolean")return{sessionId:$.sessionId,configId:$.configId,type:"boolean",value:$.value};return{sessionId:$.sessionId,configId:$.configId,value:$.value}}function V3($,Z){let X=W$({configOptions:Z.configOptions});return{models:X.models??$.models,currentModelId:X.currentModelId??$.currentModelId,configOptions:X.configOptions,modelConfigId:X.modelConfigId??$.modelConfigId}}async function e$($){try{let Z=await $.conn.setSessionConfigOption(K3({sessionId:$.sessionId,configId:$.configId,value:$.value}));return V3($.metadata,Z)}catch(Z){return console.warn(`[acp] setSessionConfigOption(${$.label}) failed; using agent default.`,Z instanceof Error?Z.message:Z),$.metadata}}async function K$($){let Z=r0($.spawn),X=[],J=0,Q=4096;Z.stderr.on("data",(x)=>{X.push(x),J+=x.length;while(J>Q*2&&X.length>1){let R=X.shift();if(R)J-=R.length}});let j=()=>{let x=Buffer.concat(X).toString("utf8");return x.length>Q?x.slice(-Q):x},W=d$($.spawn.cwd),G={current:null},V={at:Date.now()},z=W1(Z.stdin),D=j1(Z.stdout),U=Z1(z,D),B=new $1((x)=>({requestPermission:async(R)=>{let b=R.options,S=b?.find((n)=>n.kind==="allow_once")??b?.find((n)=>n.kind==="allow_always");if(S)return{outcome:{outcome:"selected",optionId:S.optionId}};return{outcome:{outcome:"cancelled"}}},sessionUpdate:async(R)=>{V.at=Date.now();let b=G.current;if(b)await b.onSessionUpdate(R)},readTextFile:W.readTextFile}),U),k=()=>Z.kill();if($.signal)$.signal.addEventListener("abort",k);let q=()=>{if($.signal)$.signal.removeEventListener("abort",k)},M=async(x,R)=>{if(q(),Z.kill(),X1(x))throw new X0(x.message??"Authentication required",R);if(x instanceof Error&&x.message==="ACP connection closed"){if($.signal?.aborted)throw new L("ACP session cancelled: abort signal received");let b=await Promise.race([Z.processCrashed,Z.waitForExit,new Promise((S)=>setTimeout(()=>S({stderr:""}),500))]);throw new L(`Runtime process exited before completion: ${b.stderr?.trim()??""}`)}throw x},Y;try{Y=await B.initialize({protocolVersion:Y0,clientCapabilities:t0,clientInfo:J1()})}catch(x){throw await M(x,[]),Error("unreachable")}try{Q1(Y.protocolVersion)}catch(x){throw q(),Z.kill(),x}let P=Y.authMethods?.map((x)=>x.id)??[],O;try{O=await B.newSession({cwd:$.spawn.cwd,mcpServers:$.mcpServers??[]})}catch(x){throw await M(x,P),Error("unreachable")}let A=O.sessionId,f=W$(O),F=$.modelId??null;if(F&&F!==f.currentModelId)if(f.modelConfigId){let x=f;if(f=await e$({conn:B,sessionId:A,configId:f.modelConfigId,value:F,metadata:f,label:`${f.modelConfigId}=${F}`}),f===x)try{await B.unstable_setSessionModel({sessionId:A,modelId:F})}catch(R){console.warn(`[acp] setSessionModel(${F}) failed; using agent default.`,R instanceof Error?R.message:R)}}else try{await B.unstable_setSessionModel({sessionId:A,modelId:F})}catch(x){console.warn(`[acp] setSessionModel(${F}) failed; using agent default.`,x instanceof Error?x.message:x)}for(let[x,R]of Object.entries($.configValues??{}).sort(([b],[S])=>b.localeCompare(S))){if(x===f.modelConfigId)continue;if(!f.configOptions.some((b)=>b.id===x))continue;f=await e$({conn:B,sessionId:A,configId:x,value:R,metadata:f,label:`${x}=${String(R)}`})}q();let K=!1,C=async()=>{if(K)return;K=!0,Z.kill(),await Promise.race([Z.waitForExit,new Promise((x)=>setTimeout(x,2000))])},w=!1;return Z.processCrashed.then(()=>{w=!0}),{sessionId:A,availableModels:f.models,currentModelId:f.currentModelId,availableConfigOptions:f.configOptions,isAlive:()=>!K&&!w,dispose:C,prompt:(x)=>Y3({proc:Z,conn:B,sessionId:A,activeAdapter:G,lastUpdate:V,collectStderrTail:j},x)}}async function Y3($,Z){let{proc:X,conn:J,sessionId:Q,activeAdapter:j,lastUpdate:W}=$,G={boardId:Z.boardId,cardId:Z.cardId,runId:Z.runId},V=o$({ctx:G,appendText:Z.appendText,appendToolActivity:Z.appendToolActivity});j.current=V,W.at=Date.now();let z=!1,D=null,U=()=>X.kill();Z.signal.addEventListener("abort",U);try{let H=async()=>{let M=Z.heartbeatMs??X3,Y=Math.min(J3,Math.max(50,Math.floor(M/2)));D=setInterval(()=>{let O=Z.isAwaitingInput?.()??!1;if(O){W.at=Date.now();return}if(r$({nowMs:Date.now(),lastUpdateMs:W.at,windowMs:M,awaitingInput:O}))z=!0,J.cancel({sessionId:Q}).catch(()=>{return}),X.kill()},Y);let P=W3(Z.prompt.systemPrompt,Z.prompt.messages);await J.prompt({sessionId:Q,prompt:P}),V.flush({completeOpenToolCalls:!0})},B=X.waitForExit.then((M)=>({kind:"exited",exit:M})),k=H().then(()=>({kind:"ok"})).catch(async(M)=>{if(M instanceof Error&&M.message==="ACP connection closed")return{kind:"crashed",exit:await Promise.race([X.processCrashed,X.waitForExit])};throw M}),q=await Promise.race([k,X.processCrashed.then((M)=>({kind:"crashed",exit:M})),B]);if(z)throw new o;if(q.kind==="crashed")throw new L(`Runtime process exited before completion: ${q.exit.stderr.trim()}`);if(q.kind==="exited"){if(Z.signal.aborted)throw new L("ACP session cancelled: abort signal received");if(z)throw new o;throw new L(`Runtime process exited unexpectedly: ${q.exit.stderr.trim()}`)}return{modelReported:"unknown",finalText:V.getFinalText(),toolCallsExecuted:V.getToolCallCount(),stderrTail:$.collectStderrTail(),updateTypeCounts:V.getUpdateTypeCounts()}}finally{if(D!==null)clearInterval(D);Z.signal.removeEventListener("abort",U),V.flush(),j.current=null}}var M3=30000;async function K1($){let Z=r0($.spawn),X=W1(Z.stdin),J=j1(Z.stdout),Q=Z1(X,J),W=new $1((z)=>({requestPermission:async()=>({outcome:{outcome:"cancelled"}}),sessionUpdate:async()=>{return},readTextFile:async()=>({content:""})}),Q),G=()=>Z.kill();if($.signal)$.signal.addEventListener("abort",G);let V=null;try{let z=(async()=>{let U=await W.initialize({protocolVersion:Y0,clientCapabilities:t0,clientInfo:J1()});Q1(U.protocolVersion);let H=U.authMethods?.map((Y)=>Y.id)??[],B;try{B=await W.newSession({cwd:$.spawn.cwd,mcpServers:[]})}catch(Y){if(X1(Y))throw new X0(Y.message??"Authentication required",H);throw Y}let k=W$(B),q=k.models??[],M=k.currentModelId;return W.cancel({sessionId:B.sessionId}).catch(()=>{return}),{models:q,configOptions:k.configOptions,currentModelId:M}})(),D=new Promise((U,H)=>{V=setTimeout(()=>{H(new o)},M3)});return await Promise.race([z,D])}catch(z){if(z instanceof Error&&z.message==="ACP connection closed"){if($.signal?.aborted)throw new L("ACP model discovery cancelled: abort signal received");throw new L("Runtime process exited during model discovery.")}throw z}finally{if(V)clearTimeout(V);if($.signal)$.signal.removeEventListener("abort",G);Z.kill()}}var E=new Map,p=null;function A0($){$.then((Z)=>Z.dispose()).catch(()=>{return})}function G3(){if(p!==null)return;p=setInterval(()=>{let $=Date.now();for(let[Z,X]of E){if(X.inUse)continue;if($-X.lastUsed<600000)continue;E.delete(Z),A0(X.sessionPromise)}if(E.size===0&&p!==null)clearInterval(p),p=null},60000),p.unref?.()}function z3(){if(E.size<200)return;let $=null,Z=1/0;for(let[J,Q]of E){if(Q.inUse)continue;if(Q.lastUsed>=Z)continue;$=J,Z=Q.lastUsed}if(!$)return;let X=E.get($);if(!X)return;E.delete($),A0(X.sessionPromise)}function V$($,Z,X){return`${$}:${Z}:${X}`}async function Y$($){let Z=E.get($.key);if(Z)if(Z.fingerprint===$.fingerprint){Z.inUse=!0,Z.lastUsed=Date.now();try{let J=await Z.sessionPromise;if(J.isAlive())return J}catch{}E.delete($.key)}else E.delete($.key),A0(Z.sessionPromise);let X=$.open();z3(),E.set($.key,{key:$.key,fingerprint:$.fingerprint,sessionPromise:X,lastUsed:Date.now(),inUse:!0}),G3();try{return await X}catch(J){throw E.delete($.key),J}}function M$($,Z=!1){let X=E.get($);if(!X)return;if(Z){E.delete($),A0(X.sessionPromise);return}X.inUse=!1,X.lastUsed=Date.now()}async function D3(){let $=Array.from(E.values());if(E.clear(),p!==null)clearInterval(p),p=null;await Promise.all($.map((Z)=>Z.sessionPromise.then((X)=>X.dispose()).catch(()=>{return})))}function G$($){let Z=Object.entries($.configValues??{}),X=Z.length>0?JSON.stringify(Object.fromEntries(Z.sort(([J],[Q])=>J.localeCompare(Q)))):"";return[$.adapterId,$.configId??"",$.modelId??"",X,$.mcpUrl??"",$.runFolder??"",$.toolSurface??"full"].join(" ")}var V1=!1;function H3(){if(V1)return;V1=!0;let $=()=>{D3()};process.once("beforeExit",$),process.once("SIGTERM",$),process.once("SIGINT",$)}H3();import M0 from"node:fs/promises";import q3 from"node:path";import{z as N}from"zod";var Y1=N.object({id:N.string(),name:N.string().nullable().default(null),description:N.string().nullable().default(null)}),z1=N.union([N.string(),N.boolean()]),B3=N.object({value:N.string(),name:N.string(),description:N.string().nullable().default(null),group:N.string().nullable().default(null),groupName:N.string().nullable().default(null)}),M1=N.object({id:N.string(),name:N.string(),category:N.string().nullable().default(null),description:N.string().nullable().default(null),type:N.enum(["select","boolean"]),currentValue:z1.nullable().default(null),options:N.array(B3).default([])}),z$=N.record(N.string(),z1).default({}),U3=N.discriminatedUnion("kind",[N.object({id:N.string(),kind:N.literal("registered"),adapterId:N.enum(["claude-code","opencode","codex","copilot"]),displayName:N.string(),runFolder:N.string().nullable(),defaultModelId:N.string().nullable().default(null),defaultConfigValues:z$,discoveredModels:N.array(Y1).default([]),discoveredConfigOptions:N.array(M1).default([]),discoveredAt:N.string().nullable().default(null)}),N.object({id:N.string(),kind:N.literal("custom"),displayName:N.string(),command:N.string(),args:N.array(N.string()),env:N.record(N.string(),N.string()),runFolder:N.string().nullable(),defaultModelId:N.string().nullable().default(null),defaultConfigValues:z$,discoveredModels:N.array(Y1).default([]),discoveredConfigOptions:N.array(M1).default([]),discoveredAt:N.string().nullable().default(null)})]),G1=N.object({runtimeConfigId:N.string(),runFolder:N.string().nullable(),modelId:N.string().nullable().default(null),configValues:z$}),f0=N.object({version:N.literal(1),globalDefaultConfigId:N.string().nullable(),runtimes:N.array(U3),boardOverrides:N.record(N.string(),G1).default({}),cardOverrides:N.record(N.string(),G1).default({})});import N3 from"node:os";import _0 from"node:path";function x3(){let $=N3.homedir();switch(process.platform){case"win32":return process.env.APPDATA??_0.join($,"AppData","Roaming");case"darwin":return _0.join($,"Library","Application Support");default:return process.env.XDG_CONFIG_HOME??_0.join($,".config")}}function O0(){let $=process.env.IDEAWAVE_RUNTIME_CONFIG_PATH;if($)return $;return _0.join(x3(),"ideawave","runtimes.json")}var F3={version:1,globalDefaultConfigId:null,runtimes:[],boardOverrides:{},cardOverrides:{}},D1=new Map;function k3($){return D1.get($)??Promise.resolve()}function C3($,Z){D1.set($,Z)}async function v($){let Z=$??O0(),X;try{X=await M0.readFile(Z,"utf-8")}catch(Q){if(Q.code==="ENOENT")return structuredClone(F3);throw Q}return f0.parse(JSON.parse(X))}async function h($,Z){let X=Z??O0(),J=k3(X).then(async()=>{let Q=await v(X),j=structuredClone(Q),W=await $(j);return await P3(j,X),W});return C3(X,J.catch(()=>{return})),J}async function P3($,Z){let X=Z??O0(),J=q3.dirname(X);await M0.mkdir(J,{recursive:!0});let Q=`${X}.tmp.${process.pid}.${Math.random().toString(36).slice(2)}`,j=JSON.stringify($,null,2);await M0.writeFile(Q,j,"utf-8");try{await M0.rename(Q,X)}catch(W){throw await M0.unlink(Q).catch(()=>{return}),W}}import f3 from"node:fs/promises";import w0 from"node:os";var H1="claude",D$="@agentclientprotocol/claude-agent-acp",B1=["claude-agent-acp","claude-code-acp"],U1="Install Claude Code from https://github.com/anthropics/claude-code",N1={id:"claude-code",displayName:"Claude Code",kind:"registered",resolveSpawnConfig:async()=>{if(!await _(H1))throw new L(`Claude Code CLI not found. ${U1}`);let Z=x0(D$);if(Z)return{command:Z.command,args:Z.args,env:y(Z.env)};let X=J0(D$),J=X?.status==="installing";for(let j of B1){let W=await _(j);if(W&&!J&&X?.status!=="failed")return{command:W,args:[],env:y()}}let Q=X?.status==="failed"?`Claude Code ACP bridge install failed. ${X.error??"Restart IdeaWave CLI so it can warm the pinned bridge again."}`:"Claude Code ACP bridge is not ready. Keep IdeaWave CLI running and retry in a moment.";throw new L(Q)},checkReadiness:async()=>{if(!await _(H1))return{status:"not_installed",detail:U1};let Z=J0(D$);if(Z?.status==="ready")return{status:"ready"};if(Z?.status==="installing")return{status:"not_running",detail:"Installing Claude Code ACP bridge..."};if(Z?.status==="failed")return{status:"not_running",detail:Z.error??"Claude Code ACP bridge install failed. Restart IdeaWave CLI so it can warm the pinned bridge again."};for(let X of B1)if(await _(X))return{status:"ready"};return{status:"not_running",detail:"Installing Claude Code ACP bridge..."}},authHint:"Run `claude login` (or set ANTHROPIC_API_KEY)."};var x1="codex",H$="@zed-industries/codex-acp",q1="codex-acp",F1="Install Codex CLI from https://developers.openai.com/codex/cli",k1={id:"codex",displayName:"Codex",kind:"registered",resolveSpawnConfig:async()=>{if(!await _(x1))throw new L(`Codex CLI not found. ${F1}`);let Z=x0(H$);if(Z)return{command:Z.command,args:Z.args,env:y(Z.env)};let X=J0(H$),J=X?.status==="installing",Q=await _(q1);if(Q&&!J&&X?.status!=="failed")return{command:Q,args:[],env:y()};let j=X?.status==="failed"?`Codex ACP bridge install failed. ${X.error??"Restart IdeaWave CLI so it can warm the pinned bridge again."}`:"Codex ACP bridge is not ready. Keep IdeaWave CLI running and retry in a moment.";throw new L(j)},checkReadiness:async()=>{if(!await _(x1))return{status:"not_installed",detail:F1};let Z=J0(H$);if(Z?.status==="ready")return{status:"ready"};if(Z?.status==="installing")return{status:"not_running",detail:"Installing Codex ACP bridge..."};if(Z?.status==="failed")return{status:"not_running",detail:Z.error??"Codex ACP bridge install failed. Restart IdeaWave CLI so it can warm the pinned bridge again."};if(await _(q1))return{status:"ready"};return{status:"not_running",detail:"Installing Codex ACP bridge..."}},authHint:"Run `codex login` (or set OPENAI_API_KEY)."};var C1="copilot",P1="copilot-acp",L1="npm i -g @github/copilot",A1={id:"copilot",displayName:"GitHub Copilot",kind:"registered",resolveSpawnConfig:async()=>{let $=await _(C1);if($)return{command:$,args:["--acp"],env:y()};let Z=await _(P1);if(Z)return{command:Z,args:[],env:y()};throw new L(`Copilot not installed. Run: ${L1}`)},checkReadiness:async()=>{if(await _(C1))return{status:"ready"};if(await _(P1))return{status:"ready"};return{status:"not_installed",detail:L1}},authHint:"Run `gh auth login` (or `copilot login`)."};import L3 from"node:fs/promises";function A3($){return $.startsWith("/")||/^[a-zA-Z]:[\\/]/.test($)}async function f1($){if(A3($))try{return await L3.access($),!0}catch{return!1}return await _($)!==null}var _1={id:"custom",displayName:"Custom ACP",kind:"custom",resolveSpawnConfig:async($)=>{if($.kind!=="custom")throw new L("customAdapter received non-custom entry");if(!await f1($.command))throw new L(`custom: command "${$.command}" not found`);return{command:$.command,args:$.args,env:y($.env)}},checkReadiness:async($)=>{if($.kind!=="custom")return{status:"unknown"};return await f1($.command)?{status:"ready"}:{status:"not_installed"}}};var O1="opencode",w1="Install OpenCode from https://opencode.ai",R1={id:"opencode",displayName:"OpenCode",kind:"registered",resolveSpawnConfig:async()=>{let $=await _(O1);if(!$)throw new L(`OpenCode not installed. ${w1}`);return{command:$,args:["acp"],env:y()}},checkReadiness:async()=>{if(!await _(O1))return{status:"not_installed",detail:w1};return{status:"ready"}},authHint:"Run `opencode auth login <provider>` (or set the provider's API-key env var)."};var b1={"claude-code":N1,opencode:R1,codex:k1,copilot:A1};function c($){if($.kind==="custom")return _1;let Z=b1[$.adapterId];if(!Z)throw Error(`Unknown registered adapter: ${$.adapterId}`);return Z}function Q0(){return Object.values(b1)}async function E1($){let Z={version:1,globalDefaultConfigId:null,runtimes:[],boardOverrides:{},cardOverrides:{}},X;try{X=await f3.readFile($,"utf-8")}catch(J){if(J.code==="ENOENT")return structuredClone(Z);throw J}return f0.parse(JSON.parse(X))}function B$($,Z,X){return{...$.defaultConfigValues??{},...Z?.configValues??{},...X?.configValues??{}}}async function e($){let Z=$.configPath?await E1($.configPath):await v(),X=$.cardId?Z.cardOverrides[$.cardId]:void 0;if(X?.runtimeConfigId){let Q=Z.runtimes.find((j)=>j.id===X.runtimeConfigId);if(Q)return{entry:Q,adapter:c(Q),runFolder:X.runFolder??Q.runFolder??w0.homedir(),modelId:X.modelId??Q.defaultModelId??null,configValues:B$(Q,void 0,X)}}let J=Z.boardOverrides[$.boardId];if(J?.runtimeConfigId){let Q=Z.runtimes.find((j)=>j.id===J.runtimeConfigId);if(Q)return{entry:Q,adapter:c(Q),runFolder:J.runFolder??Q.runFolder??w0.homedir(),modelId:J.modelId??Q.defaultModelId??null,configValues:B$(Q,J)}}if(Z.globalDefaultConfigId){let Q=Z.runtimes.find((j)=>j.id===Z.globalDefaultConfigId);if(Q)return{entry:Q,adapter:c(Q),runFolder:X?.runFolder??J?.runFolder??Q.runFolder??w0.homedir(),modelId:X?.modelId??J?.modelId??Q.defaultModelId??null,configValues:B$(Q,J,X)}}throw new s}async function T1($,Z={}){let J=(Z.configPath?await E1(Z.configPath):await v()).runtimes.find((Q)=>Q.id===$);if(!J)return null;return{entry:J,adapter:c(J),runFolder:J.runFolder??w0.homedir(),modelId:Z.modelId??J.defaultModelId??null,configValues:{...J.defaultConfigValues??{},...Z.configValues??{}}}}var $0=new Map,U$=new Set;function N$($){$0.get($)?.controller.abort()}function y1($){for(let[Z,X]of $0){if(X.frontendSessionId!==$)continue;X.controller.abort(),$0.delete(Z)}}function I1(){for(let $ of $0.values())$.controller.abort();$0.clear()}function v1($,Z,X,J){if((!Z.availableModels||Z.availableModels.length===0)&&Z.availableConfigOptions.length===0)return;h((Q)=>{let j=Q.runtimes.find((W)=>W.id===$);if(j){if(Z.availableModels&&Z.availableModels.length>0)j.discoveredModels=q0({adapterId:X,models:Z.availableModels,currentModelId:Z.currentModelId??J});if(Z.availableConfigOptions.length>0)j.discoveredConfigOptions=Z.availableConfigOptions}}).catch(()=>{return})}function _3($,Z){return`${$}
|
|
19
19
|
|
|
20
20
|
Runtime run folder:
|
|
21
21
|
- Effective run folder: ${Z}
|
|
@@ -25,4 +25,4 @@ Runtime run folder:
|
|
|
25
25
|
Connection: close\r
|
|
26
26
|
Content-Length: ${Buffer.byteLength(X)}\r
|
|
27
27
|
\r
|
|
28
|
-
${X}`),$.destroy()}function l3($,Z){return new Promise((X,J)=>{let Q=(W)=>{$.off("listening",j),J(W)},j=()=>{$.off("error",Q);let W=$.address();if(!W||typeof W==="string"){J(Error("Failed to read CLI bridge server address"));return}X(W.port)};$.once("error",Q),$.once("listening",j),$.listen(Z,"127.0.0.1")})}function a1($){return new Promise((Z,X)=>{$.close((J)=>{if(J)X(J);else Z()})})}async function n1($,Z=q$()){let X=new Map,J=new Map,Q=$.maxMessageSize??T3,j=!1,W=(H)=>{let B=J.get(H);if(B)clearTimeout(B);J.delete(H)},G=(H)=>{J.delete(H);let{activeRunIds:B}=$.registry.disconnect(H);$.onDisconnect(H,B)},V=(H,B)=>{let k={type:"cli_status",connected:!0,runtime:$.runtime,mcpReady:$.mcpReady(),error:B};if(H){let q=X.get(H);if(q)j0(q,k);return}for(let q of X.values())j0(q,k)},z=(H,B)=>{let k=B.sessionId.trim(),q=B.userId.trim();if(!k||!q){j0(H,{type:"cli_status",connected:!1,runtime:$.runtime,mcpReady:$.mcpReady(),error:"Missing frontend session or user id"}),H.close();return}W(k);let M=X.get(k);if(M&&M!==H)M.close();H.data.frontendSessionId=k,X.set(k,H),$.registry.register({frontendSessionId:k,userId:q,tools:B.tools,send:(Y)=>j0(H,Y)}),V(k),$.onToolMetadata?.(k)},D=(H,B)=>{let k=$.registry.getUserId(H);if(!k)return;switch(B.type){case"bridge_ping":{let q=X.get(H);if(q)j0(q,{type:"bridge_pong"});break}case"tool_metadata":$.registry.setMetadata(H,B.tools),$.onToolMetadata?.(H);break;case"tool_result":$.registry.handleToolResult(H,B);break;case"run_turn":$.registry.markRunActive(H,B.run.runId,B.run.cardId),$.onRunTurn(H,{...B.run,frontendSessionId:H,userId:k});break;case"cancel_run":if($.registry.ownsRun(H,B.runId))$.onCancelRun(H,B.runId);break;case"runtime_rpc":$.onRuntimeRpc(H,B);break;case"warmup_run":$.onWarmupRun(H,{...B,frontendSessionId:H},k);break;default:break}},U=(H,B)=>{let k=u1(H.headers.origin),q=l1(k),M={Vary:"Origin"};if(q&&k)M["Access-Control-Allow-Origin"]=k,M["Access-Control-Allow-Private-Network"]="true";if(H.method==="OPTIONS"){B.writeHead(204,{...M,"Access-Control-Allow-Methods":"GET, OPTIONS","Access-Control-Allow-Headers":"*","Access-Control-Max-Age":"600"}),B.end();return}if(H.method!=="GET"){B.writeHead(405,M),B.end();return}B.writeHead(200,{...M,"content-type":"application/json","cache-control":"no-store"}),B.end(JSON.stringify({ideawave:!0,runtime:$.runtime,mcpReady:$.mcpReady()}))};for(let H of Z){let B=b3((M,Y)=>{if(new URL(M.url??"/","http://127.0.0.1").pathname===S1){U(M,Y);return}Y.writeHead(404,{"content-type":"text/plain"}),Y.end("IdeaWave CLI bridge")}),k=new E3({noServer:!0,maxPayload:Q}),q=(M)=>{if(M.idleTimer)clearTimeout(M.idleTimer);M.idleTimer=setTimeout(()=>{M.close()},y3),M.idleTimer.unref?.()};k.on("connection",(M)=>{let Y=M;Y.data={id:R3(),frontendSessionId:null},Y.idleTimer=null,q(Y),Y.on("ping",()=>q(Y)),Y.on("pong",()=>q(Y)),Y.on("error",()=>{if(Y.idleTimer)clearTimeout(Y.idleTimer);Y.close()}),Y.on("message",(P)=>{q(Y);let O=c3(P);if(Q>0&&O.length>Q)return;let A=R0(O);if(!A)return;let f=Y.data.frontendSessionId;if(!f){if(A.type==="bridge_probe")j0(Y,{type:"bridge_probe_ack",protocolVersion:A.protocolVersion});else if(A.type==="frontend_hello")z(Y,A);return}D(f,A)}),Y.on("close",()=>{if(Y.idleTimer)clearTimeout(Y.idleTimer);let P=Y.data.frontendSessionId;if(!P)return;if(X.get(P)!==Y)return;if(X.delete(P),W(P),j)return;J.set(P,setTimeout(()=>G(P),I3))})}),B.on("upgrade",(M,Y,P)=>{if(new URL(M.url??"/","http://127.0.0.1").pathname!==x$){p1(Y,404,"Not Found");return}if(!l1(u1(M.headers.origin))){p1(Y,403,"Forbidden");return}k.handleUpgrade(M,Y,P,(A)=>{k.emit("connection",A,M)})});try{let M=await l3(B,H);return{url:`ws://127.0.0.1:${M}${x$}`,port:M,sendStatus:V,close:async()=>{j=!0;for(let Y of J.values())clearTimeout(Y);J.clear();for(let Y of k.clients){let P=Y;if(P.idleTimer)clearTimeout(P.idleTimer);P.close()}k.close(),await a1(B).catch(()=>{return})}}}catch(M){if(k.close(),await a1(B).catch(()=>{return}),M?.code==="EADDRINUSE")continue;throw M}}throw Error(`No free IdeaWave CLI bridge port found in ${Z[0]}-${Z.at(-1)}`)}import{Server as X6}from"@modelcontextprotocol/sdk/server/index.js";import{CallToolRequestSchema as J6,ListToolsRequestSchema as Q6}from"@modelcontextprotocol/sdk/types.js";import{randomUUID as p3}from"node:crypto";import{createServer as a3}from"node:http";import{StreamableHTTPServerTransport as d3}from"@modelcontextprotocol/sdk/server/streamableHttp.js";import{isInitializeRequest as r1}from"@modelcontextprotocol/sdk/types.js";var s1=new Set(["127.0.0.1","localhost","[::1]","::1"]);function u3($){let Z=$.trim();if(Z.length===0)return null;if(Z.startsWith("[")){let j=Z.indexOf("]");if(j===-1)return null;let W=Z.slice(0,j+1),G=Z.slice(j+1);if(G.length===0)return{host:W,port:null};if(!G.startsWith(":"))return null;let V=Number(G.slice(1));if(!Number.isInteger(V)||V<=0||V>65535)return null;return{host:W,port:V}}let X=Z.lastIndexOf(":");if(X===-1)return{host:Z,port:null};let J=Z.slice(X+1),Q=Number(J);if(!Number.isInteger(Q)||Q<=0||Q>65535)return{host:Z,port:null};return{host:Z.slice(0,X),port:Q}}function o1($){return s1.has($)||s1.has($.toLowerCase())}function t1($,Z){let X=$.headers.host;if(typeof X!=="string")return{ok:!1,status:403,reason:"missing-host"};let J=u3(X);if(!J)return{ok:!1,status:403,reason:"bad-host"};if(!o1(J.host))return{ok:!1,status:403,reason:"non-loopback-host"};if(J.port!==null&&J.port!==Z)return{ok:!1,status:403,reason:"host-port-mismatch"};let Q=$.headers.origin;if(typeof Q==="string"&&Q.length>0){if(Q!=="null"){let j;try{j=new URL(Q)}catch{return{ok:!1,status:403,reason:"bad-origin"}}if(!o1(j.hostname))return{ok:!1,status:403,reason:"non-loopback-origin"}}}return{ok:!0}}var e1="/mcp",i3=43274,Z2=16777216,n3=1800000,s3=300000,$2=200,o3=-32700,d=-32600,F$=-32603;function g($,Z,X,J,Q){$.statusCode=Z,$.setHeader("content-type","application/json"),$.end(JSON.stringify({jsonrpc:"2.0",error:{code:X,message:J,...Q?{data:Q}:{}},id:null}))}class k$ extends Error{constructor(){super("expected application/json");this.name="BadContentTypeError"}}class C$ extends Error{constructor(){super(`body exceeds ${Z2} bytes`);this.name="BodyTooLargeError"}}async function t3($){if($.method&&$.method!=="POST")return;let Z=$.headers["content-type"];if(typeof Z==="string"&&Z.trim().length>0){if(!Z.toLowerCase().includes("application/json"))throw new k$}let X=0,J=[];for await(let j of $){let W=typeof j==="string"?Buffer.from(j):j;if(X+=W.byteLength,X>Z2)throw new C$;J.push(W)}if(J.length===0)return;let Q=Buffer.concat(J).toString("utf8");if(Q.length===0)return;return JSON.parse(Q)}function r3($){if(Array.isArray($))return $.some(r1);return r1($)}function e3($){if(typeof $==="number")return $;let Z=process.env.IDEAWAVE_MCP_PORT;if(Z&&Z.trim().length>0){let X=Number(Z);if(Number.isInteger(X)&&X>=0&&X<65536)return X}return i3}async function $6($,Z){let X=(Q)=>new Promise((j,W)=>{let G=(z)=>{$.off("listening",V),W(z)},V=()=>{$.off("error",G),j()};$.once("error",G),$.once("listening",V),$.listen(Q,"127.0.0.1")});try{await X(Z)}catch(Q){if(Q.code!=="EADDRINUSE"||Z===0)throw Q;console.warn(`[mcp] port ${Z} in use, falling back to an OS-assigned port; external CLIs will need to re-paste the install command from Settings → External agents`),await X(0)}let J=$.address();if(J==null||typeof J==="string")throw Error("[mcp] failed to read bound address");return J.port}function Z6($,Z){if(Z&&typeof Z==="object"&&!Array.isArray(Z)){let J=Z.params;if(J&&typeof J==="object"&&!Array.isArray(J)){let Q=J.clientInfo;if(Q&&typeof Q==="object"&&!Array.isArray(Q)){let j=Q.name;if(typeof j==="string"&&j.trim().length>0)return j.trim()}}}let X=$.headers["user-agent"];if(typeof X==="string"&&X.trim().length>0)return X.trim();return null}async function X2($){let Z=new Map,X=0;function J(z){let D=Z.get(z);if(D)D.lastActivity=Date.now()}async function Q(z,D){Z.delete(z),await D.transport.close().catch(()=>{return})}async function j(){let z=null,D=1/0;for(let[U,H]of Z)if(H.lastActivity<D)D=H.lastActivity,z=U;if(z!==null){let U=Z.get(z);if(U)console.warn(`[mcp] session cap of ${$2} reached; evicting ${z}`),await Q(z,U)}}let W=setInterval(()=>{let z=Date.now()-n3;for(let[D,U]of Z)if(U.lastActivity<z)console.log(`[mcp] sweeping idle session ${D}`),Q(D,U)},s3);W.unref?.();let G=a3((z,D)=>{(async()=>{let U;try{U=new URL(z.url??"/","http://127.0.0.1").pathname}catch{g(D,400,d,"bad-url");return}if(U!==e1){g(D,404,d,"not-found");return}let H=z.method??"POST";if(H!=="POST"&&H!=="GET"&&H!=="DELETE"){D.setHeader("Allow","GET, POST, DELETE"),g(D,405,d,"method-not-allowed");return}let B=t1(z,X);if(!B.ok){g(D,B.status,d,"forbidden",{reason:B.reason});return}let k=z.headers["mcp-session-id"],q=typeof k==="string"?k:void 0;if(H==="GET"||H==="DELETE"){if(!q||!Z.has(q)){g(D,400,d,"no-session",{hint:"Open a POST initialize first to receive an Mcp-Session-Id."});return}let P=Z.get(q).transport;J(q);try{await P.handleRequest(z,D)}catch(O){if(console.error("[mcp] transport.handleRequest (GET/DELETE) threw",O),!D.headersSent)g(D,500,F$,"internal")}return}let M;try{M=await t3(z)}catch(P){if(P instanceof k$){g(D,415,d,"expected-application-json");return}if(P instanceof C$){g(D,413,d,"body-too-large");return}g(D,400,o3,"invalid-json");return}let Y;if(q&&Z.has(q))Y=Z.get(q).transport,J(q);else if(!q&&M!==void 0&&r3(M)){let P=z.headers["x-ideawave-user-id"],O=typeof P==="string"&&P.trim().length>0?P.trim():null,A=z.headers["x-ideawave-frontend-session-id"],f=typeof A==="string"&&A.trim().length>0?A.trim():null,F=z.headers["x-ideawave-tool-surface"],K=typeof F==="string"&&F.trim()==="read"?"read":"full",C=z.headers["x-ideawave-card-id"],w=typeof C==="string"&&C.trim().length>0?C.trim():null,x=z.headers["x-ideawave-board-id"],R=typeof x==="string"&&x.trim().length>0?x.trim():null,b=z.headers["x-ideawave-run-folder"],S=typeof b==="string"&&b.trim().length>0?b.trim():null,n=Z6(z,M);if(Z.size>=$2)await j();let Z0=null,m=new d3({sessionIdGenerator:()=>p3(),onsessioninitialized:(l)=>{Z0=l,Z.set(l,{transport:m,lastActivity:Date.now()})},onsessionclosed:(l)=>{Z.delete(l)}});m.onclose=()=>{if(m.sessionId)Z.delete(m.sessionId)};try{await $.buildServer({userId:O,frontendSessionId:f,boardId:R,toolSurface:K,cardId:w,runFolder:S,clientHint:n}).connect(m)}catch(l){if(console.error("[mcp] buildServer/connect failed",l),Z0)Z.delete(Z0);await m.close().catch(()=>{return}),g(D,500,F$,"server-init-failed");return}Y=m}else{g(D,400,d,"no-session",{hint:"Non-initialize requests must include a valid Mcp-Session-Id header obtained from a prior initialize."});return}try{await Y.handleRequest(z,D,M)}catch(P){if(console.error("[mcp] transport.handleRequest threw",P),!D.headersSent)g(D,500,F$,"internal")}})().catch((U)=>{console.error("[mcp] unhandled request handler error:",U)})});return X=await $6(G,e3($.port)),{url:`http://127.0.0.1:${X}${e1}`,port:X,close:async()=>{clearInterval(W);for(let z of Z.values())await z.transport.close().catch(()=>{return});Z.clear(),await new Promise((z)=>G.close(()=>z()))}}}var j6={name:"ideawave-board",version:"1.0.0"},E0="list_run_folder",T0="read_run_folder_file",j2=new Set(["list_boards","get_board","list_nodes","list_edges","inspect_node","list_context_sources","read_context_card","read_context_file","read_context_image","list_context_folder","get_board_wiki","search_board","get_card_messages","list_agent_cards","inspect_neighbors",E0,T0]),W6=[{name:E0,description:"List visible files and folders under the selected runtime run folder. Hidden/config folders are omitted.",inputSchema:{type:"object",properties:{path:{type:"string",description:"Optional relative path inside the selected run folder. Defaults to the run-folder root."}},additionalProperties:!1}},{name:T0,description:"Read a text file inside the selected runtime run folder. Hidden/config paths and outside-folder paths are denied.",inputSchema:{type:"object",properties:{path:{type:"string",description:"Relative path to a text file inside the selected run folder."}},required:["path"],additionalProperties:!1}}];function P$($){return{content:[{type:"text",text:$}]}}function K6($){return P$(JSON.stringify($,null,2))}function J2($){return{...P$($),isError:!0}}function V6($,Z,X="full"){return $.getMetadata(Z).filter((J)=>X!=="read"||j2.has(C0(J.name))).map((J)=>({name:J.name,description:J.description,inputSchema:J.inputSchema}))}function Y6($,Z,X={}){let J=V6($,Z,X.toolSurface??"full");return X.includeRunFolderTools?[...J,...W6]:J}function Q2($,Z){if(!$||typeof $!=="object"||Array.isArray($))return null;let X=$[Z];return typeof X==="string"?X:null}async function M6($,Z,X){if(X&&X.trim().length>0)return X.trim();if(!$||!Z)throw Error("Run-folder tools require an active IdeaWave board and card.");return(await e({boardId:$,cardId:Z})).runFolder}async function G6($){let Z=await M6($.boardId,$.cardId,$.runFolder);if($.name===E0){let X=Q2($.args,"path")??"",J=await a$(Z,X);return K6(J)}if($.name===T0){let X=Q2($.args,"path");if(!X)return J2("read_run_folder_file requires a path.");return P$(await k0(Z,X))}return J2(`Unknown local tool "${$.name}"`)}function z6($,Z,X,J,Q,j="full"){let W=new X6(j6,{capabilities:{tools:{}}});return W.setRequestHandler(Q6,async()=>({tools:Y6($,Z,{includeRunFolderTools:Boolean(Q||X&&J),toolSurface:j})})),W.setRequestHandler(J6,async(G)=>{let{name:V,arguments:z}=G.params,D=C0(V);try{if(D===E0||D===T0)return await G6({name:D,args:z,boardId:X,cardId:J,runFolder:Q});if(j==="read"&&!j2.has(D))throw Error(`Tool "${V}" is not available in read-only mode`);let U=await $.callTool(Z,D,z??{},void 0,J??void 0);if(U&&typeof U==="object"&&Array.isArray(U.content))return U;return{content:[{type:"text",text:typeof U==="string"?U:JSON.stringify(U)}]}}catch(U){return{content:[{type:"text",text:U instanceof Error?U.message:"Tool call failed"}],isError:!0}}}),W}function W2($,Z){return X2({buildServer:(X)=>z6($,X.frontendSessionId,X.boardId,X.cardId,X.runFolder,X.toolSurface),port:Z})}import I0 from"node:fs/promises";import A$ from"node:os";import v0 from"node:path";var K2=new Map,L$=new Map,D6=30000;function y0($,Z){let X=K2.get($);if(X&&X.expiresAt>Date.now())return Promise.resolve(X.value);let J=L$.get($);if(J)return J;let Q=Z().then((j)=>{return K2.set($,{value:j,expiresAt:Date.now()+D6}),j}).finally(()=>{L$.delete($)});return L$.set($,Q),Q}async function V2($){let Z=await Promise.all(Q0().map((j)=>y0(j.id,async()=>({adapterId:j.id,...await j.checkReadiness({id:"__probe__",kind:"registered",adapterId:j.id,displayName:j.displayName,runFolder:null,defaultModelId:null,defaultConfigValues:{},discoveredModels:[],discoveredConfigOptions:[],discoveredAt:null})})))),J=(await v($)).runtimes.filter((j)=>j.kind==="custom"),Q=await Promise.all(J.map((j)=>y0(`custom:${j.id}`,async()=>({adapterId:j.id,...await c(j).checkReadiness(j)}))));return[...Z,...Q]}async function Y2($,Z){let X=Q0().find((j)=>j.id===$);if(X)return y0(X.id,async()=>({adapterId:X.id,...await X.checkReadiness({id:"__probe__",kind:"registered",adapterId:X.id,displayName:X.displayName,runFolder:null,defaultModelId:null,defaultConfigValues:{},discoveredModels:[],discoveredConfigOptions:[],discoveredAt:null})}));let Q=(await v(Z)).runtimes.find((j)=>j.id===$);if(!Q||Q.kind!=="custom")return null;return y0(`custom:${Q.id}`,async()=>({adapterId:Q.id,...await c(Q).checkReadiness(Q)}))}function M2(){return`cfg_${crypto.randomUUID().replace(/-/g,"").slice(0,16)}`}async function H6(){let[$,Z]=await Promise.all([v(),V2()]),X=$.globalDefaultConfigId?$.runtimes.find((J)=>J.id===$.globalDefaultConfigId)??null:null;return{runtimes:$.runtimes,globalDefaultConfigId:$.globalDefaultConfigId,globalDefaultEffective:X?{runFolder:X.runFolder??A$.homedir(),modelId:X.defaultModelId??null,configValues:X.defaultConfigValues??{}}:null,knownAdapters:Q0().map((J)=>({id:J.id,displayName:J.displayName})),readiness:Z}}async function B6($){try{let Z=await e({boardId:$.boardId,cardId:$.cardId});return{resolved:{configId:Z.entry.id,adapterId:Z.entry.kind==="registered"?Z.entry.adapterId:null,displayName:Z.entry.displayName,runFolder:Z.runFolder,modelId:Z.modelId,configValues:Z.configValues}}}catch(Z){if(Z instanceof s)return{resolved:null};throw Z}}async function U6($){return h((Z)=>{let X=Z.runtimes.find((j)=>j.kind==="registered"&&j.adapterId===$.adapterId);if(X)return X;let J=Q0().find((j)=>j.id===$.adapterId);if(!J)throw Error(`Unknown adapterId: ${$.adapterId}`);let Q={id:M2(),kind:"registered",adapterId:$.adapterId,displayName:J.displayName,runFolder:null,defaultModelId:null,defaultConfigValues:{},discoveredModels:[],discoveredConfigOptions:[],discoveredAt:null};return Z.runtimes.push(Q),Q})}async function N6($){return h((Z)=>{let X={id:M2(),kind:"custom",displayName:$.displayName,command:$.command,args:$.args,env:$.env,runFolder:$.runFolder,defaultModelId:null,defaultConfigValues:{},discoveredModels:[],discoveredConfigOptions:[],discoveredAt:null};return Z.runtimes.push(X),X})}async function x6($){await h((Z)=>{if(Z.runtimes=Z.runtimes.filter((X)=>X.id!==$.runtimeConfigId),Z.globalDefaultConfigId===$.runtimeConfigId)Z.globalDefaultConfigId=null;for(let X of Object.keys(Z.boardOverrides))if(Z.boardOverrides[X].runtimeConfigId===$.runtimeConfigId)delete Z.boardOverrides[X];for(let X of Object.keys(Z.cardOverrides))if(Z.cardOverrides[X].runtimeConfigId===$.runtimeConfigId)delete Z.cardOverrides[X]})}async function q6($){await h((Z)=>{let X=Z.runtimes.find((J)=>J.id===$.runtimeConfigId);if(!X)throw Error(`Runtime config not found: ${$.runtimeConfigId}`);if(Z.globalDefaultConfigId=$.runtimeConfigId,$.runFolder!==void 0)X.runFolder=$.runFolder;if($.modelId!==void 0)X.defaultModelId=$.modelId;if($.configValues!==void 0)X.defaultConfigValues=$.configValues})}async function F6($){$.boardId,await h((Z)=>{if($.runtimeConfigId===null){delete Z.cardOverrides[$.cardId];return}if(!Z.runtimes.some((Q)=>Q.id===$.runtimeConfigId))throw Error(`Runtime config not found: ${$.runtimeConfigId}`);let J=Z.cardOverrides[$.cardId];Z.cardOverrides[$.cardId]={runtimeConfigId:$.runtimeConfigId,runFolder:$.runFolder!==void 0?$.runFolder:J?.runFolder??null,modelId:$.modelId!==void 0?$.modelId:J?.modelId??null,configValues:$.configValues!==void 0?$.configValues:J?.configValues??{}}})}async function k6($){let X=(await v()).runtimes.find((W)=>W.id===$.runtimeConfigId);if(!X)throw Error(`Runtime config not found: ${$.runtimeConfigId}`);let J=c(X),Q=await J.resolveSpawnConfig(X),j=X.runFolder??A$.homedir();try{let W=await K1({spawn:{...Q,cwd:j}}),G=q0({adapterId:J.id,models:W.models,currentModelId:W.currentModelId}),V=new Date().toISOString();return await h((z)=>{let D=z.runtimes.find((U)=>U.id===$.runtimeConfigId);if(D)D.discoveredModels=G,D.discoveredConfigOptions=W.configOptions,D.discoveredAt=V}),{ok:!0,models:G,configOptions:W.configOptions,currentModelId:W.currentModelId}}catch(W){if(W instanceof X0)throw Error(`${W.message}${J.authHint?` ${J.authHint}`:""}`);if(W instanceof L||W instanceof o)throw Error(W.message);throw W}}async function C6($){let Z=A$.homedir(),X=$.path?.trim()?$.path.trim():Z,J=v0.resolve(X),Q=!0;try{if(!(await I0.stat(J)).isDirectory())J=v0.dirname(J)}catch{Q=!1,J=Z}let j=[];try{j=(await I0.readdir(J,{withFileTypes:!0})).filter((V)=>V.isDirectory()&&!V.name.startsWith(".")).map((V)=>({name:V.name,path:v0.join(J,V.name)})).sort((V,z)=>V.name.localeCompare(z.name))}catch{j=[]}let W=v0.dirname(J);return{path:J,parent:W===J?null:W,home:Z,exists:Q,entries:j}}var G2={list:H6,getEffectiveForCard:B6,registerRegistered:U6,registerCustom:N6,unregister:x6,setUserDefault:q6,setCardOverride:F6,refreshModels:k6,browseFolder:C6};var P6=[{id:"claude-code",displayName:"Claude Code"},{id:"codex",displayName:"Codex"},{id:"copilot",displayName:"Copilot"},{id:"opencode",displayName:"OpenCode"}];function L6(){return`cfg_${crypto.randomUUID().replace(/-/g,"").slice(0,16)}`}async function z2(){try{let $=await v();if($.globalDefaultConfigId){if($.runtimes.some((X)=>X.id===$.globalDefaultConfigId))return $.globalDefaultConfigId}for(let Z of P6){if((await Y2(Z.id))?.status!=="ready")continue;let J=null;if(await h((Q)=>{if(Q.globalDefaultConfigId){if(Q.runtimes.some((G)=>G.id===Q.globalDefaultConfigId)){J=Q.globalDefaultConfigId;return}}let j=Q.runtimes.find((W)=>W.kind==="registered"&&W.adapterId===Z.id);if(!j){let W={id:L6(),kind:"registered",adapterId:Z.id,displayName:Z.displayName,runFolder:null,defaultModelId:null,defaultConfigValues:{},discoveredModels:[],discoveredConfigOptions:[],discoveredAt:null};Q.runtimes.push(W),j=W}Q.globalDefaultConfigId=j.id,J=j.id}),J)return console.log(`[ideawave-cli] using agent runtime: ${Z.displayName}`),J}return console.warn("[ideawave-cli] no agent runtime detected on PATH (tried claude, codex, gh copilot, opencode). Install one to run agents."),null}catch($){return console.warn("[ideawave-cli] runtime auto-seed failed",$),null}}function A6(){if(typeof crypto<"u"&&"randomUUID"in crypto)return`call_${crypto.randomUUID()}`;return`call_${Date.now()}_${Math.random().toString(16).slice(2)}`}class f${sessions=new Map;createCallId;toolCallTimeoutMs;constructor($={}){this.createCallId=$.createCallId??A6,this.toolCallTimeoutMs=$.toolCallTimeoutMs??120000}register($){let Z=this.sessions.get($.frontendSessionId);if(Z){if(Z.userId!==$.userId)this.rejectPending(Z,"frontend session user changed"),Z.activeRunIds.clear();this.sessions.set($.frontendSessionId,{frontendSessionId:$.frontendSessionId,userId:$.userId,tools:$.tools,send:$.send,pendingToolCalls:Z.pendingToolCalls,activeRunIds:Z.activeRunIds,activeRunCardIds:Z.activeRunCardIds});return}this.sessions.set($.frontendSessionId,{frontendSessionId:$.frontendSessionId,userId:$.userId,tools:$.tools,send:$.send,pendingToolCalls:new Map,activeRunIds:new Set,activeRunCardIds:new Map})}disconnect($,Z="frontend session disconnected"){let X=this.sessions.get($);if(!X)return{activeRunIds:[]};return this.sessions.delete($),this.rejectPending(X,Z),{activeRunIds:Array.from(X.activeRunIds)}}sessionCount(){return this.sessions.size}has($){return this.sessions.has($)}getUserId($){return this.sessions.get($)?.userId??null}setMetadata($,Z){let X=this.sessions.get($);if(X)X.tools=Z}getMetadata($){if(!$)return[];return this.sessions.get($)?.tools??[]}sendToSession($,Z){let X=this.sessions.get($);if(!X)return!1;return X.send(Z),!0}callTool($,Z,X,J,Q){if(!$)return Promise.reject(Error("MCP tool call is missing a frontend session id"));let j=this.sessions.get($);if(!j)return Promise.reject(Error(`Frontend session ${$} is not connected`));let W=this.createCallId();return new Promise((G,V)=>{let z=setTimeout(()=>{j.pendingToolCalls.delete(W),V(Error(`Tool "${Z}" timed out waiting for the frontend`))},this.toolCallTimeoutMs),D=J??(Q?this.getActiveRunIdForCard(j,Q):this.getSoleActiveRunId(j)),U=Q??(D?j.activeRunCardIds.get(D):void 0);j.pendingToolCalls.set(W,{resolve:G,reject:V,timer:z}),j.send({type:"tool_call",callId:W,name:Z,args:X,...D?{runId:D}:{},...U?{cardId:U}:{}})})}handleToolResult($,Z){let X=this.sessions.get($);if(!X)return;let J=X.pendingToolCalls.get(Z.callId);if(!J)return;if(X.pendingToolCalls.delete(Z.callId),clearTimeout(J.timer),Z.ok)J.resolve(Z.result);else J.reject(Error(Z.error??"Tool execution failed"))}markRunActive($,Z,X){let J=this.sessions.get($);if(!J)return;if(J.activeRunIds.add(Z),X)J.activeRunCardIds.set(Z,X)}markRunSettled($,Z){let X=this.sessions.get($);if(!X)return;X.activeRunIds.delete(Z),X.activeRunCardIds.delete(Z)}ownsRun($,Z){return this.sessions.get($)?.activeRunIds.has(Z)??!1}rejectPending($,Z){for(let X of $.pendingToolCalls.values())clearTimeout(X.timer),X.reject(Error(Z));$.pendingToolCalls.clear()}getSoleActiveRunId($){return $.activeRunIds.size===1?$.activeRunIds.values().next().value:void 0}getActiveRunIdForCard($,Z){let X=[...$.activeRunCardIds.entries()].filter(([J,Q])=>$.activeRunIds.has(J)&&Q===Z).map(([J])=>J);return X.length===1?X[0]:void 0}}function f6(){return"0.0.34".trim()?"0.0.34":"dev"}var i=process.argv[2];if(i==="--help"||i==="-h"||i==="help")console.log("ideawave — host the local bridge for a running ideawave.app session."),console.log(""),console.log("Usage:"),console.log(" ideawave Serve the browser bridge and MCP forwarder."),console.log(" ideawave --version Print the CLI version."),process.exit(0);if(i==="--version"||i==="-v"||i==="version")console.log(f6()),process.exit(0);if(i)console.error(`Unknown command: ${i}`),console.error("Run `ideawave --help` for usage."),process.exit(1);var z0=new f$,B0=null;function D0($,Z){z0.sendToSession($,Z)}async function _6($,Z){try{let X=G2[Z.method],J=await X(Z.params);D0($,{type:"runtime_rpc_result",requestId:Z.requestId,ok:!0,result:J})}catch(X){D0($,{type:"runtime_rpc_result",requestId:Z.requestId,ok:!1,error:X instanceof Error?X.message:String(X)})}}var H0=!1,G0=null;async function _$(){if(H0)return;if(G0)return G0;return G0=(async()=>{try{let $=await W2(z0);b$({...$,requestedPort:$.port}),console.log(`[ideawave-cli] MCP endpoint for the agent: ${$.url}`);try{let Z=await l$();for(let X of Z)if(X.status==="failed")console.warn(`[ideawave-cli] ACP bridge setup failed for ${X.packageName}: ${X.error??"unknown error"}`)}catch(Z){console.warn("[ideawave-cli] ACP bridge setup skipped:",Z instanceof Error?Z.message:String(Z))}await z2(),H0=!0,console.log("[ideawave-cli] ready — agent runs will drive the board."),B0?.sendStatus()}catch($){console.error("[ideawave-cli] failed to start agent host:",$),B0?.sendStatus(void 0,$ instanceof Error?$.message:String($))}finally{G0=null}})(),G0}if(process.env.APP_ENV!=="development"){let $=await i1();if($)console.error(`[ideawave-cli] An IdeaWave CLI is already running at ${$}.`),console.error("[ideawave-cli] Only one CLI per machine is supported — it already serves every browser tab. Exiting."),process.exit(0)}B0=await n1({registry:z0,runtime:null,mcpReady:()=>H0,onToolMetadata:($)=>{console.log(`[ideawave-cli] session ${$} advertised ${z0.getMetadata($).length} tools`),_$()},onRunTurn:($,Z)=>{console.log(`[ideawave-cli] running agent turn for card ${Z.cardId} (session ${$})`),(async()=>{if(await _$(),!H0){D0($,{type:"run_settled",runId:Z.runId,cardId:Z.cardId,result:{assistantContent:"",model:null,latencyMs:0,errorText:"IdeaWave CLI agent host is not ready",partialText:null}});return}await g1(Z,(X)=>D0($,X))})().catch(()=>{return}).finally(()=>z0.markRunSettled($,Z.runId))},onCancelRun:($,Z)=>{N$(Z)},onRuntimeRpc:($,Z)=>{_6($,Z)},onWarmupRun:($,Z,X)=>{(async()=>{if(await _$(),!H0)return{status:"retryable",error:"agent host is not ready"};return h1({cardId:Z.cardId,boardId:Z.boardId,userId:X,frontendSessionId:$})})().then((J)=>{if(!Z.warmupId)return;D0($,{type:"warmup_result",warmupId:Z.warmupId,cardId:Z.cardId,...J})})},onDisconnect:($,Z)=>{y1($);for(let X of Z)N$(X);console.log(`[ideawave-cli] browser session ${$} disconnected`)}});console.log(`[ideawave-cli] browser bridge listening on ${B0.url}`);console.log("[ideawave-cli] waiting for ideawave.app browser sessions…");var D2=!1;for(let $ of["SIGINT","SIGTERM"])process.once($,()=>{if(D2)return;D2=!0,I1(),(async()=>{await B0?.close().catch(()=>{return}),process.exit(0)})()});
|
|
28
|
+
${X}`),$.destroy()}function l3($,Z){return new Promise((X,J)=>{let Q=(W)=>{$.off("listening",j),J(W)},j=()=>{$.off("error",Q);let W=$.address();if(!W||typeof W==="string"){J(Error("Failed to read CLI bridge server address"));return}X(W.port)};$.once("error",Q),$.once("listening",j),$.listen(Z,"127.0.0.1")})}function a1($){return new Promise((Z,X)=>{$.close((J)=>{if(J)X(J);else Z()})})}async function n1($,Z=q$()){let X=new Map,J=new Map,Q=$.maxMessageSize??T3,j=!1,W=(H)=>{let B=J.get(H);if(B)clearTimeout(B);J.delete(H)},G=(H)=>{J.delete(H);let{activeRunIds:B}=$.registry.disconnect(H);$.onDisconnect(H,B)},V=(H,B)=>{let k={type:"cli_status",connected:!0,runtime:$.runtime,mcpReady:$.mcpReady(),error:B};if(H){let q=X.get(H);if(q)j0(q,k);return}for(let q of X.values())j0(q,k)},z=(H,B)=>{let k=B.sessionId.trim(),q=B.userId.trim();if(!k||!q){j0(H,{type:"cli_status",connected:!1,runtime:$.runtime,mcpReady:$.mcpReady(),error:"Missing frontend session or user id"}),H.close();return}W(k);let M=X.get(k);if(M&&M!==H)M.close();H.data.frontendSessionId=k,X.set(k,H),$.registry.register({frontendSessionId:k,userId:q,tools:B.tools,send:(Y)=>j0(H,Y)}),V(k),$.onToolMetadata?.(k)},D=(H,B)=>{let k=$.registry.getUserId(H);if(!k)return;switch(B.type){case"bridge_ping":{let q=X.get(H);if(q)j0(q,{type:"bridge_pong"});break}case"tool_metadata":$.registry.setMetadata(H,B.tools),$.onToolMetadata?.(H);break;case"tool_result":$.registry.handleToolResult(H,B);break;case"run_turn":$.registry.markRunActive(H,B.run.runId,B.run.cardId),$.onRunTurn(H,{...B.run,frontendSessionId:H,userId:k});break;case"cancel_run":if($.registry.ownsRun(H,B.runId))$.onCancelRun(H,B.runId);break;case"runtime_rpc":$.onRuntimeRpc(H,B);break;case"warmup_run":$.onWarmupRun(H,{...B,frontendSessionId:H},k);break;default:break}},U=(H,B)=>{let k=u1(H.headers.origin),q=l1(k),M={Vary:"Origin"};if(q&&k)M["Access-Control-Allow-Origin"]=k,M["Access-Control-Allow-Private-Network"]="true";if(H.method==="OPTIONS"){B.writeHead(204,{...M,"Access-Control-Allow-Methods":"GET, OPTIONS","Access-Control-Allow-Headers":"*","Access-Control-Max-Age":"600"}),B.end();return}if(H.method!=="GET"){B.writeHead(405,M),B.end();return}B.writeHead(200,{...M,"content-type":"application/json","cache-control":"no-store"}),B.end(JSON.stringify({ideawave:!0,runtime:$.runtime,mcpReady:$.mcpReady()}))};for(let H of Z){let B=b3((M,Y)=>{if(new URL(M.url??"/","http://127.0.0.1").pathname===S1){U(M,Y);return}Y.writeHead(404,{"content-type":"text/plain"}),Y.end("IdeaWave CLI bridge")}),k=new E3({noServer:!0,maxPayload:Q}),q=(M)=>{if(M.idleTimer)clearTimeout(M.idleTimer);M.idleTimer=setTimeout(()=>{M.close()},y3),M.idleTimer.unref?.()};k.on("connection",(M)=>{let Y=M;Y.data={id:R3(),frontendSessionId:null},Y.idleTimer=null,q(Y),Y.on("ping",()=>q(Y)),Y.on("pong",()=>q(Y)),Y.on("error",()=>{if(Y.idleTimer)clearTimeout(Y.idleTimer);Y.close()}),Y.on("message",(P)=>{q(Y);let O=c3(P);if(Q>0&&O.length>Q)return;let A=R0(O);if(!A)return;let f=Y.data.frontendSessionId;if(!f){if(A.type==="bridge_probe")j0(Y,{type:"bridge_probe_ack",protocolVersion:A.protocolVersion});else if(A.type==="frontend_hello")z(Y,A);return}D(f,A)}),Y.on("close",()=>{if(Y.idleTimer)clearTimeout(Y.idleTimer);let P=Y.data.frontendSessionId;if(!P)return;if(X.get(P)!==Y)return;if(X.delete(P),W(P),j)return;J.set(P,setTimeout(()=>G(P),I3))})}),B.on("upgrade",(M,Y,P)=>{if(new URL(M.url??"/","http://127.0.0.1").pathname!==x$){p1(Y,404,"Not Found");return}if(!l1(u1(M.headers.origin))){p1(Y,403,"Forbidden");return}k.handleUpgrade(M,Y,P,(A)=>{k.emit("connection",A,M)})});try{let M=await l3(B,H);return{url:`ws://127.0.0.1:${M}${x$}`,port:M,sendStatus:V,close:async()=>{j=!0;for(let Y of J.values())clearTimeout(Y);J.clear();for(let Y of k.clients){let P=Y;if(P.idleTimer)clearTimeout(P.idleTimer);P.close()}k.close(),await a1(B).catch(()=>{return})}}}catch(M){if(k.close(),await a1(B).catch(()=>{return}),M?.code==="EADDRINUSE")continue;throw M}}throw Error(`No free IdeaWave CLI bridge port found in ${Z[0]}-${Z.at(-1)}`)}import{Server as X6}from"@modelcontextprotocol/sdk/server/index.js";import{CallToolRequestSchema as J6,ListToolsRequestSchema as Q6}from"@modelcontextprotocol/sdk/types.js";import{randomUUID as p3}from"node:crypto";import{createServer as a3}from"node:http";import{StreamableHTTPServerTransport as d3}from"@modelcontextprotocol/sdk/server/streamableHttp.js";import{isInitializeRequest as r1}from"@modelcontextprotocol/sdk/types.js";var s1=new Set(["127.0.0.1","localhost","[::1]","::1"]);function u3($){let Z=$.trim();if(Z.length===0)return null;if(Z.startsWith("[")){let j=Z.indexOf("]");if(j===-1)return null;let W=Z.slice(0,j+1),G=Z.slice(j+1);if(G.length===0)return{host:W,port:null};if(!G.startsWith(":"))return null;let V=Number(G.slice(1));if(!Number.isInteger(V)||V<=0||V>65535)return null;return{host:W,port:V}}let X=Z.lastIndexOf(":");if(X===-1)return{host:Z,port:null};let J=Z.slice(X+1),Q=Number(J);if(!Number.isInteger(Q)||Q<=0||Q>65535)return{host:Z,port:null};return{host:Z.slice(0,X),port:Q}}function o1($){return s1.has($)||s1.has($.toLowerCase())}function t1($,Z){let X=$.headers.host;if(typeof X!=="string")return{ok:!1,status:403,reason:"missing-host"};let J=u3(X);if(!J)return{ok:!1,status:403,reason:"bad-host"};if(!o1(J.host))return{ok:!1,status:403,reason:"non-loopback-host"};if(J.port!==null&&J.port!==Z)return{ok:!1,status:403,reason:"host-port-mismatch"};let Q=$.headers.origin;if(typeof Q==="string"&&Q.length>0){if(Q!=="null"){let j;try{j=new URL(Q)}catch{return{ok:!1,status:403,reason:"bad-origin"}}if(!o1(j.hostname))return{ok:!1,status:403,reason:"non-loopback-origin"}}}return{ok:!0}}var e1="/mcp",i3=43274,Z2=16777216,n3=1800000,s3=300000,$2=200,o3=-32700,d=-32600,F$=-32603;function g($,Z,X,J,Q){$.statusCode=Z,$.setHeader("content-type","application/json"),$.end(JSON.stringify({jsonrpc:"2.0",error:{code:X,message:J,...Q?{data:Q}:{}},id:null}))}class k$ extends Error{constructor(){super("expected application/json");this.name="BadContentTypeError"}}class C$ extends Error{constructor(){super(`body exceeds ${Z2} bytes`);this.name="BodyTooLargeError"}}async function t3($){if($.method&&$.method!=="POST")return;let Z=$.headers["content-type"];if(typeof Z==="string"&&Z.trim().length>0){if(!Z.toLowerCase().includes("application/json"))throw new k$}let X=0,J=[];for await(let j of $){let W=typeof j==="string"?Buffer.from(j):j;if(X+=W.byteLength,X>Z2)throw new C$;J.push(W)}if(J.length===0)return;let Q=Buffer.concat(J).toString("utf8");if(Q.length===0)return;return JSON.parse(Q)}function r3($){if(Array.isArray($))return $.some(r1);return r1($)}function e3($){if(typeof $==="number")return $;let Z=process.env.IDEAWAVE_MCP_PORT;if(Z&&Z.trim().length>0){let X=Number(Z);if(Number.isInteger(X)&&X>=0&&X<65536)return X}return i3}async function $6($,Z){let X=(Q)=>new Promise((j,W)=>{let G=(z)=>{$.off("listening",V),W(z)},V=()=>{$.off("error",G),j()};$.once("error",G),$.once("listening",V),$.listen(Q,"127.0.0.1")});try{await X(Z)}catch(Q){if(Q.code!=="EADDRINUSE"||Z===0)throw Q;console.warn(`[mcp] port ${Z} in use, falling back to an OS-assigned port; external CLIs will need to re-paste the install command from Settings → External agents`),await X(0)}let J=$.address();if(J==null||typeof J==="string")throw Error("[mcp] failed to read bound address");return J.port}function Z6($,Z){if(Z&&typeof Z==="object"&&!Array.isArray(Z)){let J=Z.params;if(J&&typeof J==="object"&&!Array.isArray(J)){let Q=J.clientInfo;if(Q&&typeof Q==="object"&&!Array.isArray(Q)){let j=Q.name;if(typeof j==="string"&&j.trim().length>0)return j.trim()}}}let X=$.headers["user-agent"];if(typeof X==="string"&&X.trim().length>0)return X.trim();return null}async function X2($){let Z=new Map,X=0;function J(z){let D=Z.get(z);if(D)D.lastActivity=Date.now()}async function Q(z,D){Z.delete(z),await D.transport.close().catch(()=>{return})}async function j(){let z=null,D=1/0;for(let[U,H]of Z)if(H.lastActivity<D)D=H.lastActivity,z=U;if(z!==null){let U=Z.get(z);if(U)console.warn(`[mcp] session cap of ${$2} reached; evicting ${z}`),await Q(z,U)}}let W=setInterval(()=>{let z=Date.now()-n3;for(let[D,U]of Z)if(U.lastActivity<z)console.log(`[mcp] sweeping idle session ${D}`),Q(D,U)},s3);W.unref?.();let G=a3((z,D)=>{(async()=>{let U;try{U=new URL(z.url??"/","http://127.0.0.1").pathname}catch{g(D,400,d,"bad-url");return}if(U!==e1){g(D,404,d,"not-found");return}let H=z.method??"POST";if(H!=="POST"&&H!=="GET"&&H!=="DELETE"){D.setHeader("Allow","GET, POST, DELETE"),g(D,405,d,"method-not-allowed");return}let B=t1(z,X);if(!B.ok){g(D,B.status,d,"forbidden",{reason:B.reason});return}let k=z.headers["mcp-session-id"],q=typeof k==="string"?k:void 0;if(H==="GET"||H==="DELETE"){if(!q||!Z.has(q)){g(D,400,d,"no-session",{hint:"Open a POST initialize first to receive an Mcp-Session-Id."});return}let P=Z.get(q).transport;J(q);try{await P.handleRequest(z,D)}catch(O){if(console.error("[mcp] transport.handleRequest (GET/DELETE) threw",O),!D.headersSent)g(D,500,F$,"internal")}return}let M;try{M=await t3(z)}catch(P){if(P instanceof k$){g(D,415,d,"expected-application-json");return}if(P instanceof C$){g(D,413,d,"body-too-large");return}g(D,400,o3,"invalid-json");return}let Y;if(q&&Z.has(q))Y=Z.get(q).transport,J(q);else if(!q&&M!==void 0&&r3(M)){let P=z.headers["x-ideawave-user-id"],O=typeof P==="string"&&P.trim().length>0?P.trim():null,A=z.headers["x-ideawave-frontend-session-id"],f=typeof A==="string"&&A.trim().length>0?A.trim():null,F=z.headers["x-ideawave-tool-surface"],K=typeof F==="string"&&F.trim()==="read"?"read":"full",C=z.headers["x-ideawave-card-id"],w=typeof C==="string"&&C.trim().length>0?C.trim():null,x=z.headers["x-ideawave-board-id"],R=typeof x==="string"&&x.trim().length>0?x.trim():null,b=z.headers["x-ideawave-run-folder"],S=typeof b==="string"&&b.trim().length>0?b.trim():null,n=Z6(z,M);if(Z.size>=$2)await j();let Z0=null,m=new d3({sessionIdGenerator:()=>p3(),onsessioninitialized:(l)=>{Z0=l,Z.set(l,{transport:m,lastActivity:Date.now()})},onsessionclosed:(l)=>{Z.delete(l)}});m.onclose=()=>{if(m.sessionId)Z.delete(m.sessionId)};try{await $.buildServer({userId:O,frontendSessionId:f,boardId:R,toolSurface:K,cardId:w,runFolder:S,clientHint:n}).connect(m)}catch(l){if(console.error("[mcp] buildServer/connect failed",l),Z0)Z.delete(Z0);await m.close().catch(()=>{return}),g(D,500,F$,"server-init-failed");return}Y=m}else{g(D,400,d,"no-session",{hint:"Non-initialize requests must include a valid Mcp-Session-Id header obtained from a prior initialize."});return}try{await Y.handleRequest(z,D,M)}catch(P){if(console.error("[mcp] transport.handleRequest threw",P),!D.headersSent)g(D,500,F$,"internal")}})().catch((U)=>{console.error("[mcp] unhandled request handler error:",U)})});return X=await $6(G,e3($.port)),{url:`http://127.0.0.1:${X}${e1}`,port:X,close:async()=>{clearInterval(W);for(let z of Z.values())await z.transport.close().catch(()=>{return});Z.clear(),await new Promise((z)=>G.close(()=>z()))}}}var j6={name:"ideawave-board",version:"1.0.0"},E0="list_run_folder",T0="read_run_folder_file",j2=new Set(["list_boards","get_board","list_nodes","list_edges","inspect_node","list_context_sources","read_context_card","read_context_file","read_context_image","list_context_folder","get_board_wiki","search_board","get_card_messages","list_agent_cards","inspect_neighbors",E0,T0]),W6=[{name:E0,description:"List visible files and folders under the selected runtime run folder. Hidden/config folders are omitted.",inputSchema:{type:"object",properties:{path:{type:"string",description:"Optional relative path inside the selected run folder. Defaults to the run-folder root."}},additionalProperties:!1}},{name:T0,description:"Read a text file inside the selected runtime run folder. Hidden/config paths and outside-folder paths are denied.",inputSchema:{type:"object",properties:{path:{type:"string",description:"Relative path to a text file inside the selected run folder."}},required:["path"],additionalProperties:!1}}];function P$($){return{content:[{type:"text",text:$}]}}function K6($){return P$(JSON.stringify($,null,2))}function J2($){return{...P$($),isError:!0}}function V6($,Z,X="full"){return $.getMetadata(Z).filter((J)=>X!=="read"||j2.has(C0(J.name))).map((J)=>({name:J.name,description:J.description,inputSchema:J.inputSchema}))}function Y6($,Z,X={}){let J=V6($,Z,X.toolSurface??"full");return X.includeRunFolderTools?[...J,...W6]:J}function Q2($,Z){if(!$||typeof $!=="object"||Array.isArray($))return null;let X=$[Z];return typeof X==="string"?X:null}async function M6($,Z,X){if(X&&X.trim().length>0)return X.trim();if(!$||!Z)throw Error("Run-folder tools require an active IdeaWave board and card.");return(await e({boardId:$,cardId:Z})).runFolder}async function G6($){let Z=await M6($.boardId,$.cardId,$.runFolder);if($.name===E0){let X=Q2($.args,"path")??"",J=await a$(Z,X);return K6(J)}if($.name===T0){let X=Q2($.args,"path");if(!X)return J2("read_run_folder_file requires a path.");return P$(await k0(Z,X))}return J2(`Unknown local tool "${$.name}"`)}function z6($,Z,X,J,Q,j="full"){let W=new X6(j6,{capabilities:{tools:{}}});return W.setRequestHandler(Q6,async()=>({tools:Y6($,Z,{includeRunFolderTools:Boolean(Q||X&&J),toolSurface:j})})),W.setRequestHandler(J6,async(G)=>{let{name:V,arguments:z}=G.params,D=C0(V);try{if(D===E0||D===T0)return await G6({name:D,args:z,boardId:X,cardId:J,runFolder:Q});if(j==="read"&&!j2.has(D))throw Error(`Tool "${V}" is not available in read-only mode`);let U=await $.callTool(Z,D,z??{},void 0,J??void 0);if(U&&typeof U==="object"&&Array.isArray(U.content))return U;return{content:[{type:"text",text:typeof U==="string"?U:JSON.stringify(U)}]}}catch(U){return{content:[{type:"text",text:U instanceof Error?U.message:"Tool call failed"}],isError:!0}}}),W}function W2($,Z){return X2({buildServer:(X)=>z6($,X.frontendSessionId,X.boardId,X.cardId,X.runFolder,X.toolSurface),port:Z})}import I0 from"node:fs/promises";import A$ from"node:os";import v0 from"node:path";var K2=new Map,L$=new Map,D6=30000;function y0($,Z){let X=K2.get($);if(X&&X.expiresAt>Date.now())return Promise.resolve(X.value);let J=L$.get($);if(J)return J;let Q=Z().then((j)=>{return K2.set($,{value:j,expiresAt:Date.now()+D6}),j}).finally(()=>{L$.delete($)});return L$.set($,Q),Q}async function V2($){let Z=await Promise.all(Q0().map((j)=>y0(j.id,async()=>({adapterId:j.id,...await j.checkReadiness({id:"__probe__",kind:"registered",adapterId:j.id,displayName:j.displayName,runFolder:null,defaultModelId:null,defaultConfigValues:{},discoveredModels:[],discoveredConfigOptions:[],discoveredAt:null})})))),J=(await v($)).runtimes.filter((j)=>j.kind==="custom"),Q=await Promise.all(J.map((j)=>y0(`custom:${j.id}`,async()=>({adapterId:j.id,...await c(j).checkReadiness(j)}))));return[...Z,...Q]}async function Y2($,Z){let X=Q0().find((j)=>j.id===$);if(X)return y0(X.id,async()=>({adapterId:X.id,...await X.checkReadiness({id:"__probe__",kind:"registered",adapterId:X.id,displayName:X.displayName,runFolder:null,defaultModelId:null,defaultConfigValues:{},discoveredModels:[],discoveredConfigOptions:[],discoveredAt:null})}));let Q=(await v(Z)).runtimes.find((j)=>j.id===$);if(!Q||Q.kind!=="custom")return null;return y0(`custom:${Q.id}`,async()=>({adapterId:Q.id,...await c(Q).checkReadiness(Q)}))}function M2(){return`cfg_${crypto.randomUUID().replace(/-/g,"").slice(0,16)}`}async function H6(){let[$,Z]=await Promise.all([v(),V2()]),X=$.globalDefaultConfigId?$.runtimes.find((J)=>J.id===$.globalDefaultConfigId)??null:null;return{runtimes:$.runtimes,globalDefaultConfigId:$.globalDefaultConfigId,globalDefaultEffective:X?{runFolder:X.runFolder??A$.homedir(),modelId:X.defaultModelId??null,configValues:X.defaultConfigValues??{}}:null,knownAdapters:Q0().map((J)=>({id:J.id,displayName:J.displayName})),readiness:Z}}async function B6($){try{let Z=await e({boardId:$.boardId,cardId:$.cardId});return{resolved:{configId:Z.entry.id,adapterId:Z.entry.kind==="registered"?Z.entry.adapterId:null,displayName:Z.entry.displayName,runFolder:Z.runFolder,modelId:Z.modelId,configValues:Z.configValues}}}catch(Z){if(Z instanceof s)return{resolved:null};throw Z}}async function U6($){return h((Z)=>{let X=Z.runtimes.find((j)=>j.kind==="registered"&&j.adapterId===$.adapterId);if(X)return X;let J=Q0().find((j)=>j.id===$.adapterId);if(!J)throw Error(`Unknown adapterId: ${$.adapterId}`);let Q={id:M2(),kind:"registered",adapterId:$.adapterId,displayName:J.displayName,runFolder:null,defaultModelId:null,defaultConfigValues:{},discoveredModels:[],discoveredConfigOptions:[],discoveredAt:null};return Z.runtimes.push(Q),Q})}async function N6($){return h((Z)=>{let X={id:M2(),kind:"custom",displayName:$.displayName,command:$.command,args:$.args,env:$.env,runFolder:$.runFolder,defaultModelId:null,defaultConfigValues:{},discoveredModels:[],discoveredConfigOptions:[],discoveredAt:null};return Z.runtimes.push(X),X})}async function x6($){await h((Z)=>{if(Z.runtimes=Z.runtimes.filter((X)=>X.id!==$.runtimeConfigId),Z.globalDefaultConfigId===$.runtimeConfigId)Z.globalDefaultConfigId=null;for(let X of Object.keys(Z.boardOverrides))if(Z.boardOverrides[X].runtimeConfigId===$.runtimeConfigId)delete Z.boardOverrides[X];for(let X of Object.keys(Z.cardOverrides))if(Z.cardOverrides[X].runtimeConfigId===$.runtimeConfigId)delete Z.cardOverrides[X]})}async function q6($){await h((Z)=>{let X=Z.runtimes.find((J)=>J.id===$.runtimeConfigId);if(!X)throw Error(`Runtime config not found: ${$.runtimeConfigId}`);if(Z.globalDefaultConfigId=$.runtimeConfigId,$.runFolder!==void 0)X.runFolder=$.runFolder;if($.modelId!==void 0)X.defaultModelId=$.modelId;if($.configValues!==void 0)X.defaultConfigValues=$.configValues})}async function F6($){$.boardId,await h((Z)=>{if($.runtimeConfigId===null){delete Z.cardOverrides[$.cardId];return}if(!Z.runtimes.some((Q)=>Q.id===$.runtimeConfigId))throw Error(`Runtime config not found: ${$.runtimeConfigId}`);let J=Z.cardOverrides[$.cardId];Z.cardOverrides[$.cardId]={runtimeConfigId:$.runtimeConfigId,runFolder:$.runFolder!==void 0?$.runFolder:J?.runFolder??null,modelId:$.modelId!==void 0?$.modelId:J?.modelId??null,configValues:$.configValues!==void 0?$.configValues:J?.configValues??{}}})}async function k6($){let X=(await v()).runtimes.find((W)=>W.id===$.runtimeConfigId);if(!X)throw Error(`Runtime config not found: ${$.runtimeConfigId}`);let J=c(X),Q=await J.resolveSpawnConfig(X),j=X.runFolder??A$.homedir();try{let W=await K1({spawn:{...Q,cwd:j}}),G=q0({adapterId:J.id,models:W.models,currentModelId:W.currentModelId}),V=new Date().toISOString();return await h((z)=>{let D=z.runtimes.find((U)=>U.id===$.runtimeConfigId);if(D)D.discoveredModels=G,D.discoveredConfigOptions=W.configOptions,D.discoveredAt=V}),{ok:!0,models:G,configOptions:W.configOptions,currentModelId:W.currentModelId}}catch(W){if(W instanceof X0)throw Error(`${W.message}${J.authHint?` ${J.authHint}`:""}`);if(W instanceof L||W instanceof o)throw Error(W.message);throw W}}async function C6($){let Z=A$.homedir(),X=$.path?.trim()?$.path.trim():Z,J=v0.resolve(X),Q=!0;try{if(!(await I0.stat(J)).isDirectory())J=v0.dirname(J)}catch{Q=!1,J=Z}let j=[];try{j=(await I0.readdir(J,{withFileTypes:!0})).filter((V)=>V.isDirectory()&&!V.name.startsWith(".")).map((V)=>({name:V.name,path:v0.join(J,V.name)})).sort((V,z)=>V.name.localeCompare(z.name))}catch{j=[]}let W=v0.dirname(J);return{path:J,parent:W===J?null:W,home:Z,exists:Q,entries:j}}var G2={list:H6,getEffectiveForCard:B6,registerRegistered:U6,registerCustom:N6,unregister:x6,setUserDefault:q6,setCardOverride:F6,refreshModels:k6,browseFolder:C6};var P6=[{id:"claude-code",displayName:"Claude Code"},{id:"codex",displayName:"Codex"},{id:"copilot",displayName:"Copilot"},{id:"opencode",displayName:"OpenCode"}];function L6(){return`cfg_${crypto.randomUUID().replace(/-/g,"").slice(0,16)}`}async function z2(){try{let $=await v();if($.globalDefaultConfigId){if($.runtimes.some((X)=>X.id===$.globalDefaultConfigId))return $.globalDefaultConfigId}for(let Z of P6){if((await Y2(Z.id))?.status!=="ready")continue;let J=null;if(await h((Q)=>{if(Q.globalDefaultConfigId){if(Q.runtimes.some((G)=>G.id===Q.globalDefaultConfigId)){J=Q.globalDefaultConfigId;return}}let j=Q.runtimes.find((W)=>W.kind==="registered"&&W.adapterId===Z.id);if(!j){let W={id:L6(),kind:"registered",adapterId:Z.id,displayName:Z.displayName,runFolder:null,defaultModelId:null,defaultConfigValues:{},discoveredModels:[],discoveredConfigOptions:[],discoveredAt:null};Q.runtimes.push(W),j=W}Q.globalDefaultConfigId=j.id,J=j.id}),J)return console.log(`[ideawave-cli] using agent runtime: ${Z.displayName}`),J}return console.warn("[ideawave-cli] no agent runtime detected on PATH (tried claude, codex, gh copilot, opencode). Install one to run agents."),null}catch($){return console.warn("[ideawave-cli] runtime auto-seed failed",$),null}}function A6(){if(typeof crypto<"u"&&"randomUUID"in crypto)return`call_${crypto.randomUUID()}`;return`call_${Date.now()}_${Math.random().toString(16).slice(2)}`}class f${sessions=new Map;createCallId;toolCallTimeoutMs;constructor($={}){this.createCallId=$.createCallId??A6,this.toolCallTimeoutMs=$.toolCallTimeoutMs??120000}register($){let Z=this.sessions.get($.frontendSessionId);if(Z){if(Z.userId!==$.userId)this.rejectPending(Z,"frontend session user changed"),Z.activeRunIds.clear();this.sessions.set($.frontendSessionId,{frontendSessionId:$.frontendSessionId,userId:$.userId,tools:$.tools,send:$.send,pendingToolCalls:Z.pendingToolCalls,activeRunIds:Z.activeRunIds,activeRunCardIds:Z.activeRunCardIds});return}this.sessions.set($.frontendSessionId,{frontendSessionId:$.frontendSessionId,userId:$.userId,tools:$.tools,send:$.send,pendingToolCalls:new Map,activeRunIds:new Set,activeRunCardIds:new Map})}disconnect($,Z="frontend session disconnected"){let X=this.sessions.get($);if(!X)return{activeRunIds:[]};return this.sessions.delete($),this.rejectPending(X,Z),{activeRunIds:Array.from(X.activeRunIds)}}sessionCount(){return this.sessions.size}has($){return this.sessions.has($)}getUserId($){return this.sessions.get($)?.userId??null}setMetadata($,Z){let X=this.sessions.get($);if(X)X.tools=Z}getMetadata($){if(!$)return[];return this.sessions.get($)?.tools??[]}sendToSession($,Z){let X=this.sessions.get($);if(!X)return!1;return X.send(Z),!0}callTool($,Z,X,J,Q){if(!$)return Promise.reject(Error("MCP tool call is missing a frontend session id"));let j=this.sessions.get($);if(!j)return Promise.reject(Error(`Frontend session ${$} is not connected`));let W=this.createCallId();return new Promise((G,V)=>{let z=setTimeout(()=>{j.pendingToolCalls.delete(W),V(Error(`Tool "${Z}" timed out waiting for the frontend`))},this.toolCallTimeoutMs),D=J??(Q?this.getActiveRunIdForCard(j,Q):this.getSoleActiveRunId(j)),U=Q??(D?j.activeRunCardIds.get(D):void 0);j.pendingToolCalls.set(W,{resolve:G,reject:V,timer:z}),j.send({type:"tool_call",callId:W,name:Z,args:X,...D?{runId:D}:{},...U?{cardId:U}:{}})})}handleToolResult($,Z){let X=this.sessions.get($);if(!X)return;let J=X.pendingToolCalls.get(Z.callId);if(!J)return;if(X.pendingToolCalls.delete(Z.callId),clearTimeout(J.timer),Z.ok)J.resolve(Z.result);else J.reject(Error(Z.error??"Tool execution failed"))}markRunActive($,Z,X){let J=this.sessions.get($);if(!J)return;if(J.activeRunIds.add(Z),X)J.activeRunCardIds.set(Z,X)}markRunSettled($,Z){let X=this.sessions.get($);if(!X)return;X.activeRunIds.delete(Z),X.activeRunCardIds.delete(Z)}ownsRun($,Z){return this.sessions.get($)?.activeRunIds.has(Z)??!1}rejectPending($,Z){for(let X of $.pendingToolCalls.values())clearTimeout(X.timer),X.reject(Error(Z));$.pendingToolCalls.clear()}getSoleActiveRunId($){return $.activeRunIds.size===1?$.activeRunIds.values().next().value:void 0}getActiveRunIdForCard($,Z){let X=[...$.activeRunCardIds.entries()].filter(([J,Q])=>$.activeRunIds.has(J)&&Q===Z).map(([J])=>J);return X.length===1?X[0]:void 0}}function f6(){return"0.0.35".trim()?"0.0.35":"dev"}var i=process.argv[2];if(i==="--help"||i==="-h"||i==="help")console.log("ideawave — host the local bridge for a running ideawave.app session."),console.log(""),console.log("Usage:"),console.log(" ideawave Serve the browser bridge and MCP forwarder."),console.log(" ideawave --version Print the CLI version."),process.exit(0);if(i==="--version"||i==="-v"||i==="version")console.log(f6()),process.exit(0);if(i)console.error(`Unknown command: ${i}`),console.error("Run `ideawave --help` for usage."),process.exit(1);var z0=new f$,B0=null;function D0($,Z){z0.sendToSession($,Z)}async function _6($,Z){try{let X=G2[Z.method],J=await X(Z.params);D0($,{type:"runtime_rpc_result",requestId:Z.requestId,ok:!0,result:J})}catch(X){D0($,{type:"runtime_rpc_result",requestId:Z.requestId,ok:!1,error:X instanceof Error?X.message:String(X)})}}var H0=!1,G0=null;async function _$(){if(H0)return;if(G0)return G0;return G0=(async()=>{try{let $=await W2(z0);b$({...$,requestedPort:$.port}),console.log(`[ideawave-cli] MCP endpoint for the agent: ${$.url}`);try{let Z=await l$();for(let X of Z)if(X.status==="failed")console.warn(`[ideawave-cli] ACP bridge setup failed for ${X.packageName}: ${X.error??"unknown error"}`)}catch(Z){console.warn("[ideawave-cli] ACP bridge setup skipped:",Z instanceof Error?Z.message:String(Z))}await z2(),H0=!0,console.log("[ideawave-cli] ready — agent runs will drive the board."),B0?.sendStatus()}catch($){console.error("[ideawave-cli] failed to start agent host:",$),B0?.sendStatus(void 0,$ instanceof Error?$.message:String($))}finally{G0=null}})(),G0}if(process.env.APP_ENV!=="development"){let $=await i1();if($)console.error(`[ideawave-cli] An IdeaWave CLI is already running at ${$}.`),console.error("[ideawave-cli] Only one CLI per machine is supported — it already serves every browser tab. Exiting."),process.exit(0)}B0=await n1({registry:z0,runtime:null,mcpReady:()=>H0,onToolMetadata:($)=>{console.log(`[ideawave-cli] session ${$} advertised ${z0.getMetadata($).length} tools`),_$()},onRunTurn:($,Z)=>{console.log(`[ideawave-cli] running agent turn for card ${Z.cardId} (session ${$})`),(async()=>{if(await _$(),!H0){D0($,{type:"run_settled",runId:Z.runId,cardId:Z.cardId,result:{assistantContent:"",model:null,latencyMs:0,errorText:"IdeaWave CLI agent host is not ready",partialText:null}});return}await g1(Z,(X)=>D0($,X))})().catch(()=>{return}).finally(()=>z0.markRunSettled($,Z.runId))},onCancelRun:($,Z)=>{N$(Z)},onRuntimeRpc:($,Z)=>{_6($,Z)},onWarmupRun:($,Z,X)=>{(async()=>{if(await _$(),!H0)return{status:"retryable",error:"agent host is not ready"};return h1({cardId:Z.cardId,boardId:Z.boardId,userId:X,frontendSessionId:$})})().then((J)=>{if(!Z.warmupId)return;D0($,{type:"warmup_result",warmupId:Z.warmupId,cardId:Z.cardId,...J})})},onDisconnect:($,Z)=>{y1($);for(let X of Z)N$(X);console.log(`[ideawave-cli] browser session ${$} disconnected`)}});console.log(`[ideawave-cli] browser bridge listening on ${B0.url}`);console.log("[ideawave-cli] waiting for ideawave.app browser sessions…");var D2=!1;for(let $ of["SIGINT","SIGTERM"])process.once($,()=>{if(D2)return;D2=!0,I1(),(async()=>{await B0?.close().catch(()=>{return}),process.exit(0)})()});
|