@popmelt.com/core 0.7.1 → 0.7.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/cli.mjs CHANGED
@@ -1,5 +1,5 @@
1
1
  #!/usr/bin/env node
2
- import{spawn as Jo}from"child_process";import{readFile as Bo,unlink as Fo}from"fs/promises";import{join as jo}from"path";import{execFileSync as et,spawn as wn}from"child_process";import{createHash as Pn,randomUUID as Se}from"crypto";import{mkdir as wo,readFile as Sn,readdir as So,stat as bo,unlink as xo,writeFile as Ae}from"fs/promises";import{createServer as ko}from"http";import{tmpdir as Po}from"os";import{basename as bn,dirname as Eo,join as me}from"path";import{fileURLToPath as Ro}from"url";function St(s,t){return`<!DOCTYPE html>
2
+ import{spawn as Jo}from"child_process";import{readFile as Bo,unlink as Fo}from"fs/promises";import{join as jo}from"path";import{execFileSync as et,spawn as Sn}from"child_process";import{createHash as Pn,randomUUID as Se}from"crypto";import{mkdir as wo,readFile as bn,readdir as So,stat as bo,unlink as xo,writeFile as Ae}from"fs/promises";import{createServer as ko}from"http";import{tmpdir as Po}from"os";import{basename as gt,dirname as Eo,join as me}from"path";import{fileURLToPath as Ro}from"url";function bt(s,t){return`<!DOCTYPE html>
3
3
  <html>
4
4
  <head>
5
5
  <meta charset="utf-8">
@@ -26,12 +26,12 @@ import{spawn as Jo}from"child_process";import{readFile as Bo,unlink as Fo}from"f
26
26
  });
27
27
  </script>
28
28
  </body>
29
- </html>`}import{spawn as qn}from"child_process";import{createInterface as Yn}from"readline";var Kn=new Set([".md",".txt",".json",".ts",".tsx",".js",".jsx",".css",".scss",".html",".xml",".yaml",".yml",".toml",".ini",".cfg",".conf",".sh",".bash",".zsh",".py",".rb",".go",".rs",".java",".c",".h",".cpp",".hpp",".swift",".kt",".sql",".graphql",".svg",".env",".gitignore",".prettierrc",".eslintrc"]),bt=1e5;function Qn(s){let t=s.input?.file_path||s.input?.path;if(!t)return;let n=t.includes(".")?`.${t.split(".").pop().toLowerCase()}`:"";if(!Kn.has(n))return;let e;if(s.name==="Write"&&typeof s.input?.content=="string"?e=s.input.content:s.name==="Edit"&&typeof s.input?.new_string=="string"&&(e=s.input.new_string),!!e)return e.length>bt?e.slice(0,bt)+`
30
- \u2026[truncated]`:e}function Be(s,t){let{prompt:n,projectRoot:e,maxTurns:o=40,maxBudgetUsd:i=1,allowedTools:a=["Read","Edit","Write","Glob","Grep","Bash"],claudePath:c="claude",resumeSessionId:m,model:P,timeoutMs:g=3e5,onEvent:B}=t,z=[];m?z.push("--resume",m,"-p",n):z.push("-p",n),z.push("--output-format","stream-json","--verbose","--max-turns",String(o),"--max-budget-usd",String(i)),P&&z.push("--model",P);for(let S of a)z.push("--allowedTools",S);let O=qn(c,z,{cwd:e,stdio:["ignore","pipe","pipe"],env:{...process.env,ANTHROPIC_API_KEY:void 0,CLAUDECODE:void 0}}),L=new Promise(S=>{let X,W=[],Q=[],Z=[],V=!1,q="",D,_,I=!1,k=setTimeout(()=>{I=!0,O.kill("SIGTERM"),setTimeout(()=>{try{O.kill("SIGKILL")}catch{}},5e3)},g),y=Yn({input:O.stdout}),C=new Set;y.on("line",x=>{if(x.trim())try{let b=JSON.parse(x);b.session_id&&!X&&(X=b.session_id);let H=b.type??(b.event?.type?`event.${b.event.type}`:"unknown");if(C.add(H),b.type==="result"&&b.result&&W.length===0){let E=typeof b.result=="string"?b.result:"";E&&(W.push(E),B?.({type:"delta",jobId:s,text:E},s))}if(b.type==="assistant"&&Array.isArray(b.message?.content)){typeof b.error=="string"&&(D=b.error);for(let E of b.message.content){if(E.type==="text"&&E.text&&(W.push(E.text),B?.({type:"delta",jobId:s,text:E.text},s)),E.type==="tool_use"&&E.name){let ie=E.input?.file_path||E.input?.path||void 0,re=Qn(E);if(B?.({type:"tool_use",jobId:s,tool:E.name,...ie?{file:ie}:{},...re?{content:re}:{}},s),E.name==="Edit"&&E.input?.file_path){let ne={tool:"Edit",file_path:E.input.file_path,old_string:E.input.old_string,new_string:E.input.new_string,replace_all:E.input.replace_all};Q.push(ne),Z.push({sequence:Z.length,occurredAt:Date.now(),provider:"claude",source:"claude_tool_use",tool:"Edit",operation:"modify",filePath:ne.file_path,oldString:typeof ne.old_string=="string"?ne.old_string:void 0,newString:typeof ne.new_string=="string"?ne.new_string:void 0,replaceAll:typeof ne.replace_all=="boolean"?ne.replace_all:void 0})}else E.name==="Write"&&E.input?.file_path&&(Q.push({tool:"Write",file_path:E.input.file_path,content:E.input.content}),Z.push({sequence:Z.length,occurredAt:Date.now(),provider:"claude",source:"claude_tool_use",tool:"Write",operation:"create",filePath:E.input.file_path,content:typeof E.input.content=="string"?E.input.content:void 0}))}E.type==="thinking"&&E.thinking&&B?.({type:"thinking",jobId:s,text:E.thinking},s)}}b.type==="result"&&b.is_error===!0&&(V=!0,typeof b.error=="string"&&(D=b.error),typeof b.result=="string"&&(_=b.result)),b.type==="user"&&b.tool_use_result?.file?.filePath&&B?.({type:"tool_use",jobId:s,tool:"Read",file:b.tool_use_result.file.filePath},s)}catch{}});let N=[];O.stderr?.on("data",x=>{N.push(x.toString())}),O.on("close",x=>{if(clearTimeout(k),y.close(),W.length===0&&C.size>0&&console.warn(`[Claude:${s}] No text captured. Event types seen: ${[...C].join(", ")}`),I)V=!0,q=`Timed out after ${Math.round(g/6e4)} minutes`;else if(x!==0&&x!==null){V=!0;let H=N.join("").trim(),E=W.join("").trim(),ie=W.length===0&&C.size>0?` (no text captured, event types: ${[...C].join(", ")})`:"";q=H||_||E||`Claude process exited with code ${x}${ie}`}V&&!q&&_&&(q=_);let b=Vn(q,D);S({sessionId:X,text:W.join(""),success:!V,error:V?b.message:void 0,errorCode:V?b.code:void 0,recoverableError:V?b.recoverable:void 0,fileEdits:Q.length>0?Q:void 0,editEvents:Z.length>0?Z:void 0})}),O.on("error",x=>{clearTimeout(k),V=!0,q=x.message,S({sessionId:X,text:W.join(""),success:!1,error:q,errorCode:"claude_spawn_error",recoverableError:!1,fileEdits:Q.length>0?Q:void 0,editEvents:Z.length>0?Z:void 0})})});return{process:O,result:L}}function Vn(s,t){let n=s.trim()||"Claude CLI failed",e=`${t??""} ${n}`.toLowerCase();return e.includes("authentication_failed")||e.includes("not logged in")||e.includes("please run /login")||e.includes("api key")?{message:n,code:"claude_auth_required",recoverable:!0}:e.includes("model")&&(e.includes("not found")||e.includes("unknown")||e.includes("unavailable"))?{message:n,code:"claude_model_unavailable",recoverable:!0}:{message:n,code:t||"claude_error",recoverable:!1}}import{spawn as Zn,spawnSync as It}from"child_process";import{readFileSync as es,statSync as ts}from"fs";import{basename as ns,extname as ss,isAbsolute as os,relative as Tt,resolve as rs}from"path";import{createInterface as is}from"readline";function xt(s){let t=s.match(/^\/bin\/(?:ba)?sh\s+-\w+\s+"(.*)"$/s);return t?t[1].replace(/\\"/g,'"').replace(/\\\\/g,"\\"):s}var kt=new Set([".md",".txt",".json",".ts",".tsx",".js",".jsx",".css",".scss",".html",".xml",".yaml",".yml",".toml",".ini",".cfg",".conf",".sh",".bash",".zsh",".py",".rb",".go",".rs",".java",".c",".h",".cpp",".hpp",".swift",".kt",".sql",".graphql",".svg",".env",".gitignore",".prettierrc",".eslintrc"]),st=2e5,as=1e5,Mt=2e5;function Ct(s){let t=ns(s).toLowerCase(),n=ss(s).toLowerCase();return kt.has(n)||kt.has(t)}function $t(s,t){let n=os(t)?t:rs(s,t),e=Tt(s,n);return e===""||e.startsWith("..")?null:n}function Fe(s,t){if(!Ct(t))return;let n=$t(s,t);if(n)try{let e=ts(n);return e.isFile()?e.size>st?{exists:!0,truncated:!0}:{exists:!0,content:es(n,"utf8")}:{exists:!1}}catch{return{exists:!1}}}function cs(s,t){for(let n of t){let e=s[n];if(typeof e=="string")return e}}function ls(s){let t=s?.toLowerCase()??"";return t.includes("add")||t.includes("create")?"create":t.includes("delete")||t.includes("remove")?"delete":t.includes("rename")||t.includes("move")?"rename":t.includes("modify")||t.includes("update")||t.includes("edit")?"modify":"unknown"}function Pt(s){return cs(s,["patch","diff","unifiedDiff","unified_diff"])}function ot(s,t){return s.length<=t?{text:s}:{text:s.slice(0,t),truncated:!0}}function Et(s,t){let n=It("git",t,{cwd:s,encoding:"utf8",timeout:5e3,maxBuffer:Mt+65536});return n.status!==0?null:n.stdout??""}function ds(s){let t=new Set;for(let n of s.split(`
31
- `)){if(n.length<4)continue;let e=n.slice(3).trim();if(!e)continue;e.startsWith('"')&&e.endsWith('"')&&(e=e.slice(1,-1));let o=e.split(" -> ");for(let i of o)i&&t.add(i)}return[...t]}function Rt(s){let t=Et(s,["status","--porcelain=v1","-uall"]);if(t===null)return;let n=Et(s,["diff","--no-ext-diff","HEAD","--"])??"",e=ot(n,Mt);return{status:t,changedFiles:ds(t),...e.text?{diff:e.text}:{},...e.truncated?{diffTruncated:!0}:{}}}function us(s,t){return!s||!t?!1:s.status!==t.status||(s.diff??"")!==(t.diff??"")}function ps(s,t){if(!Ct(t))return;let n=$t(s,t);if(!n)return;let e=Tt(s,n);if(e===""||e.startsWith(".."))return;let o=It("git",["show",`HEAD:${e}`],{cwd:s,encoding:"utf8",timeout:5e3,maxBuffer:st+65536});if(o.status!==0)return{exists:!1};let i=o.stdout??"",a=ot(i,st);return{exists:!0,content:a.text,...a.truncated?{truncated:!0}:{}}}function fs(s,t,n){return t.map(e=>{let o=n.has(e)?n.get(e):ps(s,e),i=Fe(s,e);return{filePath:e,...o?{before:o}:{},...i?{after:i}:{}}})}function je(s,t){let{prompt:n,projectRoot:e,screenshotPath:o,resumeSessionId:i,model:a,onEvent:c,sandboxMode:m="workspace-write",configOverrides:P=[]}=t,g=[];if(i){g.push("exec","resume","--skip-git-repo-check","--json","-c",`sandbox_mode="${m}"`);for(let O of P)g.push("-c",O);a&&g.push("-m",a),g.push(i),g.push(n),o&&g.push("--image",o)}else{g.push("exec","--skip-git-repo-check","--json","--sandbox",m);for(let O of P)g.push("-c",O);a&&g.push("-m",a),g.push(n),o&&g.push("--image",o)}let B=Zn("codex",g,{cwd:e,stdio:["ignore","pipe","pipe"],env:{...process.env,CLAUDECODE:void 0}}),z=new Promise(O=>{let L,S=[],X=[],W=[],Q=new Map,Z=new Map,V=!1,q="",D=is({input:B.stdout}),_=new Set;D.on("line",k=>{if(k.trim())try{let y=JSON.parse(k),C=y.type??"unknown";if(_.add(C),C==="thread.started"&&y.thread_id&&!L&&(L=y.thread_id),(C==="item.agentMessage.delta"||C==="item/agentMessage/delta")&&y.delta?.text&&(S.push(y.delta.text),c?.({type:"delta",jobId:s,text:y.delta.text},s)),(C==="item.reasoning.delta"||C==="item/reasoning/delta")&&y.delta?.text&&c?.({type:"thinking",jobId:s,text:y.delta.text},s),(C==="item.started"||C==="item/started")&&y.item){let N=y.item.type;if(N==="command_execution"){let x=y.item.command,b=x?xt(x):void 0,H=Rt(e),E=new Map;for(let re of H?.changedFiles??[])E.set(re,Fe(e,re));typeof y.item.id=="string"&&Z.set(y.item.id,{...x?{rawCommand:x}:{},...b?{command:b}:{},...H?{beforeWorktree:H}:{},beforeSnapshots:E});let ie=b?`Bash: ${b.split(`
32
- `)[0].slice(0,80)}`:"Bash";X.push(ie),c?.({type:"tool_use",jobId:s,tool:"Bash",...b?{content:b}:{}},s)}else if(N==="file_change"){let x=y.item.filename||y.item.path;x&&!Q.has(x)&&Q.set(x,Fe(e,x)),X.push(x?`Edit ${x.split("/").pop()}`:"Edit"),c?.({type:"tool_use",jobId:s,tool:"Edit",...x?{file:x}:{}},s)}else if(N==="file_read"){let x=y.item.filename||y.item.path;X.push(x?`Read ${x.split("/").pop()}`:"Read"),c?.({type:"tool_use",jobId:s,tool:"Read",...x?{file:x}:{}},s)}else if(N==="web_search")X.push("WebSearch"),c?.({type:"tool_use",jobId:s,tool:"WebSearch"},s);else if(N==="mcp_tool_call"){let x=y.item.tool_name||y.item.name||"MCP";X.push(x),c?.({type:"tool_use",jobId:s,tool:x},s)}}if((C==="item.completed"||C==="item/completed")&&y.item){if(y.item.type==="agent_message"){let N=y.item.text;typeof N=="string"&&N&&(S.push(N),c?.({type:"delta",jobId:s,text:N},s))}else if(y.item.type==="reasoning"){let N=y.item.text;typeof N=="string"&&N&&c?.({type:"thinking",jobId:s,text:N},s)}else if(y.item.type==="file_change"&&Array.isArray(y.item.changes))for(let N of y.item.changes){let x=N.path||N.filename,b=N.kind==="add"?"Write":"Edit";if(x){X.push(`${b} ${x.split("/").pop()}`),c?.({type:"tool_use",jobId:s,tool:b,file:x},s);let H=N,E=typeof H.kind=="string"?H.kind:void 0,ie=ls(E),re=Q.get(x),ne=Fe(e,x);W.push({sequence:W.length,occurredAt:Date.now(),provider:"codex",source:"codex_file_change",tool:"FileChange",operation:ie,filePath:x,...re?{before:re}:{},...ne?{after:ne}:{},...Pt(H)?{patch:Pt(H)}:{},...E?{rawKind:E}:{}}),Q.set(x,ne)}}else if(y.item.type==="command_execution"){let N=typeof y.item.id=="string"?y.item.id:void 0,x=N?Z.get(N):void 0,b=typeof y.item.command=="string"?y.item.command:x?.rawCommand,H=b?xt(b):x?.command,E=Rt(e);if(us(x?.beforeWorktree,E)){let ie=[...new Set([...x?.beforeWorktree?.changedFiles??[],...E?.changedFiles??[]])],re=typeof y.item.aggregated_output=="string"?ot(y.item.aggregated_output,as):void 0,ne=typeof y.item.exit_code=="number"?y.item.exit_code:y.item.exit_code===null?null:void 0,Ne=typeof y.item.status=="string"?y.item.status:void 0;W.push({sequence:W.length,occurredAt:Date.now(),provider:"codex",source:"codex_command_execution",tool:"Bash",operation:"command",...b?{rawCommand:b}:{},...H?{command:H}:{},...ne!==void 0?{exitCode:ne}:{},...Ne?{status:Ne}:{},...re?{output:re.text}:{},...re?.truncated?{outputTruncated:!0}:{},...x?.beforeWorktree?{beforeWorktree:x.beforeWorktree}:{},...E?{afterWorktree:E}:{},...ie.length>0?{touchedFiles:fs(e,ie,x?.beforeSnapshots??new Map)}:{}})}N&&Z.delete(N)}}C==="turn.failed"&&(V=!0,q=y.error?.message||y.message||"Turn failed")}catch{}});let I=[];B.stderr?.on("data",k=>{I.push(k.toString())}),B.on("close",k=>{D.close(),S.length===0&&_.size>0&&console.warn(`[Codex:${s}] No text captured. Event types seen: ${[..._].join(", ")}`),k!==0&&k!==null&&(V=!0,q=I.join("")||`Codex process exited with code ${k}`),O({sessionId:L,text:S.join(""),success:!V,error:V?q:void 0,editEvents:W.length>0?W:void 0,toolsUsed:X.length>0?X:void 0})}),B.on("error",k=>{V=!0,q=k.message,O({sessionId:L,text:S.join(""),success:!1,error:q,editEvents:W.length>0?W:void 0,toolsUsed:X.length>0?X:void 0})})});return{process:B,result:z}}import{spawn as hs}from"child_process";import{createInterface as ms}from"readline";function _t(s,t){let n=s.trim()||"Copilot CLI failed",e=n.toLowerCase();return e.includes("no authentication information")||e.includes("not authenticated")||e.includes("/login")?{code:"copilot_auth_required",recoverable:!0,message:"Copilot is not authenticated. Run `copilot login`, set `COPILOT_GITHUB_TOKEN`, or refresh `gh auth login`."}:e.includes("model")&&(e.includes("not available")||e.includes("not found")||e.includes("disabled")||e.includes("policy"))?{code:"copilot_model_unavailable",recoverable:!0,message:t?`Copilot rejected ${t}. Choose another model or ask your administrator to enable it.`:n}:e.includes("mcp")&&(e.includes("blocked")||e.includes("allowlist"))?{code:"copilot_mcp_blocked",recoverable:!0,message:"Copilot blocked the Popmelt MCP server. Ask your administrator to add it to the enterprise allowlist."}:e.includes("permission")||e.includes("denied")||e.includes("not allowed")?{code:"copilot_permission_denied",recoverable:!0,message:n}:{code:"copilot_error",recoverable:!1,message:n}}function Ot(s,t){let{prompt:n,projectRoot:e,resumeSessionId:o,model:i,timeoutMs:a=3e5,onEvent:c,copilotPath:m="copilot"}=t,P=["--no-color","--no-auto-update","--output-format","json","--stream","on","--available-tools=read,write,shell","--allow-all-tools","--deny-tool=shell(git push)","--deny-tool=shell(rm)","--deny-tool=shell(sudo)","--no-ask-user"];o&&P.push(`--resume=${o}`),i&&P.push("--model",i),P.push("-p",n);let g=hs(m,P,{cwd:e,stdio:["ignore","pipe","pipe"],env:{...process.env,ANTHROPIC_API_KEY:void 0,CLAUDECODE:void 0}}),B=new Promise(z=>{let O,L=[],S=[],X=!1,W="",Q=!1,Z=setTimeout(()=>{Q=!0,g.kill("SIGTERM"),setTimeout(()=>{try{g.kill("SIGKILL")}catch{}},5e3)},a),V=ms({input:g.stdout}),q=new Set,D=[];V.on("line",I=>{if(I.trim())try{let k=JSON.parse(I),y=Pe(k);q.add(y),D.length<5&&vs(y)&&D.push(I.slice(0,800));let C=ke(k,["sessionId","session_id","id"],["session","conversation","thread"]);C&&!O&&(O=C);let N=gs(k,L.length===0);N&&(L.push(N),c?.({type:"delta",jobId:s,text:N},s));let x=ys(k);x&&c?.({type:"thinking",jobId:s,text:x},s);let b=ws(k);b&&(S.push(b.label),c?.({type:"tool_use",jobId:s,tool:b.tool,...b.file?{file:b.file}:{},...b.content?{content:b.content}:{}},s));let H=Ss(k);H&&(X=!0,W=H)}catch{}});let _=[];g.stderr?.on("data",I=>{_.push(I.toString())}),g.on("close",I=>{clearTimeout(Z),V.close(),L.length===0&&q.size>0&&(console.warn(`[Copilot:${s}] No text captured. Event types seen: ${[...q].join(", ")}`),D.length>0&&console.warn(`[Copilot:${s}] Sample text-bearing candidates: ${D.join(`
33
- `)}`)),Q?(X=!0,W=`Timed out after ${Math.round(a/6e4)} minutes`):I!==0&&I!==null&&(X=!0,W=_.join("").trim()||W||`Copilot process exited with code ${I}`);let k=X?_t(W,i):null;z({sessionId:O,text:L.join(""),success:!X,error:k?.message,errorCode:k?.code,recoverableError:k?.recoverable,provider:"copilot",toolsUsed:S.length>0?S:void 0})}),g.on("error",I=>{clearTimeout(Z);let k=_t(I.message,i);z({sessionId:O,text:L.join(""),success:!1,error:k.message,errorCode:k.code,recoverableError:k.recoverable,provider:"copilot",toolsUsed:S.length>0?S:void 0})})});return{process:g,result:B}}function Pe(s){if(typeof s.type=="string")return s.type;let t=s.update;return t&&typeof t.sessionUpdate=="string"?t.sessionUpdate:typeof s.event=="string"?s.event:"unknown"}function gs(s,t=!1){let n=Pe(s).toLowerCase(),e=s.update,o=e?.content;if(e?.sessionUpdate==="agent_message_chunk"&&o?.type==="text"&&typeof o.text=="string")return o.text;if(n==="assistant.message_delta"||n.includes("agent_message.delta"))return pe(s.delta)??pe(s.content)??pe(s.message);if(n==="assistant.message"&&t)return pe(s.message)??pe(s.content)??pe(s);if(n==="result"&&t)return pe(s.result)??pe(s.output)??pe(s.message)??pe(s);let i=s.content;if(i?.type==="text"&&typeof i.text=="string")return i.text;if(typeof s.text=="string"&&rt(s))return s.text;if(typeof s.message=="string"&&rt(s))return s.message;if(typeof s.delta=="string"&&rt(s))return s.delta}function pe(s){if(typeof s=="string")return s;if(!s)return;if(Array.isArray(s)){let n=s.map(pe).filter(e=>!!e);return n.length>0?n.join(""):void 0}if(typeof s!="object")return;let t=s;if(typeof t.text=="string")return t.text;if(typeof t.content=="string")return t.content;for(let n of["delta","content","message","messages","parts","items","output","result","data","value","body","markdown","text_delta","textDelta"]){let e=pe(t[n]);if(e)return e}}function vs(s){let t=s.toLowerCase();return t.includes("assistant.message")||t==="result"}function ys(s){if(typeof s.thinking=="string")return s.thinking;if(typeof s.reasoning=="string")return s.reasoning;let t=s.update;if(t?.sessionUpdate==="reasoning_chunk"){let n=t.content;if(typeof n?.text=="string")return n.text}}function ws(s){let t=ke(s,["tool","toolName","tool_name","name"]),n=ke(s,["command","cmd"]),e=ke(s,["file","path","filename","filePath"]),o=Pe(s);if(n||o.includes("shell"))return{tool:"Bash",label:n?`Bash: ${n.split(`
34
- `)[0].slice(0,80)}`:"Bash",...n?{content:n}:{}};if(t)return{tool:t,label:e?`${t} ${e.split("/").pop()}`:t,...e?{file:e}:{}};if(e&&(o.includes("file")||o.includes("write")||o.includes("edit"))){let i=o.includes("write")?"Write":o.includes("read")?"Read":"Edit";return{tool:i,label:`${i} ${e.split("/").pop()}`,file:e}}}function Ss(s){if(typeof s.error=="string")return s.error;if(s.error&&typeof s.error=="object"){let t=s.error;if(typeof t.message=="string")return t.message}if(typeof s.message=="string"&&Pe(s).toLowerCase().includes("error"))return s.message}function ke(s,t,n=[]){for(let e of t){let o=s[e];if(typeof o=="string")return o}for(let[e,o]of Object.entries(s)){if(!o||typeof o!="object"||n.length>0&&!n.some(a=>e.toLowerCase().includes(a)))continue;let i=ke(o,t);if(i)return i}}function rt(s){let t=Pe(s).toLowerCase();return(typeof s.role=="string"?s.role.toLowerCase():"")==="assistant"||t.includes("assistant")||t.includes("agent_message")}import{execFile as bs}from"child_process";import{copyFile as At,mkdir as Dt,readdir as xs,readFile as ks,writeFile as Ps}from"fs/promises";import{join as ye}from"path";var Ue=class{constructor(t){this.projectRoot=t;let n=ye(t,".popmelt");this.decisionsDir=ye(n,"decisions"),this.screenshotsDir=ye(n,"screenshots")}async persist(t,n,e){try{await Dt(this.decisionsDir,{recursive:!0}),await Dt(this.screenshotsDir,{recursive:!0});try{await At(n,ye(this.screenshotsDir,`s-${t.id}.webp`))}catch{}for(let o=0;o<e.length;o++)try{let i=t.pastedImagePaths[o];i&&await At(e[o],ye(this.screenshotsDir,i.replace("screenshots/","")))}catch{}await Ps(ye(this.decisionsDir,`d-${t.id}.json`),JSON.stringify(t,null,2))}catch(o){console.error("[DecisionStore] Failed to persist decision record:",o)}}async listDecisionIds(){try{return(await xs(this.decisionsDir)).filter(n=>n.startsWith("d-")&&n.endsWith(".json")).map(n=>n.slice(2,-5))}catch{return[]}}async loadDecision(t){try{let n=await ks(ye(this.decisionsDir,`d-${t}.json`),"utf-8");return JSON.parse(n)}catch{return null}}async loadDecisions(t){return(await Promise.all(t.map(e=>this.loadDecision(e)))).filter(e=>e!==null)}captureGitDiff(t){return new Promise(n=>{bs("git",["diff","HEAD"],{cwd:t,timeout:5e3,maxBuffer:1024*1024},(e,o)=>{if(e){n(null);return}n(o||null)})})}};import{mkdir as Ee,readFile as Nt,writeFile as Re}from"fs/promises";import{join as Ie}from"path";var Es=80,Rs=24,Jt=1e3,ze=class{constructor(t){this.evalsDir=Ie(t,".popmelt","evals"),this.suitePath=Ie(this.evalsDir,"evals.json"),this.runsPath=Ie(this.evalsDir,"runs.json"),this.screenshotsDir=Ie(this.evalsDir,"screenshots")}async loadSuite(){try{let t=await Nt(this.suitePath,"utf-8"),n=JSON.parse(t);if(n?.version===1&&Array.isArray(n.cases))return n}catch{}return{version:1,updatedAt:0,cases:[]}}async loadRuns(){try{let t=await Nt(this.runsPath,"utf-8"),n=JSON.parse(t);if(n?.version===1&&Array.isArray(n.runs))return n}catch{}return{version:1,updatedAt:0,runs:[]}}async loadSuiteWithRuns(){let[t,n]=await Promise.all([this.loadSuite(),this.loadRuns()]);return{...t,runs:n.runs}}async loadSuiteWithCurrentRuns(t){let n=await this.loadSuiteWithRuns();return t?{...n,runs:n.runs.filter(e=>e.codeFingerprint===t)}:n}async writeSynthesized(t){let n=await this.loadSuite(),e=new Map(n.cases.map(a=>[a.id,a])),o=new Set;for(let a of t){let c=e.get(a.id);c&&(c.title!==a.title||c.prompt!==a.prompt||JSON.stringify(c.assertions)!==JSON.stringify(a.assertions))&&o.add(a.id),e.set(a.id,a)}let i={version:1,updatedAt:Date.now(),cases:[...e.values()]};return await Ee(this.evalsDir,{recursive:!0}),await Re(this.suitePath,JSON.stringify(i,null,2)),o.size>0&&await this.invalidateRunsForEvals(o),i}async updateCase(t,n){let e=await this.loadSuite(),o=n.title!==void 0||n.assertions!==void 0,i=e.cases.map(c=>c.id===t?{...c,...n.status?{status:n.status}:{},...n.title?{title:n.title}:{},...n.assertions?{assertions:n.assertions}:{}}:c),a=i.find(c=>c.id===t)??null;return a?(await Ee(this.evalsDir,{recursive:!0}),await Re(this.suitePath,JSON.stringify({version:1,updatedAt:Date.now(),cases:i},null,2)),o&&await this.invalidateRunsForEvals(new Set([t])),a):null}async invalidateRunsForEvals(t){let e=(await this.loadRuns()).runs.filter(i=>!t.has(i.evalId)),o={version:1,updatedAt:Date.now(),runs:e};return await Ee(this.evalsDir,{recursive:!0}),await Re(this.runsPath,JSON.stringify(o,null,2)),o}async writeRun(t){let n=await this.loadRuns(),e=[t,...n.runs.filter(i=>i.id!==t.id)].slice(0,200),o={version:1,updatedAt:Date.now(),runs:e};return await Ee(this.evalsDir,{recursive:!0}),await Re(this.runsPath,JSON.stringify(o,null,2)),o}async writeRunScreenshot(t,n){return await Ee(this.screenshotsDir,{recursive:!0}),await Re(Ie(this.screenshotsDir,`eval-${t}.webp`),n),`evals/screenshots/eval-${t}.webp`}async findCase(t){return(await this.loadSuite()).cases.find(e=>e.id===t)??null}};function Bt(s){let t=s.match(/<evals>\s*([\s\S]*?)\s*<\/evals>/);if(!t?.[1])return null;try{let n=JSON.parse(t[1]),e=Array.isArray(n)?n:n&&typeof n=="object"&&Array.isArray(n.evals)?n.evals:null;return e?Ut(e):null}catch{return null}}function Ft(s){let t=s.match(/<eval_proposals>\s*([\s\S]*?)\s*<\/eval_proposals>/);if(!t?.[1])return null;try{let n=JSON.parse(t[1]),e=Array.isArray(n)?n:n&&typeof n=="object"&&Array.isArray(n.evals)?n.evals:null;return e?Ut(e,"pending"):null}catch{return null}}function jt(s,t){let n=s.match(/<eval_run_result>\s*([\s\S]*?)\s*<\/eval_run_result>/);if(n?.[1])try{let e=JSON.parse(n[1]),o=_s(e.assertionResults,t),i=Os(o);return{status:i,summary:he(e.summary)||Wt(i),assertionResults:o,tooling:As(e.tooling)}}catch{}return{status:"needs_review",summary:Le(s.trim()||"The evaluator did not return a structured result.",1e3),assertionResults:t.map(e=>({assertion:e,status:"needs_review",summary:"No structured assertion result was returned.",evidence:[]})),tooling:{chromeDevtools:"unknown",evidenceTypes:["screenshot"]}}}function Ut(s,t="approved"){let n=[];return s.forEach((e,o)=>{if(!e||typeof e!="object")return;let i=e,a=he(i.title),c=he(i.prompt),m=he(i.rationale),P=be(i.assertions);if(!a||!c||P.length===0)return;let g=Ms(i.category),B=Ds(he(i.id))||`eval-${Ht(a).slice(0,32)||o+1}`,z=Cs(i.status)??t;n.push({id:B,title:a,status:z,category:g,prompt:c,assertions:P,rationale:m||"Inferred from Popmelt design history.",sourceDecisionIds:be(i.sourceDecisionIds),sourceThreadIds:be(i.sourceThreadIds)})}),n}function zt(s){let t=[...s.decisions].sort((e,o)=>o.createdAt-e.createdAt).slice(0,Es),n=Object.values(s.threads.threads).sort((e,o)=>o.updatedAt-e.updatedAt).slice(0,Rs);return`You are helping a designer turn organic Popmelt annotation history into a project eval suite.
29
+ </html>`}import{spawn as qn}from"child_process";import{createInterface as Yn}from"readline";var Kn=new Set([".md",".txt",".json",".ts",".tsx",".js",".jsx",".css",".scss",".html",".xml",".yaml",".yml",".toml",".ini",".cfg",".conf",".sh",".bash",".zsh",".py",".rb",".go",".rs",".java",".c",".h",".cpp",".hpp",".swift",".kt",".sql",".graphql",".svg",".env",".gitignore",".prettierrc",".eslintrc"]),xt=1e5;function Qn(s){let t=s.input?.file_path||s.input?.path;if(!t)return;let n=t.includes(".")?`.${t.split(".").pop().toLowerCase()}`:"";if(!Kn.has(n))return;let e;if(s.name==="Write"&&typeof s.input?.content=="string"?e=s.input.content:s.name==="Edit"&&typeof s.input?.new_string=="string"&&(e=s.input.new_string),!!e)return e.length>xt?e.slice(0,xt)+`
30
+ \u2026[truncated]`:e}function Be(s,t){let{prompt:n,projectRoot:e,maxTurns:o=40,maxBudgetUsd:i=1,allowedTools:a=["Read","Edit","Write","Glob","Grep","Bash"],claudePath:c="claude",resumeSessionId:m,model:P,timeoutMs:v=3e5,onEvent:B}=t,z=[];m?z.push("--resume",m,"-p",n):z.push("-p",n),z.push("--output-format","stream-json","--verbose","--max-turns",String(o),"--max-budget-usd",String(i)),P&&z.push("--model",P);for(let S of a)z.push("--allowedTools",S);let A=qn(c,z,{cwd:e,stdio:["ignore","pipe","pipe"],env:{...process.env,ANTHROPIC_API_KEY:void 0,CLAUDECODE:void 0}}),W=new Promise(S=>{let X,H=[],K=[],Z=[],Q=!1,V="",N,_,R=!1,k=setTimeout(()=>{R=!0,A.kill("SIGTERM"),setTimeout(()=>{try{A.kill("SIGKILL")}catch{}},5e3)},v),w=Yn({input:A.stdout}),C=new Set;w.on("line",x=>{if(x.trim())try{let b=JSON.parse(x);b.session_id&&!X&&(X=b.session_id);let G=b.type??(b.event?.type?`event.${b.event.type}`:"unknown");if(C.add(G),b.type==="result"&&b.result&&H.length===0){let E=typeof b.result=="string"?b.result:"";E&&(H.push(E),B?.({type:"delta",jobId:s,text:E},s))}if(b.type==="assistant"&&Array.isArray(b.message?.content)){typeof b.error=="string"&&(N=b.error);for(let E of b.message.content){if(E.type==="text"&&E.text&&(H.push(E.text),B?.({type:"delta",jobId:s,text:E.text},s)),E.type==="tool_use"&&E.name){let ie=E.input?.file_path||E.input?.path||void 0,re=Qn(E);if(B?.({type:"tool_use",jobId:s,tool:E.name,...ie?{file:ie}:{},...re?{content:re}:{}},s),E.name==="Edit"&&E.input?.file_path){let ne={tool:"Edit",file_path:E.input.file_path,old_string:E.input.old_string,new_string:E.input.new_string,replace_all:E.input.replace_all};K.push(ne),Z.push({sequence:Z.length,occurredAt:Date.now(),provider:"claude",source:"claude_tool_use",tool:"Edit",operation:"modify",filePath:ne.file_path,oldString:typeof ne.old_string=="string"?ne.old_string:void 0,newString:typeof ne.new_string=="string"?ne.new_string:void 0,replaceAll:typeof ne.replace_all=="boolean"?ne.replace_all:void 0})}else E.name==="Write"&&E.input?.file_path&&(K.push({tool:"Write",file_path:E.input.file_path,content:E.input.content}),Z.push({sequence:Z.length,occurredAt:Date.now(),provider:"claude",source:"claude_tool_use",tool:"Write",operation:"create",filePath:E.input.file_path,content:typeof E.input.content=="string"?E.input.content:void 0}))}E.type==="thinking"&&E.thinking&&B?.({type:"thinking",jobId:s,text:E.thinking},s)}}b.type==="result"&&b.is_error===!0&&(Q=!0,typeof b.error=="string"&&(N=b.error),typeof b.result=="string"&&(_=b.result)),b.type==="user"&&b.tool_use_result?.file?.filePath&&B?.({type:"tool_use",jobId:s,tool:"Read",file:b.tool_use_result.file.filePath},s)}catch{}});let J=[];A.stderr?.on("data",x=>{J.push(x.toString())}),A.on("close",x=>{if(clearTimeout(k),w.close(),H.length===0&&C.size>0&&console.warn(`[Claude:${s}] No text captured. Event types seen: ${[...C].join(", ")}`),R)Q=!0,V=`Timed out after ${Math.round(v/6e4)} minutes`;else if(x!==0&&x!==null){Q=!0;let G=J.join("").trim(),E=H.join("").trim(),ie=H.length===0&&C.size>0?` (no text captured, event types: ${[...C].join(", ")})`:"";V=G||_||E||`Claude process exited with code ${x}${ie}`}Q&&!V&&_&&(V=_);let b=Vn(V,N);S({sessionId:X,text:H.join(""),success:!Q,error:Q?b.message:void 0,errorCode:Q?b.code:void 0,recoverableError:Q?b.recoverable:void 0,fileEdits:K.length>0?K:void 0,editEvents:Z.length>0?Z:void 0})}),A.on("error",x=>{clearTimeout(k),Q=!0,V=x.message,S({sessionId:X,text:H.join(""),success:!1,error:V,errorCode:"claude_spawn_error",recoverableError:!1,fileEdits:K.length>0?K:void 0,editEvents:Z.length>0?Z:void 0})})});return{process:A,result:W}}function Vn(s,t){let n=s.trim()||"Claude CLI failed",e=`${t??""} ${n}`.toLowerCase();return e.includes("authentication_failed")||e.includes("not logged in")||e.includes("please run /login")||e.includes("api key")?{message:n,code:"claude_auth_required",recoverable:!0}:e.includes("model")&&(e.includes("not found")||e.includes("unknown")||e.includes("unavailable"))?{message:n,code:"claude_model_unavailable",recoverable:!0}:{message:n,code:t||"claude_error",recoverable:!1}}import{spawn as Zn,spawnSync as Tt}from"child_process";import{readFileSync as es,statSync as ts}from"fs";import{basename as ns,extname as ss,isAbsolute as os,relative as Mt,resolve as rs}from"path";import{createInterface as is}from"readline";function kt(s){let t=s.match(/^\/bin\/(?:ba)?sh\s+-\w+\s+"(.*)"$/s);return t?t[1].replace(/\\"/g,'"').replace(/\\\\/g,"\\"):s}var Pt=new Set([".md",".txt",".json",".ts",".tsx",".js",".jsx",".css",".scss",".html",".xml",".yaml",".yml",".toml",".ini",".cfg",".conf",".sh",".bash",".zsh",".py",".rb",".go",".rs",".java",".c",".h",".cpp",".hpp",".swift",".kt",".sql",".graphql",".svg",".env",".gitignore",".prettierrc",".eslintrc"]),st=2e5,as=1e5,Ct=2e5;function $t(s){let t=ns(s).toLowerCase(),n=ss(s).toLowerCase();return Pt.has(n)||Pt.has(t)}function _t(s,t){let n=os(t)?t:rs(s,t),e=Mt(s,n);return e===""||e.startsWith("..")?null:n}function Fe(s,t){if(!$t(t))return;let n=_t(s,t);if(n)try{let e=ts(n);return e.isFile()?e.size>st?{exists:!0,truncated:!0}:{exists:!0,content:es(n,"utf8")}:{exists:!1}}catch{return{exists:!1}}}function cs(s,t){for(let n of t){let e=s[n];if(typeof e=="string")return e}}function ls(s){let t=s?.toLowerCase()??"";return t.includes("add")||t.includes("create")?"create":t.includes("delete")||t.includes("remove")?"delete":t.includes("rename")||t.includes("move")?"rename":t.includes("modify")||t.includes("update")||t.includes("edit")?"modify":"unknown"}function Et(s){return cs(s,["patch","diff","unifiedDiff","unified_diff"])}function ot(s,t){return s.length<=t?{text:s}:{text:s.slice(0,t),truncated:!0}}function Rt(s,t){let n=Tt("git",t,{cwd:s,encoding:"utf8",timeout:5e3,maxBuffer:Ct+65536});return n.status!==0?null:n.stdout??""}function ds(s){let t=new Set;for(let n of s.split(`
31
+ `)){if(n.length<4)continue;let e=n.slice(3).trim();if(!e)continue;e.startsWith('"')&&e.endsWith('"')&&(e=e.slice(1,-1));let o=e.split(" -> ");for(let i of o)i&&t.add(i)}return[...t]}function It(s){let t=Rt(s,["status","--porcelain=v1","-uall"]);if(t===null)return;let n=Rt(s,["diff","--no-ext-diff","HEAD","--"])??"",e=ot(n,Ct);return{status:t,changedFiles:ds(t),...e.text?{diff:e.text}:{},...e.truncated?{diffTruncated:!0}:{}}}function us(s,t){return!s||!t?!1:s.status!==t.status||(s.diff??"")!==(t.diff??"")}function ps(s,t){if(!$t(t))return;let n=_t(s,t);if(!n)return;let e=Mt(s,n);if(e===""||e.startsWith(".."))return;let o=Tt("git",["show",`HEAD:${e}`],{cwd:s,encoding:"utf8",timeout:5e3,maxBuffer:st+65536});if(o.status!==0)return{exists:!1};let i=o.stdout??"",a=ot(i,st);return{exists:!0,content:a.text,...a.truncated?{truncated:!0}:{}}}function fs(s,t,n){return t.map(e=>{let o=n.has(e)?n.get(e):ps(s,e),i=Fe(s,e);return{filePath:e,...o?{before:o}:{},...i?{after:i}:{}}})}function je(s,t){let{prompt:n,projectRoot:e,screenshotPath:o,resumeSessionId:i,model:a,onEvent:c,sandboxMode:m="workspace-write",configOverrides:P=[]}=t,v=[];if(i){v.push("exec","resume","--skip-git-repo-check","--json","-c",`sandbox_mode="${m}"`);for(let A of P)v.push("-c",A);a&&v.push("-m",a),v.push(i),v.push(n),o&&v.push("--image",o)}else{v.push("exec","--skip-git-repo-check","--json","--sandbox",m);for(let A of P)v.push("-c",A);a&&v.push("-m",a),v.push(n),o&&v.push("--image",o)}let B=Zn("codex",v,{cwd:e,stdio:["ignore","pipe","pipe"],env:{...process.env,CLAUDECODE:void 0}}),z=new Promise(A=>{let W,S=[],X=[],H=[],K=new Map,Z=new Map,Q=!1,V="",N=is({input:B.stdout}),_=new Set;N.on("line",k=>{if(k.trim())try{let w=JSON.parse(k),C=w.type??"unknown";if(_.add(C),C==="thread.started"&&w.thread_id&&!W&&(W=w.thread_id),(C==="item.agentMessage.delta"||C==="item/agentMessage/delta")&&w.delta?.text&&(S.push(w.delta.text),c?.({type:"delta",jobId:s,text:w.delta.text},s)),(C==="item.reasoning.delta"||C==="item/reasoning/delta")&&w.delta?.text&&c?.({type:"thinking",jobId:s,text:w.delta.text},s),(C==="item.started"||C==="item/started")&&w.item){let J=w.item.type;if(J==="command_execution"){let x=w.item.command,b=x?kt(x):void 0,G=It(e),E=new Map;for(let re of G?.changedFiles??[])E.set(re,Fe(e,re));typeof w.item.id=="string"&&Z.set(w.item.id,{...x?{rawCommand:x}:{},...b?{command:b}:{},...G?{beforeWorktree:G}:{},beforeSnapshots:E});let ie=b?`Bash: ${b.split(`
32
+ `)[0].slice(0,80)}`:"Bash";X.push(ie),c?.({type:"tool_use",jobId:s,tool:"Bash",...b?{content:b}:{}},s)}else if(J==="file_change"){let x=w.item.filename||w.item.path;x&&!K.has(x)&&K.set(x,Fe(e,x)),X.push(x?`Edit ${x.split("/").pop()}`:"Edit"),c?.({type:"tool_use",jobId:s,tool:"Edit",...x?{file:x}:{}},s)}else if(J==="file_read"){let x=w.item.filename||w.item.path;X.push(x?`Read ${x.split("/").pop()}`:"Read"),c?.({type:"tool_use",jobId:s,tool:"Read",...x?{file:x}:{}},s)}else if(J==="web_search")X.push("WebSearch"),c?.({type:"tool_use",jobId:s,tool:"WebSearch"},s);else if(J==="mcp_tool_call"){let x=w.item.tool_name||w.item.name||"MCP";X.push(x),c?.({type:"tool_use",jobId:s,tool:x},s)}}if((C==="item.completed"||C==="item/completed")&&w.item){if(w.item.type==="agent_message"){let J=w.item.text;typeof J=="string"&&J&&(S.push(J),c?.({type:"delta",jobId:s,text:J},s))}else if(w.item.type==="reasoning"){let J=w.item.text;typeof J=="string"&&J&&c?.({type:"thinking",jobId:s,text:J},s)}else if(w.item.type==="file_change"&&Array.isArray(w.item.changes))for(let J of w.item.changes){let x=J.path||J.filename,b=J.kind==="add"?"Write":"Edit";if(x){X.push(`${b} ${x.split("/").pop()}`),c?.({type:"tool_use",jobId:s,tool:b,file:x},s);let G=J,E=typeof G.kind=="string"?G.kind:void 0,ie=ls(E),re=K.get(x),ne=Fe(e,x);H.push({sequence:H.length,occurredAt:Date.now(),provider:"codex",source:"codex_file_change",tool:"FileChange",operation:ie,filePath:x,...re?{before:re}:{},...ne?{after:ne}:{},...Et(G)?{patch:Et(G)}:{},...E?{rawKind:E}:{}}),K.set(x,ne)}}else if(w.item.type==="command_execution"){let J=typeof w.item.id=="string"?w.item.id:void 0,x=J?Z.get(J):void 0,b=typeof w.item.command=="string"?w.item.command:x?.rawCommand,G=b?kt(b):x?.command,E=It(e);if(us(x?.beforeWorktree,E)){let ie=[...new Set([...x?.beforeWorktree?.changedFiles??[],...E?.changedFiles??[]])],re=typeof w.item.aggregated_output=="string"?ot(w.item.aggregated_output,as):void 0,ne=typeof w.item.exit_code=="number"?w.item.exit_code:w.item.exit_code===null?null:void 0,Ne=typeof w.item.status=="string"?w.item.status:void 0;H.push({sequence:H.length,occurredAt:Date.now(),provider:"codex",source:"codex_command_execution",tool:"Bash",operation:"command",...b?{rawCommand:b}:{},...G?{command:G}:{},...ne!==void 0?{exitCode:ne}:{},...Ne?{status:Ne}:{},...re?{output:re.text}:{},...re?.truncated?{outputTruncated:!0}:{},...x?.beforeWorktree?{beforeWorktree:x.beforeWorktree}:{},...E?{afterWorktree:E}:{},...ie.length>0?{touchedFiles:fs(e,ie,x?.beforeSnapshots??new Map)}:{}})}J&&Z.delete(J)}}C==="turn.failed"&&(Q=!0,V=w.error?.message||w.message||"Turn failed")}catch{}});let R=[];B.stderr?.on("data",k=>{R.push(k.toString())}),B.on("close",k=>{N.close(),S.length===0&&_.size>0&&console.warn(`[Codex:${s}] No text captured. Event types seen: ${[..._].join(", ")}`),k!==0&&k!==null&&(Q=!0,V=R.join("")||`Codex process exited with code ${k}`),A({sessionId:W,text:S.join(""),success:!Q,error:Q?V:void 0,editEvents:H.length>0?H:void 0,toolsUsed:X.length>0?X:void 0})}),B.on("error",k=>{Q=!0,V=k.message,A({sessionId:W,text:S.join(""),success:!1,error:V,editEvents:H.length>0?H:void 0,toolsUsed:X.length>0?X:void 0})})});return{process:B,result:z}}import{spawn as hs}from"child_process";import{createInterface as ms}from"readline";function Ot(s,t){let n=s.trim()||"Copilot CLI failed",e=n.toLowerCase();return e.includes("no authentication information")||e.includes("not authenticated")||e.includes("/login")?{code:"copilot_auth_required",recoverable:!0,message:"Copilot is not authenticated. Run `copilot login`, set `COPILOT_GITHUB_TOKEN`, or refresh `gh auth login`."}:e.includes("model")&&(e.includes("not available")||e.includes("not found")||e.includes("disabled")||e.includes("policy"))?{code:"copilot_model_unavailable",recoverable:!0,message:t?`Copilot rejected ${t}. Choose another model or ask your administrator to enable it.`:n}:e.includes("mcp")&&(e.includes("blocked")||e.includes("allowlist"))?{code:"copilot_mcp_blocked",recoverable:!0,message:"Copilot blocked the Popmelt MCP server. Ask your administrator to add it to the enterprise allowlist."}:e.includes("permission")||e.includes("denied")||e.includes("not allowed")?{code:"copilot_permission_denied",recoverable:!0,message:n}:{code:"copilot_error",recoverable:!1,message:n}}function At(s,t){let{prompt:n,projectRoot:e,resumeSessionId:o,model:i,timeoutMs:a=3e5,onEvent:c,copilotPath:m="copilot"}=t,P=["--no-color","--no-auto-update","--output-format","json","--stream","on","--available-tools=read,write,shell","--allow-all-tools","--deny-tool=shell(git push)","--deny-tool=shell(rm)","--deny-tool=shell(sudo)","--no-ask-user"];o&&P.push(`--resume=${o}`),i&&P.push("--model",i),P.push("-p",n);let v=hs(m,P,{cwd:e,stdio:["ignore","pipe","pipe"],env:{...process.env,ANTHROPIC_API_KEY:void 0,CLAUDECODE:void 0}}),B=new Promise(z=>{let A,W=[],S=[],X=!1,H="",K=!1,Z=setTimeout(()=>{K=!0,v.kill("SIGTERM"),setTimeout(()=>{try{v.kill("SIGKILL")}catch{}},5e3)},a),Q=ms({input:v.stdout}),V=new Set,N=[];Q.on("line",R=>{if(R.trim())try{let k=JSON.parse(R),w=Pe(k);V.add(w),N.length<5&&vs(w)&&N.push(R.slice(0,800));let C=ke(k,["sessionId","session_id","id"],["session","conversation","thread"]);C&&!A&&(A=C);let J=gs(k,W.length===0);J&&(W.push(J),c?.({type:"delta",jobId:s,text:J},s));let x=ys(k);x&&c?.({type:"thinking",jobId:s,text:x},s);let b=ws(k);b&&(S.push(b.label),c?.({type:"tool_use",jobId:s,tool:b.tool,...b.file?{file:b.file}:{},...b.content?{content:b.content}:{}},s));let G=Ss(k);G&&(X=!0,H=G)}catch{}});let _=[];v.stderr?.on("data",R=>{_.push(R.toString())}),v.on("close",R=>{clearTimeout(Z),Q.close(),W.length===0&&V.size>0&&(console.warn(`[Copilot:${s}] No text captured. Event types seen: ${[...V].join(", ")}`),N.length>0&&console.warn(`[Copilot:${s}] Sample text-bearing candidates: ${N.join(`
33
+ `)}`)),K?(X=!0,H=`Timed out after ${Math.round(a/6e4)} minutes`):R!==0&&R!==null&&(X=!0,H=_.join("").trim()||H||`Copilot process exited with code ${R}`);let k=X?Ot(H,i):null;z({sessionId:A,text:W.join(""),success:!X,error:k?.message,errorCode:k?.code,recoverableError:k?.recoverable,provider:"copilot",toolsUsed:S.length>0?S:void 0})}),v.on("error",R=>{clearTimeout(Z);let k=Ot(R.message,i);z({sessionId:A,text:W.join(""),success:!1,error:k.message,errorCode:k.code,recoverableError:k.recoverable,provider:"copilot",toolsUsed:S.length>0?S:void 0})})});return{process:v,result:B}}function Pe(s){if(typeof s.type=="string")return s.type;let t=s.update;return t&&typeof t.sessionUpdate=="string"?t.sessionUpdate:typeof s.event=="string"?s.event:"unknown"}function gs(s,t=!1){let n=Pe(s).toLowerCase(),e=s.update,o=e?.content;if(e?.sessionUpdate==="agent_message_chunk"&&o?.type==="text"&&typeof o.text=="string")return o.text;if(n==="assistant.message_delta"||n.includes("agent_message.delta"))return pe(s.delta)??pe(s.content)??pe(s.message);if(n==="assistant.message"&&t)return pe(s.message)??pe(s.content)??pe(s);if(n==="result"&&t)return pe(s.result)??pe(s.output)??pe(s.message)??pe(s);let i=s.content;if(i?.type==="text"&&typeof i.text=="string")return i.text;if(typeof s.text=="string"&&rt(s))return s.text;if(typeof s.message=="string"&&rt(s))return s.message;if(typeof s.delta=="string"&&rt(s))return s.delta}function pe(s){if(typeof s=="string")return s;if(!s)return;if(Array.isArray(s)){let n=s.map(pe).filter(e=>!!e);return n.length>0?n.join(""):void 0}if(typeof s!="object")return;let t=s;if(typeof t.text=="string")return t.text;if(typeof t.content=="string")return t.content;for(let n of["delta","content","message","messages","parts","items","output","result","data","value","body","markdown","text_delta","textDelta"]){let e=pe(t[n]);if(e)return e}}function vs(s){let t=s.toLowerCase();return t.includes("assistant.message")||t==="result"}function ys(s){if(typeof s.thinking=="string")return s.thinking;if(typeof s.reasoning=="string")return s.reasoning;let t=s.update;if(t?.sessionUpdate==="reasoning_chunk"){let n=t.content;if(typeof n?.text=="string")return n.text}}function ws(s){let t=ke(s,["tool","toolName","tool_name","name"]),n=ke(s,["command","cmd"]),e=ke(s,["file","path","filename","filePath"]),o=Pe(s);if(n||o.includes("shell"))return{tool:"Bash",label:n?`Bash: ${n.split(`
34
+ `)[0].slice(0,80)}`:"Bash",...n?{content:n}:{}};if(t)return{tool:t,label:e?`${t} ${e.split("/").pop()}`:t,...e?{file:e}:{}};if(e&&(o.includes("file")||o.includes("write")||o.includes("edit"))){let i=o.includes("write")?"Write":o.includes("read")?"Read":"Edit";return{tool:i,label:`${i} ${e.split("/").pop()}`,file:e}}}function Ss(s){if(typeof s.error=="string")return s.error;if(s.error&&typeof s.error=="object"){let t=s.error;if(typeof t.message=="string")return t.message}if(typeof s.message=="string"&&Pe(s).toLowerCase().includes("error"))return s.message}function ke(s,t,n=[]){for(let e of t){let o=s[e];if(typeof o=="string")return o}for(let[e,o]of Object.entries(s)){if(!o||typeof o!="object"||n.length>0&&!n.some(a=>e.toLowerCase().includes(a)))continue;let i=ke(o,t);if(i)return i}}function rt(s){let t=Pe(s).toLowerCase();return(typeof s.role=="string"?s.role.toLowerCase():"")==="assistant"||t.includes("assistant")||t.includes("agent_message")}import{execFile as bs}from"child_process";import{copyFile as Dt,mkdir as Nt,readdir as xs,readFile as ks,writeFile as Ps}from"fs/promises";import{join as ye}from"path";var Ue=class{constructor(t){this.projectRoot=t;let n=ye(t,".popmelt");this.decisionsDir=ye(n,"decisions"),this.screenshotsDir=ye(n,"screenshots")}async persist(t,n,e){try{await Nt(this.decisionsDir,{recursive:!0}),await Nt(this.screenshotsDir,{recursive:!0});try{await Dt(n,ye(this.screenshotsDir,`s-${t.id}.webp`))}catch{}for(let o=0;o<e.length;o++)try{let i=t.pastedImagePaths[o];i&&await Dt(e[o],ye(this.screenshotsDir,i.replace("screenshots/","")))}catch{}await Ps(ye(this.decisionsDir,`d-${t.id}.json`),JSON.stringify(t,null,2))}catch(o){console.error("[DecisionStore] Failed to persist decision record:",o)}}async listDecisionIds(){try{return(await xs(this.decisionsDir)).filter(n=>n.startsWith("d-")&&n.endsWith(".json")).map(n=>n.slice(2,-5))}catch{return[]}}async loadDecision(t){try{let n=await ks(ye(this.decisionsDir,`d-${t}.json`),"utf-8");return JSON.parse(n)}catch{return null}}async loadDecisions(t){return(await Promise.all(t.map(e=>this.loadDecision(e)))).filter(e=>e!==null)}captureGitDiff(t){return new Promise(n=>{bs("git",["diff","HEAD"],{cwd:t,timeout:5e3,maxBuffer:1024*1024},(e,o)=>{if(e){n(null);return}n(o||null)})})}};import{mkdir as Ee,readFile as Jt,writeFile as Re}from"fs/promises";import{join as Ie}from"path";var Es=80,Rs=24,Bt=1e3,ze=class{constructor(t){this.evalsDir=Ie(t,".popmelt","evals"),this.suitePath=Ie(this.evalsDir,"evals.json"),this.runsPath=Ie(this.evalsDir,"runs.json"),this.screenshotsDir=Ie(this.evalsDir,"screenshots")}async loadSuite(){try{let t=await Jt(this.suitePath,"utf-8"),n=JSON.parse(t);if(n?.version===1&&Array.isArray(n.cases))return n}catch{}return{version:1,updatedAt:0,cases:[]}}async loadRuns(){try{let t=await Jt(this.runsPath,"utf-8"),n=JSON.parse(t);if(n?.version===1&&Array.isArray(n.runs))return n}catch{}return{version:1,updatedAt:0,runs:[]}}async loadSuiteWithRuns(){let[t,n]=await Promise.all([this.loadSuite(),this.loadRuns()]);return{...t,runs:n.runs}}async loadSuiteWithCurrentRuns(t){let n=await this.loadSuiteWithRuns();return t?{...n,runs:n.runs.filter(e=>e.codeFingerprint===t)}:n}async writeSynthesized(t){let n=await this.loadSuite(),e=new Map(n.cases.map(a=>[a.id,a])),o=new Set;for(let a of t){let c=e.get(a.id);c&&(c.title!==a.title||c.prompt!==a.prompt||JSON.stringify(c.assertions)!==JSON.stringify(a.assertions))&&o.add(a.id),e.set(a.id,a)}let i={version:1,updatedAt:Date.now(),cases:[...e.values()]};return await Ee(this.evalsDir,{recursive:!0}),await Re(this.suitePath,JSON.stringify(i,null,2)),o.size>0&&await this.invalidateRunsForEvals(o),i}async updateCase(t,n){let e=await this.loadSuite(),o=n.title!==void 0||n.assertions!==void 0,i=e.cases.map(c=>c.id===t?{...c,...n.status?{status:n.status}:{},...n.title?{title:n.title}:{},...n.assertions?{assertions:n.assertions}:{}}:c),a=i.find(c=>c.id===t)??null;return a?(await Ee(this.evalsDir,{recursive:!0}),await Re(this.suitePath,JSON.stringify({version:1,updatedAt:Date.now(),cases:i},null,2)),o&&await this.invalidateRunsForEvals(new Set([t])),a):null}async invalidateRunsForEvals(t){let e=(await this.loadRuns()).runs.filter(i=>!t.has(i.evalId)),o={version:1,updatedAt:Date.now(),runs:e};return await Ee(this.evalsDir,{recursive:!0}),await Re(this.runsPath,JSON.stringify(o,null,2)),o}async writeRun(t){let n=await this.loadRuns(),e=[t,...n.runs.filter(i=>i.id!==t.id)].slice(0,200),o={version:1,updatedAt:Date.now(),runs:e};return await Ee(this.evalsDir,{recursive:!0}),await Re(this.runsPath,JSON.stringify(o,null,2)),o}async writeRunScreenshot(t,n){return await Ee(this.screenshotsDir,{recursive:!0}),await Re(Ie(this.screenshotsDir,`eval-${t}.webp`),n),`evals/screenshots/eval-${t}.webp`}async findCase(t){return(await this.loadSuite()).cases.find(e=>e.id===t)??null}};function Ft(s){let t=s.match(/<evals>\s*([\s\S]*?)\s*<\/evals>/);if(!t?.[1])return null;try{let n=JSON.parse(t[1]),e=Array.isArray(n)?n:n&&typeof n=="object"&&Array.isArray(n.evals)?n.evals:null;return e?zt(e):null}catch{return null}}function jt(s){let t=s.match(/<eval_proposals>\s*([\s\S]*?)\s*<\/eval_proposals>/);if(!t?.[1])return null;try{let n=JSON.parse(t[1]),e=Array.isArray(n)?n:n&&typeof n=="object"&&Array.isArray(n.evals)?n.evals:null;return e?zt(e,"pending"):null}catch{return null}}function Ut(s,t){let n=s.match(/<eval_run_result>\s*([\s\S]*?)\s*<\/eval_run_result>/);if(n?.[1])try{let e=JSON.parse(n[1]),o=_s(e.assertionResults,t),i=Os(o);return{status:i,summary:he(e.summary)||Ht(i),assertionResults:o,tooling:As(e.tooling)}}catch{}return{status:"needs_review",summary:Le(s.trim()||"The evaluator did not return a structured result.",1e3),assertionResults:t.map(e=>({assertion:e,status:"needs_review",summary:"No structured assertion result was returned.",evidence:[]})),tooling:{chromeDevtools:"unknown",evidenceTypes:["screenshot"]}}}function zt(s,t="approved"){let n=[];return s.forEach((e,o)=>{if(!e||typeof e!="object")return;let i=e,a=he(i.title),c=he(i.prompt),m=he(i.rationale),P=be(i.assertions);if(!a||!c||P.length===0)return;let v=Ms(i.category),B=Ds(he(i.id))||`eval-${Gt(a).slice(0,32)||o+1}`,z=Cs(i.status)??t;n.push({id:B,title:a,status:z,category:v,prompt:c,assertions:P,rationale:m||"Inferred from Popmelt design history.",sourceDecisionIds:be(i.sourceDecisionIds),sourceThreadIds:be(i.sourceThreadIds)})}),n}function Lt(s){let t=[...s.decisions].sort((e,o)=>o.createdAt-e.createdAt).slice(0,Es),n=Object.values(s.threads.threads).sort((e,o)=>o.updatedAt-e.updatedAt).slice(0,Rs);return`You are helping a designer turn organic Popmelt annotation history into a project eval suite.
35
35
 
36
36
  The designer does NOT work eval-first. They use annotations and conversation threads to zhuzh the UI over time. Your job is to infer useful eval proposals from that trail. The developer will reject, refine, or approve proposals in this thread. Do not write files or run commands.
37
37
 
@@ -87,7 +87,7 @@ Allowed categories: component, layout, copy, interaction, visual, workflow.
87
87
  Return only the <evals> block after approval.
88
88
 
89
89
  ## First Response Structured Block
90
- At the end of your first proposal response, output a JSON array inside <eval_proposals> tags using the same schema, but with "status": "pending".`}function Lt(s){return`You are running a Popmelt design eval against the current page. This is an evaluation-only task: do not edit files, do not write code, and do not create annotations.
90
+ At the end of your first proposal response, output a JSON array inside <eval_proposals> tags using the same schema, but with "status": "pending".`}function Wt(s){return`You are running a Popmelt design eval against the current page. This is an evaluation-only task: do not edit files, do not write code, and do not create annotations.
91
91
 
92
92
  ## Eval
93
93
  \`\`\`json
@@ -148,16 +148,16 @@ Thread: ${s.threadId??"none"}
148
148
  Created: ${new Date(s.createdAt).toISOString()}
149
149
  URL: ${s.url||"unknown"}
150
150
  Instructions:
151
- ${Le(o||"(none)",Jt)}
151
+ ${Le(o||"(none)",Bt)}
152
152
  Resolutions:
153
- ${Le(i||"(none)",Jt)}
153
+ ${Le(i||"(none)",Bt)}
154
154
  Files: ${a.length>0?a.join(", "):"none"}`}function Ts(s){let n=s.messages.slice(0,2).concat(s.messages.length>4?s.messages.slice(-2):s.messages.slice(2)).map(e=>{let o=e.feedbackSummary??e.replyToQuestion??e.question??e.responseText??"",i=e.resolutions?.map(a=>`${a.status}: ${a.summary}`).join("; ")??"";return`- ${e.role}: ${Le(o||i||"(no text)",500)}`}).join(`
155
155
  `);return`### Thread ${s.id}
156
156
  Messages: ${s.messages.length}
157
157
  Updated: ${new Date(s.updatedAt).toISOString()}
158
- ${n}`}function he(s){return typeof s=="string"?s.trim():""}function be(s){return Array.isArray(s)?s.filter(t=>typeof t=="string"&&t.trim().length>0).map(t=>t.trim()):[]}function Ms(s){let t=he(s);return t==="component"||t==="layout"||t==="copy"||t==="interaction"||t==="visual"||t==="workflow"?t:"visual"}function Cs(s){let t=he(s);return t==="pending"||t==="approved"||t==="rejected"?t:null}function $s(s){let t=he(s);return t==="pass"||t==="fail"||t==="needs_review"?t:null}function _s(s,t){let e=(Array.isArray(s)?s:[]).map(o=>{if(!o||typeof o!="object")return null;let i=o,a=he(i.assertion);return a?{assertion:a,status:$s(i.status)??"needs_review",summary:he(i.summary)||Wt("needs_review"),evidence:be(i.evidence)}:null}).filter(o=>o!==null);return e.length>0?e:t.map(o=>({assertion:o,status:"needs_review",summary:"No structured assertion result was returned.",evidence:[]}))}function Os(s){return s.some(t=>t.status==="fail")?"fail":s.some(t=>t.status==="needs_review")?"needs_review":"pass"}function As(s){if(!s||typeof s!="object")return{chromeDevtools:"unknown",evidenceTypes:["screenshot"]};let t=s,n=he(t.chromeDevtools);return{chromeDevtools:n==="available"||n==="unavailable"||n==="unknown"?n:"unknown",evidenceTypes:be(t.evidenceTypes).length>0?be(t.evidenceTypes):["screenshot"]}}function Wt(s){return s==="pass"?"The current page appears to satisfy this eval.":s==="fail"?"The current page appears to violate this eval.":"The current page needs human review for this eval."}function Ds(s){return Ht(s).slice(0,48)}function Ht(s){return s.toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/^-+|-+$/g,"")}function Le(s,t){return s.length>t?`${s.slice(0,t)}...`:s}import{readFile as Gt,writeFile as ve}from"fs/promises";import{join as it}from"path";var te="[Materializer]",Ns={version:1,materializedIds:[],lastRunAt:null,lastRunDecisionIds:[],lastRunError:null};function Js(s){return Array.isArray(s)?s.map((t,n)=>typeof t=="string"?{id:Math.random().toString(16).slice(2,10),scope:"general",text:t,sources:[]}:t&&typeof t=="object"&&typeof t.text=="string"?t:{id:Math.random().toString(16).slice(2,10),scope:"general",text:String(t),sources:[]}):[]}function Te(s){let t=[];for(let n of s){if(!n||typeof n!="object")continue;let e=n;if(typeof e.id!="string"||typeof e.text!="string"){console.warn(`${te} Dropping rule missing id or text:`,JSON.stringify(n).slice(0,120));continue}t.push({id:e.id,scope:typeof e.scope=="string"?e.scope:"general",text:e.text,sources:Array.isArray(e.sources)?e.sources.filter(o=>typeof o=="string"):[]})}return t.length>30&&console.warn(`${te} Rule count ${t.length} exceeds cap of 30`),t}var We=class{constructor(t,n,e={}){this.projectRoot=t;this.decisionStore=n;this.options=e;this.cachedIndex=null;this.running=!1;let o=it(t,".popmelt");this.indexPath=it(o,"materialized.json"),this.modelPath=it(o,"model.json")}get isRunning(){return this.running}async loadModel(){try{let t=await Gt(this.modelPath,"utf-8"),n=JSON.parse(t);return Array.isArray(n.rules)&&(n.rules=Js(n.rules)),n}catch{return null}}async addComponent(t){let n=await this.loadModel();n||(n={tokens:{},components:{},rules:[]}),(!n.components||typeof n.components!="object")&&(n.components={});let e=n.components;return e[t]?{added:!1,alreadyExists:!0}:(e[t]={description:""},await ve(this.modelPath,JSON.stringify(n,null,2)),console.log(`${te} Added component "${t}" to model`),{added:!0,alreadyExists:!1})}async updateToken(t,n){let e=await this.loadModel();e||(e={tokens:{},components:{},rules:[]});let o=t.split("."),i=e;for(let m=0;m<o.length-1;m++){let P=o[m];(!i[P]||typeof i[P]!="object")&&(i[P]={}),i=i[P]}let a=o[o.length-1],c;try{c=JSON.parse(n)}catch{c=null}if(c&&typeof c=="object"&&c!==null&&"value"in c)i[a]=c;else{let m=i[a];m&&typeof m=="object"&&m!==null&&"value"in m?m.value=n:i[a]=n}return await ve(this.modelPath,JSON.stringify(e,null,2)),console.log(`${te} Updated token "${t}" \u2192 "${n.slice(0,80)}"`),{updated:!0}}async removeToken(t){let n=await this.loadModel();if(!n)return{removed:!1};let e=t.split("."),o=n;for(let a=0;a<e.length-1;a++){let c=e[a];if(!o[c]||typeof o[c]!="object")return{removed:!1};o=o[c]}let i=e[e.length-1];return i in o?(delete o[i],await ve(this.modelPath,JSON.stringify(n,null,2)),console.log(`${te} Removed token "${t}" from model`),{removed:!0}):{removed:!1}}async removeComponent(t){let n=await this.loadModel();if(!n)return{removed:!1};let e=n.components;return!e||!e[t]?{removed:!1}:(delete e[t],await ve(this.modelPath,JSON.stringify(n,null,2)),console.log(`${te} Removed component "${t}" from model`),{removed:!0})}async getUnmaterializedPatternDecisions(){let t=await this.loadIndex(),n=new Set(t.materializedIds),o=(await this.decisionStore.listDecisionIds()).filter(a=>!n.has(a));return o.length===0?[]:(await this.decisionStore.loadDecisions(o)).filter(a=>a.resolutions.some(c=>(c.finalScope??c.inferredScope)?.breadth==="pattern"))}async run(t={}){if(this.running)return{processedIds:[],success:!0,error:"Already running"};this.running=!0;try{let n=await this.getUnmaterializedPatternDecisions();if(n.length===0)return{processedIds:[],success:!0};let e=n.map(g=>g.id);console.log(`${te} Processing ${e.length} pattern-scoped decision(s): ${e.join(", ")}`),this.options.onEvent?.({type:"materialize_started",decisionIds:e});let o=await this.loadModel(),i=Bs(n,o),a=!0,c;try{let g=`mat-${Date.now()}`,{provider:B,result:z}=this.spawnModelAgent(g,i,t,{maxTurns:this.options.maxTurns??5,maxBudgetUsd:this.options.maxBudgetUsd??.5,allowedTools:["Read"]}),O=await z;if(!O.success)a=!1,c=O.error,console.error(`${te} ${B} spawn error:`,c);else{let L=He(O.text);L?(Array.isArray(L.rules)&&(L.rules=Te(L.rules)),await ve(this.modelPath,JSON.stringify(L,null,2)),console.log(`${te} Successfully materialized ${e.length} decision(s) with ${B} \u2192 ${this.modelPath}`)):(a=!1,c="No <model> block found in response",console.error(`${te} ${c}`))}}catch(g){a=!1,c=g instanceof Error?g.message:String(g),console.error(`${te} Error:`,c)}let m=await this.loadIndex(),P=new Set(m.materializedIds);for(let g of e)P.add(g);return m.materializedIds=[...P],m.lastRunAt=Date.now(),m.lastRunDecisionIds=e,m.lastRunError=c??null,await this.persistIndex(m),this.options.onEvent?.({type:"materialize_done",decisionIds:e,success:a,error:c}),{processedIds:e,success:a,error:c}}finally{this.running=!1}}async consolidate(t={}){if(this.running)return{success:!1,error:"Already running"};this.running=!0;try{let n=await this.loadModel();if(!n)return{success:!1,error:"No model exists"};let e=Fs(n),{provider:o,result:i}=this.spawnModelAgent(`consolidate-${Date.now()}`,e,t,{maxTurns:this.options.maxTurns??3,maxBudgetUsd:this.options.maxBudgetUsd??.3,allowedTools:[]}),a=await i;if(!a.success)return console.error(`${te} ${o} consolidation spawn error:`,a.error),{success:!1,error:a.error};let c=He(a.text);return c?(Array.isArray(c.rules)&&(c.rules=Te(c.rules)),!c.tokens&&n.tokens&&(c.tokens=n.tokens),!c.components&&n.components&&(c.components=n.components),await ve(this.modelPath,JSON.stringify(c,null,2)),console.log(`${te} Consolidation complete with ${o} \u2192 ${this.modelPath}`),{success:!0}):(console.error(`${te} No <model> block in consolidation response`),{success:!1,error:"No <model> block found"})}catch(n){let e=n instanceof Error?n.message:String(n);return console.error(`${te} Consolidation error:`,e),{success:!1,error:e}}finally{this.running=!1}}async writeModel(t){Array.isArray(t.rules)&&(t.rules=Te(t.rules)),await ve(this.modelPath,JSON.stringify(t,null,2)),console.log(`${te} Model written \u2192 ${this.modelPath}`)}async loadIndex(){if(this.cachedIndex)return this.cachedIndex;try{let t=await Gt(this.indexPath,"utf-8"),n=JSON.parse(t);return this.cachedIndex=n,n}catch{return this.cachedIndex={...Ns,materializedIds:[],lastRunDecisionIds:[]},this.cachedIndex}}async persistIndex(t){this.cachedIndex=t;try{await ve(this.indexPath,JSON.stringify(t,null,2))}catch(n){console.error(`${te} Failed to write index:`,n)}}spawnModelAgent(t,n,e,o){let i=e.provider??this.options.provider??"claude",a=this.options.onEvent?(c,m)=>this.options.onEvent?.(c):void 0;return i==="codex"?{provider:"codex",...je(t,{prompt:n,projectRoot:this.projectRoot,model:e.model??this.options.model,timeoutMs:this.options.timeoutMs,sandboxMode:"read-only",onEvent:a})}:(i==="copilot"&&console.warn(`${te} Copilot materialization is not available yet; falling back to Claude.`),{provider:"claude",...Be(t,{prompt:n,projectRoot:this.projectRoot,maxTurns:o.maxTurns,maxBudgetUsd:o.maxBudgetUsd,allowedTools:o.allowedTools,claudePath:this.options.claudePath??"claude",model:i==="claude"?e.model??this.options.model:void 0,timeoutMs:this.options.timeoutMs,onEvent:a})})}};function He(s){let t=s.match(/<model>\s*([\s\S]*?)\s*<\/model>/);if(!t?.[1])return null;try{let n=JSON.parse(t[1]);return typeof n!="object"||n===null||Array.isArray(n)?null:n}catch{return null}}function Bs(s,t){let n=s.map(i=>{let c=i.resolutions.filter(g=>(g.finalScope??g.inferredScope)?.breadth==="pattern").map(g=>{let z=(g.finalScope??g.inferredScope)?.target??"unknown",O=g.filesModified?.join(", ")??"none";return`- **${g.summary}** [scope: pattern/${z}]
159
- Files modified: ${O}`}).join(`
160
- `),m=i.annotations.map(g=>g.instruction).filter(Boolean).join(`
158
+ ${n}`}function he(s){return typeof s=="string"?s.trim():""}function be(s){return Array.isArray(s)?s.filter(t=>typeof t=="string"&&t.trim().length>0).map(t=>t.trim()):[]}function Ms(s){let t=he(s);return t==="component"||t==="layout"||t==="copy"||t==="interaction"||t==="visual"||t==="workflow"?t:"visual"}function Cs(s){let t=he(s);return t==="pending"||t==="approved"||t==="rejected"?t:null}function $s(s){let t=he(s);return t==="pass"||t==="fail"||t==="needs_review"?t:null}function _s(s,t){let e=(Array.isArray(s)?s:[]).map(o=>{if(!o||typeof o!="object")return null;let i=o,a=he(i.assertion);return a?{assertion:a,status:$s(i.status)??"needs_review",summary:he(i.summary)||Ht("needs_review"),evidence:be(i.evidence)}:null}).filter(o=>o!==null);return e.length>0?e:t.map(o=>({assertion:o,status:"needs_review",summary:"No structured assertion result was returned.",evidence:[]}))}function Os(s){return s.some(t=>t.status==="fail")?"fail":s.some(t=>t.status==="needs_review")?"needs_review":"pass"}function As(s){if(!s||typeof s!="object")return{chromeDevtools:"unknown",evidenceTypes:["screenshot"]};let t=s,n=he(t.chromeDevtools);return{chromeDevtools:n==="available"||n==="unavailable"||n==="unknown"?n:"unknown",evidenceTypes:be(t.evidenceTypes).length>0?be(t.evidenceTypes):["screenshot"]}}function Ht(s){return s==="pass"?"The current page appears to satisfy this eval.":s==="fail"?"The current page appears to violate this eval.":"The current page needs human review for this eval."}function Ds(s){return Gt(s).slice(0,48)}function Gt(s){return s.toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/^-+|-+$/g,"")}function Le(s,t){return s.length>t?`${s.slice(0,t)}...`:s}import{readFile as Xt,writeFile as ve}from"fs/promises";import{join as it}from"path";var te="[Materializer]",Ns={version:1,materializedIds:[],lastRunAt:null,lastRunDecisionIds:[],lastRunError:null};function Js(s){return Array.isArray(s)?s.map((t,n)=>typeof t=="string"?{id:Math.random().toString(16).slice(2,10),scope:"general",text:t,sources:[]}:t&&typeof t=="object"&&typeof t.text=="string"?t:{id:Math.random().toString(16).slice(2,10),scope:"general",text:String(t),sources:[]}):[]}function Te(s){let t=[];for(let n of s){if(!n||typeof n!="object")continue;let e=n;if(typeof e.id!="string"||typeof e.text!="string"){console.warn(`${te} Dropping rule missing id or text:`,JSON.stringify(n).slice(0,120));continue}t.push({id:e.id,scope:typeof e.scope=="string"?e.scope:"general",text:e.text,sources:Array.isArray(e.sources)?e.sources.filter(o=>typeof o=="string"):[]})}return t.length>30&&console.warn(`${te} Rule count ${t.length} exceeds cap of 30`),t}var We=class{constructor(t,n,e={}){this.projectRoot=t;this.decisionStore=n;this.options=e;this.cachedIndex=null;this.running=!1;let o=it(t,".popmelt");this.indexPath=it(o,"materialized.json"),this.modelPath=it(o,"model.json")}get isRunning(){return this.running}async loadModel(){try{let t=await Xt(this.modelPath,"utf-8"),n=JSON.parse(t);return Array.isArray(n.rules)&&(n.rules=Js(n.rules)),n}catch{return null}}async addComponent(t){let n=await this.loadModel();n||(n={tokens:{},components:{},rules:[]}),(!n.components||typeof n.components!="object")&&(n.components={});let e=n.components;return e[t]?{added:!1,alreadyExists:!0}:(e[t]={description:""},await ve(this.modelPath,JSON.stringify(n,null,2)),console.log(`${te} Added component "${t}" to model`),{added:!0,alreadyExists:!1})}async updateToken(t,n){let e=await this.loadModel();e||(e={tokens:{},components:{},rules:[]});let o=t.split("."),i=e;for(let m=0;m<o.length-1;m++){let P=o[m];(!i[P]||typeof i[P]!="object")&&(i[P]={}),i=i[P]}let a=o[o.length-1],c;try{c=JSON.parse(n)}catch{c=null}if(c&&typeof c=="object"&&c!==null&&"value"in c)i[a]=c;else{let m=i[a];m&&typeof m=="object"&&m!==null&&"value"in m?m.value=n:i[a]=n}return await ve(this.modelPath,JSON.stringify(e,null,2)),console.log(`${te} Updated token "${t}" \u2192 "${n.slice(0,80)}"`),{updated:!0}}async removeToken(t){let n=await this.loadModel();if(!n)return{removed:!1};let e=t.split("."),o=n;for(let a=0;a<e.length-1;a++){let c=e[a];if(!o[c]||typeof o[c]!="object")return{removed:!1};o=o[c]}let i=e[e.length-1];return i in o?(delete o[i],await ve(this.modelPath,JSON.stringify(n,null,2)),console.log(`${te} Removed token "${t}" from model`),{removed:!0}):{removed:!1}}async removeComponent(t){let n=await this.loadModel();if(!n)return{removed:!1};let e=n.components;return!e||!e[t]?{removed:!1}:(delete e[t],await ve(this.modelPath,JSON.stringify(n,null,2)),console.log(`${te} Removed component "${t}" from model`),{removed:!0})}async getUnmaterializedPatternDecisions(){let t=await this.loadIndex(),n=new Set(t.materializedIds),o=(await this.decisionStore.listDecisionIds()).filter(a=>!n.has(a));return o.length===0?[]:(await this.decisionStore.loadDecisions(o)).filter(a=>a.resolutions.some(c=>(c.finalScope??c.inferredScope)?.breadth==="pattern"))}async run(t={}){if(this.running)return{processedIds:[],success:!0,error:"Already running"};this.running=!0;try{let n=await this.getUnmaterializedPatternDecisions();if(n.length===0)return{processedIds:[],success:!0};let e=n.map(v=>v.id);console.log(`${te} Processing ${e.length} pattern-scoped decision(s): ${e.join(", ")}`),this.options.onEvent?.({type:"materialize_started",decisionIds:e});let o=await this.loadModel(),i=Bs(n,o),a=!0,c;try{let v=`mat-${Date.now()}`,{provider:B,result:z}=this.spawnModelAgent(v,i,t,{maxTurns:this.options.maxTurns??5,maxBudgetUsd:this.options.maxBudgetUsd??.5,allowedTools:["Read"]}),A=await z;if(!A.success)a=!1,c=A.error,console.error(`${te} ${B} spawn error:`,c);else{let W=He(A.text);W?(Array.isArray(W.rules)&&(W.rules=Te(W.rules)),await ve(this.modelPath,JSON.stringify(W,null,2)),console.log(`${te} Successfully materialized ${e.length} decision(s) with ${B} \u2192 ${this.modelPath}`)):(a=!1,c="No <model> block found in response",console.error(`${te} ${c}`))}}catch(v){a=!1,c=v instanceof Error?v.message:String(v),console.error(`${te} Error:`,c)}let m=await this.loadIndex(),P=new Set(m.materializedIds);for(let v of e)P.add(v);return m.materializedIds=[...P],m.lastRunAt=Date.now(),m.lastRunDecisionIds=e,m.lastRunError=c??null,await this.persistIndex(m),this.options.onEvent?.({type:"materialize_done",decisionIds:e,success:a,error:c}),{processedIds:e,success:a,error:c}}finally{this.running=!1}}async consolidate(t={}){if(this.running)return{success:!1,error:"Already running"};this.running=!0;try{let n=await this.loadModel();if(!n)return{success:!1,error:"No model exists"};let e=Fs(n),{provider:o,result:i}=this.spawnModelAgent(`consolidate-${Date.now()}`,e,t,{maxTurns:this.options.maxTurns??3,maxBudgetUsd:this.options.maxBudgetUsd??.3,allowedTools:[]}),a=await i;if(!a.success)return console.error(`${te} ${o} consolidation spawn error:`,a.error),{success:!1,error:a.error};let c=He(a.text);return c?(Array.isArray(c.rules)&&(c.rules=Te(c.rules)),!c.tokens&&n.tokens&&(c.tokens=n.tokens),!c.components&&n.components&&(c.components=n.components),await ve(this.modelPath,JSON.stringify(c,null,2)),console.log(`${te} Consolidation complete with ${o} \u2192 ${this.modelPath}`),{success:!0}):(console.error(`${te} No <model> block in consolidation response`),{success:!1,error:"No <model> block found"})}catch(n){let e=n instanceof Error?n.message:String(n);return console.error(`${te} Consolidation error:`,e),{success:!1,error:e}}finally{this.running=!1}}async writeModel(t){Array.isArray(t.rules)&&(t.rules=Te(t.rules)),await ve(this.modelPath,JSON.stringify(t,null,2)),console.log(`${te} Model written \u2192 ${this.modelPath}`)}async loadIndex(){if(this.cachedIndex)return this.cachedIndex;try{let t=await Xt(this.indexPath,"utf-8"),n=JSON.parse(t);return this.cachedIndex=n,n}catch{return this.cachedIndex={...Ns,materializedIds:[],lastRunDecisionIds:[]},this.cachedIndex}}async persistIndex(t){this.cachedIndex=t;try{await ve(this.indexPath,JSON.stringify(t,null,2))}catch(n){console.error(`${te} Failed to write index:`,n)}}spawnModelAgent(t,n,e,o){let i=e.provider??this.options.provider??"claude",a=this.options.onEvent?(c,m)=>this.options.onEvent?.(c):void 0;return i==="codex"?{provider:"codex",...je(t,{prompt:n,projectRoot:this.projectRoot,model:e.model??this.options.model,timeoutMs:this.options.timeoutMs,sandboxMode:"read-only",onEvent:a})}:(i==="copilot"&&console.warn(`${te} Copilot materialization is not available yet; falling back to Claude.`),{provider:"claude",...Be(t,{prompt:n,projectRoot:this.projectRoot,maxTurns:o.maxTurns,maxBudgetUsd:o.maxBudgetUsd,allowedTools:o.allowedTools,claudePath:this.options.claudePath??"claude",model:i==="claude"?e.model??this.options.model:void 0,timeoutMs:this.options.timeoutMs,onEvent:a})})}};function He(s){let t=s.match(/<model>\s*([\s\S]*?)\s*<\/model>/);if(!t?.[1])return null;try{let n=JSON.parse(t[1]);return typeof n!="object"||n===null||Array.isArray(n)?null:n}catch{return null}}function Bs(s,t){let n=s.map(i=>{let c=i.resolutions.filter(v=>(v.finalScope??v.inferredScope)?.breadth==="pattern").map(v=>{let z=(v.finalScope??v.inferredScope)?.target??"unknown",A=v.filesModified?.join(", ")??"none";return`- **${v.summary}** [scope: pattern/${z}]
159
+ Files modified: ${A}`}).join(`
160
+ `),m=i.annotations.map(v=>v.instruction).filter(Boolean).join(`
161
161
  `),P=i.gitDiff?`
162
162
  \`\`\`diff
163
163
  ${i.gitDiff.slice(0,2e3)}
@@ -280,7 +280,7 @@ Output the complete model inside <model> tags. Preserve tokens and components as
280
280
 
281
281
  <model>
282
282
  { "tokens": { ... }, "components": { ... }, "rules": [ ... ] }
283
- </model>`}function Xt(s){return`You are a design system curator reviewing a project's design model. Your job is to propose improvements to the rules \u2014 merging duplicates, filling gaps, removing noise.
283
+ </model>`}function qt(s){return`You are a design system curator reviewing a project's design model. Your job is to propose improvements to the rules \u2014 merging duplicates, filling gaps, removing noise.
284
284
 
285
285
  ## Current Model
286
286
  \`\`\`json
@@ -313,47 +313,47 @@ These are my proposed rule changes. Would you like to approve all of them, adjus
313
313
  </question>
314
314
 
315
315
  ## After approval
316
- When the developer approves (says "yes", "looks good", "go ahead", etc.), output the complete updated model inside <model> tags \u2014 the full JSON with tokens, components, and the revised rules array. Do NOT output partial models. Preserve tokens and components as-is unless the developer asks to change them.`}import{execFile as js}from"child_process";import{readFile as ct}from"fs/promises";import{homedir as Xe}from"os";import{join as ue}from"path";var lt=/popmelt/i,Us=/chrome|devtools/i;function we(){return{found:!1,name:null,scope:null,disabled:!1}}function xe(s,t,n=!1){return{found:!0,name:s,scope:t,disabled:n}}function Me(s){for(let t of Object.keys(s))if(lt.test(t))return t;return null}async function Ge(s){try{let t=await ct(s,"utf-8");return JSON.parse(t)}catch{return null}}async function dt(s){let t=Xe(),n=await Ge(ue(t,".claude.json"));if(n&&typeof n=="object"){let o=n;if(o.mcpServers&&typeof o.mcpServers=="object"){let i=Me(o.mcpServers);if(i)return xe(i,"user")}if(o.projects&&typeof o.projects=="object"){let a=o.projects[s];if(a&&typeof a=="object"){let c=a;if(c.mcpServers&&typeof c.mcpServers=="object"){let m=Me(c.mcpServers);if(m){let P=Array.isArray(c.disabledMcpjsonServers)&&c.disabledMcpjsonServers.some(g=>lt.test(g));return xe(m,"project",P)}}}}}let e=await Ge(ue(s,".mcp.json"));if(e&&typeof e=="object"){let o=e;if(o.mcpServers&&typeof o.mcpServers=="object"){let a=Me(o.mcpServers);if(a){let c=await qt(s,a);return xe(a,"mcp.json",c)}}let i=Me(o);if(i&&i!=="mcpServers"){let a=await qt(s,i);return xe(i,"mcp.json",a)}}return we()}async function qt(s,t){let n=ue(s,".claude","settings.local.json"),e=await Ge(n);if(e&&typeof e=="object"){let o=e;if(Array.isArray(o.disabledMcpjsonServers))return o.disabledMcpjsonServers.some(i=>i===t)}return!1}var zs=/^\[mcp_servers\.([^\]]+)\]/;function Ls(s){return s.replace(/^"(.+)"$/,"$1")}function Ws(s){return/\.tools\./.test(s)||/"\.tools\./.test(s)}function Kt(s){let t=s.split(`
316
+ When the developer approves (says "yes", "looks good", "go ahead", etc.), output the complete updated model inside <model> tags \u2014 the full JSON with tokens, components, and the revised rules array. Do NOT output partial models. Preserve tokens and components as-is unless the developer asks to change them.`}import{execFile as js}from"child_process";import{readFile as ct}from"fs/promises";import{homedir as Xe}from"os";import{join as ue}from"path";var lt=/popmelt/i,Us=/chrome|devtools/i;function we(){return{found:!1,name:null,scope:null,disabled:!1}}function xe(s,t,n=!1){return{found:!0,name:s,scope:t,disabled:n}}function Me(s){for(let t of Object.keys(s))if(lt.test(t))return t;return null}async function Ge(s){try{let t=await ct(s,"utf-8");return JSON.parse(t)}catch{return null}}async function dt(s){let t=Xe(),n=await Ge(ue(t,".claude.json"));if(n&&typeof n=="object"){let o=n;if(o.mcpServers&&typeof o.mcpServers=="object"){let i=Me(o.mcpServers);if(i)return xe(i,"user")}if(o.projects&&typeof o.projects=="object"){let a=o.projects[s];if(a&&typeof a=="object"){let c=a;if(c.mcpServers&&typeof c.mcpServers=="object"){let m=Me(c.mcpServers);if(m){let P=Array.isArray(c.disabledMcpjsonServers)&&c.disabledMcpjsonServers.some(v=>lt.test(v));return xe(m,"project",P)}}}}}let e=await Ge(ue(s,".mcp.json"));if(e&&typeof e=="object"){let o=e;if(o.mcpServers&&typeof o.mcpServers=="object"){let a=Me(o.mcpServers);if(a){let c=await Yt(s,a);return xe(a,"mcp.json",c)}}let i=Me(o);if(i&&i!=="mcpServers"){let a=await Yt(s,i);return xe(i,"mcp.json",a)}}return we()}async function Yt(s,t){let n=ue(s,".claude","settings.local.json"),e=await Ge(n);if(e&&typeof e=="object"){let o=e;if(Array.isArray(o.disabledMcpjsonServers))return o.disabledMcpjsonServers.some(i=>i===t)}return!1}var zs=/^\[mcp_servers\.([^\]]+)\]/;function Ls(s){return s.replace(/^"(.+)"$/,"$1")}function Ws(s){return/\.tools\./.test(s)||/"\.tools\./.test(s)}function Qt(s){let t=s.split(`
317
317
  `),n=[],e=null;for(let o of t){let i=o.match(zs);i?(e&&n.push({name:e.name,body:e.bodyLines.join(`
318
318
  `)}),e={name:i[1],bodyLines:[]}):o.startsWith("[")?e&&(n.push({name:e.name,body:e.bodyLines.join(`
319
319
  `)}),e=null):e&&e.bodyLines.push(o)}return e&&n.push({name:e.name,body:e.bodyLines.join(`
320
- `)}),n}function Qt(s){return/enabled\s*=\s*false/i.test(s)}async function ut(s){let t=process.env.CODEX_HOME||ue(Xe(),".codex"),n=await Yt(ue(t,"config.toml"),"user");if(n.found)return n;let e=await Yt(ue(s,".codex","config.toml"),"project");return e.found?e:we()}async function Hs(s,t){let n=process.env.CODEX_HOME||ue(Xe(),".codex"),e=new Set;for(let o of[ue(n,"config.toml"),ue(s,".codex","config.toml")])try{let i=await ct(o,"utf-8");for(let a of Kt(i)){if(Ws(a.name)||Qt(a.body))continue;let c=Ls(a.name);t.test(c)&&e.add(c)}}catch{}return[...e]}async function Vt(s){return Hs(s,Us)}async function pt(s,t="copilot"){let n=await Gs(t);if(n.found)return n;let e=process.env.COPILOT_HOME||ue(Xe(),".copilot"),o=await at(ue(e,"mcp-config.json"),"user");if(o.found)return o;let i=await at(ue(s,".mcp.json"),"mcp.json");if(i.found)return i;let a=await at(ue(s,".github","mcp.json"),"mcp.json");return a.found?a:we()}async function Gs(s){try{let t=await new Promise((n,e)=>{js(s,["mcp","list","--json"],{encoding:"utf-8",timeout:5e3},(o,i)=>{if(o){e(o);return}n(i)})});return Zt(JSON.parse(t),"user")}catch{return we()}}async function at(s,t){let n=await Ge(s);return Zt(n,t)}function Zt(s,t){if(!s||typeof s!="object")return we();let n=s,e=n.mcpServers&&typeof n.mcpServers=="object"?n.mcpServers:n.servers&&typeof n.servers=="object"?n.servers:n,o=Me(e);return o?xe(o,t):we()}async function Yt(s,t){try{let n=await ct(s,"utf-8"),e=Kt(n);for(let o of e)if(lt.test(o.name)){let i=Qt(o.body);return xe(o.name,t,i)}}catch{}return we()}import{execFile as en}from"child_process";import{mkdir as Xs,readFile as tn,writeFile as nn}from"fs/promises";import{homedir as sn}from"os";import{dirname as qs,join as ft}from"path";var ht="https://mcp.popmelt.com/mcp";async function on(s=ht){let t=ft(sn(),".claude.json"),n;try{let o=await tn(t,"utf-8");n=JSON.parse(o)}catch{n={}}(!n.mcpServers||typeof n.mcpServers!="object")&&(n.mcpServers={});let e=n.mcpServers;for(let o of Object.keys(e))if(/popmelt/i.test(o))return{installed:!1,provider:"claude",scope:null,reason:"already_configured"};return e.popmelt={type:"http",url:s},await nn(t,JSON.stringify(n,null,2)+`
321
- `,"utf-8"),{installed:!0,provider:"claude",scope:"user"}}async function rn(s=ht){let t=process.env.CODEX_HOME||ft(sn(),".codex"),n=ft(t,"config.toml"),e;try{e=await tn(n,"utf-8")}catch{e=""}if(/\[mcp_servers\.[^\]]*popmelt[^\]]*\]/i.test(e))return{installed:!1,provider:"codex",scope:null,reason:"already_configured"};await Xs(qs(n),{recursive:!0});let o=`
320
+ `)}),n}function Vt(s){return/enabled\s*=\s*false/i.test(s)}async function ut(s){let t=process.env.CODEX_HOME||ue(Xe(),".codex"),n=await Kt(ue(t,"config.toml"),"user");if(n.found)return n;let e=await Kt(ue(s,".codex","config.toml"),"project");return e.found?e:we()}async function Hs(s,t){let n=process.env.CODEX_HOME||ue(Xe(),".codex"),e=new Set;for(let o of[ue(n,"config.toml"),ue(s,".codex","config.toml")])try{let i=await ct(o,"utf-8");for(let a of Qt(i)){if(Ws(a.name)||Vt(a.body))continue;let c=Ls(a.name);t.test(c)&&e.add(c)}}catch{}return[...e]}async function Zt(s){return Hs(s,Us)}async function pt(s,t="copilot"){let n=await Gs(t);if(n.found)return n;let e=process.env.COPILOT_HOME||ue(Xe(),".copilot"),o=await at(ue(e,"mcp-config.json"),"user");if(o.found)return o;let i=await at(ue(s,".mcp.json"),"mcp.json");if(i.found)return i;let a=await at(ue(s,".github","mcp.json"),"mcp.json");return a.found?a:we()}async function Gs(s){try{let t=await new Promise((n,e)=>{js(s,["mcp","list","--json"],{encoding:"utf-8",timeout:5e3},(o,i)=>{if(o){e(o);return}n(i)})});return en(JSON.parse(t),"user")}catch{return we()}}async function at(s,t){let n=await Ge(s);return en(n,t)}function en(s,t){if(!s||typeof s!="object")return we();let n=s,e=n.mcpServers&&typeof n.mcpServers=="object"?n.mcpServers:n.servers&&typeof n.servers=="object"?n.servers:n,o=Me(e);return o?xe(o,t):we()}async function Kt(s,t){try{let n=await ct(s,"utf-8"),e=Qt(n);for(let o of e)if(lt.test(o.name)){let i=Vt(o.body);return xe(o.name,t,i)}}catch{}return we()}import{execFile as tn}from"child_process";import{mkdir as Xs,readFile as nn,writeFile as sn}from"fs/promises";import{homedir as on}from"os";import{dirname as qs,join as ft}from"path";var ht="https://mcp.popmelt.com/mcp";async function rn(s=ht){let t=ft(on(),".claude.json"),n;try{let o=await nn(t,"utf-8");n=JSON.parse(o)}catch{n={}}(!n.mcpServers||typeof n.mcpServers!="object")&&(n.mcpServers={});let e=n.mcpServers;for(let o of Object.keys(e))if(/popmelt/i.test(o))return{installed:!1,provider:"claude",scope:null,reason:"already_configured"};return e.popmelt={type:"http",url:s},await sn(t,JSON.stringify(n,null,2)+`
321
+ `,"utf-8"),{installed:!0,provider:"claude",scope:"user"}}async function an(s=ht){let t=process.env.CODEX_HOME||ft(on(),".codex"),n=ft(t,"config.toml"),e;try{e=await nn(n,"utf-8")}catch{e=""}if(/\[mcp_servers\.[^\]]*popmelt[^\]]*\]/i.test(e))return{installed:!1,provider:"codex",scope:null,reason:"already_configured"};await Xs(qs(n),{recursive:!0});let o=`
322
322
  [mcp_servers.popmelt]
323
323
  url = "${s}"
324
- `;return await nn(n,e+o,"utf-8"),{installed:!0,provider:"codex",scope:"user"}}async function an(s=ht,t="copilot"){let n=await Ys(t);if(Ks(n))return{installed:!1,provider:"copilot",scope:null,reason:"already_configured"};try{return await new Promise((e,o)=>{en(t,["mcp","add","--transport","http","--json","popmelt",s],{encoding:"utf-8",timeout:1e4},i=>{if(i){o(i);return}e()})}),{installed:!0,provider:"copilot",scope:"user"}}catch(e){return{installed:!1,provider:"copilot",scope:null,reason:e instanceof Error?e.message:"install_failed"}}}async function Ys(s){try{let t=await new Promise((n,e)=>{en(s,["mcp","list","--json"],{encoding:"utf-8",timeout:5e3},(o,i)=>{if(o){e(o);return}n(i)})});return JSON.parse(t)}catch{return null}}function Ks(s){if(!s||typeof s!="object")return!1;let t=s,n=t.mcpServers&&typeof t.mcpServers=="object"?t.mcpServers:t;return Object.keys(n).some(e=>/popmelt/i.test(e))}async function qe(s){let n=(s.headers["content-type"]||"").match(/boundary=(?:"([^"]+)"|([^\s;]+))/);if(!n)throw new Error("Missing multipart boundary");let e=n[1]||n[2],o=await Qs(s),i=Buffer.from(`--${e}`),a=Buffer.from(`--${e}--`),c,m,P,g,B,z,O,L,S,X,W,Q,Z=[],V=[],q=0,D=[];for(;q<o.length;){let _=o.indexOf(i,q);if(_===-1)break;let I=_+i.length;if(o.slice(_,_+a.length).equals(a))break;let k=I;o[k]===13&&o[k+1]===10&&(k+=2);let y=o.indexOf(`\r
324
+ `;return await sn(n,e+o,"utf-8"),{installed:!0,provider:"codex",scope:"user"}}async function cn(s=ht,t="copilot"){let n=await Ys(t);if(Ks(n))return{installed:!1,provider:"copilot",scope:null,reason:"already_configured"};try{return await new Promise((e,o)=>{tn(t,["mcp","add","--transport","http","--json","popmelt",s],{encoding:"utf-8",timeout:1e4},i=>{if(i){o(i);return}e()})}),{installed:!0,provider:"copilot",scope:"user"}}catch(e){return{installed:!1,provider:"copilot",scope:null,reason:e instanceof Error?e.message:"install_failed"}}}async function Ys(s){try{let t=await new Promise((n,e)=>{tn(s,["mcp","list","--json"],{encoding:"utf-8",timeout:5e3},(o,i)=>{if(o){e(o);return}n(i)})});return JSON.parse(t)}catch{return null}}function Ks(s){if(!s||typeof s!="object")return!1;let t=s,n=t.mcpServers&&typeof t.mcpServers=="object"?t.mcpServers:t;return Object.keys(n).some(e=>/popmelt/i.test(e))}async function qe(s){let n=(s.headers["content-type"]||"").match(/boundary=(?:"([^"]+)"|([^\s;]+))/);if(!n)throw new Error("Missing multipart boundary");let e=n[1]||n[2],o=await Qs(s),i=Buffer.from(`--${e}`),a=Buffer.from(`--${e}--`),c,m,P,v,B,z,A,W,S,X,H,K,Z=[],Q=[],V=0,N=[];for(;V<o.length;){let _=o.indexOf(i,V);if(_===-1)break;let R=_+i.length;if(o.slice(_,_+a.length).equals(a))break;let k=R;o[k]===13&&o[k+1]===10&&(k+=2);let w=o.indexOf(`\r
325
325
  \r
326
- `,k);if(y===-1)break;let C=o.slice(k,y).toString("utf-8"),N=y+4,x=o.indexOf(i,N),b=x!==-1?x-2:o.length;D.push({headers:C,body:o.slice(N,b)}),q=x!==-1?x:o.length}for(let _ of D){let I=_.headers.match(/name="([^"]+)"/);if(!I)continue;let k=I[1];if(k==="screenshot")c=_.body;else if(k==="feedback")m=_.body.toString("utf-8");else if(k==="color")P=_.body.toString("utf-8");else if(k==="provider")g=_.body.toString("utf-8");else if(k==="model")B=_.body.toString("utf-8");else if(k==="goal")z=_.body.toString("utf-8");else if(k==="pageUrl")O=_.body.toString("utf-8");else if(k==="viewport")L=_.body.toString("utf-8");else if(k==="planId")S=_.body.toString("utf-8");else if(k==="manifest")X=_.body.toString("utf-8");else if(k==="tasks")W=_.body.toString("utf-8");else if(k==="sourceId")Q=_.body.toString("utf-8");else if(k.startsWith("screenshot-")){let y=k.slice(11);try{let C=decodeURIComponent(y);V.push({pathname:C,data:_.body})}catch{}}else if(k.startsWith("image-")){let y=k.split("-"),C=parseInt(y[y.length-1],10),N=y.slice(1,-1).join("-");N&&!isNaN(C)&&Z.push({annotationId:N,index:C,data:_.body})}}if(!c)throw new Error("Missing screenshot field");return m||(m=""),{screenshot:c,feedback:m,color:P,provider:g,model:B,goal:z,pageUrl:O,viewport:L,planId:S,manifest:X,tasks:W,sourceId:Q,pastedImages:Z,pageScreenshots:V}}function Qs(s){return new Promise((t,n)=>{let e=[];s.on("data",o=>e.push(o)),s.on("end",()=>t(Buffer.concat(e))),s.on("error",n)})}function Ce(s,t){let n=[];if(s.annotations.length>0){n.push("## Annotations");for(let e of s.annotations){let o=e.elements.map(c=>{let m=[c.selector];return c.reactComponent&&m.push(`(${c.reactComponent})`),m.join(" ")}).join(", "),i=e.instruction||"No text";n.push(`- id=${e.id} [${e.type}] ${i} \u2192 Elements: ${o||"none"}`);let a=t?.[e.id];if(a&&a.length>0)for(let c of a)n.push(` Attached image: use the Read tool to view ${c}`)}}if(s.styleModifications.length>0){n.push(""),n.push("## Style Changes (make permanent in source)"),n.push("The developer previewed these CSS changes via inline style overrides. Find the corresponding styles in the source files and update them so the changes persist:");for(let e of s.styleModifications){let o=e.element?.reactComponent?`(${e.element.reactComponent})`:"";for(let i of e.changes)n.push(`- ${e.selector} ${o}: ${i.property} ${i.original} \u2192 ${i.modified}`)}}if(s.spacingTokenChanges?.length){n.push(""),n.push("## Spacing Token Changes"),n.push("The developer adjusted these spacing tokens. Apply each change to the source code:");for(let e of s.spacingTokenChanges){n.push(`
326
+ `,k);if(w===-1)break;let C=o.slice(k,w).toString("utf-8"),J=w+4,x=o.indexOf(i,J),b=x!==-1?x-2:o.length;N.push({headers:C,body:o.slice(J,b)}),V=x!==-1?x:o.length}for(let _ of N){let R=_.headers.match(/name="([^"]+)"/);if(!R)continue;let k=R[1];if(k==="screenshot")c=_.body;else if(k==="feedback")m=_.body.toString("utf-8");else if(k==="color")P=_.body.toString("utf-8");else if(k==="provider")v=_.body.toString("utf-8");else if(k==="model")B=_.body.toString("utf-8");else if(k==="goal")z=_.body.toString("utf-8");else if(k==="pageUrl")A=_.body.toString("utf-8");else if(k==="viewport")W=_.body.toString("utf-8");else if(k==="planId")S=_.body.toString("utf-8");else if(k==="manifest")X=_.body.toString("utf-8");else if(k==="tasks")H=_.body.toString("utf-8");else if(k==="sourceId")K=_.body.toString("utf-8");else if(k.startsWith("screenshot-")){let w=k.slice(11);try{let C=decodeURIComponent(w);Q.push({pathname:C,data:_.body})}catch{}}else if(k.startsWith("image-")){let w=k.split("-"),C=parseInt(w[w.length-1],10),J=w.slice(1,-1).join("-");J&&!isNaN(C)&&Z.push({annotationId:J,index:C,data:_.body})}}if(!c)throw new Error("Missing screenshot field");return m||(m=""),{screenshot:c,feedback:m,color:P,provider:v,model:B,goal:z,pageUrl:A,viewport:W,planId:S,manifest:X,tasks:H,sourceId:K,pastedImages:Z,pageScreenshots:Q}}function Qs(s){return new Promise((t,n)=>{let e=[];s.on("data",o=>e.push(o)),s.on("end",()=>t(Buffer.concat(e))),s.on("error",n)})}function Ce(s,t){let n=[];if(s.annotations.length>0){n.push("## Annotations");for(let e of s.annotations){let o=e.elements.map(c=>{let m=[c.selector];return c.reactComponent&&m.push(`(${c.reactComponent})`),m.join(" ")}).join(", "),i=e.instruction||"No text";n.push(`- id=${e.id} [${e.type}] ${i} \u2192 Elements: ${o||"none"}`);let a=t?.[e.id];if(a&&a.length>0)for(let c of a)n.push(` Attached image: use the Read tool to view ${c}`)}}if(s.styleModifications.length>0){n.push(""),n.push("## Style Changes (make permanent in source)"),n.push("The developer previewed these CSS changes via inline style overrides. Find the corresponding styles in the source files and update them so the changes persist:");for(let e of s.styleModifications){let o=e.element?.reactComponent?`(${e.element.reactComponent})`:"";for(let i of e.changes)n.push(`- ${e.selector} ${o}: ${i.property} ${i.original} \u2192 ${i.modified}`)}}if(s.spacingTokenChanges?.length){n.push(""),n.push("## Spacing Token Changes"),n.push("The developer adjusted these spacing tokens. Apply each change to the source code:");for(let e of s.spacingTokenChanges){n.push(`
327
327
  ### ${e.tokenName}: ${e.originalPx}px \u2192 ${e.newPx}px`);for(let o of e.affectedElements){let i=o.reactComponent?` (${o.reactComponent})`:"";o.matchedClass&&o.suggestedClass?n.push(`- ${o.selector}${i}: \`${o.matchedClass}\` \u2192 \`${o.suggestedClass}\``):n.push(`- ${o.selector}${i}: ${o.property} ${e.originalPx}px \u2192 ${e.newPx}px`),n.push(` class="${o.className}"`)}}}if(s.inspectedElement){let e=s.inspectedElement;n.push(""),n.push("## Inspected Element"),n.push("The developer has this element selected in the inspector:");let o=[e.selector];e.reactComponent&&o.push(`(${e.reactComponent})`),e.context&&o.push(`in ${e.context}`),e.textContent&&o.push(`"${e.textContent.slice(0,80)}"`),n.push(`- ${o.join(" ")}`)}return n.join(`
328
- `)}function cn(s,t,n){let e=[],i=new Set(t.annotations.map(a=>a.pathname).filter(Boolean)).size>1;if(e.push("You are reviewing a UI screenshot with developer annotations."),e.push(""),!i&&n?.provider!=="codex"&&(e.push(`IMPORTANT: First, use the Read tool to view the screenshot at: ${s}`),e.push("")),e.push(`The developer annotated their running app at ${t.url} (${t.viewport.width}x${t.viewport.height}).`),n?.threadHistory&&n.threadHistory.length>0){e.push(""),e.push("## Previous Conversation");let a=0;for(let c of n.threadHistory)if(c.role==="human")a++,c.replyToQuestion?(e.push(`### Round ${a} (human) \u2014 reply`),e.push(`"${c.replyToQuestion}"`)):(e.push(`### Round ${a} (human)`),c.feedbackSummary&&e.push(`Annotations: ${c.feedbackSummary}`),c.annotationIds&&c.annotationIds.length>0&&e.push(`Annotation IDs: ${c.annotationIds.join(", ")}`));else if(c.question)e.push(`### Round ${a} (assistant) \u2014 question`),e.push(`"${c.question}"`);else{if(e.push(`### Round ${a} (assistant)`),c.responseText&&e.push(`Response: ${c.responseText}`),c.resolutions&&c.resolutions.length>0)for(let m of c.resolutions){let P=m.finalScope??m.inferredScope,g=P?` [${P.breadth} ${P.target}]`:"";e.push(`- ${m.annotationId}: ${m.status}${g} \u2014 ${m.summary}`),m.filesModified&&m.filesModified.length>0&&e.push(` Files: ${m.filesModified.join(", ")}`)}c.toolsUsed&&c.toolsUsed.length>0&&e.push(`Tools used: ${c.toolsUsed.join(", ")}`)}e.push(""),e.push("The current round is shown in full below.")}if(n?.designModel){e.push(""),e.push("## Established Design Policies"),e.push("This project has an established design model (stored in .popmelt/model.json), extracted from the developer's previous design decisions. When making changes, follow these patterns unless the developer explicitly overrides them. When asked about design tokens, component patterns, or design decisions, reference this model as the authoritative source.");let a=n.designModel.rules;if(Array.isArray(a)&&a.length>0)if(e.push(""),e.push("Rules:"),a.length>0&&typeof a[0]=="object"&&a[0]!==null&&"scope"in a[0]){let g=new Map;for(let B of a)if(typeof B=="object"&&B!==null&&"text"in B){let z=B,O=z.scope||"general";g.has(O)||g.set(O,[]),g.get(O).push(z.text)}for(let[B,z]of g){e.push(`**${B.charAt(0).toUpperCase()+B.slice(1)}**`);for(let O of z)e.push(`- ${O}`)}}else for(let g of a)typeof g=="string"&&e.push(`- ${g}`);let c=n.designModel.tokens;c&&typeof c=="object"&&(e.push(""),e.push("Design tokens:"),e.push("```json"),e.push(JSON.stringify(c,null,2)),e.push("```"));let m=n.designModel.components;m&&typeof m=="object"&&(e.push(""),e.push("Component patterns:"),e.push("```json"),e.push(JSON.stringify(m,null,2)),e.push("```")),e.push(""),e.push("### Novel Pattern Detection"),e.push("When you make a design decision that has no matching policy in the model above (e.g., styling a component type not yet in the model, choosing a color with no token, picking spacing with no rule), flag it:"),e.push("<novel>"),e.push('[{"category":"component","element":"button","decision":"Used 8px border-radius, 12px 24px padding","reason":"No button pattern in design model"}]'),e.push("</novel>"),e.push('- `category`: "token" (color, spacing, typography), "component" (UI component pattern), or "element" (specific element style)'),e.push("- `element`: What you are styling or creating"),e.push("- `decision`: What you decided to do (specific values)"),e.push("- `reason`: Why this is novel (what is missing from the model)"),e.push("Still do the work \u2014 just flag it so the developer can review and set policy.")}if(n?.designModel||(e.push(""),e.push("## Design Context"),e.push("This project uses Popmelt for design governance. Design decisions are stored in .popmelt/decisions/ (JSON files). A materialized design model may exist at .popmelt/model.json. When the developer asks about design tokens, patterns, or past decisions, check these files first before searching source code.")),i){let a=n?.screenshotPaths??{},c=new Map;for(let m of t.annotations){let P=m.pathname||new URL(t.url).pathname;c.has(P)||c.set(P,[]),c.get(P).push(m)}e.push(""),e.push("The developer annotated multiple pages. Each page section includes its own screenshot where available.");for(let[m,P]of c){e.push(""),e.push(`## Page: ${m}`);let g=a[m];g&&n?.provider!=="codex"?e.push(`IMPORTANT: Use the Read tool to view the screenshot at: ${g}`):!g&&n?.provider!=="codex"&&(Object.keys(a).length===0?(e.push(`IMPORTANT: Use the Read tool to view the screenshot at: ${s}`),e.push("(This screenshot shows the page the developer was on when submitting \u2014 not this page)")):e.push("(No screenshot available for this page \u2014 work from the annotation text and selectors)"));let B={...t,annotations:P,styleModifications:t.styleModifications},z=Ce(B,n?.imagePaths);z&&(e.push(""),e.push(z))}}else{let a=Ce(t,n?.imagePaths);a&&(e.push(""),e.push(a))}return e.push(""),e.push("Follow the developer's instructions. If they ask for changes, apply them to the source files \u2014 the dev server has HMR so changes appear immediately. If they ask a question or request analysis, respond in text without modifying code."),e.push(""),e.push("IMPORTANT: If any elements you modify have a `data-pm` attribute, preserve it in the source. This attribute tracks annotation positions."),e.push(""),e.push("## Resolution"),e.push("After completing all work, output a resolution block listing what you did for each annotation:"),e.push("<resolution>"),e.push('[{"annotationId":"<id>","status":"resolved","summary":"<what you did>","filesModified":["<file>"],"declaredScope":{"breadth":"...","target":"..."},"inferredScope":{"breadth":"...","target":"..."}}]'),e.push("</resolution>"),e.push(`Use status "resolved" when the change is complete, or "needs_review" if you're unsure about the result.`),e.push(""),e.push("### Scope classification"),e.push("Each resolution MUST include scope fields:"),e.push("- `declaredScope`: What scope the user's instruction text implies. null if no signal."),e.push("- `inferredScope`: What scope the change actually has, based on what you modified."),e.push("Scope has two dimensions:"),e.push('- `breadth`: "instance" (just this occurrence) or "pattern" (all similar occurrences)'),e.push('- `target`: "element" (a specific DOM element), "component" (a React/UI component), or "token" (a design token \u2014 color, spacing, typography)'),e.push('Note: "instance" + "token" is invalid \u2014 tokens are inherently patterns.'),e.push("If you cannot confidently determine scope, set it to null."),e.push(""),e.push("## Questions"),e.push("If the annotation text is unclear, ambiguous, gibberish, or you are unsure what the developer wants, output a question:"),e.push('<question>What do you mean by "..."?</question>'),e.push("Do NOT guess what unclear instructions mean \u2014 ask instead."),e.push("You may output BOTH a <resolution> for clear annotations AND a <question> for unclear ones in the same response."),e.join(`
329
- `)}function ln(s){return s.match(/<question>\s*([\s\S]*?)\s*<\/question>/)?.[1]??null}function dn(s,t,n,e){let o=[];o.push("You are continuing work on a UI based on the developer's reply to your question."),o.push(""),n!=="codex"&&o.push(`IMPORTANT: First, use the Read tool to view the screenshot at: ${s}`);let i=t.find(a=>a.role==="human"&&a.feedbackContext);if(i?.feedbackContext&&(o.push(""),o.push(i.feedbackContext)),t.length>0){o.push(""),o.push("## Conversation History");let a=0;for(let c of t)c.role==="human"?(a++,c.replyToQuestion?(o.push(`### Round ${a} (human) \u2014 reply`),o.push(`"${c.replyToQuestion}"`)):(o.push(`### Round ${a} (human)`),c.feedbackSummary&&o.push(`Annotations: ${c.feedbackSummary}`))):c.question?(o.push(`### Round ${a} (assistant) \u2014 question`),o.push(`"${c.question}"`)):(o.push(`### Round ${a} (assistant)`),c.responseText&&o.push(`Response: ${c.responseText}`))}if(o.push(""),o.push("The developer answered your question. Continue working based on their reply."),o.push("Follow their instructions \u2014 apply code changes only if requested. The dev server has HMR so changes appear immediately."),o.push(""),o.push("IMPORTANT: If any elements you modify have a `data-pm` attribute, preserve it in the source. This attribute tracks annotation positions."),e&&e.length>0){o.push(""),o.push("## Attached Images"),o.push("The developer attached reference images with their reply:");for(let a of e)o.push(`Attached image: use the Read tool to view the image at: ${a}`)}return o.push(""),o.push("## Resolution"),o.push("After completing all work, output a resolution block listing what you did for each annotation:"),o.push("<resolution>"),o.push('[{"annotationId":"<id>","status":"resolved","summary":"<what you did>","filesModified":["<file>"],"declaredScope":{"breadth":"...","target":"..."},"inferredScope":{"breadth":"...","target":"..."}}]'),o.push("</resolution>"),o.push(`Use status "resolved" when the change is complete, or "needs_review" if you're unsure about the result.`),o.push(""),o.push("### Scope classification"),o.push("Each resolution MUST include scope fields:"),o.push("- `declaredScope`: What scope the user's instruction text implies. null if no signal."),o.push("- `inferredScope`: What scope the change actually has, based on what you modified."),o.push("Scope has two dimensions:"),o.push('- `breadth`: "instance" (just this occurrence) or "pattern" (all similar occurrences)'),o.push('- `target`: "element" (a specific DOM element), "component" (a React/UI component), or "token" (a design token \u2014 color, spacing, typography)'),o.push('Note: "instance" + "token" is invalid \u2014 tokens are inherently patterns.'),o.push("If you cannot confidently determine scope, set it to null."),o.push('If the developer\'s reply corrects a prior scope classification (e.g., "this should apply everywhere" or "only fix this one"), set `finalScope` on your resolution to reflect their correction and apply the change at the corrected scope.'),o.push(""),o.push("## Questions"),o.push("If you still need clarification, output:"),o.push("<question>Your question here</question>"),o.push("You may output BOTH a <resolution> and a <question> in the same response."),o.join(`
330
- `)}function Vs(s){if(typeof s!="object"||s===null)return!1;let t=s;return(t.breadth==="instance"||t.breadth==="pattern")&&(t.target==="element"||t.target==="component"||t.target==="token")}function Zs(s){if(typeof s!="object"||s===null||typeof s.annotationId!="string"||s.status!=="resolved"&&s.status!=="needs_review"||typeof s.summary!="string")return!1;let t=s;for(let n of["declaredScope","inferredScope","finalScope"])if(t[n]!==void 0&&t[n]!==null&&!Vs(t[n]))return!1;return!0}function un(s){let t=s.match(/<resolution>\s*([\s\S]*?)\s*<\/resolution>/);if(!t||!t[1])return[];try{let n=JSON.parse(t[1]);return Array.isArray(n)?n.filter(Zs):[]}catch{return[]}}function pn(s){let t=s.match(/<novel>\s*([\s\S]*?)\s*<\/novel>/);if(!t?.[1])return[];try{let n=JSON.parse(t[1]);return Array.isArray(n)?n.filter(e=>{if(typeof e!="object"||e===null)return!1;let o=e;return(o.category==="token"||o.category==="component"||o.category==="element")&&typeof o.element=="string"&&typeof o.decision=="string"&&typeof o.reason=="string"}):[]}catch{return[]}}import{execFile as eo}from"child_process";import{readdir as to,readFile as no,stat as so}from"fs/promises";import{homedir as $e}from"os";import{join as ge}from"path";var Ke=[{id:"claude-opus-4-6",label:"Opus 4.6",source:"static"},{id:"claude-sonnet-4-6",label:"Sonn 4.6",source:"static"}],_e=[{id:"gpt-5.4",label:"GPT 5.4",source:"static"},{id:"gpt-5.3-codex",label:"Codex 5.3",source:"static"},{id:"gpt-5.3-codex-spark",label:"Spark 5.3",source:"static"}],Oe={id:"",label:"Default",source:"cli"},oo=/^\s*-\s+"([^"]+)"/;function ro(s){let t=[],n=!1;for(let e of s.split(`
331
- `)){if(/^\s*`model`:/.test(e)){n=!0;continue}if(n&&/^\s*`[^`]+`:/.test(e))break;if(!n)continue;let o=e.match(oo);if(!o?.[1])continue;let i=o[1];t.push({id:i,label:mt(i),source:"cli"})}return Qe([Oe,...t])}function io(s){let t=JSON.parse(s);return(Array.isArray(t.models)?t.models:[]).filter(e=>e.visibility===void 0||e.visibility==="list").sort((e,o)=>{let i=typeof e.priority=="number"?e.priority:Number.MAX_SAFE_INTEGER,a=typeof o.priority=="number"?o.priority:Number.MAX_SAFE_INTEGER;return i-a}).flatMap(e=>{let o=typeof e.slug=="string"?e.slug:typeof e.id=="string"?e.id:"";if(!o)return[];let i=typeof e.display_name=="string"&&e.display_name.trim()?e.display_name.trim():mt(o);return[{id:o,label:i,source:"cli"}]})}function ao(s){let t=s.match(/\bclaude-[a-z0-9]+(?:[-.][a-z0-9]+)*\b/gi)??[];return Qe(t.map(n=>({id:n,label:mt(n),source:"cli"})))}function mt(s){if(!s)return"Default";let t={"gpt-5.5":"GPT 5.5","gpt-5.4":"GPT 5.4","gpt-5.4-mini":"Mini 5.4","gpt-5.3-codex":"Codex 5.3","gpt-5.3-codex-spark":"Spark 5.3","gpt-5.2-codex":"Codex 5.2","gpt-5.2":"GPT 5.2","gpt-5.1":"GPT 5.1","gpt-5-mini":"Mini 5","gpt-4.1":"GPT 4.1","claude-opus-4.7":"Opus 4.7","claude-opus-4.6":"Opus 4.6","claude-opus-4.6-fast":"Opus 4.6 Fast","claude-opus-4.5":"Opus 4.5","claude-sonnet-4.6":"Sonn 4.6","claude-sonnet-4.5":"Sonn 4.5","claude-sonnet-4":"Sonn 4","claude-haiku-4.5":"Haiku 4.5","claude-fable":"Fable"},n=s.replace(/-\d{8,}$/,"");if(t[s])return t[s];if(t[n])return t[n];let e=n.match(/^claude-(opus|sonnet|haiku|fable)(?:-(\d+)[-.](\d+))?(?:-(fast))?$/);if(e){let i=e[1]==="sonnet"?"Sonn":po(e[1]),a=e[2]&&e[3]?` ${e[2]}.${e[3]}`:"";return`${i}${a}${e[4]?" Fast":""}`}let o=n.match(/^gpt-(\d+)(?:[.-](\d+))?(?:-(mini|codex|spark))?$/);if(o){let i=o[2]?`${o[1]}.${o[2]}`:o[1];return o[3]==="mini"?`Mini ${i}`:o[3]==="codex"?`Codex ${i}`:o[3]==="spark"?`Spark ${i}`:`GPT ${i}`}return n}async function fn(s="copilot"){let t=await Ye(s,["help","config"]),n=ro(t);return n.length>1?n:[Oe]}async function hn(s="codex"){let t=await Ye(s,["debug","models"]).catch(async()=>{let e=process.env.CODEX_HOME||ge($e(),".codex");return gn(ge(e,"models_cache.json"))}),n=io(t);return n.length>0?Qe([...n,..._e]):_e}async function mn(s="claude"){let t=ge($e(),".claude"),n=[ge($e(),".claude","stats-cache.json"),ge($e(),".claude","settings.json"),ge($e(),".claude.json"),ge(t,"cache","changelog.md")],[e,o,i]=await Promise.all([Ye(s,["--help"]).catch(()=>""),Ye(s,["--version"]).catch(()=>""),co(t).catch(()=>[])]),a=uo([...n,...i]),c=await Promise.all(a.map(P=>gn(P).catch(()=>""))),m=ao([e,o,...c].join(`
332
- `));return Qe([...Ke,...m])}function Ye(s,t,n=5e3){return new Promise((e,o)=>{eo(s,t,{encoding:"utf-8",timeout:n},(i,a)=>{if(i){o(i);return}e(a)})})}async function gn(s){return no(s,"utf-8")}async function co(s){let t=[ge(s,"projects"),ge(s,"telemetry")],n=[];for(let e of t)await vn(e,n,4);return n.sort((e,o)=>o.mtimeMs-e.mtimeMs).slice(0,80).map(e=>e.path)}async function vn(s,t,n){if(n<0)return;let e;try{e=await to(s,{withFileTypes:!0})}catch{return}await Promise.all(e.map(async o=>{let i=ge(s,o.name);if(o.isDirectory()){await vn(i,t,n-1);return}if(!(!o.isFile()||!lo(o.name)))try{let a=await so(i);if(a.size>2e6)return;t.push({path:i,mtimeMs:a.mtimeMs})}catch{}}))}function lo(s){return s.endsWith(".json")||s.endsWith(".jsonl")||s.endsWith(".md")||s.endsWith(".txt")}function Qe(s){let t=new Set,n=[];for(let e of s)t.has(e.id)||(t.add(e.id),n.push(e));return n}function uo(s){return[...new Set(s)]}function po(s){return s.charAt(0).toUpperCase()+s.slice(1)}var Ve=class{constructor(t=5){this.queue=[];this.activeJobs=new Map;this.activeProcesses=new Map;this.listeners=new Set;this.processor=null;this.eventBuffers=new Map;this.accumulators=new Map;this.cleanupTimers=new Map;this.maxConcurrency=t}setProcessor(t){this.processor=t}get active(){let t=this.activeJobs.values().next();return t.done?null:t.value}get allActive(){return Array.from(this.activeJobs.values())}get activeCount(){return this.activeJobs.size}get depth(){return this.queue.length}get isRunning(){return this.activeJobs.size>0}setActiveProcess(t,n){n?this.activeProcesses.set(t,n):this.activeProcesses.delete(t)}enqueue(t){return this.queue.push(t),this.processNext(),this.queue.length+this.activeJobs.size}addListener(t){return this.listeners.add(t),()=>this.listeners.delete(t)}broadcast(t,n,e){let o=this.bufferEvent(n,t);for(let i of this.listeners)i(o,n,e)}bufferEvent(t,n){let e=this.eventBuffers.get(t);e||(e={events:[],nextSeq:0},this.eventBuffers.set(t,e));let o={...n,seq:e.nextSeq++};return e.events.push(o),e.events.length>1e4&&e.events.splice(0,e.events.length-1e4),o}getBufferedEvents(t,n=-1){let e=this.eventBuffers.get(t),o=this.accumulators.get(t)??{response:"",thinking:""},i=this.activeJobs.has(t);if(!e)return null;let a=n<0?e.events:e.events.filter(c=>c.seq>n);return{jobId:t,events:a,currentSeq:e.nextSeq-1,accumulated:{...o},jobActive:i}}accumulateText(t,n,e){let o=this.accumulators.get(t);o||(o={response:"",thinking:""},this.accumulators.set(t,o)),o[n]+=e}getAccumulated(t){return this.accumulators.get(t)??null}scheduleBufferCleanup(t){let n=this.cleanupTimers.get(t);n&&clearTimeout(n);let e=setTimeout(()=>{this.eventBuffers.delete(t),this.accumulators.delete(t),this.cleanupTimers.delete(t)},6e4);this.cleanupTimers.set(t,e)}cancelJob(t){let n=this.activeProcesses.get(t),e=this.activeJobs.get(t);return!n||!e?!1:(n.kill("SIGTERM"),this.activeProcesses.delete(t),this.activeJobs.delete(t),e.status="error",e.error="Cancelled by user",this.broadcast({type:"error",jobId:e.id,threadId:e.threadId,message:"Cancelled by user",cancelled:!0,provider:e.provider,model:e.model},e.id,e.sourceId),this.processNext(),!0)}cancelActive(){if(this.activeJobs.size===0)return!1;let t=Array.from(this.activeJobs.keys());for(let n of t)this.cancelJob(n);return!0}destroy(){for(let t of this.activeProcesses.values())t.kill("SIGTERM");this.activeProcesses.clear(),this.activeJobs.clear(),this.queue=[],this.listeners.clear(),this.eventBuffers.clear(),this.accumulators.clear();for(let t of this.cleanupTimers.values())clearTimeout(t);this.cleanupTimers.clear()}async destroyAsync(t=1e4){let n=Array.from(this.activeProcesses.values());if(this.queue=[],this.listeners.clear(),n.length===0){this.activeProcesses.clear(),this.activeJobs.clear();return}for(let e of n)try{e.kill("SIGTERM")}catch{}await Promise.all(n.map(e=>new Promise(o=>{let i=!1,a=()=>{i||(i=!0,o())};e.on("exit",a),e.on("error",a),setTimeout(()=>{if(!i){try{e.kill("SIGKILL")}catch{}setTimeout(a,500)}},t)}))),this.activeProcesses.clear(),this.activeJobs.clear(),this.eventBuffers.clear(),this.accumulators.clear();for(let e of this.cleanupTimers.values())clearTimeout(e);this.cleanupTimers.clear()}processNext(){for(;this.activeJobs.size<this.maxConcurrency&&this.queue.length>0&&this.processor;){let t=this.queue.shift();this.activeJobs.set(t.id,t),t.status="running",this.broadcast({type:"job_started",jobId:t.id,position:0,threadId:t.threadId},t.id,t.sourceId),this.processor(t).catch(n=>{t.status="error",t.error=n instanceof Error?n.message:String(n),this.broadcast({type:"error",jobId:t.id,threadId:t.threadId,message:t.error,provider:t.provider,model:t.model},t.id,t.sourceId)}).finally(()=>{this.activeJobs.delete(t.id),this.activeProcesses.delete(t.id),this.scheduleBufferCleanup(t.id),this.processNext(),this.activeJobs.size===0&&this.queue.length===0&&this.broadcast({type:"queue_drained"},t.id)})}}};import{mkdir as fo,readFile as ho,writeFile as mo}from"fs/promises";import{dirname as go,join as vo}from"path";var yo={version:1,threads:{}},Ze=class{constructor(t){this.cache=null;this.writeChain=Promise.resolve();this.filePath=vo(t,".popmelt","threads.json")}async load(){if(this.cache)return this.cache;try{let t=await ho(this.filePath,"utf-8"),n=JSON.parse(t);if(n&&n.version===1&&n.threads)return this.cache=n,this.cache}catch{}return this.cache={...yo,threads:{}},this.cache}async getThread(t){return(await this.load()).threads[t]??null}async findContinuationThread(t){if(t.length===0)return null;let n=await this.load(),e=new Set(t);for(let o of Object.values(n.threads))if(o.elementIdentifiers.some(a=>e.has(a)))return o;return null}async createThread(t,n){let e=await this.load(),o={id:t,createdAt:Date.now(),updatedAt:Date.now(),elementIdentifiers:n,messages:[]};return e.threads[t]=o,await this.persist(),o}async appendMessage(t,n){let o=(await this.load()).threads[t];o&&(o.messages.push(n),o.updatedAt=Date.now(),await this.persist())}async addElementIdentifiers(t,n){let o=(await this.load()).threads[t];if(!o)return;let i=new Set(o.elementIdentifiers);for(let a of n)i.has(a)||o.elementIdentifiers.push(a);o.updatedAt=Date.now(),await this.persist()}async listRecent(t=5){let n=await this.load(),e=Object.values(n.threads);return e.sort((o,i)=>i.updatedAt-o.updatedAt),e.slice(0,t).map(o=>{let a=o.messages.find(c=>c.role==="human")?.feedbackSummary||"[thread]";return{id:o.id,createdAt:o.createdAt,updatedAt:o.updatedAt,preview:a,messageCount:o.messages.length,elementIdentifiers:o.elementIdentifiers}})}async getThreadHistory(t,n=6){let e=await this.getThread(t);return!e||e.messages.length===0?[]:e.messages.length<=n?e.messages:[e.messages[0],...e.messages.slice(-(n-1))]}async persist(){this.writeChain=this.writeChain.then(async()=>{if(this.cache)try{await fo(go(this.filePath),{recursive:!0}),await mo(this.filePath,JSON.stringify(this.cache,null,2))}catch(t){console.error("[ThreadStore] Failed to persist:",t)}}),await this.writeChain}};var yn="0.7.1";var Io=1111,To=["Read","Edit","Write","Glob","Grep","Bash","WebFetch","WebSearch","Bash(curl:*)"],Mo=1800*1e3,Co=3600*1e3;function En(s){if(!s)return!1;try{let t=new URL(s);return t.hostname==="localhost"||t.hostname.endsWith(".localhost")||t.hostname==="127.0.0.1"||t.hostname==="::1"||t.hostname==="[::1]"}catch{return!1}}function $o(s){let t=[];for(let n of s)if(n.type==="delta"){let e=n.text;if(!e)continue;let o=t[t.length-1];o&&o.kind==="text"?o.text+=e:t.push({kind:"text",text:e})}else if(n.type==="tool_use"){let e=n.tool||"",o=n.file??void 0,i=n.content??void 0,a=o?o.split("/").pop()??o:void 0,c;switch(e){case"Read":c=a?`Reading ${a}`:"Reading file";break;case"Edit":c=a?`Editing ${a}`:"Editing file";break;case"Write":c=a?`Writing ${a}`:"Writing file";break;case"Bash":c=i?i.split(`
333
- `)[0].trim().slice(0,60):"Running command";break;case"Glob":c="Searching files";break;case"Grep":c="Searching code";break;case"WebFetch":c="Fetching page";break;case"WebSearch":c="Searching web";break;default:c=e?`Using ${e}`:"tool";break}let m=o??i??void 0,P=t[t.length-1];P&&P.kind==="tool_group"&&P.tool===e?P.items.push({label:c,detail:m}):t.push({kind:"tool_group",tool:e,items:[{label:c,detail:m}]})}return t}function _o(s,t){let n=s.headers.origin;En(n)&&(t.setHeader("Access-Control-Allow-Origin",n),t.setHeader("Access-Control-Allow-Methods","GET, POST, PATCH, DELETE, OPTIONS"),t.setHeader("Access-Control-Allow-Headers","Content-Type"))}function v(s,t,n){s.writeHead(t,{"Content-Type":"application/json"}),s.end(JSON.stringify(n))}function Oo(s,t){if(!s)return t;let n=s.match(/^#?([0-9a-f]{2})([0-9a-f]{2})([0-9a-f]{2})$/i);if(!n)return t;let[,e,o,i]=n;return`\x1B[38;2;${parseInt(e,16)};${parseInt(o,16)};${parseInt(i,16)}m${t}\x1B[0m`}function gt(s,t){try{s.res.write(`event: ${t.type}
328
+ `)}function ln(s,t,n){let e=[],i=new Set(t.annotations.map(a=>a.pathname).filter(Boolean)).size>1;if(e.push("You are reviewing a UI screenshot with developer annotations."),e.push(""),!i&&n?.provider!=="codex"&&(e.push(`IMPORTANT: First, use the Read tool to view the screenshot at: ${s}`),e.push("")),e.push(`The developer annotated their running app at ${t.url} (${t.viewport.width}x${t.viewport.height}).`),n?.threadHistory&&n.threadHistory.length>0){e.push(""),e.push("## Previous Conversation");let a=0;for(let c of n.threadHistory)if(c.role==="human")a++,c.replyToQuestion?(e.push(`### Round ${a} (human) \u2014 reply`),e.push(`"${c.replyToQuestion}"`)):(e.push(`### Round ${a} (human)`),c.feedbackSummary&&e.push(`Annotations: ${c.feedbackSummary}`),c.annotationIds&&c.annotationIds.length>0&&e.push(`Annotation IDs: ${c.annotationIds.join(", ")}`));else if(c.question)e.push(`### Round ${a} (assistant) \u2014 question`),e.push(`"${c.question}"`);else{if(e.push(`### Round ${a} (assistant)`),c.responseText&&e.push(`Response: ${c.responseText}`),c.resolutions&&c.resolutions.length>0)for(let m of c.resolutions){let P=m.finalScope??m.inferredScope,v=P?` [${P.breadth} ${P.target}]`:"";e.push(`- ${m.annotationId}: ${m.status}${v} \u2014 ${m.summary}`),m.filesModified&&m.filesModified.length>0&&e.push(` Files: ${m.filesModified.join(", ")}`)}c.toolsUsed&&c.toolsUsed.length>0&&e.push(`Tools used: ${c.toolsUsed.join(", ")}`)}e.push(""),e.push("The current round is shown in full below.")}if(n?.designModel){e.push(""),e.push("## Established Design Policies"),e.push("This project has an established design model (stored in .popmelt/model.json), extracted from the developer's previous design decisions. When making changes, follow these patterns unless the developer explicitly overrides them. When asked about design tokens, component patterns, or design decisions, reference this model as the authoritative source.");let a=n.designModel.rules;if(Array.isArray(a)&&a.length>0)if(e.push(""),e.push("Rules:"),a.length>0&&typeof a[0]=="object"&&a[0]!==null&&"scope"in a[0]){let v=new Map;for(let B of a)if(typeof B=="object"&&B!==null&&"text"in B){let z=B,A=z.scope||"general";v.has(A)||v.set(A,[]),v.get(A).push(z.text)}for(let[B,z]of v){e.push(`**${B.charAt(0).toUpperCase()+B.slice(1)}**`);for(let A of z)e.push(`- ${A}`)}}else for(let v of a)typeof v=="string"&&e.push(`- ${v}`);let c=n.designModel.tokens;c&&typeof c=="object"&&(e.push(""),e.push("Design tokens:"),e.push("```json"),e.push(JSON.stringify(c,null,2)),e.push("```"));let m=n.designModel.components;m&&typeof m=="object"&&(e.push(""),e.push("Component patterns:"),e.push("```json"),e.push(JSON.stringify(m,null,2)),e.push("```")),e.push(""),e.push("### Novel Pattern Detection"),e.push("When you make a design decision that has no matching policy in the model above (e.g., styling a component type not yet in the model, choosing a color with no token, picking spacing with no rule), flag it:"),e.push("<novel>"),e.push('[{"category":"component","element":"button","decision":"Used 8px border-radius, 12px 24px padding","reason":"No button pattern in design model"}]'),e.push("</novel>"),e.push('- `category`: "token" (color, spacing, typography), "component" (UI component pattern), or "element" (specific element style)'),e.push("- `element`: What you are styling or creating"),e.push("- `decision`: What you decided to do (specific values)"),e.push("- `reason`: Why this is novel (what is missing from the model)"),e.push("Still do the work \u2014 just flag it so the developer can review and set policy.")}if(n?.designModel||(e.push(""),e.push("## Design Context"),e.push("This project uses Popmelt for design governance. Design decisions are stored in .popmelt/decisions/ (JSON files). A materialized design model may exist at .popmelt/model.json. When the developer asks about design tokens, patterns, or past decisions, check these files first before searching source code.")),i){let a=n?.screenshotPaths??{},c=new Map;for(let m of t.annotations){let P=m.pathname||new URL(t.url).pathname;c.has(P)||c.set(P,[]),c.get(P).push(m)}e.push(""),e.push("The developer annotated multiple pages. Each page section includes its own screenshot where available.");for(let[m,P]of c){e.push(""),e.push(`## Page: ${m}`);let v=a[m];v&&n?.provider!=="codex"?e.push(`IMPORTANT: Use the Read tool to view the screenshot at: ${v}`):!v&&n?.provider!=="codex"&&(Object.keys(a).length===0?(e.push(`IMPORTANT: Use the Read tool to view the screenshot at: ${s}`),e.push("(This screenshot shows the page the developer was on when submitting \u2014 not this page)")):e.push("(No screenshot available for this page \u2014 work from the annotation text and selectors)"));let B={...t,annotations:P,styleModifications:t.styleModifications},z=Ce(B,n?.imagePaths);z&&(e.push(""),e.push(z))}}else{let a=Ce(t,n?.imagePaths);a&&(e.push(""),e.push(a))}return e.push(""),e.push("Follow the developer's instructions. If they ask for changes, apply them to the source files \u2014 the dev server has HMR so changes appear immediately. If they ask a question or request analysis, respond in text without modifying code."),e.push(""),e.push("IMPORTANT: If any elements you modify have a `data-pm` attribute, preserve it in the source. This attribute tracks annotation positions."),e.push(""),e.push("## Resolution"),e.push("After completing all work, output a resolution block listing what you did for each annotation:"),e.push("<resolution>"),e.push('[{"annotationId":"<id>","status":"resolved","summary":"<what you did>","filesModified":["<file>"],"declaredScope":{"breadth":"...","target":"..."},"inferredScope":{"breadth":"...","target":"..."}}]'),e.push("</resolution>"),e.push(`Use status "resolved" when the change is complete, or "needs_review" if you're unsure about the result.`),e.push(""),e.push("### Scope classification"),e.push("Each resolution MUST include scope fields:"),e.push("- `declaredScope`: What scope the user's instruction text implies. null if no signal."),e.push("- `inferredScope`: What scope the change actually has, based on what you modified."),e.push("Scope has two dimensions:"),e.push('- `breadth`: "instance" (just this occurrence) or "pattern" (all similar occurrences)'),e.push('- `target`: "element" (a specific DOM element), "component" (a React/UI component), or "token" (a design token \u2014 color, spacing, typography)'),e.push('Note: "instance" + "token" is invalid \u2014 tokens are inherently patterns.'),e.push("If you cannot confidently determine scope, set it to null."),e.push(""),e.push("## Questions"),e.push("If the annotation text is unclear, ambiguous, gibberish, or you are unsure what the developer wants, output a question:"),e.push('<question>What do you mean by "..."?</question>'),e.push("Do NOT guess what unclear instructions mean \u2014 ask instead."),e.push("You may output BOTH a <resolution> for clear annotations AND a <question> for unclear ones in the same response."),e.join(`
329
+ `)}function dn(s){return s.match(/<question>\s*([\s\S]*?)\s*<\/question>/)?.[1]??null}function un(s,t,n,e){let o=[];o.push("You are continuing work on a UI based on the developer's reply to your question."),o.push(""),n!=="codex"&&o.push(`IMPORTANT: First, use the Read tool to view the screenshot at: ${s}`);let i=t.find(a=>a.role==="human"&&a.feedbackContext);if(i?.feedbackContext&&(o.push(""),o.push(i.feedbackContext)),t.length>0){o.push(""),o.push("## Conversation History");let a=0;for(let c of t)c.role==="human"?(a++,c.replyToQuestion?(o.push(`### Round ${a} (human) \u2014 reply`),o.push(`"${c.replyToQuestion}"`)):(o.push(`### Round ${a} (human)`),c.feedbackSummary&&o.push(`Annotations: ${c.feedbackSummary}`))):c.question?(o.push(`### Round ${a} (assistant) \u2014 question`),o.push(`"${c.question}"`)):(o.push(`### Round ${a} (assistant)`),c.responseText&&o.push(`Response: ${c.responseText}`))}if(o.push(""),o.push("The developer answered your question. Continue working based on their reply."),o.push("Follow their instructions \u2014 apply code changes only if requested. The dev server has HMR so changes appear immediately."),o.push(""),o.push("IMPORTANT: If any elements you modify have a `data-pm` attribute, preserve it in the source. This attribute tracks annotation positions."),e&&e.length>0){o.push(""),o.push("## Attached Images"),o.push("The developer attached reference images with their reply:");for(let a of e)o.push(`Attached image: use the Read tool to view the image at: ${a}`)}return o.push(""),o.push("## Resolution"),o.push("After completing all work, output a resolution block listing what you did for each annotation:"),o.push("<resolution>"),o.push('[{"annotationId":"<id>","status":"resolved","summary":"<what you did>","filesModified":["<file>"],"declaredScope":{"breadth":"...","target":"..."},"inferredScope":{"breadth":"...","target":"..."}}]'),o.push("</resolution>"),o.push(`Use status "resolved" when the change is complete, or "needs_review" if you're unsure about the result.`),o.push(""),o.push("### Scope classification"),o.push("Each resolution MUST include scope fields:"),o.push("- `declaredScope`: What scope the user's instruction text implies. null if no signal."),o.push("- `inferredScope`: What scope the change actually has, based on what you modified."),o.push("Scope has two dimensions:"),o.push('- `breadth`: "instance" (just this occurrence) or "pattern" (all similar occurrences)'),o.push('- `target`: "element" (a specific DOM element), "component" (a React/UI component), or "token" (a design token \u2014 color, spacing, typography)'),o.push('Note: "instance" + "token" is invalid \u2014 tokens are inherently patterns.'),o.push("If you cannot confidently determine scope, set it to null."),o.push('If the developer\'s reply corrects a prior scope classification (e.g., "this should apply everywhere" or "only fix this one"), set `finalScope` on your resolution to reflect their correction and apply the change at the corrected scope.'),o.push(""),o.push("## Questions"),o.push("If you still need clarification, output:"),o.push("<question>Your question here</question>"),o.push("You may output BOTH a <resolution> and a <question> in the same response."),o.join(`
330
+ `)}function Vs(s){if(typeof s!="object"||s===null)return!1;let t=s;return(t.breadth==="instance"||t.breadth==="pattern")&&(t.target==="element"||t.target==="component"||t.target==="token")}function Zs(s){if(typeof s!="object"||s===null||typeof s.annotationId!="string"||s.status!=="resolved"&&s.status!=="needs_review"||typeof s.summary!="string")return!1;let t=s;for(let n of["declaredScope","inferredScope","finalScope"])if(t[n]!==void 0&&t[n]!==null&&!Vs(t[n]))return!1;return!0}function pn(s){let t=s.match(/<resolution>\s*([\s\S]*?)\s*<\/resolution>/);if(!t||!t[1])return[];try{let n=JSON.parse(t[1]);return Array.isArray(n)?n.filter(Zs):[]}catch{return[]}}function fn(s){let t=s.match(/<novel>\s*([\s\S]*?)\s*<\/novel>/);if(!t?.[1])return[];try{let n=JSON.parse(t[1]);return Array.isArray(n)?n.filter(e=>{if(typeof e!="object"||e===null)return!1;let o=e;return(o.category==="token"||o.category==="component"||o.category==="element")&&typeof o.element=="string"&&typeof o.decision=="string"&&typeof o.reason=="string"}):[]}catch{return[]}}import{execFile as eo}from"child_process";import{readdir as to,readFile as no,stat as so}from"fs/promises";import{homedir as $e}from"os";import{join as ge}from"path";var Ke=[{id:"claude-opus-4-6",label:"Opus 4.6",source:"static"},{id:"claude-sonnet-4-6",label:"Sonn 4.6",source:"static"}],_e=[{id:"gpt-5.4",label:"GPT 5.4",source:"static"},{id:"gpt-5.3-codex",label:"Codex 5.3",source:"static"},{id:"gpt-5.3-codex-spark",label:"Spark 5.3",source:"static"}],Oe={id:"",label:"Default",source:"cli"},oo=/^\s*-\s+"([^"]+)"/;function ro(s){let t=[],n=!1;for(let e of s.split(`
331
+ `)){if(/^\s*`model`:/.test(e)){n=!0;continue}if(n&&/^\s*`[^`]+`:/.test(e))break;if(!n)continue;let o=e.match(oo);if(!o?.[1])continue;let i=o[1];t.push({id:i,label:mt(i),source:"cli"})}return Qe([Oe,...t])}function io(s){let t=JSON.parse(s);return(Array.isArray(t.models)?t.models:[]).filter(e=>e.visibility===void 0||e.visibility==="list").sort((e,o)=>{let i=typeof e.priority=="number"?e.priority:Number.MAX_SAFE_INTEGER,a=typeof o.priority=="number"?o.priority:Number.MAX_SAFE_INTEGER;return i-a}).flatMap(e=>{let o=typeof e.slug=="string"?e.slug:typeof e.id=="string"?e.id:"";if(!o)return[];let i=typeof e.display_name=="string"&&e.display_name.trim()?e.display_name.trim():mt(o);return[{id:o,label:i,source:"cli"}]})}function ao(s){let t=s.match(/\bclaude-[a-z0-9]+(?:[-.][a-z0-9]+)*\b/gi)??[];return Qe(t.map(n=>({id:n,label:mt(n),source:"cli"})))}function mt(s){if(!s)return"Default";let t={"gpt-5.5":"GPT 5.5","gpt-5.4":"GPT 5.4","gpt-5.4-mini":"Mini 5.4","gpt-5.3-codex":"Codex 5.3","gpt-5.3-codex-spark":"Spark 5.3","gpt-5.2-codex":"Codex 5.2","gpt-5.2":"GPT 5.2","gpt-5.1":"GPT 5.1","gpt-5-mini":"Mini 5","gpt-4.1":"GPT 4.1","claude-opus-4.7":"Opus 4.7","claude-opus-4.6":"Opus 4.6","claude-opus-4.6-fast":"Opus 4.6 Fast","claude-opus-4.5":"Opus 4.5","claude-sonnet-4.6":"Sonn 4.6","claude-sonnet-4.5":"Sonn 4.5","claude-sonnet-4":"Sonn 4","claude-haiku-4.5":"Haiku 4.5","claude-fable":"Fable"},n=s.replace(/-\d{8,}$/,"");if(t[s])return t[s];if(t[n])return t[n];let e=n.match(/^claude-(opus|sonnet|haiku|fable)(?:-(\d+)[-.](\d+))?(?:-(fast))?$/);if(e){let i=e[1]==="sonnet"?"Sonn":po(e[1]),a=e[2]&&e[3]?` ${e[2]}.${e[3]}`:"";return`${i}${a}${e[4]?" Fast":""}`}let o=n.match(/^gpt-(\d+)(?:[.-](\d+))?(?:-(mini|codex|spark))?$/);if(o){let i=o[2]?`${o[1]}.${o[2]}`:o[1];return o[3]==="mini"?`Mini ${i}`:o[3]==="codex"?`Codex ${i}`:o[3]==="spark"?`Spark ${i}`:`GPT ${i}`}return n}async function hn(s="copilot"){let t=await Ye(s,["help","config"]),n=ro(t);return n.length>1?n:[Oe]}async function mn(s="codex"){let t=await Ye(s,["debug","models"]).catch(async()=>{let e=process.env.CODEX_HOME||ge($e(),".codex");return vn(ge(e,"models_cache.json"))}),n=io(t);return n.length>0?Qe([...n,..._e]):_e}async function gn(s="claude"){let t=ge($e(),".claude"),n=[ge($e(),".claude","stats-cache.json"),ge($e(),".claude","settings.json"),ge($e(),".claude.json"),ge(t,"cache","changelog.md")],[e,o,i]=await Promise.all([Ye(s,["--help"]).catch(()=>""),Ye(s,["--version"]).catch(()=>""),co(t).catch(()=>[])]),a=uo([...n,...i]),c=await Promise.all(a.map(P=>vn(P).catch(()=>""))),m=ao([e,o,...c].join(`
332
+ `));return Qe([...Ke,...m])}function Ye(s,t,n=5e3){return new Promise((e,o)=>{eo(s,t,{encoding:"utf-8",timeout:n},(i,a)=>{if(i){o(i);return}e(a)})})}async function vn(s){return no(s,"utf-8")}async function co(s){let t=[ge(s,"projects"),ge(s,"telemetry")],n=[];for(let e of t)await yn(e,n,4);return n.sort((e,o)=>o.mtimeMs-e.mtimeMs).slice(0,80).map(e=>e.path)}async function yn(s,t,n){if(n<0)return;let e;try{e=await to(s,{withFileTypes:!0})}catch{return}await Promise.all(e.map(async o=>{let i=ge(s,o.name);if(o.isDirectory()){await yn(i,t,n-1);return}if(!(!o.isFile()||!lo(o.name)))try{let a=await so(i);if(a.size>2e6)return;t.push({path:i,mtimeMs:a.mtimeMs})}catch{}}))}function lo(s){return s.endsWith(".json")||s.endsWith(".jsonl")||s.endsWith(".md")||s.endsWith(".txt")}function Qe(s){let t=new Set,n=[];for(let e of s)t.has(e.id)||(t.add(e.id),n.push(e));return n}function uo(s){return[...new Set(s)]}function po(s){return s.charAt(0).toUpperCase()+s.slice(1)}var Ve=class{constructor(t=5){this.queue=[];this.activeJobs=new Map;this.activeProcesses=new Map;this.listeners=new Set;this.processor=null;this.eventBuffers=new Map;this.accumulators=new Map;this.cleanupTimers=new Map;this.maxConcurrency=t}setProcessor(t){this.processor=t}get active(){let t=this.activeJobs.values().next();return t.done?null:t.value}get allActive(){return Array.from(this.activeJobs.values())}get activeCount(){return this.activeJobs.size}get depth(){return this.queue.length}get isRunning(){return this.activeJobs.size>0}setActiveProcess(t,n){n?this.activeProcesses.set(t,n):this.activeProcesses.delete(t)}enqueue(t){return this.queue.push(t),this.processNext(),this.queue.length+this.activeJobs.size}addListener(t){return this.listeners.add(t),()=>this.listeners.delete(t)}broadcast(t,n,e){let o=this.bufferEvent(n,t);for(let i of this.listeners)i(o,n,e)}bufferEvent(t,n){let e=this.eventBuffers.get(t);e||(e={events:[],nextSeq:0},this.eventBuffers.set(t,e));let o={...n,seq:e.nextSeq++};return e.events.push(o),e.events.length>1e4&&e.events.splice(0,e.events.length-1e4),o}getBufferedEvents(t,n=-1){let e=this.eventBuffers.get(t),o=this.accumulators.get(t)??{response:"",thinking:""},i=this.activeJobs.has(t);if(!e)return null;let a=n<0?e.events:e.events.filter(c=>c.seq>n);return{jobId:t,events:a,currentSeq:e.nextSeq-1,accumulated:{...o},jobActive:i}}accumulateText(t,n,e){let o=this.accumulators.get(t);o||(o={response:"",thinking:""},this.accumulators.set(t,o)),o[n]+=e}getAccumulated(t){return this.accumulators.get(t)??null}scheduleBufferCleanup(t){let n=this.cleanupTimers.get(t);n&&clearTimeout(n);let e=setTimeout(()=>{this.eventBuffers.delete(t),this.accumulators.delete(t),this.cleanupTimers.delete(t)},6e4);this.cleanupTimers.set(t,e)}cancelJob(t){let n=this.activeProcesses.get(t),e=this.activeJobs.get(t);return!n||!e?!1:(n.kill("SIGTERM"),this.activeProcesses.delete(t),this.activeJobs.delete(t),e.status="error",e.error="Cancelled by user",this.broadcast({type:"error",jobId:e.id,threadId:e.threadId,message:"Cancelled by user",cancelled:!0,provider:e.provider,model:e.model},e.id,e.sourceId),this.processNext(),!0)}cancelActive(){if(this.activeJobs.size===0)return!1;let t=Array.from(this.activeJobs.keys());for(let n of t)this.cancelJob(n);return!0}destroy(){for(let t of this.activeProcesses.values())t.kill("SIGTERM");this.activeProcesses.clear(),this.activeJobs.clear(),this.queue=[],this.listeners.clear(),this.eventBuffers.clear(),this.accumulators.clear();for(let t of this.cleanupTimers.values())clearTimeout(t);this.cleanupTimers.clear()}async destroyAsync(t=1e4){let n=Array.from(this.activeProcesses.values());if(this.queue=[],this.listeners.clear(),n.length===0){this.activeProcesses.clear(),this.activeJobs.clear();return}for(let e of n)try{e.kill("SIGTERM")}catch{}await Promise.all(n.map(e=>new Promise(o=>{let i=!1,a=()=>{i||(i=!0,o())};e.on("exit",a),e.on("error",a),setTimeout(()=>{if(!i){try{e.kill("SIGKILL")}catch{}setTimeout(a,500)}},t)}))),this.activeProcesses.clear(),this.activeJobs.clear(),this.eventBuffers.clear(),this.accumulators.clear();for(let e of this.cleanupTimers.values())clearTimeout(e);this.cleanupTimers.clear()}processNext(){for(;this.activeJobs.size<this.maxConcurrency&&this.queue.length>0&&this.processor;){let t=this.queue.shift();this.activeJobs.set(t.id,t),t.status="running",this.broadcast({type:"job_started",jobId:t.id,position:0,threadId:t.threadId},t.id,t.sourceId),this.processor(t).catch(n=>{t.status="error",t.error=n instanceof Error?n.message:String(n),this.broadcast({type:"error",jobId:t.id,threadId:t.threadId,message:t.error,provider:t.provider,model:t.model},t.id,t.sourceId)}).finally(()=>{this.activeJobs.delete(t.id),this.activeProcesses.delete(t.id),this.scheduleBufferCleanup(t.id),this.processNext(),this.activeJobs.size===0&&this.queue.length===0&&this.broadcast({type:"queue_drained"},t.id)})}}};import{mkdir as fo,readFile as ho,writeFile as mo}from"fs/promises";import{dirname as go,join as vo}from"path";var yo={version:1,threads:{}},Ze=class{constructor(t){this.cache=null;this.writeChain=Promise.resolve();this.filePath=vo(t,".popmelt","threads.json")}async load(){if(this.cache)return this.cache;try{let t=await ho(this.filePath,"utf-8"),n=JSON.parse(t);if(n&&n.version===1&&n.threads)return this.cache=n,this.cache}catch{}return this.cache={...yo,threads:{}},this.cache}async getThread(t){return(await this.load()).threads[t]??null}async findContinuationThread(t){if(t.length===0)return null;let n=await this.load(),e=new Set(t);for(let o of Object.values(n.threads))if(o.elementIdentifiers.some(a=>e.has(a)))return o;return null}async createThread(t,n){let e=await this.load(),o={id:t,createdAt:Date.now(),updatedAt:Date.now(),elementIdentifiers:n,messages:[]};return e.threads[t]=o,await this.persist(),o}async appendMessage(t,n){let o=(await this.load()).threads[t];o&&(o.messages.push(n),o.updatedAt=Date.now(),await this.persist())}async addElementIdentifiers(t,n){let o=(await this.load()).threads[t];if(!o)return;let i=new Set(o.elementIdentifiers);for(let a of n)i.has(a)||o.elementIdentifiers.push(a);o.updatedAt=Date.now(),await this.persist()}async listRecent(t=5){let n=await this.load(),e=Object.values(n.threads);return e.sort((o,i)=>i.updatedAt-o.updatedAt),e.slice(0,t).map(o=>{let a=o.messages.find(c=>c.role==="human")?.feedbackSummary||"[thread]";return{id:o.id,createdAt:o.createdAt,updatedAt:o.updatedAt,preview:a,messageCount:o.messages.length,elementIdentifiers:o.elementIdentifiers}})}async getThreadHistory(t,n=6){let e=await this.getThread(t);return!e||e.messages.length===0?[]:e.messages.length<=n?e.messages:[e.messages[0],...e.messages.slice(-(n-1))]}async persist(){this.writeChain=this.writeChain.then(async()=>{if(this.cache)try{await fo(go(this.filePath),{recursive:!0}),await mo(this.filePath,JSON.stringify(this.cache,null,2))}catch(t){console.error("[ThreadStore] Failed to persist:",t)}}),await this.writeChain}};var wn="0.7.3";var Io=1111,To=["Read","Edit","Write","Glob","Grep","Bash","WebFetch","WebSearch","Bash(curl:*)"],Mo=1800*1e3,Co=3600*1e3;function En(s){if(!s)return!1;try{let t=new URL(s);return t.hostname==="localhost"||t.hostname.endsWith(".localhost")||t.hostname==="127.0.0.1"||t.hostname==="::1"||t.hostname==="[::1]"}catch{return!1}}function $o(s){let t=[];for(let n of s)if(n.type==="delta"){let e=n.text;if(!e)continue;let o=t[t.length-1];o&&o.kind==="text"?o.text+=e:t.push({kind:"text",text:e})}else if(n.type==="tool_use"){let e=n.tool||"",o=n.file??void 0,i=n.content??void 0,a=o?o.split("/").pop()??o:void 0,c;switch(e){case"Read":c=a?`Reading ${a}`:"Reading file";break;case"Edit":c=a?`Editing ${a}`:"Editing file";break;case"Write":c=a?`Writing ${a}`:"Writing file";break;case"Bash":c=i?i.split(`
333
+ `)[0].trim().slice(0,60):"Running command";break;case"Glob":c="Searching files";break;case"Grep":c="Searching code";break;case"WebFetch":c="Fetching page";break;case"WebSearch":c="Searching web";break;default:c=e?`Using ${e}`:"tool";break}let m=o??i??void 0,P=t[t.length-1];P&&P.kind==="tool_group"&&P.tool===e?P.items.push({label:c,detail:m}):t.push({kind:"tool_group",tool:e,items:[{label:c,detail:m}]})}return t}function _o(s,t){let n=s.headers.origin;En(n)&&(t.setHeader("Access-Control-Allow-Origin",n),t.setHeader("Access-Control-Allow-Methods","GET, POST, PATCH, DELETE, OPTIONS"),t.setHeader("Access-Control-Allow-Headers","Content-Type, X-Popmelt-Project-Id"))}function y(s,t,n){s.writeHead(t,{"Content-Type":"application/json"}),s.end(JSON.stringify(n))}function Oo(s,t){let n=s.headers["x-popmelt-project-id"];return(Array.isArray(n)?n[0]:n)||t.searchParams.get("projectId")||null}function Ao(s,t){if(!s)return t;let n=s.match(/^#?([0-9a-f]{2})([0-9a-f]{2})([0-9a-f]{2})$/i);if(!n)return t;let[,e,o,i]=n;return`\x1B[38;2;${parseInt(e,16)};${parseInt(o,16)};${parseInt(i,16)}m${t}\x1B[0m`}function vt(s,t){try{s.res.write(`event: ${t.type}
334
334
  data: ${JSON.stringify(t)}
335
335
 
336
- `)}catch{}}function De(s){return s==="claude"||s==="codex"||s==="copilot"?s:void 0}async function Ao(s){try{let t=new AbortController,n=setTimeout(()=>t.abort(),500),e=await fetch(`http://127.0.0.1:${s}/status`,{signal:t.signal});return clearTimeout(n),e.ok?await e.json():null}catch{return null}}function Do(s,t){return new Promise((n,e)=>{let o=a=>{s.removeListener("listening",i),e(a)},i=()=>{s.removeListener("error",o),n()};s.once("error",o),s.once("listening",i),s.listen(t,"127.0.0.1")})}function No(s){return/^[A-Za-z0-9_-]+$/.test(s)?`mcp_servers.${/^[A-Za-z0-9_]+$/.test(s)?s:`"${s}"`}.default_tools_approval_mode="approve"`:null}function xn(s){try{let t=et("git",["rev-parse","HEAD"],{cwd:s,encoding:"utf-8",stdio:["ignore","pipe","ignore"],timeout:5e3}).trim(),n=["--",".",":(exclude).popmelt/**",":(exclude)**/.popmelt/**"],e=et("git",["status","--porcelain","--untracked-files=all",...n],{cwd:s,encoding:"utf-8",stdio:["ignore","pipe","ignore"],timeout:5e3,maxBuffer:1024*1024}),o=et("git",["diff","HEAD",...n],{cwd:s,encoding:"utf-8",stdio:["ignore","pipe","ignore"],timeout:5e3,maxBuffer:2*1024*1024});return Pn("sha256").update(`${t}
336
+ `)}catch{}}function De(s){return s==="claude"||s==="codex"||s==="copilot"?s:void 0}async function Do(s){try{let t=new AbortController,n=setTimeout(()=>t.abort(),500),e=await fetch(`http://127.0.0.1:${s}/status`,{signal:t.signal});return clearTimeout(n),e.ok?await e.json():null}catch{return null}}function No(s,t){return new Promise((n,e)=>{let o=a=>{s.removeListener("listening",i),e(a)},i=()=>{s.removeListener("error",o),n()};s.once("error",o),s.once("listening",i),s.listen(t,"127.0.0.1")})}function xn(s){try{let t=et("git",["rev-parse","HEAD"],{cwd:s,encoding:"utf-8",stdio:["ignore","pipe","ignore"],timeout:5e3}).trim(),n=["--",".",":(exclude).popmelt/**",":(exclude)**/.popmelt/**"],e=et("git",["status","--porcelain","--untracked-files=all",...n],{cwd:s,encoding:"utf-8",stdio:["ignore","pipe","ignore"],timeout:5e3,maxBuffer:1024*1024}),o=et("git",["diff","HEAD",...n],{cwd:s,encoding:"utf-8",stdio:["ignore","pipe","ignore"],timeout:5e3,maxBuffer:2*1024*1024});return Pn("sha256").update(`${t}
337
337
  ${e}
338
- ${o}`).digest("hex").slice(0,16)}catch{return}}async function vt(s={}){let t=s.port??Io,n=s.projectRoot??process.cwd(),e=Pn("sha256").update(n).digest("hex").slice(0,12),o=s.devOrigin??(process.env.PORT?`http://localhost:${process.env.PORT}`:null),i=s.tempDir??me(Po(),"popmelt-bridge"),a=s.maxTurns??40,c=s.maxBudgetUsd??1,m=[...s.allowedTools??To],P=s.claudePath??"claude",g=s.copilotPath??"copilot",B=s.provider??"claude",z=s.timeoutMs,O=s.restartMode??"embedded",L=t,S={},X={claude:P,codex:"codex",copilot:g};for(let[r,l]of Object.entries(X))try{let p=l.includes("/")?l:et("which",[l],{encoding:"utf-8"}).trim();S[r]={available:!0,path:p}}catch{S[r]={available:!1,path:null}}function W(r,l){return new Promise(p=>{let u=wn(l,["--version"],{stdio:["ignore","ignore","ignore"]}),f=!1,d=J=>{f||(f=!0,p(J))},w=setTimeout(()=>{u.kill("SIGTERM"),d(!0)},5e3);u.on("error",()=>{clearTimeout(w),d(!1)}),u.on("close",J=>{clearTimeout(w),d(J===0)})})}await Promise.all([(async()=>{S.claude&&(S.claude.models=S.claude.available&&S.claude.path?await mn(S.claude.path).catch(()=>Ke):Ke)})(),(async()=>{S.codex&&(S.codex.models=S.codex.available&&S.codex.path?await hn(S.codex.path).catch(()=>_e):_e)})(),(async()=>{S.copilot&&(S.copilot.models=S.copilot.available&&S.copilot.path?await fn(S.copilot.path).catch(()=>[Oe]):[Oe])})()]);let[Q,Z,V]=await Promise.all([dt(n),ut(n),pt(n,S.copilot?.path??g)]),q=await Vt(n);S.claude&&(S.claude.mcp=Q),S.codex&&(S.codex.mcp=Z),S.copilot&&(S.copilot.mcp=V),Q.found&&Q.name&&m.push(`mcp__${Q.name}__*`),await wo(i,{recursive:!0}),kn(i).catch(()=>{});let D=new Ve(1),_=new Set,I=new Ze(n),k=new Ue(n),y=new ze(n),C=new We(n,k,{claudePath:P,provider:B,timeoutMs:z,onEvent:r=>{for(let l of _)gt(l,r)}}),N=20,x=300*1e3,b=[],H=new Map,E=null,ie;D.addListener((r,l,p)=>{for(let u of _)(!p||!u.sourceId||u.sourceId===p)&&gt(u,r)}),D.setProcessor(async r=>{let l=r._replyPrompt,p=r._replyImagePaths,u=r.provider??B,f;if(r.threadId){let R=await I.getThread(r.threadId);if(R){for(let T=R.messages.length-1;T>=0;T--)if(R.messages[T].sessionId){f=R.messages[T].sessionId;break}}}let d;if(f&&l){let R=(await I.getThread(r.threadId))?.messages.filter(se=>se.role==="human").pop();if(d=R?.replyToQuestion||R?.feedbackSummary||"",p&&p.length>0){d+=`
338
+ ${o}`).digest("hex").slice(0,16)}catch{return}}async function yt(s={}){let t=s.port??Io,n=s.projectRoot??process.cwd(),e=Pn("sha256").update(n).digest("hex").slice(0,12),o=s.devOrigin??(process.env.PORT?`http://localhost:${process.env.PORT}`:null),i=s.tempDir??me(Po(),"popmelt-bridge"),a=s.maxTurns??40,c=s.maxBudgetUsd??1,m=[...s.allowedTools??To],P=s.claudePath??"claude",v=s.copilotPath??"copilot",B=s.provider??"claude",z=s.timeoutMs,A=s.restartMode??"embedded",W=t,S={},X={claude:P,codex:"codex",copilot:v};for(let[r,l]of Object.entries(X))try{let p=l.includes("/")?l:et("which",[l],{encoding:"utf-8"}).trim();S[r]={available:!0,path:p}}catch{S[r]={available:!1,path:null}}function H(r,l){return new Promise(p=>{let d=Sn(l,["--version"],{stdio:["ignore","ignore","ignore"]}),f=!1,u=O=>{f||(f=!0,p(O))},g=setTimeout(()=>{d.kill("SIGTERM"),u(!0)},5e3);d.on("error",()=>{clearTimeout(g),u(!1)}),d.on("close",O=>{clearTimeout(g),u(O===0)})})}await Promise.all([(async()=>{S.claude&&(S.claude.models=S.claude.available&&S.claude.path?await gn(S.claude.path).catch(()=>Ke):Ke)})(),(async()=>{S.codex&&(S.codex.models=S.codex.available&&S.codex.path?await mn(S.codex.path).catch(()=>_e):_e)})(),(async()=>{S.copilot&&(S.copilot.models=S.copilot.available&&S.copilot.path?await hn(S.copilot.path).catch(()=>[Oe]):[Oe])})()]);let[K,Z,Q]=await Promise.all([dt(n),ut(n),pt(n,S.copilot?.path??v)]),V=await Zt(n);S.claude&&(S.claude.mcp=K),S.codex&&(S.codex.mcp=Z),S.copilot&&(S.copilot.mcp=Q),K.found&&K.name&&m.push(`mcp__${K.name}__*`),await wo(i,{recursive:!0}),kn(i).catch(()=>{});let N=new Ve(1),_=new Set,R=new Ze(n),k=new Ue(n),w=new ze(n),C=new We(n,k,{claudePath:P,provider:B,timeoutMs:z,onEvent:r=>{for(let l of _)vt(l,r)}}),J=20,x=300*1e3,b=[],G=new Map,E=null,ie;N.addListener((r,l,p)=>{for(let d of _)(!p||!d.sourceId||d.sourceId===p)&&vt(d,r)}),N.setProcessor(async r=>{let l=r._replyPrompt,p=r._replyImagePaths,d=r.provider??B,f;if(r.threadId){let I=await R.getThread(r.threadId);if(I){for(let T=I.messages.length-1;T>=0;T--)if(I.messages[T].sessionId){f=I.messages[T].sessionId;break}}}let u;if(f&&l){let I=(await R.getThread(r.threadId))?.messages.filter(se=>se.role==="human").pop();if(u=I?.replyToQuestion||I?.feedbackSummary||"",p&&p.length>0){u+=`
339
339
 
340
- The developer attached reference images with their reply:`;for(let se of p)d+=`
341
- Attached image: use the Read tool to view the image at: ${se}`}r.kind==="synthesize"||r.kind==="synthesize_evals"||r.kind==="run_eval"?d+=`
340
+ The developer attached reference images with their reply:`;for(let se of p)u+=`
341
+ Attached image: use the Read tool to view the image at: ${se}`}r.kind==="synthesize"||r.kind==="synthesize_evals"||r.kind==="run_eval"?u+=`
342
342
 
343
- Continue the project workflow from the previous instructions. If the developer approved the proposals, emit the final structured block requested earlier.`:d+=`
343
+ Continue the project workflow from the previous instructions. If the developer approved the proposals, emit the final structured block requested earlier.`:u+=`
344
344
 
345
- After completing work, output a <resolution> block with declaredScope and inferredScope. If the developer corrected scope, set finalScope. If unclear, output a <question> block.`}else if(f)d=Ce(r.feedback,r.imagePaths)+`
345
+ After completing work, output a <resolution> block with declaredScope and inferredScope. If the developer corrected scope, set finalScope. If unclear, output a <question> block.`}else if(f)u=Ce(r.feedback,r.imagePaths)+`
346
346
 
347
347
  Follow the developer's instructions. If they ask for changes, apply them to the source files.
348
348
 
349
- After completing work, output a <resolution> block with declaredScope and inferredScope. If unclear, output a <question> block.`+(u!=="codex"?`
349
+ After completing work, output a <resolution> block with declaredScope and inferredScope. If unclear, output a <question> block.`+(d!=="codex"?`
350
350
 
351
- IMPORTANT: First, use the Read tool to view the updated screenshot at: ${r.screenshotPath}`:"");else{let R=!l&&r.threadId?await I.getThreadHistory(r.threadId):void 0,T=l?null:await C.loadModel();d=l??cn(r.screenshotPath,r.feedback,{threadHistory:R&&R.length>0?R:void 0,provider:u,imagePaths:r.imagePaths,designModel:T??void 0,screenshotPaths:r.screenshotPaths})}let w=Oo(r.color,`[\u22B9 ${L}:${r.id}]`),J=r.screenshotPaths&&Object.keys(r.screenshotPaths).length>0?`${Object.keys(r.screenshotPaths).length} pages [${Object.keys(r.screenshotPaths).join(", ")}]`:r.screenshotPath;console.log(`${w} Reviewing ${J} (provider: ${u})${r.threadId?` (thread: ${r.threadId})`:""}${f?` (resuming: ${f.slice(0,8)})`:""}`);let j=(R,T)=>{R.type==="delta"&&"text"in R?D.accumulateText(T,"response",R.text):R.type==="thinking"&&"text"in R&&D.accumulateText(T,"thinking",R.text),D.broadcast(R,T,r.sourceId)},{process:Y,result:F}=u==="codex"?je(r.id,{prompt:d,projectRoot:n,screenshotPath:r.screenshotPath,resumeSessionId:f,model:r.model,configOverrides:r.kind==="run_eval"?q.map(No).filter(R=>typeof R=="string"):[],onEvent:j}):u==="copilot"?Ot(r.id,{prompt:d,projectRoot:n,resumeSessionId:f,model:r.model,timeoutMs:z,copilotPath:S.copilot?.path??g,onEvent:j}):Be(r.id,{prompt:d,projectRoot:n,maxTurns:a,maxBudgetUsd:c,allowedTools:m,claudePath:P,resumeSessionId:f,model:r.model,timeoutMs:z,onEvent:j});D.setActiveProcess(r.id,Y);let h=await F;if(r.result=h.text,h.success){console.log(`${w} Iteration complete`),h.fileEdits&&h.fileEdits.length>0&&console.log(`${w} Captured ${h.fileEdits.length} file edit(s): ${h.fileEdits.map($=>`${$.tool} ${$.file_path}`).join(", ")}`),h.editEvents&&h.editEvents.length>0&&console.log(`${w} Captured ${h.editEvents.length} edit event(s): ${h.editEvents.map($=>`${$.operation} ${$.filePath}`).join(", ")}`),r.status="done";let R=ln(h.text),T=un(h.text);if(T.length>0&&r.annotationIds&&r.annotationIds.length>0){let $=new Set(r.annotationIds);T.every(A=>$.has(A.annotationId))||(T=T.map((A,U)=>({...A,annotationId:r.annotationIds[U%r.annotationIds.length]})))}let se=h.fileEdits&&h.fileEdits.length>0?h.fileEdits.map($=>`${$.tool} ${$.file_path.split("/").pop()}`):h.toolsUsed,ee=D.getBufferedEvents(r.id),ae=ee?$o(ee.events):void 0;if(r.threadId&&await I.appendMessage(r.threadId,{role:"assistant",timestamp:Date.now(),jobId:r.id,responseText:h.text,resolutions:T.length>0?T:void 0,question:R??void 0,sessionId:h.sessionId,toolsUsed:se,editEvents:h.editEvents&&h.editEvents.length>0?h.editEvents:void 0,segments:ae&&ae.length>0?ae:void 0,model:r.model,provider:u}),r.kind!=="run_eval"&&k.captureGitDiff(n).then(async $=>{let K=Date.now(),A=r.imagePaths?Object.values(r.imagePaths).flat():[],U=[];if(r.imagePaths)for(let[le,M]of Object.entries(r.imagePaths))for(let oe=0;oe<M.length;oe++)U.push(`screenshots/p-${r.id}-${le}-${oe}.webp`);await k.persist({version:1,id:r.id,createdAt:r.createdAt,completedAt:K,durationMs:K-r.createdAt,url:r.feedback.url,viewport:r.feedback.viewport,screenshotPath:`screenshots/s-${r.id}.webp`,pastedImagePaths:U,annotations:r.feedback.annotations,styleModifications:r.feedback.styleModifications,inspectedElement:r.feedback.inspectedElement,provider:u,model:r.model,sessionId:h.sessionId,threadId:r.threadId,responseText:h.text,resolutions:T.length>0?T:[],question:R??void 0,fileEdits:h.fileEdits??[],editEvents:h.editEvents??[],toolsUsed:se,gitDiff:$},r.screenshotPath,A)}).catch(()=>{}),T.length>0){let $=T.some(K=>(K.finalScope??K.inferredScope)?.breadth==="pattern");$&&u!=="copilot"?C.run({provider:u,model:r.model}).catch(()=>{}):$&&console.log(`${w} Skipping materialization for Copilot provider`)}if(r.kind==="synthesize"&&h.text){let $=He(h.text);if($){Array.isArray($.rules)&&($.rules=Te($.rules));let K=await C.loadModel();K&&(!$.tokens&&K.tokens&&($.tokens=K.tokens),!$.components&&K.components&&($.components=K.components)),await C.writeModel($),console.log(`${w} Synthesize: model.json updated`)}}else if(r.kind==="synthesize_evals"&&h.text){let $=Bt(h.text),K=Ft(h.text),A=$??K;if(A&&A.length>0){let U=await y.writeSynthesized(A);console.log(`${w} Eval synthesis: wrote ${A.length} ${$?"approved":"pending"} eval(s) \u2192 .popmelt/evals/evals.json (${U.cases.length} total)`)}}else if(r.kind==="run_eval"&&h.text&&r.evalId&&r.evalRunId){let $=await y.findCase(r.evalId),K=r._evalRunScope;if($&&K){let A=jt(h.text,$.assertions),U=Date.now();await y.writeRun({id:r.evalRunId,evalId:r.evalId,threadId:r.threadId,createdAt:r.createdAt,completedAt:U,durationMs:U-r.createdAt,scope:K,provider:u,model:r.model,screenshotPath:r._evalScreenshotPath,codeFingerprint:xn(n),rawResponse:h.text,...A}),console.log(`${w} Eval run: ${$.title} \u2192 ${A.status}`)}}R&&(console.log(`${w} \u{1F4AC} Question detected: "${R.slice(0,120)}" \u2192 broadcasting to ${_.size} SSE clients (threadId=${r.threadId??r.id}, annotationIds=${r.annotationIds?.join(",")??"none"})`),D.broadcast({type:"question",jobId:r.id,threadId:r.threadId??r.id,question:R,annotationIds:r.annotationIds},r.id,r.sourceId));let ce=pn(h.text);ce.length>0&&(console.log(`${w} Novel pattern(s): ${ce.map($=>`${$.category}/${$.element}`).join(", ")}`),D.broadcast({type:"novel_patterns",jobId:r.id,patterns:ce,threadId:r.threadId},r.id,r.sourceId)),D.broadcast({type:"done",jobId:r.id,success:!0,resolutions:T.length>0?T:void 0,responseText:h.text,threadId:r.threadId},r.id,r.sourceId),b.push({id:r.id,status:"done",completedAt:Date.now(),threadId:r.threadId,annotationIds:r.annotationIds})}else console.error(`${w} Error: ${h.error}`),r.status="error",r.error=h.error,r.threadId&&await I.appendMessage(r.threadId,{role:"assistant",timestamp:Date.now(),jobId:r.id,error:h.error||"Unknown error",model:r.model,provider:u}),D.broadcast({type:"error",jobId:r.id,threadId:r.threadId,message:h.error||"Unknown error",provider:u,model:r.model,recoverable:h.recoverableError,code:h.errorCode},r.id,r.sourceId),b.push({id:r.id,status:"error",completedAt:Date.now(),error:h.error,threadId:r.threadId,annotationIds:r.annotationIds});let G=Date.now()-x;for(;b.length>0&&(b[0].completedAt<G||b.length>N);)b.shift()});let re=ko(async(r,l)=>{if(_o(r,l),r.method==="OPTIONS"){l.writeHead(204),l.end();return}let p=r.url||"/",u=p.split("?")[0],f=new URL(p,`http://127.0.0.1:${L}`),d=f.pathname;try{if(r.method==="POST"&&d==="/send")await ne(r,l);else if(r.method==="GET"&&d==="/events")Rn(r,l);else if(r.method==="GET"&&d==="/status")In(l);else if(r.method==="PATCH"&&d==="/config")await Tn(r,l);else if(r.method==="POST"&&d==="/shutdown"){if(O!=="detached"){v(l,409,{ok:!1,error:"This Popmelt bridge is embedded in the host dev server. Restart the dev server to reload bridge code.",restartMode:O});return}v(l,200,{ok:!0}),setTimeout(()=>process.exit(0),100)}else if(r.method==="POST"&&d==="/restart"){if(O!=="detached"){v(l,409,{ok:!1,error:"This Popmelt bridge is embedded in the host dev server. Restart the dev server to reload bridge code.",restartMode:O});return}v(l,200,{ok:!0,restarting:!0}),setTimeout(()=>{for(let w of _)try{w.res.end()}catch{}re.close(()=>{wn(process.execPath,process.argv.slice(1),{detached:!0,stdio:"ignore",cwd:process.cwd(),env:process.env}).unref(),setTimeout(()=>process.exit(0),200)})},100)}else if(r.method==="GET"&&d==="/capabilities")v(l,200,{providers:S});else if(r.method==="POST"&&d==="/mcp/install")await Nn(r,l);else if(r.method==="POST"&&d==="/reply")await Ne(r,l);else if(r.method==="POST"&&d==="/cancel")Mn(r,l);else if(r.method==="POST"&&d==="/materialize")await Cn(r,l);else if(r.method==="POST"&&d==="/model/component")await Jn(r,l);else if(r.method==="DELETE"&&d==="/model/component")await Bn(r,l);else if(r.method==="PATCH"&&d==="/model/token")await Fn(r,l);else if(r.method==="DELETE"&&d==="/model/token")await jn(r,l);else if(r.method==="POST"&&d==="/model/consolidate"){let{provider:w,model:J}=await tt(r);if(w==="copilot"){v(l,400,{error:"Model consolidation is not available for Copilot yet."});return}C.consolidate({provider:w,model:J}).catch(j=>console.error("[Bridge] Consolidation error:",j)),v(l,200,{started:!0})}else if(r.method==="POST"&&d==="/model/synthesize")await $n(r,l);else if(r.method==="POST"&&d==="/evals/synthesize")await _n(r,l);else if(r.method==="GET"&&d==="/evals"){let w=await y.loadSuiteWithCurrentRuns(xn(n));v(l,200,w)}else if(r.method==="POST"&&d.startsWith("/evals/")&&d.endsWith("/run")){let w=d.slice(7,-4);await On(w,r,l)}else if(r.method==="PATCH"&&d.startsWith("/evals/"))await An(d.slice(7),r,l);else if(r.method==="GET"&&d==="/model"){let w=await C.loadModel();v(l,200,{model:w})}else if(r.method==="GET"&&d.startsWith("/jobs/")&&d.endsWith("/events")){let w=d.slice(6,d.length-7),J=parseInt(f.searchParams.get("afterSeq")??"-1",10),j=D.getBufferedEvents(w,isNaN(J)?-1:J);j?v(l,200,j):v(l,404,{error:"Unknown job"})}else if(r.method==="GET"&&u.startsWith("/files/"))await Wn(u.slice(7),l);else if(r.method==="GET"&&d==="/threads/recent"){let w=Math.min(Math.max(parseInt(f.searchParams.get("limit")??"5",10)||5,1),20),J=await I.listRecent(w);v(l,200,J)}else if(r.method==="GET"&&d.startsWith("/thread/")){let w=d.slice(8);await Hn(w,l)}else r.method==="GET"&&(d==="/canvas"||d==="/canvas/")?Un(r,l):r.method==="GET"&&d==="/canvas/manifest"?await zn(l):r.method==="GET"&&d==="/canvas/app.mjs"?await Ln(l):v(l,404,{error:"Not found"})}catch(w){console.error("[Bridge] Request error:",w),v(l,500,{error:w instanceof Error?w.message:"Internal error"})}});async function ne(r,l){let{screenshot:p,feedback:u,color:f,provider:d,model:w,sourceId:J,pastedImages:j,pageScreenshots:Y}=await qe(r),F;try{F=JSON.parse(u)}catch{v(l,400,{error:"Invalid feedback JSON"});return}let h=Se().slice(0,8),G={};if(Y.length>0)for(let M of Y){let oe=encodeURIComponent(M.pathname),de=me(i,`screenshot-${h}-${oe}.webp`);await Ae(de,M.data),G[M.pathname]=de}let R=me(i,`screenshot-${h}.webp`);await Ae(R,p);let T={};if(j.length>0)for(let M of j){let oe=me(i,`pasted-${h}-${M.annotationId}-${M.index}.webp`);await Ae(oe,M.data),T[M.annotationId]||(T[M.annotationId]=[]),T[M.annotationId].push(oe)}let se=F.annotations.map(M=>M.linkedSelector?M.pathname?`${M.pathname}:${M.linkedSelector}`:M.linkedSelector:null).filter(M=>!!M),ee;if(se.length>0){let M=await I.findContinuationThread(se);M?(ee=M.id,await I.addElementIdentifiers(ee,se)):ee=(await I.createThread(h,se)).id}else ee=(await I.createThread(h,[])).id;let ae=F.annotations.map(M=>M.id),ce=Object.keys(G).length>0,$={id:h,status:"queued",screenshotPath:R,feedback:F,createdAt:Date.now(),color:f,threadId:ee,annotationIds:ae,provider:De(d),model:w||void 0,...Object.keys(T).length>0?{imagePaths:T}:{},sourceId:J||void 0,...ce?{screenshotPaths:G}:{}},K=new Set(F.annotations.map(M=>M.pathname).filter(Boolean)),A;if(K.size>1){let M=new Map;for(let oe of F.annotations){let de=oe.pathname||"(unknown)";M.has(de)||M.set(de,[]),M.get(de).push(oe.instruction||`[${oe.type}]`)}A=[...M.entries()].map(([oe,de])=>`\`${oe}\`
351
+ IMPORTANT: First, use the Read tool to view the updated screenshot at: ${r.screenshotPath}`:"");else{let I=!l&&r.threadId?await R.getThreadHistory(r.threadId):void 0,T=l?null:await C.loadModel();u=l??ln(r.screenshotPath,r.feedback,{threadHistory:I&&I.length>0?I:void 0,provider:d,imagePaths:r.imagePaths,designModel:T??void 0,screenshotPaths:r.screenshotPaths})}let g=Ao(r.color,`[\u22B9 ${W}:${r.id}]`),O=r.screenshotPaths&&Object.keys(r.screenshotPaths).length>0?`${Object.keys(r.screenshotPaths).length} pages [${Object.keys(r.screenshotPaths).join(", ")}]`:r.screenshotPath;console.log(`${g} Reviewing ${O} (provider: ${d})${r.threadId?` (thread: ${r.threadId})`:""}${f?` (resuming: ${f.slice(0,8)})`:""}`);let F=(I,T)=>{I.type==="delta"&&"text"in I?N.accumulateText(T,"response",I.text):I.type==="thinking"&&"text"in I&&N.accumulateText(T,"thinking",I.text),N.broadcast(I,T,r.sourceId)},{process:q,result:j}=d==="codex"?je(r.id,{prompt:u,projectRoot:n,screenshotPath:r.screenshotPath,resumeSessionId:f,model:r.model,onEvent:F}):d==="copilot"?At(r.id,{prompt:u,projectRoot:n,resumeSessionId:f,model:r.model,timeoutMs:z,copilotPath:S.copilot?.path??v,onEvent:F}):Be(r.id,{prompt:u,projectRoot:n,maxTurns:a,maxBudgetUsd:c,allowedTools:m,claudePath:P,resumeSessionId:f,model:r.model,timeoutMs:z,onEvent:F});N.setActiveProcess(r.id,q);let h=await j;if(r.result=h.text,h.success){console.log(`${g} Iteration complete`),h.fileEdits&&h.fileEdits.length>0&&console.log(`${g} Captured ${h.fileEdits.length} file edit(s): ${h.fileEdits.map($=>`${$.tool} ${$.file_path}`).join(", ")}`),h.editEvents&&h.editEvents.length>0&&console.log(`${g} Captured ${h.editEvents.length} edit event(s): ${h.editEvents.map($=>`${$.operation} ${$.filePath}`).join(", ")}`),r.status="done";let I=dn(h.text),T=pn(h.text);if(T.length>0&&r.annotationIds&&r.annotationIds.length>0){let $=new Set(r.annotationIds);T.every(D=>$.has(D.annotationId))||(T=T.map((D,U)=>({...D,annotationId:r.annotationIds[U%r.annotationIds.length]})))}let se=h.fileEdits&&h.fileEdits.length>0?h.fileEdits.map($=>`${$.tool} ${$.file_path.split("/").pop()}`):h.toolsUsed,ee=N.getBufferedEvents(r.id),ae=ee?$o(ee.events):void 0;if(r.threadId&&await R.appendMessage(r.threadId,{role:"assistant",timestamp:Date.now(),jobId:r.id,responseText:h.text,resolutions:T.length>0?T:void 0,question:I??void 0,sessionId:h.sessionId,toolsUsed:se,editEvents:h.editEvents&&h.editEvents.length>0?h.editEvents:void 0,segments:ae&&ae.length>0?ae:void 0,model:r.model,provider:d}),r.kind!=="run_eval"&&k.captureGitDiff(n).then(async $=>{let Y=Date.now(),D=r.imagePaths?Object.values(r.imagePaths).flat():[],U=[];if(r.imagePaths)for(let[le,M]of Object.entries(r.imagePaths))for(let oe=0;oe<M.length;oe++)U.push(`screenshots/p-${r.id}-${le}-${oe}.webp`);await k.persist({version:1,id:r.id,createdAt:r.createdAt,completedAt:Y,durationMs:Y-r.createdAt,url:r.feedback.url,viewport:r.feedback.viewport,screenshotPath:`screenshots/s-${r.id}.webp`,pastedImagePaths:U,annotations:r.feedback.annotations,styleModifications:r.feedback.styleModifications,inspectedElement:r.feedback.inspectedElement,provider:d,model:r.model,sessionId:h.sessionId,threadId:r.threadId,responseText:h.text,resolutions:T.length>0?T:[],question:I??void 0,fileEdits:h.fileEdits??[],editEvents:h.editEvents??[],toolsUsed:se,gitDiff:$},r.screenshotPath,D)}).catch(()=>{}),T.length>0){let $=T.some(Y=>(Y.finalScope??Y.inferredScope)?.breadth==="pattern");$&&d!=="copilot"?C.run({provider:d,model:r.model}).catch(()=>{}):$&&console.log(`${g} Skipping materialization for Copilot provider`)}if(r.kind==="synthesize"&&h.text){let $=He(h.text);if($){Array.isArray($.rules)&&($.rules=Te($.rules));let Y=await C.loadModel();Y&&(!$.tokens&&Y.tokens&&($.tokens=Y.tokens),!$.components&&Y.components&&($.components=Y.components)),await C.writeModel($),console.log(`${g} Synthesize: model.json updated`)}}else if(r.kind==="synthesize_evals"&&h.text){let $=Ft(h.text),Y=jt(h.text),D=$??Y;if(D&&D.length>0){let U=await w.writeSynthesized(D);console.log(`${g} Eval synthesis: wrote ${D.length} ${$?"approved":"pending"} eval(s) \u2192 .popmelt/evals/evals.json (${U.cases.length} total)`)}}else if(r.kind==="run_eval"&&h.text&&r.evalId&&r.evalRunId){let $=await w.findCase(r.evalId),Y=r._evalRunScope;if($&&Y){let D=Ut(h.text,$.assertions),U=Date.now();await w.writeRun({id:r.evalRunId,evalId:r.evalId,threadId:r.threadId,createdAt:r.createdAt,completedAt:U,durationMs:U-r.createdAt,scope:Y,provider:d,model:r.model,screenshotPath:r._evalScreenshotPath,codeFingerprint:xn(n),rawResponse:h.text,...D}),console.log(`${g} Eval run: ${$.title} \u2192 ${D.status}`)}}I&&(console.log(`${g} \u{1F4AC} Question detected: "${I.slice(0,120)}" \u2192 broadcasting to ${_.size} SSE clients (threadId=${r.threadId??r.id}, annotationIds=${r.annotationIds?.join(",")??"none"})`),N.broadcast({type:"question",jobId:r.id,threadId:r.threadId??r.id,question:I,annotationIds:r.annotationIds},r.id,r.sourceId));let ce=fn(h.text);ce.length>0&&(console.log(`${g} Novel pattern(s): ${ce.map($=>`${$.category}/${$.element}`).join(", ")}`),N.broadcast({type:"novel_patterns",jobId:r.id,patterns:ce,threadId:r.threadId},r.id,r.sourceId)),N.broadcast({type:"done",jobId:r.id,success:!0,resolutions:T.length>0?T:void 0,responseText:h.text,threadId:r.threadId},r.id,r.sourceId),b.push({id:r.id,status:"done",completedAt:Date.now(),threadId:r.threadId,annotationIds:r.annotationIds})}else console.error(`${g} Error: ${h.error}`),r.status="error",r.error=h.error,r.threadId&&await R.appendMessage(r.threadId,{role:"assistant",timestamp:Date.now(),jobId:r.id,error:h.error||"Unknown error",model:r.model,provider:d}),N.broadcast({type:"error",jobId:r.id,threadId:r.threadId,message:h.error||"Unknown error",provider:d,model:r.model,recoverable:h.recoverableError,code:h.errorCode},r.id,r.sourceId),b.push({id:r.id,status:"error",completedAt:Date.now(),error:h.error,threadId:r.threadId,annotationIds:r.annotationIds});let L=Date.now()-x;for(;b.length>0&&(b[0].completedAt<L||b.length>J);)b.shift()});let re=ko(async(r,l)=>{if(_o(r,l),r.method==="OPTIONS"){l.writeHead(204),l.end();return}let p=r.url||"/",d=p.split("?")[0],f=new URL(p,`http://127.0.0.1:${W}`),u=f.pathname;if(r.headers.origin){if(!En(r.headers.origin)){y(l,403,{error:"Popmelt bridge only accepts browser requests from localhost origins."});return}let g=Oo(r,f);if(g!==e){y(l,409,{error:"Popmelt bridge/project mismatch",bridgeProjectId:e,requestProjectId:g,projectName:gt(n)});return}}try{if(r.method==="POST"&&u==="/send")await ne(r,l);else if(r.method==="GET"&&u==="/events")Rn(r,l);else if(r.method==="GET"&&u==="/status")In(l);else if(r.method==="PATCH"&&u==="/config")await Tn(r,l);else if(r.method==="POST"&&u==="/shutdown"){if(A!=="detached"){y(l,409,{ok:!1,error:"This Popmelt bridge is embedded in the host dev server. Restart the dev server to reload bridge code.",restartMode:A});return}y(l,200,{ok:!0}),setTimeout(()=>process.exit(0),100)}else if(r.method==="POST"&&u==="/restart"){if(A!=="detached"){y(l,409,{ok:!1,error:"This Popmelt bridge is embedded in the host dev server. Restart the dev server to reload bridge code.",restartMode:A});return}y(l,200,{ok:!0,restarting:!0}),setTimeout(()=>{for(let g of _)try{g.res.end()}catch{}re.close(()=>{Sn(process.execPath,process.argv.slice(1),{detached:!0,stdio:"ignore",cwd:process.cwd(),env:process.env}).unref(),setTimeout(()=>process.exit(0),200)})},100)}else if(r.method==="GET"&&u==="/capabilities")y(l,200,{providers:S});else if(r.method==="POST"&&u==="/mcp/install")await Nn(r,l);else if(r.method==="POST"&&u==="/reply")await Ne(r,l);else if(r.method==="POST"&&u==="/cancel")Mn(r,l);else if(r.method==="POST"&&u==="/materialize")await Cn(r,l);else if(r.method==="POST"&&u==="/model/component")await Jn(r,l);else if(r.method==="DELETE"&&u==="/model/component")await Bn(r,l);else if(r.method==="PATCH"&&u==="/model/token")await Fn(r,l);else if(r.method==="DELETE"&&u==="/model/token")await jn(r,l);else if(r.method==="POST"&&u==="/model/consolidate"){let{provider:g,model:O}=await tt(r);if(g==="copilot"){y(l,400,{error:"Model consolidation is not available for Copilot yet."});return}C.consolidate({provider:g,model:O}).catch(F=>console.error("[Bridge] Consolidation error:",F)),y(l,200,{started:!0})}else if(r.method==="POST"&&u==="/model/synthesize")await $n(r,l);else if(r.method==="POST"&&u==="/evals/synthesize")await _n(r,l);else if(r.method==="GET"&&u==="/evals"){let g=await w.loadSuiteWithCurrentRuns(xn(n));y(l,200,g)}else if(r.method==="POST"&&u.startsWith("/evals/")&&u.endsWith("/run")){let g=u.slice(7,-4);await On(g,r,l)}else if(r.method==="PATCH"&&u.startsWith("/evals/"))await An(u.slice(7),r,l);else if(r.method==="GET"&&u==="/model"){let g=await C.loadModel();y(l,200,{model:g})}else if(r.method==="GET"&&u.startsWith("/jobs/")&&u.endsWith("/events")){let g=u.slice(6,u.length-7),O=parseInt(f.searchParams.get("afterSeq")??"-1",10),F=N.getBufferedEvents(g,isNaN(O)?-1:O);F?y(l,200,F):y(l,404,{error:"Unknown job"})}else if(r.method==="GET"&&d.startsWith("/files/"))await Wn(d.slice(7),l);else if(r.method==="GET"&&u==="/threads/recent"){let g=Math.min(Math.max(parseInt(f.searchParams.get("limit")??"5",10)||5,1),20),O=await R.listRecent(g);y(l,200,O)}else if(r.method==="GET"&&u.startsWith("/thread/")){let g=u.slice(8);await Hn(g,l)}else r.method==="GET"&&(u==="/canvas"||u==="/canvas/")?Un(r,l):r.method==="GET"&&u==="/canvas/manifest"?await zn(l):r.method==="GET"&&u==="/canvas/app.mjs"?await Ln(l):y(l,404,{error:"Not found"})}catch(g){console.error("[Bridge] Request error:",g),y(l,500,{error:g instanceof Error?g.message:"Internal error"})}});async function ne(r,l){let{screenshot:p,feedback:d,color:f,provider:u,model:g,sourceId:O,pastedImages:F,pageScreenshots:q}=await qe(r),j;try{j=JSON.parse(d)}catch{y(l,400,{error:"Invalid feedback JSON"});return}let h=Se().slice(0,8),L={};if(q.length>0)for(let M of q){let oe=encodeURIComponent(M.pathname),de=me(i,`screenshot-${h}-${oe}.webp`);await Ae(de,M.data),L[M.pathname]=de}let I=me(i,`screenshot-${h}.webp`);await Ae(I,p);let T={};if(F.length>0)for(let M of F){let oe=me(i,`pasted-${h}-${M.annotationId}-${M.index}.webp`);await Ae(oe,M.data),T[M.annotationId]||(T[M.annotationId]=[]),T[M.annotationId].push(oe)}let se=j.annotations.map(M=>M.linkedSelector?M.pathname?`${M.pathname}:${M.linkedSelector}`:M.linkedSelector:null).filter(M=>!!M),ee;if(se.length>0){let M=await R.findContinuationThread(se);M?(ee=M.id,await R.addElementIdentifiers(ee,se)):ee=(await R.createThread(h,se)).id}else ee=(await R.createThread(h,[])).id;let ae=j.annotations.map(M=>M.id),ce=Object.keys(L).length>0,$={id:h,status:"queued",screenshotPath:I,feedback:j,createdAt:Date.now(),color:f,threadId:ee,annotationIds:ae,provider:De(u),model:g||void 0,...Object.keys(T).length>0?{imagePaths:T}:{},sourceId:O||void 0,...ce?{screenshotPaths:L}:{}},Y=new Set(j.annotations.map(M=>M.pathname).filter(Boolean)),D;if(Y.size>1){let M=new Map;for(let oe of j.annotations){let de=oe.pathname||"(unknown)";M.has(de)||M.set(de,[]),M.get(de).push(oe.instruction||`[${oe.type}]`)}D=[...M.entries()].map(([oe,de])=>`\`${oe}\`
352
352
  ${de.map(nt=>`- ${nt}`).join(`
353
353
  `)}`).join(`
354
- `)}else A=F.annotations.map(M=>M.instruction||`[${M.type}]`).join("; ");let U=Ce(F,Object.keys(T).length>0?T:void 0);await I.appendMessage(ee,{role:"human",timestamp:Date.now(),jobId:h,screenshotPath:R,...ce?{screenshotPaths:G}:{},...Object.keys(T).length>0?{imagePaths:T}:{},annotationIds:ae,feedbackSummary:A,feedbackContext:U||void 0});let le=D.enqueue($);v(l,200,{jobId:h,position:le,threadId:ee})}async function Ne(r,l){let p=r.headers["content-type"]||"",u,f,d,w,J,j,Y=[];if(p.includes("multipart/form-data")){let A=await qe(r),U=A.feedback?JSON.parse(A.feedback):{};u=U.threadId,f=U.reply,d=U.color,w=U.provider,J=U.model,j=U.sourceId||A.sourceId;for(let le of A.pastedImages)Y.push(le.data)}else{let A=[];for await(let M of r)A.push(typeof M=="string"?Buffer.from(M):M);let U=Buffer.concat(A).toString("utf-8"),le;try{le=JSON.parse(U)}catch{v(l,400,{error:"Invalid JSON"});return}u=le.threadId,f=le.reply,d=le.color,w=le.provider,J=le.model,j=le.sourceId}if(!u||!f){v(l,400,{error:"Missing threadId or reply"});return}if(!await I.getThread(u)){v(l,404,{error:"Thread not found"});return}let h=Se().slice(0,8),G=[];for(let A=0;A<Y.length;A++){let U=me(i,`reply-${h}-${A}.webp`);await Ae(U,Y[A]),G.push(U)}let R="";{let A=await I.getThreadHistory(u);for(let U=A.length-1;U>=0;U--)if(A[U].screenshotPath){R=A[U].screenshotPath;break}}let T=await Dn(u);if(!R&&!T){v(l,400,{error:"No screenshot available"});return}await I.appendMessage(u,{role:"human",timestamp:Date.now(),jobId:h,replyToQuestion:f,screenshotPath:R,...G.length>0?{replyImagePaths:G}:{}});let se=await I.getThreadHistory(u),ee=[];for(let A of se)if(A.annotationIds)for(let U of A.annotationIds)ee.includes(U)||ee.push(U);let ae=De(w),ce=dn(R,se,ae,G.length>0?G:void 0),$={id:h,status:"queued",screenshotPath:R,feedback:{timestamp:new Date().toISOString(),url:"",viewport:{width:0,height:0},scrollPosition:{x:0,y:0},annotations:[],styleModifications:[]},createdAt:Date.now(),color:d,threadId:u,annotationIds:ee.length>0?ee:void 0,provider:ae,model:J||void 0,sourceId:j||void 0};T&&($.kind=T),$._replyPrompt=ce,G.length>0&&($._replyImagePaths=G);let K=D.enqueue($);v(l,200,{jobId:h,position:K,threadId:u})}function Rn(r,l){l.writeHead(200,{"Content-Type":"text/event-stream","Cache-Control":"no-cache",Connection:"keep-alive"}),l.write(`event: connected
354
+ `)}else D=j.annotations.map(M=>M.instruction||`[${M.type}]`).join("; ");let U=Ce(j,Object.keys(T).length>0?T:void 0);await R.appendMessage(ee,{role:"human",timestamp:Date.now(),jobId:h,screenshotPath:I,...ce?{screenshotPaths:L}:{},...Object.keys(T).length>0?{imagePaths:T}:{},annotationIds:ae,feedbackSummary:D,feedbackContext:U||void 0});let le=N.enqueue($);y(l,200,{jobId:h,position:le,threadId:ee})}async function Ne(r,l){let p=r.headers["content-type"]||"",d,f,u,g,O,F,q=[];if(p.includes("multipart/form-data")){let D=await qe(r),U=D.feedback?JSON.parse(D.feedback):{};d=U.threadId,f=U.reply,u=U.color,g=U.provider,O=U.model,F=U.sourceId||D.sourceId;for(let le of D.pastedImages)q.push(le.data)}else{let D=[];for await(let M of r)D.push(typeof M=="string"?Buffer.from(M):M);let U=Buffer.concat(D).toString("utf-8"),le;try{le=JSON.parse(U)}catch{y(l,400,{error:"Invalid JSON"});return}d=le.threadId,f=le.reply,u=le.color,g=le.provider,O=le.model,F=le.sourceId}if(!d||!f){y(l,400,{error:"Missing threadId or reply"});return}if(!await R.getThread(d)){y(l,404,{error:"Thread not found"});return}let h=Se().slice(0,8),L=[];for(let D=0;D<q.length;D++){let U=me(i,`reply-${h}-${D}.webp`);await Ae(U,q[D]),L.push(U)}let I="";{let D=await R.getThreadHistory(d);for(let U=D.length-1;U>=0;U--)if(D[U].screenshotPath){I=D[U].screenshotPath;break}}let T=await Dn(d);if(!I&&!T){y(l,400,{error:"No screenshot available"});return}await R.appendMessage(d,{role:"human",timestamp:Date.now(),jobId:h,replyToQuestion:f,screenshotPath:I,...L.length>0?{replyImagePaths:L}:{}});let se=await R.getThreadHistory(d),ee=[];for(let D of se)if(D.annotationIds)for(let U of D.annotationIds)ee.includes(U)||ee.push(U);let ae=De(g),ce=un(I,se,ae,L.length>0?L:void 0),$={id:h,status:"queued",screenshotPath:I,feedback:{timestamp:new Date().toISOString(),url:"",viewport:{width:0,height:0},scrollPosition:{x:0,y:0},annotations:[],styleModifications:[]},createdAt:Date.now(),color:u,threadId:d,annotationIds:ee.length>0?ee:void 0,provider:ae,model:O||void 0,sourceId:F||void 0};T&&($.kind=T),$._replyPrompt=ce,L.length>0&&($._replyImagePaths=L);let Y=N.enqueue($);y(l,200,{jobId:h,position:Y,threadId:d})}function Rn(r,l){l.writeHead(200,{"Content-Type":"text/event-stream","Cache-Control":"no-cache",Connection:"keep-alive"}),l.write(`event: connected
355
355
  data: {"status":"connected"}
356
356
 
357
- `),r.headers.origin&&En(r.headers.origin)&&o!==r.headers.origin&&(o=r.headers.origin);let u=new URL(r.url||"/",`http://127.0.0.1:${L}`).searchParams.get("sourceId")||void 0,f={id:Se().slice(0,8),res:l,sourceId:u};_.add(f),r.on("close",()=>{_.delete(f)})}function In(r){let l=D.allActive;v(r,200,{ok:!0,version:yn,restartMode:O,projectId:e,projectName:bn(n),devOrigin:o,activeJob:l[0]?{id:l[0].id,status:l[0].status}:null,activeJobs:l.map(p=>({id:p.id,status:p.status,threadId:p.threadId,annotationIds:p.annotationIds,color:p.color,kind:p.kind})),queueDepth:D.depth,recentJobs:b})}async function Tn(r,l){let p=await new Promise(u=>{let f="";r.on("data",d=>{f+=d.toString()}),r.on("end",()=>u(f))});try{let u=JSON.parse(p);typeof u.devOrigin=="string"&&(o=u.devOrigin||null),v(l,200,{ok:!0,devOrigin:o})}catch{v(l,400,{error:"Invalid JSON"})}}async function Mn(r,l){let u=new URL(r.url||"/",`http://127.0.0.1:${L}`).searchParams.get("jobId"),d=(u?D.allActive.filter(J=>J.id===u):D.allActive).map(J=>J.threadId).filter(Boolean),w=u?D.cancelJob(u):D.cancelActive();for(let J of d)await I.appendMessage(J,{role:"assistant",timestamp:Date.now(),jobId:u||"",cancelled:!0});v(l,200,{cancelled:w})}async function Cn(r,l){if(C.isRunning){v(l,200,{skipped:!0,reason:"Already running"});return}let{provider:p,model:u}=await tt(r);if(p==="copilot"){v(l,400,{error:"Materialization is not available for Copilot yet."});return}let f=await C.getUnmaterializedPatternDecisions();if(f.length===0){v(l,200,{skipped:!0,reason:"No unmaterialized pattern decisions"});return}C.run({provider:p,model:u}).catch(()=>{}),v(l,200,{started:!0,decisionCount:f.length,decisionIds:f.map(d=>d.id)})}async function tt(r){try{let l=[];for await(let u of r)l.push(typeof u=="string"?Buffer.from(u):u);if(l.length===0)return{};let p=JSON.parse(Buffer.concat(l).toString());return{provider:De(typeof p.provider=="string"?p.provider:void 0),model:typeof p.model=="string"&&p.model?p.model:void 0}}catch{return{}}}async function $n(r,l){let p=await C.loadModel();if(!p){v(l,400,{error:"No model exists yet"});return}let u,f;try{let h=[];for await(let R of r)h.push(typeof R=="string"?Buffer.from(R):R);let G=JSON.parse(Buffer.concat(h).toString());u=G.provider,f=G.model}catch{}if(u==="copilot"){v(l,400,{error:"Rule synthesis is not available for Copilot yet."});return}let d=Xt(p),w=Se().slice(0,8),j=(await I.createThread(w,[])).id;H.set(j,"synthesize"),await I.appendMessage(j,{role:"human",timestamp:Date.now(),jobId:w,feedbackSummary:"Synthesize rules \u2014 review and propose improvements"});let Y={id:w,status:"queued",screenshotPath:"",feedback:{timestamp:new Date().toISOString(),url:"",viewport:{width:0,height:0},scrollPosition:{x:0,y:0},annotations:[],styleModifications:[]},createdAt:Date.now(),threadId:j,kind:"synthesize",provider:De(u),model:f||void 0};Y._replyPrompt=d;let F=D.enqueue(Y);v(l,200,{jobId:w,position:F,threadId:j})}async function _n(r,l){let{provider:p,model:u}=await tt(r);if(p==="copilot"){v(l,400,{error:"Eval synthesis is not available for Copilot yet."});return}let f=await k.listDecisionIds(),d=await k.loadDecisions(f),w=await I.load(),J=await C.loadModel(),j=await y.loadSuite(),Y=zt({decisions:d,threads:w,model:J,existingSuite:j}),F=Se().slice(0,8),G=(await I.createThread(F,[])).id;H.set(G,"synthesize_evals"),await I.appendMessage(G,{role:"human",timestamp:Date.now(),jobId:F,feedbackSummary:"Synthesize evals \u2014 infer evals from design history"});let R={id:F,status:"queued",screenshotPath:"",feedback:{timestamp:new Date().toISOString(),url:"",viewport:{width:0,height:0},scrollPosition:{x:0,y:0},annotations:[],styleModifications:[]},createdAt:Date.now(),color:"#6b7280",threadId:G,kind:"synthesize_evals",provider:p,model:u||void 0};R._replyPrompt=Y;let T=D.enqueue(R);v(l,200,{jobId:F,position:T,threadId:G})}async function On(r,l,p){let u=decodeURIComponent(r),f=await y.findCase(u);if(!f){v(p,404,{error:"Eval not found"});return}if(f.status!=="approved"){v(p,400,{error:"Only approved evals can be run"});return}let{screenshot:d,feedback:w,provider:J,model:j,sourceId:Y}=await qe(l),F=De(J)??B;if(F==="copilot"){v(p,400,{error:"Eval runs are not available for Copilot yet."});return}let h={};try{h=w?JSON.parse(w):{}}catch{v(p,400,{error:"Invalid eval run context JSON"});return}let G=Se().slice(0,8),R=Se().slice(0,8),T=me(i,`eval-run-${G}.webp`);await Ae(T,d);let se=await y.writeRunScreenshot(G,d),ee=h.viewport,ae=ee&&typeof ee=="object"?ee:{},ce={type:"current_page",url:typeof h.url=="string"?h.url:"",viewport:{width:typeof ae.width=="number"?ae.width:0,height:typeof ae.height=="number"?ae.height:0}},$=await C.loadModel(),K=F==="claude"?Q.name:F==="codex"?Z.name:null,A=F==="codex"?q.length>0:typeof K=="string"&&/chrome|devtools/i.test(K),U=Lt({evalCase:f,scope:ce,model:$,screenshotPath:T,chromeDevtoolsAvailable:A}),M=(await I.createThread(R,[])).id;H.set(M,"run_eval");let oe={timestamp:new Date().toISOString(),url:ce.url,viewport:ce.viewport,scrollPosition:{x:typeof h.scrollX=="number"?h.scrollX:0,y:typeof h.scrollY=="number"?h.scrollY:0},annotations:[],styleModifications:[]};await I.appendMessage(M,{role:"human",timestamp:Date.now(),jobId:R,screenshotPath:T,feedbackSummary:`Run eval \u2014 ${f.title}`,feedbackContext:[`Eval: ${f.title}`,"Scope: current page",`URL: ${ce.url||"unknown"}`,"Assertions:",...f.assertions.map(Xn=>`- ${Xn}`)].join(`
358
- `)});let de={id:R,status:"queued",screenshotPath:T,feedback:oe,createdAt:Date.now(),color:"#111827",threadId:M,kind:"run_eval",evalId:u,evalRunId:G,provider:F,model:j||void 0,sourceId:Y||void 0};de._replyPrompt=U,de._evalRunScope=ce,de._evalScreenshotPath=se;let nt=D.enqueue(de);v(p,200,{jobId:R,runId:G,position:nt,threadId:M,screenshotPath:se})}async function An(r,l,p){let u=decodeURIComponent(r),f={},d,w,J;try{let Y=[];for await(let h of l)Y.push(typeof h=="string"?Buffer.from(h):h);let F=JSON.parse(Buffer.concat(Y).toString());d=F.status,w=F.title,J=F.assertions}catch{v(p,400,{error:"Invalid JSON"});return}if(d!==void 0&&d!=="pending"&&d!=="approved"&&d!=="rejected"){v(p,400,{error:"Invalid status"});return}if(d&&(f.status=d),w!==void 0){if(typeof w!="string"||w.trim().length===0){v(p,400,{error:"Invalid title"});return}f.title=w.trim()}if(J!==void 0){if(!Array.isArray(J)){v(p,400,{error:"Invalid assertions"});return}let Y=J.filter(F=>typeof F=="string").map(F=>F.trim()).filter(Boolean);if(Y.length===0){v(p,400,{error:"Invalid assertions"});return}f.assertions=Y}if(!f.status&&!f.title&&!f.assertions){v(p,400,{error:"No eval changes provided"});return}let j=await y.updateCase(u,f);if(!j){v(p,404,{error:"Eval not found"});return}v(p,200,{eval:j})}async function Dn(r){let l=H.get(r);if(l)return l;let f=(await I.getThread(r))?.messages.find(d=>d.role==="human")?.feedbackSummary??"";if(f.startsWith("Synthesize evals"))return H.set(r,"synthesize_evals"),"synthesize_evals";if(f.startsWith("Synthesize rules"))return H.set(r,"synthesize"),"synthesize";if(f.startsWith("Run eval"))return H.set(r,"run_eval"),"run_eval"}async function Nn(r,l){let p=[];for await(let j of r)p.push(typeof j=="string"?Buffer.from(j):j);let u;if(p.length>0)try{u=JSON.parse(Buffer.concat(p).toString("utf-8")).serverUrl}catch{}let f=[];S.claude?.available&&S.claude.mcp&&!S.claude.mcp.found&&f.push(await on(u)),S.codex?.available&&S.codex.mcp&&!S.codex.mcp.found&&f.push(await rn(u)),S.copilot?.available&&S.copilot.mcp&&!S.copilot.mcp.found&&f.push(await an(u,S.copilot.path??g));let[d,w,J]=await Promise.all([dt(n),ut(n),pt(n,S.copilot?.path??g)]);S.claude&&(S.claude.mcp=d),S.codex&&(S.codex.mcp=w),S.copilot&&(S.copilot.mcp=J),v(l,200,{results:f,capabilities:{providers:S}})}async function Jn(r,l){let p=[];for await(let d of r)p.push(typeof d=="string"?Buffer.from(d):d);let u;try{u=JSON.parse(Buffer.concat(p).toString("utf-8"))}catch{v(l,400,{error:"Invalid JSON"});return}if(!u.name||typeof u.name!="string"){v(l,400,{error:"Missing or invalid name"});return}let f=await C.addComponent(u.name);v(l,200,f)}async function Bn(r,l){let p=[];for await(let d of r)p.push(typeof d=="string"?Buffer.from(d):d);let u;try{u=JSON.parse(Buffer.concat(p).toString("utf-8"))}catch{v(l,400,{error:"Invalid JSON"});return}if(!u.name||typeof u.name!="string"){v(l,400,{error:"Missing or invalid name"});return}let f=await C.removeComponent(u.name);v(l,200,f)}async function Fn(r,l){let p=[];for await(let d of r)p.push(typeof d=="string"?Buffer.from(d):d);let u;try{u=JSON.parse(Buffer.concat(p).toString("utf-8"))}catch{v(l,400,{error:"Invalid JSON"});return}if(!u.path||typeof u.path!="string"||typeof u.value!="string"){v(l,400,{error:"Missing or invalid path/value"});return}let f=await C.updateToken(u.path,u.value);v(l,200,f)}async function jn(r,l){let p=[];for await(let d of r)p.push(typeof d=="string"?Buffer.from(d):d);let u;try{u=JSON.parse(Buffer.concat(p).toString("utf-8"))}catch{v(l,400,{error:"Invalid JSON"});return}if(!u.path||typeof u.path!="string"){v(l,400,{error:"Missing or invalid path"});return}let f=await C.removeToken(u.path);v(l,200,f)}function Un(r,l){let p=o??"http://localhost:3000";if(!o){let f=r.headers.referer||r.headers.origin;if(f)try{let d=new URL(typeof f=="string"?f:f[0]||"");(d.hostname==="localhost"||d.hostname==="127.0.0.1")&&(p=d.origin)}catch{}}let u=St(L,p);l.writeHead(200,{"Content-Type":"text/html; charset=utf-8"}),l.end(u)}async function zn(r){let l=Date.now();if(E&&l<E.expires){v(r,200,E.data);return}try{let{scanForComponents:p}=await import("./react-scanner-ZXYS5M3Y.mjs"),{generateRenderFiles:u}=await import("./render-generator-EANIDD2E.mjs"),f=await p(n);E={data:f,expires:l+5e3},u(f,n,ie).then(d=>{ie=d}).catch(d=>console.warn("[Bridge] Render generation failed:",d)),v(r,200,f)}catch(p){console.error("[Bridge] Scanner error:",p),v(r,500,{error:"Failed to scan components"})}}async function Ln(r){let l=[me(n,"node_modules","@popmelt.com","core","dist","canvas.mjs"),me(n,"packages","popmelt","dist","canvas.mjs")];try{let p=Ro(import.meta.url);l.unshift(me(Eo(p),"canvas.mjs"))}catch{}for(let p of l)try{let u=await Sn(p,"utf-8");r.writeHead(200,{"Content-Type":"application/javascript; charset=utf-8","Access-Control-Allow-Origin":"*"}),r.end(u);return}catch{}console.error("[Bridge] Canvas bundle not found in:",l),v(r,404,{error:"Canvas bundle not found"})}async function Wn(r,l){if(!r||r.includes("/")||r.includes("\\")||r.includes("..")){v(l,400,{error:"Invalid filename"});return}try{let p=await Sn(me(i,r)),u=r.split(".").pop()?.toLowerCase(),f=u==="png"?"image/png":u==="webp"?"image/webp":u==="jpg"||u==="jpeg"?"image/jpeg":"application/octet-stream";l.writeHead(200,{"Content-Type":f,"Cache-Control":"public, max-age=3600"}),l.end(p)}catch{v(l,404,{error:"File not found"})}}function Je(r){return`/files/${bn(r)}`}async function Hn(r,l){let p=await I.getThread(r);if(!p){v(l,404,{error:"Thread not found"});return}let u=(await y.loadRuns()).runs.filter(d=>d.threadId===r).sort((d,w)=>w.completedAt-d.completedAt)[0],f=p.messages.map(({screenshotPath:d,screenshotPaths:w,imagePaths:J,replyImagePaths:j,...Y})=>({...Y,...Y.role==="assistant"&&u?{evalRun:u}:{},...d?{screenshotUrl:Je(d)}:{},...w?{screenshotUrls:Object.fromEntries(Object.entries(w).map(([F,h])=>[F,Je(h)]))}:{},...J?{imageUrls:Object.fromEntries(Object.entries(J).map(([F,h])=>[F,h.map(Je)]))}:{},...j?{replyImageUrls:j.map(Je)}:{}}));v(l,200,{id:p.id,createdAt:p.createdAt,messages:f})}let yt=9,wt=!1;for(let r=t;r<t+yt;r++)try{await Do(re,r),L=r,wt=!0,console.log(`[\u22B9 is watching :${L}]`);break}catch(l){if(l.code==="EADDRINUSE"){let p=await Ao(r);if(p&&p.projectId===e)return console.log(`[\u22B9 already watching :${r}]`),{port:r,projectId:e,close:async()=>{}};continue}throw l}if(!wt)throw new Error(`[Bridge] All ports ${t}\u2013${t+yt-1} in use`);for(let[r,l]of Object.entries(S))!l.available||!l.path||W(r,l.path).then(p=>{if(p)console.log(`[Bridge] ${r} warmed up`);else{console.warn(`[Bridge] ${r} warm-up failed \u2014 marking unavailable`),l.available=!1,l.path=null;for(let u of _)gt(u,{type:"capabilities_changed",data:{}})}});let Gn=setInterval(()=>{kn(i).catch(()=>{})},Mo);return{port:L,projectId:e,close:async()=>{clearInterval(Gn),await D.destroyAsync();for(let r of _)try{r.res.end()}catch{}return _.clear(),new Promise(r=>{re.close(()=>r())})}}}async function kn(s){try{let t=await So(s),n=Date.now();for(let e of t){let o=me(s,e);try{let i=await bo(o);n-i.mtimeMs>Co&&await xo(o)}catch{}}}catch{}}var fe="\x1B[35m[popmelt]\x1B[0m";async function Uo(){let s=process.argv.slice(2);if(s[0]==="wrap"){let t=s.indexOf("--");(t===-1||t===s.length-1)&&(console.error(`${fe} Usage: popmelt wrap -- <dev command>`),console.error(`${fe} Example: popmelt wrap -- next dev`),console.error(`${fe} Example: popmelt wrap -- astro dev`),process.exit(1));let n=s.slice(t+1);await Wo(n);return}if(s[0]==="bridge"){await zo();return}if(s[0]==="stop"){await Lo();return}console.log(`${fe} Popmelt \u2014 design collaboration for AI coding agents`),console.log(""),console.log(" popmelt wrap -- <command> Start bridge + dev server together"),console.log(" popmelt bridge Start the bridge server standalone"),console.log(" popmelt stop Stop a detached bridge"),console.log(""),console.log("Examples:"),console.log(" popmelt wrap -- next dev"),console.log(" popmelt wrap -- astro dev"),console.log(" popmelt wrap -- vite"),console.log(""),console.log("In package.json:"),console.log(' "scripts": { "dev": "popmelt wrap -- next dev" }')}async function zo(){let s=await vt({projectRoot:process.cwd()});console.log(`${fe} Bridge running on http://localhost:${s.port}`),await new Promise(t=>{let n=async()=>{console.log(`
359
- ${fe} Shutting down bridge...`),await s.close(),t()};process.on("SIGINT",n),process.on("SIGTERM",n)})}async function Lo(){let s=jo(process.cwd(),".popmelt","bridge.lock"),t;try{t=JSON.parse(await Bo(s,"utf8"))}catch{console.log(`${fe} No bridge running (no .popmelt/bridge.lock found)`);return}try{process.kill(t.pid,"SIGTERM"),console.log(`${fe} Sent SIGTERM to bridge (pid ${t.pid}, port ${t.port})`)}catch(n){if(n.code==="ESRCH")console.log(`${fe} Bridge process ${t.pid} already dead`);else throw n}try{await Fo(s)}catch{}}async function Wo(s){let t=await vt({projectRoot:process.cwd()});console.log(`${fe} Bridge running on http://localhost:${t.port}`);let[n,...e]=s;console.log(`${fe} Starting: ${s.join(" ")}`);let o=Jo(n,e,{stdio:"inherit",shell:!0,env:{...process.env,POPMELT_BRIDGE_URL:`http://localhost:${t.port}`}}),i=a=>{o.kill(a)};process.on("SIGINT",()=>i("SIGINT")),process.on("SIGTERM",()=>i("SIGTERM")),o.on("exit",async(a,c)=>{await t.close(),c?process.kill(process.pid,c):process.exit(a??0)})}Uo().catch(s=>{console.error(`${fe} Fatal:`,s),process.exit(1)});
357
+ `);let d=new URL(r.url||"/",`http://127.0.0.1:${W}`).searchParams.get("sourceId")||void 0,f={id:Se().slice(0,8),res:l,sourceId:d};_.add(f),r.on("close",()=>{_.delete(f)})}function In(r){let l=N.allActive;y(r,200,{ok:!0,version:wn,restartMode:A,projectId:e,projectName:gt(n),devOrigin:o,activeJob:l[0]?{id:l[0].id,status:l[0].status}:null,activeJobs:l.map(p=>({id:p.id,status:p.status,threadId:p.threadId,annotationIds:p.annotationIds,color:p.color,kind:p.kind})),queueDepth:N.depth,recentJobs:b})}async function Tn(r,l){let p=await new Promise(d=>{let f="";r.on("data",u=>{f+=u.toString()}),r.on("end",()=>d(f))});try{let d=JSON.parse(p);typeof d.devOrigin=="string"&&(o=d.devOrigin||null),y(l,200,{ok:!0,devOrigin:o})}catch{y(l,400,{error:"Invalid JSON"})}}async function Mn(r,l){let d=new URL(r.url||"/",`http://127.0.0.1:${W}`).searchParams.get("jobId"),u=(d?N.allActive.filter(O=>O.id===d):N.allActive).map(O=>O.threadId).filter(Boolean),g=d?N.cancelJob(d):N.cancelActive();for(let O of u)await R.appendMessage(O,{role:"assistant",timestamp:Date.now(),jobId:d||"",cancelled:!0});y(l,200,{cancelled:g})}async function Cn(r,l){if(C.isRunning){y(l,200,{skipped:!0,reason:"Already running"});return}let{provider:p,model:d}=await tt(r);if(p==="copilot"){y(l,400,{error:"Materialization is not available for Copilot yet."});return}let f=await C.getUnmaterializedPatternDecisions();if(f.length===0){y(l,200,{skipped:!0,reason:"No unmaterialized pattern decisions"});return}C.run({provider:p,model:d}).catch(()=>{}),y(l,200,{started:!0,decisionCount:f.length,decisionIds:f.map(u=>u.id)})}async function tt(r){try{let l=[];for await(let d of r)l.push(typeof d=="string"?Buffer.from(d):d);if(l.length===0)return{};let p=JSON.parse(Buffer.concat(l).toString());return{provider:De(typeof p.provider=="string"?p.provider:void 0),model:typeof p.model=="string"&&p.model?p.model:void 0}}catch{return{}}}async function $n(r,l){let p=await C.loadModel();if(!p){y(l,400,{error:"No model exists yet"});return}let d,f;try{let h=[];for await(let I of r)h.push(typeof I=="string"?Buffer.from(I):I);let L=JSON.parse(Buffer.concat(h).toString());d=L.provider,f=L.model}catch{}if(d==="copilot"){y(l,400,{error:"Rule synthesis is not available for Copilot yet."});return}let u=qt(p),g=Se().slice(0,8),F=(await R.createThread(g,[])).id;G.set(F,"synthesize"),await R.appendMessage(F,{role:"human",timestamp:Date.now(),jobId:g,feedbackSummary:"Synthesize rules \u2014 review and propose improvements"});let q={id:g,status:"queued",screenshotPath:"",feedback:{timestamp:new Date().toISOString(),url:"",viewport:{width:0,height:0},scrollPosition:{x:0,y:0},annotations:[],styleModifications:[]},createdAt:Date.now(),threadId:F,kind:"synthesize",provider:De(d),model:f||void 0};q._replyPrompt=u;let j=N.enqueue(q);y(l,200,{jobId:g,position:j,threadId:F})}async function _n(r,l){let{provider:p,model:d}=await tt(r);if(p==="copilot"){y(l,400,{error:"Eval synthesis is not available for Copilot yet."});return}let f=await k.listDecisionIds(),u=await k.loadDecisions(f),g=await R.load(),O=await C.loadModel(),F=await w.loadSuite(),q=Lt({decisions:u,threads:g,model:O,existingSuite:F}),j=Se().slice(0,8),L=(await R.createThread(j,[])).id;G.set(L,"synthesize_evals"),await R.appendMessage(L,{role:"human",timestamp:Date.now(),jobId:j,feedbackSummary:"Synthesize evals \u2014 infer evals from design history"});let I={id:j,status:"queued",screenshotPath:"",feedback:{timestamp:new Date().toISOString(),url:"",viewport:{width:0,height:0},scrollPosition:{x:0,y:0},annotations:[],styleModifications:[]},createdAt:Date.now(),color:"#6b7280",threadId:L,kind:"synthesize_evals",provider:p,model:d||void 0};I._replyPrompt=q;let T=N.enqueue(I);y(l,200,{jobId:j,position:T,threadId:L})}async function On(r,l,p){let d=decodeURIComponent(r),f=await w.findCase(d);if(!f){y(p,404,{error:"Eval not found"});return}if(f.status!=="approved"){y(p,400,{error:"Only approved evals can be run"});return}let{screenshot:u,feedback:g,provider:O,model:F,sourceId:q}=await qe(l),j=De(O)??B;if(j==="copilot"){y(p,400,{error:"Eval runs are not available for Copilot yet."});return}let h={};try{h=g?JSON.parse(g):{}}catch{y(p,400,{error:"Invalid eval run context JSON"});return}let L=Se().slice(0,8),I=Se().slice(0,8),T=me(i,`eval-run-${L}.webp`);await Ae(T,u);let se=await w.writeRunScreenshot(L,u),ee=h.viewport,ae=ee&&typeof ee=="object"?ee:{},ce={type:"current_page",url:typeof h.url=="string"?h.url:"",viewport:{width:typeof ae.width=="number"?ae.width:0,height:typeof ae.height=="number"?ae.height:0}},$=await C.loadModel(),Y=j==="claude"?K.name:j==="codex"?Z.name:null,D=j==="codex"?V.length>0:typeof Y=="string"&&/chrome|devtools/i.test(Y),U=Wt({evalCase:f,scope:ce,model:$,screenshotPath:T,chromeDevtoolsAvailable:D}),M=(await R.createThread(I,[])).id;G.set(M,"run_eval");let oe={timestamp:new Date().toISOString(),url:ce.url,viewport:ce.viewport,scrollPosition:{x:typeof h.scrollX=="number"?h.scrollX:0,y:typeof h.scrollY=="number"?h.scrollY:0},annotations:[],styleModifications:[]};await R.appendMessage(M,{role:"human",timestamp:Date.now(),jobId:I,screenshotPath:T,feedbackSummary:`Run eval \u2014 ${f.title}`,feedbackContext:[`Eval: ${f.title}`,"Scope: current page",`URL: ${ce.url||"unknown"}`,"Assertions:",...f.assertions.map(Xn=>`- ${Xn}`)].join(`
358
+ `)});let de={id:I,status:"queued",screenshotPath:T,feedback:oe,createdAt:Date.now(),color:"#111827",threadId:M,kind:"run_eval",evalId:d,evalRunId:L,provider:j,model:F||void 0,sourceId:q||void 0};de._replyPrompt=U,de._evalRunScope=ce,de._evalScreenshotPath=se;let nt=N.enqueue(de);y(p,200,{jobId:I,runId:L,position:nt,threadId:M,screenshotPath:se})}async function An(r,l,p){let d=decodeURIComponent(r),f={},u,g,O;try{let q=[];for await(let h of l)q.push(typeof h=="string"?Buffer.from(h):h);let j=JSON.parse(Buffer.concat(q).toString());u=j.status,g=j.title,O=j.assertions}catch{y(p,400,{error:"Invalid JSON"});return}if(u!==void 0&&u!=="pending"&&u!=="approved"&&u!=="rejected"){y(p,400,{error:"Invalid status"});return}if(u&&(f.status=u),g!==void 0){if(typeof g!="string"||g.trim().length===0){y(p,400,{error:"Invalid title"});return}f.title=g.trim()}if(O!==void 0){if(!Array.isArray(O)){y(p,400,{error:"Invalid assertions"});return}let q=O.filter(j=>typeof j=="string").map(j=>j.trim()).filter(Boolean);if(q.length===0){y(p,400,{error:"Invalid assertions"});return}f.assertions=q}if(!f.status&&!f.title&&!f.assertions){y(p,400,{error:"No eval changes provided"});return}let F=await w.updateCase(d,f);if(!F){y(p,404,{error:"Eval not found"});return}y(p,200,{eval:F})}async function Dn(r){let l=G.get(r);if(l)return l;let f=(await R.getThread(r))?.messages.find(u=>u.role==="human")?.feedbackSummary??"";if(f.startsWith("Synthesize evals"))return G.set(r,"synthesize_evals"),"synthesize_evals";if(f.startsWith("Synthesize rules"))return G.set(r,"synthesize"),"synthesize";if(f.startsWith("Run eval"))return G.set(r,"run_eval"),"run_eval"}async function Nn(r,l){let p=[];for await(let F of r)p.push(typeof F=="string"?Buffer.from(F):F);let d;if(p.length>0)try{d=JSON.parse(Buffer.concat(p).toString("utf-8")).serverUrl}catch{}let f=[];S.claude?.available&&S.claude.mcp&&!S.claude.mcp.found&&f.push(await rn(d)),S.codex?.available&&S.codex.mcp&&!S.codex.mcp.found&&f.push(await an(d)),S.copilot?.available&&S.copilot.mcp&&!S.copilot.mcp.found&&f.push(await cn(d,S.copilot.path??v));let[u,g,O]=await Promise.all([dt(n),ut(n),pt(n,S.copilot?.path??v)]);S.claude&&(S.claude.mcp=u),S.codex&&(S.codex.mcp=g),S.copilot&&(S.copilot.mcp=O),y(l,200,{results:f,capabilities:{providers:S}})}async function Jn(r,l){let p=[];for await(let u of r)p.push(typeof u=="string"?Buffer.from(u):u);let d;try{d=JSON.parse(Buffer.concat(p).toString("utf-8"))}catch{y(l,400,{error:"Invalid JSON"});return}if(!d.name||typeof d.name!="string"){y(l,400,{error:"Missing or invalid name"});return}let f=await C.addComponent(d.name);y(l,200,f)}async function Bn(r,l){let p=[];for await(let u of r)p.push(typeof u=="string"?Buffer.from(u):u);let d;try{d=JSON.parse(Buffer.concat(p).toString("utf-8"))}catch{y(l,400,{error:"Invalid JSON"});return}if(!d.name||typeof d.name!="string"){y(l,400,{error:"Missing or invalid name"});return}let f=await C.removeComponent(d.name);y(l,200,f)}async function Fn(r,l){let p=[];for await(let u of r)p.push(typeof u=="string"?Buffer.from(u):u);let d;try{d=JSON.parse(Buffer.concat(p).toString("utf-8"))}catch{y(l,400,{error:"Invalid JSON"});return}if(!d.path||typeof d.path!="string"||typeof d.value!="string"){y(l,400,{error:"Missing or invalid path/value"});return}let f=await C.updateToken(d.path,d.value);y(l,200,f)}async function jn(r,l){let p=[];for await(let u of r)p.push(typeof u=="string"?Buffer.from(u):u);let d;try{d=JSON.parse(Buffer.concat(p).toString("utf-8"))}catch{y(l,400,{error:"Invalid JSON"});return}if(!d.path||typeof d.path!="string"){y(l,400,{error:"Missing or invalid path"});return}let f=await C.removeToken(d.path);y(l,200,f)}function Un(r,l){let p=o??"http://localhost:3000";if(!o){let f=r.headers.referer||r.headers.origin;if(f)try{let u=new URL(typeof f=="string"?f:f[0]||"");(u.hostname==="localhost"||u.hostname==="127.0.0.1")&&(p=u.origin)}catch{}}let d=bt(W,p);l.writeHead(200,{"Content-Type":"text/html; charset=utf-8"}),l.end(d)}async function zn(r){let l=Date.now();if(E&&l<E.expires){y(r,200,E.data);return}try{let{scanForComponents:p}=await import("./react-scanner-ZXYS5M3Y.mjs"),{generateRenderFiles:d}=await import("./render-generator-EANIDD2E.mjs"),f=await p(n);E={data:f,expires:l+5e3},d(f,n,ie).then(u=>{ie=u}).catch(u=>console.warn("[Bridge] Render generation failed:",u)),y(r,200,f)}catch(p){console.error("[Bridge] Scanner error:",p),y(r,500,{error:"Failed to scan components"})}}async function Ln(r){let l=[me(n,"node_modules","@popmelt.com","core","dist","canvas.mjs"),me(n,"packages","popmelt","dist","canvas.mjs")];try{let p=Ro(import.meta.url);l.unshift(me(Eo(p),"canvas.mjs"))}catch{}for(let p of l)try{let d=await bn(p,"utf-8");r.writeHead(200,{"Content-Type":"application/javascript; charset=utf-8","Access-Control-Allow-Origin":"*"}),r.end(d);return}catch{}console.error("[Bridge] Canvas bundle not found in:",l),y(r,404,{error:"Canvas bundle not found"})}async function Wn(r,l){if(!r||r.includes("/")||r.includes("\\")||r.includes("..")){y(l,400,{error:"Invalid filename"});return}try{let p=await bn(me(i,r)),d=r.split(".").pop()?.toLowerCase(),f=d==="png"?"image/png":d==="webp"?"image/webp":d==="jpg"||d==="jpeg"?"image/jpeg":"application/octet-stream";l.writeHead(200,{"Content-Type":f,"Cache-Control":"public, max-age=3600"}),l.end(p)}catch{y(l,404,{error:"File not found"})}}function Je(r){return`/files/${gt(r)}`}async function Hn(r,l){let p=await R.getThread(r);if(!p){y(l,404,{error:"Thread not found"});return}let d=(await w.loadRuns()).runs.filter(g=>g.threadId===r).sort((g,O)=>O.completedAt-g.completedAt)[0],f=d?await w.findCase(d.evalId):null,u=p.messages.map(({screenshotPath:g,screenshotPaths:O,imagePaths:F,replyImagePaths:q,...j})=>({...j,...j.role==="assistant"&&d?{evalRun:{...d,title:f?.title}}:{},...g?{screenshotUrl:Je(g)}:{},...O?{screenshotUrls:Object.fromEntries(Object.entries(O).map(([h,L])=>[h,Je(L)]))}:{},...F?{imageUrls:Object.fromEntries(Object.entries(F).map(([h,L])=>[h,L.map(Je)]))}:{},...q?{replyImageUrls:q.map(Je)}:{}}));y(l,200,{id:p.id,createdAt:p.createdAt,messages:u})}let wt=9,St=!1;for(let r=t;r<t+wt;r++)try{await No(re,r),W=r,St=!0,console.log(`[\u22B9 is watching :${W}]`);break}catch(l){if(l.code==="EADDRINUSE"){let p=await Do(r);if(p&&p.projectId===e)return console.log(`[\u22B9 already watching :${r}]`),{port:r,projectId:e,close:async()=>{}};continue}throw l}if(!St)throw new Error(`[Bridge] All ports ${t}\u2013${t+wt-1} in use`);for(let[r,l]of Object.entries(S))!l.available||!l.path||H(r,l.path).then(p=>{if(p)console.log(`[Bridge] ${r} warmed up`);else{console.warn(`[Bridge] ${r} warm-up failed \u2014 marking unavailable`),l.available=!1,l.path=null;for(let d of _)vt(d,{type:"capabilities_changed",data:{}})}});let Gn=setInterval(()=>{kn(i).catch(()=>{})},Mo);return{port:W,projectId:e,close:async()=>{clearInterval(Gn),await N.destroyAsync();for(let r of _)try{r.res.end()}catch{}return _.clear(),new Promise(r=>{re.close(()=>r())})}}}async function kn(s){try{let t=await So(s),n=Date.now();for(let e of t){let o=me(s,e);try{let i=await bo(o);n-i.mtimeMs>Co&&await xo(o)}catch{}}}catch{}}var fe="\x1B[35m[popmelt]\x1B[0m";async function Uo(){let s=process.argv.slice(2);if(s[0]==="wrap"){let t=s.indexOf("--");(t===-1||t===s.length-1)&&(console.error(`${fe} Usage: popmelt wrap -- <dev command>`),console.error(`${fe} Example: popmelt wrap -- next dev`),console.error(`${fe} Example: popmelt wrap -- astro dev`),process.exit(1));let n=s.slice(t+1);await Wo(n);return}if(s[0]==="bridge"){await zo();return}if(s[0]==="stop"){await Lo();return}console.log(`${fe} Popmelt \u2014 design collaboration for AI coding agents`),console.log(""),console.log(" popmelt wrap -- <command> Start bridge + dev server together"),console.log(" popmelt bridge Start the bridge server standalone"),console.log(" popmelt stop Stop a detached bridge"),console.log(""),console.log("Examples:"),console.log(" popmelt wrap -- next dev"),console.log(" popmelt wrap -- astro dev"),console.log(" popmelt wrap -- vite"),console.log(""),console.log("In package.json:"),console.log(' "scripts": { "dev": "popmelt wrap -- next dev" }')}async function zo(){let s=await yt({projectRoot:process.cwd()});console.log(`${fe} Bridge running on http://localhost:${s.port}`),await new Promise(t=>{let n=async()=>{console.log(`
359
+ ${fe} Shutting down bridge...`),await s.close(),t()};process.on("SIGINT",n),process.on("SIGTERM",n)})}async function Lo(){let s=jo(process.cwd(),".popmelt","bridge.lock"),t;try{t=JSON.parse(await Bo(s,"utf8"))}catch{console.log(`${fe} No bridge running (no .popmelt/bridge.lock found)`);return}try{process.kill(t.pid,"SIGTERM"),console.log(`${fe} Sent SIGTERM to bridge (pid ${t.pid}, port ${t.port})`)}catch(n){if(n.code==="ESRCH")console.log(`${fe} Bridge process ${t.pid} already dead`);else throw n}try{await Fo(s)}catch{}}async function Wo(s){let t=await yt({projectRoot:process.cwd()});console.log(`${fe} Bridge running on http://localhost:${t.port}`);let[n,...e]=s;console.log(`${fe} Starting: ${s.join(" ")}`);let o=Jo(n,e,{stdio:"inherit",shell:!0,env:{...process.env,POPMELT_BRIDGE_URL:`http://localhost:${t.port}`}}),i=a=>{o.kill(a)};process.on("SIGINT",()=>i("SIGINT")),process.on("SIGTERM",()=>i("SIGTERM")),o.on("exit",async(a,c)=>{await t.close(),c?process.kill(process.pid,c):process.exit(a??0)})}Uo().catch(s=>{console.error(`${fe} Fatal:`,s),process.exit(1)});