@popmelt.com/core 0.6.9 → 0.7.0
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/bridge-entry.mjs +39 -36
- package/dist/cli.mjs +39 -36
- package/dist/index.mjs +22 -22
- package/dist/plugin-astro.mjs +1 -1
- package/dist/plugin-next.mjs +1 -1
- package/dist/plugin-vite.mjs +1 -1
- package/dist/server-TB2KIKX7.mjs +228 -0
- package/dist/server.d.mts +1 -0
- package/dist/server.mjs +37 -34
- package/package.json +3 -3
- package/dist/server-55K3KMLN.mjs +0 -225
package/dist/bridge-entry.mjs
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import{mkdir as
|
|
1
|
+
import{mkdir as Vs,unlink as Zs,writeFile as eo}from"fs/promises";import{join as Vt}from"path";import{execFileSync as Os,spawn as qt}from"child_process";import{createHash as _s,randomUUID as Ue}from"crypto";import{mkdir as As,readFile as Xt,readdir as Ds,stat as Ns,unlink as Fs,writeFile as je}from"fs/promises";import{createServer as Bs}from"http";import{tmpdir as Js}from"os";import{basename as Us,dirname as js,join as ue}from"path";import{fileURLToPath as Ls}from"url";function at(s,t){return`<!DOCTYPE html>
|
|
2
2
|
<html>
|
|
3
3
|
<head>
|
|
4
4
|
<meta charset="utf-8">
|
|
@@ -25,23 +25,24 @@ import{mkdir as ws,unlink as vs,writeFile as Ss}from"fs/promises";import{join as
|
|
|
25
25
|
});
|
|
26
26
|
</script>
|
|
27
27
|
</body>
|
|
28
|
-
</html>`}import{spawn as
|
|
29
|
-
\u2026[truncated]`:e}function
|
|
30
|
-
`)
|
|
31
|
-
`)}
|
|
32
|
-
`)
|
|
33
|
-
|
|
34
|
-
`)
|
|
35
|
-
`),
|
|
28
|
+
</html>`}import{spawn as xn}from"child_process";import{createInterface as bn}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"]),ct=1e5;function Pn(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>ct?e.slice(0,ct)+`
|
|
29
|
+
\u2026[truncated]`:e}function Me(s,t){let{prompt:n,projectRoot:e,maxTurns:o=40,maxBudgetUsd:i=1,allowedTools:c=["Read","Edit","Write","Glob","Grep","Bash"],claudePath:a="claude",resumeSessionId:m,model:v,timeoutMs:y=3e5,onEvent:B}=t,N=[];m?N.push("--resume",m,"-p",n):N.push("-p",n),N.push("--output-format","stream-json","--verbose","--max-turns",String(o),"--max-budget-usd",String(i)),v&&N.push("--model",v);for(let h of c)N.push("--allowedTools",h);let R=xn(a,N,{cwd:e,stdio:["ignore","pipe","pipe"],env:{...process.env,ANTHROPIC_API_KEY:void 0,CLAUDECODE:void 0}}),F=new Promise(h=>{let W,H=[],Y=[],q=[],G=!1,I="",z,b,_=!1,u=setTimeout(()=>{_=!0,R.kill("SIGTERM"),setTimeout(()=>{try{R.kill("SIGKILL")}catch{}},5e3)},y),j=bn({input:R.stdout}),E=new Set;j.on("line",C=>{if(C.trim())try{let S=JSON.parse(C);S.session_id&&!W&&(W=S.session_id);let X=S.type??(S.event?.type?`event.${S.event.type}`:"unknown");if(E.add(X),S.type==="result"&&S.result&&H.length===0){let T=typeof S.result=="string"?S.result:"";T&&(H.push(T),B?.({type:"delta",jobId:s,text:T},s))}if(S.type==="assistant"&&Array.isArray(S.message?.content)){typeof S.error=="string"&&(z=S.error);for(let T of S.message.content){if(T.type==="text"&&T.text&&(H.push(T.text),B?.({type:"delta",jobId:s,text:T.text},s)),T.type==="tool_use"&&T.name){let Z=T.input?.file_path||T.input?.path||void 0,ae=Pn(T);if(B?.({type:"tool_use",jobId:s,tool:T.name,...Z?{file:Z}:{},...ae?{content:ae}:{}},s),T.name==="Edit"&&T.input?.file_path){let oe={tool:"Edit",file_path:T.input.file_path,old_string:T.input.old_string,new_string:T.input.new_string,replace_all:T.input.replace_all};Y.push(oe),q.push({sequence:q.length,occurredAt:Date.now(),provider:"claude",source:"claude_tool_use",tool:"Edit",operation:"modify",filePath:oe.file_path,oldString:typeof oe.old_string=="string"?oe.old_string:void 0,newString:typeof oe.new_string=="string"?oe.new_string:void 0,replaceAll:typeof oe.replace_all=="boolean"?oe.replace_all:void 0})}else T.name==="Write"&&T.input?.file_path&&(Y.push({tool:"Write",file_path:T.input.file_path,content:T.input.content}),q.push({sequence:q.length,occurredAt:Date.now(),provider:"claude",source:"claude_tool_use",tool:"Write",operation:"create",filePath:T.input.file_path,content:typeof T.input.content=="string"?T.input.content:void 0}))}T.type==="thinking"&&T.thinking&&B?.({type:"thinking",jobId:s,text:T.thinking},s)}}S.type==="result"&&S.is_error===!0&&(G=!0,typeof S.error=="string"&&(z=S.error),typeof S.result=="string"&&(b=S.result)),S.type==="user"&&S.tool_use_result?.file?.filePath&&B?.({type:"tool_use",jobId:s,tool:"Read",file:S.tool_use_result.file.filePath},s)}catch{}});let w=[];R.stderr?.on("data",C=>{w.push(C.toString())}),R.on("close",C=>{if(clearTimeout(u),j.close(),H.length===0&&E.size>0&&console.warn(`[Claude:${s}] No text captured. Event types seen: ${[...E].join(", ")}`),_)G=!0,I=`Timed out after ${Math.round(y/6e4)} minutes`;else if(C!==0&&C!==null){G=!0;let X=w.join("").trim(),T=H.join("").trim(),Z=H.length===0&&E.size>0?` (no text captured, event types: ${[...E].join(", ")})`:"";I=X||b||T||`Claude process exited with code ${C}${Z}`}G&&!I&&b&&(I=b);let S=Tn(I,z);h({sessionId:W,text:H.join(""),success:!G,error:G?S.message:void 0,errorCode:G?S.code:void 0,recoverableError:G?S.recoverable:void 0,fileEdits:Y.length>0?Y:void 0,editEvents:q.length>0?q:void 0})}),R.on("error",C=>{clearTimeout(u),G=!0,I=C.message,h({sessionId:W,text:H.join(""),success:!1,error:I,errorCode:"claude_spawn_error",recoverableError:!1,fileEdits:Y.length>0?Y:void 0,editEvents:q.length>0?q:void 0})})});return{process:R,result:F}}function Tn(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 En,spawnSync as ht}from"child_process";import{readFileSync as In,statSync as Mn}from"fs";import{basename as Rn,extname as Cn,isAbsolute as $n,relative as mt,resolve as On}from"path";import{createInterface as _n}from"readline";function lt(s){let t=s.match(/^\/bin\/(?:ba)?sh\s+-\w+\s+"(.*)"$/s);return t?t[1].replace(/\\"/g,'"').replace(/\\\\/g,"\\"):s}var dt=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"]),ze=2e5,An=1e5,gt=2e5;function yt(s){let t=Rn(s).toLowerCase(),n=Cn(s).toLowerCase();return dt.has(n)||dt.has(t)}function wt(s,t){let n=$n(t)?t:On(s,t),e=mt(s,n);return e===""||e.startsWith("..")?null:n}function Re(s,t){if(!yt(t))return;let n=wt(s,t);if(n)try{let e=Mn(n);return e.isFile()?e.size>ze?{exists:!0,truncated:!0}:{exists:!0,content:In(n,"utf8")}:{exists:!1}}catch{return{exists:!1}}}function Dn(s,t){for(let n of t){let e=s[n];if(typeof e=="string")return e}}function Nn(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 ut(s){return Dn(s,["patch","diff","unifiedDiff","unified_diff"])}function We(s,t){return s.length<=t?{text:s}:{text:s.slice(0,t),truncated:!0}}function pt(s,t){let n=ht("git",t,{cwd:s,encoding:"utf8",timeout:5e3,maxBuffer:gt+65536});return n.status!==0?null:n.stdout??""}function Fn(s){let t=new Set;for(let n of s.split(`
|
|
30
|
+
`)){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 ft(s){let t=pt(s,["status","--porcelain=v1","-uall"]);if(t===null)return;let n=pt(s,["diff","--no-ext-diff","HEAD","--"])??"",e=We(n,gt);return{status:t,changedFiles:Fn(t),...e.text?{diff:e.text}:{},...e.truncated?{diffTruncated:!0}:{}}}function Bn(s,t){return!s||!t?!1:s.status!==t.status||(s.diff??"")!==(t.diff??"")}function Jn(s,t){if(!yt(t))return;let n=wt(s,t);if(!n)return;let e=mt(s,n);if(e===""||e.startsWith(".."))return;let o=ht("git",["show",`HEAD:${e}`],{cwd:s,encoding:"utf8",timeout:5e3,maxBuffer:ze+65536});if(o.status!==0)return{exists:!1};let i=o.stdout??"",c=We(i,ze);return{exists:!0,content:c.text,...c.truncated?{truncated:!0}:{}}}function Un(s,t,n){return t.map(e=>{let o=n.has(e)?n.get(e):Jn(s,e),i=Re(s,e);return{filePath:e,...o?{before:o}:{},...i?{after:i}:{}}})}function Ce(s,t){let{prompt:n,projectRoot:e,screenshotPath:o,resumeSessionId:i,model:c,onEvent:a,sandboxMode:m="workspace-write"}=t,v=[];i?(v.push("exec","resume","--skip-git-repo-check","--json","-c",`sandbox_mode="${m}"`),c&&v.push("-m",c),v.push(i),v.push(n),o&&v.push("--image",o)):(v.push("exec","--skip-git-repo-check","--json","--sandbox",m),c&&v.push("-m",c),v.push(n),o&&v.push("--image",o));let y=En("codex",v,{cwd:e,stdio:["ignore","pipe","pipe"],env:{...process.env,CLAUDECODE:void 0}}),B=new Promise(N=>{let R,F=[],h=[],W=[],H=new Map,Y=new Map,q=!1,G="",I=_n({input:y.stdout}),z=new Set;I.on("line",_=>{if(_.trim())try{let u=JSON.parse(_),j=u.type??"unknown";if(z.add(j),j==="thread.started"&&u.thread_id&&!R&&(R=u.thread_id),(j==="item.agentMessage.delta"||j==="item/agentMessage/delta")&&u.delta?.text&&(F.push(u.delta.text),a?.({type:"delta",jobId:s,text:u.delta.text},s)),(j==="item.reasoning.delta"||j==="item/reasoning/delta")&&u.delta?.text&&a?.({type:"thinking",jobId:s,text:u.delta.text},s),(j==="item.started"||j==="item/started")&&u.item){let E=u.item.type;if(E==="command_execution"){let w=u.item.command,C=w?lt(w):void 0,S=ft(e),X=new Map;for(let Z of S?.changedFiles??[])X.set(Z,Re(e,Z));typeof u.item.id=="string"&&Y.set(u.item.id,{...w?{rawCommand:w}:{},...C?{command:C}:{},...S?{beforeWorktree:S}:{},beforeSnapshots:X});let T=C?`Bash: ${C.split(`
|
|
31
|
+
`)[0].slice(0,80)}`:"Bash";h.push(T),a?.({type:"tool_use",jobId:s,tool:"Bash",...C?{content:C}:{}},s)}else if(E==="file_change"){let w=u.item.filename||u.item.path;w&&!H.has(w)&&H.set(w,Re(e,w)),h.push(w?`Edit ${w.split("/").pop()}`:"Edit"),a?.({type:"tool_use",jobId:s,tool:"Edit",...w?{file:w}:{}},s)}else if(E==="file_read"){let w=u.item.filename||u.item.path;h.push(w?`Read ${w.split("/").pop()}`:"Read"),a?.({type:"tool_use",jobId:s,tool:"Read",...w?{file:w}:{}},s)}else if(E==="web_search")h.push("WebSearch"),a?.({type:"tool_use",jobId:s,tool:"WebSearch"},s);else if(E==="mcp_tool_call"){let w=u.item.tool_name||u.item.name||"MCP";h.push(w),a?.({type:"tool_use",jobId:s,tool:w},s)}}if((j==="item.completed"||j==="item/completed")&&u.item){if(u.item.type==="agent_message"){let E=u.item.text;typeof E=="string"&&E&&(F.push(E),a?.({type:"delta",jobId:s,text:E},s))}else if(u.item.type==="reasoning"){let E=u.item.text;typeof E=="string"&&E&&a?.({type:"thinking",jobId:s,text:E},s)}else if(u.item.type==="file_change"&&Array.isArray(u.item.changes))for(let E of u.item.changes){let w=E.path||E.filename,C=E.kind==="add"?"Write":"Edit";if(w){h.push(`${C} ${w.split("/").pop()}`),a?.({type:"tool_use",jobId:s,tool:C,file:w},s);let S=E,X=typeof S.kind=="string"?S.kind:void 0,T=Nn(X),Z=H.get(w),ae=Re(e,w);W.push({sequence:W.length,occurredAt:Date.now(),provider:"codex",source:"codex_file_change",tool:"FileChange",operation:T,filePath:w,...Z?{before:Z}:{},...ae?{after:ae}:{},...ut(S)?{patch:ut(S)}:{},...X?{rawKind:X}:{}}),H.set(w,ae)}}else if(u.item.type==="command_execution"){let E=typeof u.item.id=="string"?u.item.id:void 0,w=E?Y.get(E):void 0,C=typeof u.item.command=="string"?u.item.command:w?.rawCommand,S=C?lt(C):w?.command,X=ft(e);if(Bn(w?.beforeWorktree,X)){let T=[...new Set([...w?.beforeWorktree?.changedFiles??[],...X?.changedFiles??[]])],Z=typeof u.item.aggregated_output=="string"?We(u.item.aggregated_output,An):void 0,ae=typeof u.item.exit_code=="number"?u.item.exit_code:u.item.exit_code===null?null:void 0,oe=typeof u.item.status=="string"?u.item.status:void 0;W.push({sequence:W.length,occurredAt:Date.now(),provider:"codex",source:"codex_command_execution",tool:"Bash",operation:"command",...C?{rawCommand:C}:{},...S?{command:S}:{},...ae!==void 0?{exitCode:ae}:{},...oe?{status:oe}:{},...Z?{output:Z.text}:{},...Z?.truncated?{outputTruncated:!0}:{},...w?.beforeWorktree?{beforeWorktree:w.beforeWorktree}:{},...X?{afterWorktree:X}:{},...T.length>0?{touchedFiles:Un(e,T,w?.beforeSnapshots??new Map)}:{}})}E&&Y.delete(E)}}j==="turn.failed"&&(q=!0,G=u.error?.message||u.message||"Turn failed")}catch{}});let b=[];y.stderr?.on("data",_=>{b.push(_.toString())}),y.on("close",_=>{I.close(),F.length===0&&z.size>0&&console.warn(`[Codex:${s}] No text captured. Event types seen: ${[...z].join(", ")}`),_!==0&&_!==null&&(q=!0,G=b.join("")||`Codex process exited with code ${_}`),N({sessionId:R,text:F.join(""),success:!q,error:q?G:void 0,editEvents:W.length>0?W:void 0,toolsUsed:h.length>0?h:void 0})}),y.on("error",_=>{q=!0,G=_.message,N({sessionId:R,text:F.join(""),success:!1,error:G,editEvents:W.length>0?W:void 0,toolsUsed:h.length>0?h:void 0})})});return{process:y,result:B}}import{spawn as jn}from"child_process";import{createInterface as Ln}from"readline";function vt(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 St(s,t){let{prompt:n,projectRoot:e,resumeSessionId:o,model:i,timeoutMs:c=3e5,onEvent:a,copilotPath:m="copilot"}=t,v=["--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&&v.push(`--resume=${o}`),i&&v.push("--model",i),v.push("-p",n);let y=jn(m,v,{cwd:e,stdio:["ignore","pipe","pipe"],env:{...process.env,ANTHROPIC_API_KEY:void 0,CLAUDECODE:void 0}}),B=new Promise(N=>{let R,F=[],h=[],W=!1,H="",Y=!1,q=setTimeout(()=>{Y=!0,y.kill("SIGTERM"),setTimeout(()=>{try{y.kill("SIGKILL")}catch{}},5e3)},c),G=Ln({input:y.stdout}),I=new Set,z=[];G.on("line",_=>{if(_.trim())try{let u=JSON.parse(_),j=Se(u);I.add(j),z.length<5&&Wn(j)&&z.push(_.slice(0,800));let E=ve(u,["sessionId","session_id","id"],["session","conversation","thread"]);E&&!R&&(R=E);let w=zn(u,F.length===0);w&&(F.push(w),a?.({type:"delta",jobId:s,text:w},s));let C=Hn(u);C&&a?.({type:"thinking",jobId:s,text:C},s);let S=Gn(u);S&&(h.push(S.label),a?.({type:"tool_use",jobId:s,tool:S.tool,...S.file?{file:S.file}:{},...S.content?{content:S.content}:{}},s));let X=qn(u);X&&(W=!0,H=X)}catch{}});let b=[];y.stderr?.on("data",_=>{b.push(_.toString())}),y.on("close",_=>{clearTimeout(q),G.close(),F.length===0&&I.size>0&&(console.warn(`[Copilot:${s}] No text captured. Event types seen: ${[...I].join(", ")}`),z.length>0&&console.warn(`[Copilot:${s}] Sample text-bearing candidates: ${z.join(`
|
|
32
|
+
`)}`)),Y?(W=!0,H=`Timed out after ${Math.round(c/6e4)} minutes`):_!==0&&_!==null&&(W=!0,H=b.join("").trim()||H||`Copilot process exited with code ${_}`);let u=W?vt(H,i):null;N({sessionId:R,text:F.join(""),success:!W,error:u?.message,errorCode:u?.code,recoverableError:u?.recoverable,provider:"copilot",toolsUsed:h.length>0?h:void 0})}),y.on("error",_=>{clearTimeout(q);let u=vt(_.message,i);N({sessionId:R,text:F.join(""),success:!1,error:u.message,errorCode:u.code,recoverableError:u.recoverable,provider:"copilot",toolsUsed:h.length>0?h:void 0})})});return{process:y,result:B}}function Se(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 zn(s,t=!1){let n=Se(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 ie(s.delta)??ie(s.content)??ie(s.message);if(n==="assistant.message"&&t)return ie(s.message)??ie(s.content)??ie(s);if(n==="result"&&t)return ie(s.result)??ie(s.output)??ie(s.message)??ie(s);let i=s.content;if(i?.type==="text"&&typeof i.text=="string")return i.text;if(typeof s.text=="string"&&He(s))return s.text;if(typeof s.message=="string"&&He(s))return s.message;if(typeof s.delta=="string"&&He(s))return s.delta}function ie(s){if(typeof s=="string")return s;if(!s)return;if(Array.isArray(s)){let n=s.map(ie).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=ie(t[n]);if(e)return e}}function Wn(s){let t=s.toLowerCase();return t.includes("assistant.message")||t==="result"}function Hn(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 Gn(s){let t=ve(s,["tool","toolName","tool_name","name"]),n=ve(s,["command","cmd"]),e=ve(s,["file","path","filename","filePath"]),o=Se(s);if(n||o.includes("shell"))return{tool:"Bash",label:n?`Bash: ${n.split(`
|
|
33
|
+
`)[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 qn(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"&&Se(s).toLowerCase().includes("error"))return s.message}function ve(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(c=>e.toLowerCase().includes(c)))continue;let i=ve(o,t);if(i)return i}}function He(s){let t=Se(s).toLowerCase();return(typeof s.role=="string"?s.role.toLowerCase():"")==="assistant"||t.includes("assistant")||t.includes("agent_message")}import{execFile as Xn}from"child_process";import{copyFile as xt,mkdir as bt,readdir as Yn,readFile as Qn,writeFile as Kn}from"fs/promises";import{join as ge}from"path";var $e=class{constructor(t){this.projectRoot=t;let n=ge(t,".popmelt");this.decisionsDir=ge(n,"decisions"),this.screenshotsDir=ge(n,"screenshots")}async persist(t,n,e){try{await bt(this.decisionsDir,{recursive:!0}),await bt(this.screenshotsDir,{recursive:!0});try{await xt(n,ge(this.screenshotsDir,`s-${t.id}.webp`))}catch{}for(let o=0;o<e.length;o++)try{let i=t.pastedImagePaths[o];i&&await xt(e[o],ge(this.screenshotsDir,i.replace("screenshots/","")))}catch{}await Kn(ge(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 Yn(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 Qn(ge(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=>{Xn("git",["diff","HEAD"],{cwd:t,timeout:5e3,maxBuffer:1024*1024},(e,o)=>{if(e){n(null);return}n(o||null)})})}};import{readFile as kt,writeFile as he}from"fs/promises";import{join as Ge}from"path";var Q="[Materializer]",Vn={version:1,materializedIds:[],lastRunAt:null,lastRunDecisionIds:[],lastRunError:null};function Zn(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 xe(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(`${Q} 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(`${Q} Rule count ${t.length} exceeds cap of 30`),t}var Oe=class{constructor(t,n,e={}){this.projectRoot=t;this.decisionStore=n;this.options=e;this.cachedIndex=null;this.running=!1;let o=Ge(t,".popmelt");this.indexPath=Ge(o,"materialized.json"),this.modelPath=Ge(o,"model.json")}get isRunning(){return this.running}async loadModel(){try{let t=await kt(this.modelPath,"utf-8"),n=JSON.parse(t);return Array.isArray(n.rules)&&(n.rules=Zn(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 he(this.modelPath,JSON.stringify(n,null,2)),console.log(`${Q} 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 v=o[m];(!i[v]||typeof i[v]!="object")&&(i[v]={}),i=i[v]}let c=o[o.length-1],a;try{a=JSON.parse(n)}catch{a=null}if(a&&typeof a=="object"&&a!==null&&"value"in a)i[c]=a;else{let m=i[c];m&&typeof m=="object"&&m!==null&&"value"in m?m.value=n:i[c]=n}return await he(this.modelPath,JSON.stringify(e,null,2)),console.log(`${Q} 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 c=0;c<e.length-1;c++){let a=e[c];if(!o[a]||typeof o[a]!="object")return{removed:!1};o=o[a]}let i=e[e.length-1];return i in o?(delete o[i],await he(this.modelPath,JSON.stringify(n,null,2)),console.log(`${Q} 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 he(this.modelPath,JSON.stringify(n,null,2)),console.log(`${Q} 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(c=>!n.has(c));return o.length===0?[]:(await this.decisionStore.loadDecisions(o)).filter(c=>c.resolutions.some(a=>(a.finalScope??a.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(y=>y.id);console.log(`${Q} Processing ${e.length} pattern-scoped decision(s): ${e.join(", ")}`),this.options.onEvent?.({type:"materialize_started",decisionIds:e});let o=await this.loadModel(),i=es(n,o),c=!0,a;try{let y=`mat-${Date.now()}`,{provider:B,result:N}=this.spawnModelAgent(y,i,t,{maxTurns:this.options.maxTurns??5,maxBudgetUsd:this.options.maxBudgetUsd??.5,allowedTools:["Read"]}),R=await N;if(!R.success)c=!1,a=R.error,console.error(`${Q} ${B} spawn error:`,a);else{let F=_e(R.text);F?(Array.isArray(F.rules)&&(F.rules=xe(F.rules)),await he(this.modelPath,JSON.stringify(F,null,2)),console.log(`${Q} Successfully materialized ${e.length} decision(s) with ${B} \u2192 ${this.modelPath}`)):(c=!1,a="No <model> block found in response",console.error(`${Q} ${a}`))}}catch(y){c=!1,a=y instanceof Error?y.message:String(y),console.error(`${Q} Error:`,a)}let m=await this.loadIndex(),v=new Set(m.materializedIds);for(let y of e)v.add(y);return m.materializedIds=[...v],m.lastRunAt=Date.now(),m.lastRunDecisionIds=e,m.lastRunError=a??null,await this.persistIndex(m),this.options.onEvent?.({type:"materialize_done",decisionIds:e,success:c,error:a}),{processedIds:e,success:c,error:a}}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=ts(n),{provider:o,result:i}=this.spawnModelAgent(`consolidate-${Date.now()}`,e,t,{maxTurns:this.options.maxTurns??3,maxBudgetUsd:this.options.maxBudgetUsd??.3,allowedTools:[]}),c=await i;if(!c.success)return console.error(`${Q} ${o} consolidation spawn error:`,c.error),{success:!1,error:c.error};let a=_e(c.text);return a?(Array.isArray(a.rules)&&(a.rules=xe(a.rules)),!a.tokens&&n.tokens&&(a.tokens=n.tokens),!a.components&&n.components&&(a.components=n.components),await he(this.modelPath,JSON.stringify(a,null,2)),console.log(`${Q} Consolidation complete with ${o} \u2192 ${this.modelPath}`),{success:!0}):(console.error(`${Q} 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(`${Q} Consolidation error:`,e),{success:!1,error:e}}finally{this.running=!1}}async writeModel(t){Array.isArray(t.rules)&&(t.rules=xe(t.rules)),await he(this.modelPath,JSON.stringify(t,null,2)),console.log(`${Q} Model written \u2192 ${this.modelPath}`)}async loadIndex(){if(this.cachedIndex)return this.cachedIndex;try{let t=await kt(this.indexPath,"utf-8"),n=JSON.parse(t);return this.cachedIndex=n,n}catch{return this.cachedIndex={...Vn,materializedIds:[],lastRunDecisionIds:[]},this.cachedIndex}}async persistIndex(t){this.cachedIndex=t;try{await he(this.indexPath,JSON.stringify(t,null,2))}catch(n){console.error(`${Q} Failed to write index:`,n)}}spawnModelAgent(t,n,e,o){let i=e.provider??this.options.provider??"claude",c=this.options.onEvent?(a,m)=>this.options.onEvent?.(a):void 0;return i==="codex"?{provider:"codex",...Ce(t,{prompt:n,projectRoot:this.projectRoot,model:e.model??this.options.model,timeoutMs:this.options.timeoutMs,sandboxMode:"read-only",onEvent:c})}:(i==="copilot"&&console.warn(`${Q} Copilot materialization is not available yet; falling back to Claude.`),{provider:"claude",...Me(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:c})})}};function _e(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 es(s,t){let n=s.map(i=>{let a=i.resolutions.filter(y=>(y.finalScope??y.inferredScope)?.breadth==="pattern").map(y=>{let N=(y.finalScope??y.inferredScope)?.target??"unknown",R=y.filesModified?.join(", ")??"none";return`- **${y.summary}** [scope: pattern/${N}]
|
|
34
|
+
Files modified: ${R}`}).join(`
|
|
35
|
+
`),m=i.annotations.map(y=>y.instruction).filter(Boolean).join(`
|
|
36
|
+
`),v=i.gitDiff?`
|
|
36
37
|
\`\`\`diff
|
|
37
38
|
${i.gitDiff.slice(0,2e3)}
|
|
38
39
|
\`\`\``:"";return`### Decision ${i.id} (${new Date(i.createdAt).toISOString()})
|
|
39
40
|
Page: ${i.url}
|
|
40
|
-
${
|
|
41
|
-
${
|
|
42
|
-
${
|
|
41
|
+
${a}
|
|
42
|
+
${v}
|
|
43
|
+
${m?`
|
|
43
44
|
Original instructions:
|
|
44
|
-
${
|
|
45
|
+
${m}`:""}`}).join(`
|
|
45
46
|
|
|
46
47
|
`),e=s.map(i=>i.id);return`You are extracting a local design model from accumulated design decisions.
|
|
47
48
|
|
|
@@ -49,6 +50,7 @@ ${h}`:""}`}).join(`
|
|
|
49
50
|
1. Review the current model (if any) and the new decisions below.
|
|
50
51
|
2. Determine what design tokens, component patterns, and rules to add or update.
|
|
51
52
|
3. Output the complete updated model as a JSON object inside <model> tags.
|
|
53
|
+
4. Do not edit files or run commands. Return only the <model> block.
|
|
52
54
|
|
|
53
55
|
${t?`## Current Model
|
|
54
56
|
\`\`\`json
|
|
@@ -114,7 +116,7 @@ Example:
|
|
|
114
116
|
{ "value": "8px", "property": "gap", "bindings": ["gap-2"] }
|
|
115
117
|
- property: "gap", "padding", or "margin" \u2014 which CSS property this token controls
|
|
116
118
|
- bindings: Tailwind class names (without responsive prefixes) that use this token
|
|
117
|
-
Include property and bindings when the decision context has class-level evidence.`}function
|
|
119
|
+
Include property and bindings when the decision context has class-level evidence.`}function ts(s){return`You are consolidating a design model's rules. The model has accumulated too many rules and needs cleanup.
|
|
118
120
|
|
|
119
121
|
## Current Model
|
|
120
122
|
\`\`\`json
|
|
@@ -130,6 +132,7 @@ ${JSON.stringify(s,null,2)}
|
|
|
130
132
|
- Restating tokens or components already defined elsewhere in the model
|
|
131
133
|
- Procedural instructions rather than design constraints
|
|
132
134
|
4. Output the full model with consolidated rules.
|
|
135
|
+
5. Do not edit files or run commands. Return only the <model> block.
|
|
133
136
|
|
|
134
137
|
## Rule format
|
|
135
138
|
Each rule MUST be a JSON object:
|
|
@@ -152,7 +155,7 @@ Output the complete model inside <model> tags. Preserve tokens and components as
|
|
|
152
155
|
|
|
153
156
|
<model>
|
|
154
157
|
{ "tokens": { ... }, "components": { ... }, "rules": [ ... ] }
|
|
155
|
-
</model>`}function
|
|
158
|
+
</model>`}function Pt(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.
|
|
156
159
|
|
|
157
160
|
## Current Model
|
|
158
161
|
\`\`\`json
|
|
@@ -185,42 +188,42 @@ These are my proposed rule changes. Would you like to approve all of them, adjus
|
|
|
185
188
|
</question>
|
|
186
189
|
|
|
187
190
|
## After approval
|
|
188
|
-
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
|
|
189
|
-
`),n=[],e=null;for(let o of t){let i=o.match(
|
|
191
|
+
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 ns}from"child_process";import{readFile as It}from"fs/promises";import{homedir as Xe}from"os";import{join as le}from"path";var Ye=/popmelt/i;function ye(){return{found:!1,name:null,scope:null,disabled:!1}}function we(s,t,n=!1){return{found:!0,name:s,scope:t,disabled:n}}function be(s){for(let t of Object.keys(s))if(Ye.test(t))return t;return null}async function Ae(s){try{let t=await It(s,"utf-8");return JSON.parse(t)}catch{return null}}async function Qe(s){let t=Xe(),n=await Ae(le(t,".claude.json"));if(n&&typeof n=="object"){let o=n;if(o.mcpServers&&typeof o.mcpServers=="object"){let i=be(o.mcpServers);if(i)return we(i,"user")}if(o.projects&&typeof o.projects=="object"){let c=o.projects[s];if(c&&typeof c=="object"){let a=c;if(a.mcpServers&&typeof a.mcpServers=="object"){let m=be(a.mcpServers);if(m){let v=Array.isArray(a.disabledMcpjsonServers)&&a.disabledMcpjsonServers.some(y=>Ye.test(y));return we(m,"project",v)}}}}}let e=await Ae(le(s,".mcp.json"));if(e&&typeof e=="object"){let o=e;if(o.mcpServers&&typeof o.mcpServers=="object"){let c=be(o.mcpServers);if(c){let a=await Tt(s,c);return we(c,"mcp.json",a)}}let i=be(o);if(i&&i!=="mcpServers"){let c=await Tt(s,i);return we(i,"mcp.json",c)}}return ye()}async function Tt(s,t){let n=le(s,".claude","settings.local.json"),e=await Ae(n);if(e&&typeof e=="object"){let o=e;if(Array.isArray(o.disabledMcpjsonServers))return o.disabledMcpjsonServers.some(i=>i===t)}return!1}var ss=/^\[mcp_servers\.([^\]]+)\]/;function os(s){let t=s.split(`
|
|
192
|
+
`),n=[],e=null;for(let o of t){let i=o.match(ss);i?(e&&n.push({name:e.name,body:e.bodyLines.join(`
|
|
190
193
|
`)}),e={name:i[1],bodyLines:[]}):o.startsWith("[")?e&&(n.push({name:e.name,body:e.bodyLines.join(`
|
|
191
194
|
`)}),e=null):e&&e.bodyLines.push(o)}return e&&n.push({name:e.name,body:e.bodyLines.join(`
|
|
192
|
-
`)}),n}function
|
|
193
|
-
`,"utf-8"),{installed:!0,provider:"claude",scope:"user"}}async function
|
|
195
|
+
`)}),n}function rs(s){return/enabled\s*=\s*false/i.test(s)}async function Ke(s){let t=process.env.CODEX_HOME||le(Xe(),".codex"),n=await Et(le(t,"config.toml"),"user");if(n.found)return n;let e=await Et(le(s,".codex","config.toml"),"project");return e.found?e:ye()}async function Ve(s,t="copilot"){let n=await is(t);if(n.found)return n;let e=process.env.COPILOT_HOME||le(Xe(),".copilot"),o=await qe(le(e,"mcp-config.json"),"user");if(o.found)return o;let i=await qe(le(s,".mcp.json"),"mcp.json");if(i.found)return i;let c=await qe(le(s,".github","mcp.json"),"mcp.json");return c.found?c:ye()}async function is(s){try{let t=await new Promise((n,e)=>{ns(s,["mcp","list","--json"],{encoding:"utf-8",timeout:5e3},(o,i)=>{if(o){e(o);return}n(i)})});return Mt(JSON.parse(t),"user")}catch{return ye()}}async function qe(s,t){let n=await Ae(s);return Mt(n,t)}function Mt(s,t){if(!s||typeof s!="object")return ye();let n=s,e=n.mcpServers&&typeof n.mcpServers=="object"?n.mcpServers:n.servers&&typeof n.servers=="object"?n.servers:n,o=be(e);return o?we(o,t):ye()}async function Et(s,t){try{let n=await It(s,"utf-8"),e=os(n);for(let o of e)if(Ye.test(o.name)){let i=rs(o.body);return we(o.name,t,i)}}catch{}return ye()}import{execFile as Rt}from"child_process";import{mkdir as as,readFile as Ct,writeFile as $t}from"fs/promises";import{homedir as Ot}from"os";import{dirname as cs,join as Ze}from"path";var et="https://mcp.popmelt.com/mcp";async function _t(s=et){let t=Ze(Ot(),".claude.json"),n;try{let o=await Ct(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 $t(t,JSON.stringify(n,null,2)+`
|
|
196
|
+
`,"utf-8"),{installed:!0,provider:"claude",scope:"user"}}async function At(s=et){let t=process.env.CODEX_HOME||Ze(Ot(),".codex"),n=Ze(t,"config.toml"),e;try{e=await Ct(n,"utf-8")}catch{e=""}if(/\[mcp_servers\.[^\]]*popmelt[^\]]*\]/i.test(e))return{installed:!1,provider:"codex",scope:null,reason:"already_configured"};await as(cs(n),{recursive:!0});let o=`
|
|
194
197
|
[mcp_servers.popmelt]
|
|
195
198
|
url = "${s}"
|
|
196
|
-
`;return await
|
|
199
|
+
`;return await $t(n,e+o,"utf-8"),{installed:!0,provider:"codex",scope:"user"}}async function Dt(s=et,t="copilot"){let n=await ls(t);if(ds(n))return{installed:!1,provider:"copilot",scope:null,reason:"already_configured"};try{return await new Promise((e,o)=>{Rt(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 ls(s){try{let t=await new Promise((n,e)=>{Rt(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 ds(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 tt(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 us(s),i=Buffer.from(`--${e}`),c=Buffer.from(`--${e}--`),a,m,v,y,B,N,R,F,h,W,H,Y,q=[],G=[],I=0,z=[];for(;I<o.length;){let b=o.indexOf(i,I);if(b===-1)break;let _=b+i.length;if(o.slice(b,b+c.length).equals(c))break;let u=_;o[u]===13&&o[u+1]===10&&(u+=2);let j=o.indexOf(`\r
|
|
197
200
|
\r
|
|
198
|
-
`,
|
|
201
|
+
`,u);if(j===-1)break;let E=o.slice(u,j).toString("utf-8"),w=j+4,C=o.indexOf(i,w),S=C!==-1?C-2:o.length;z.push({headers:E,body:o.slice(w,S)}),I=C!==-1?C:o.length}for(let b of z){let _=b.headers.match(/name="([^"]+)"/);if(!_)continue;let u=_[1];if(u==="screenshot")a=b.body;else if(u==="feedback")m=b.body.toString("utf-8");else if(u==="color")v=b.body.toString("utf-8");else if(u==="provider")y=b.body.toString("utf-8");else if(u==="model")B=b.body.toString("utf-8");else if(u==="goal")N=b.body.toString("utf-8");else if(u==="pageUrl")R=b.body.toString("utf-8");else if(u==="viewport")F=b.body.toString("utf-8");else if(u==="planId")h=b.body.toString("utf-8");else if(u==="manifest")W=b.body.toString("utf-8");else if(u==="tasks")H=b.body.toString("utf-8");else if(u==="sourceId")Y=b.body.toString("utf-8");else if(u.startsWith("screenshot-")){let j=u.slice(11);try{let E=decodeURIComponent(j);G.push({pathname:E,data:b.body})}catch{}}else if(u.startsWith("image-")){let j=u.split("-"),E=parseInt(j[j.length-1],10),w=j.slice(1,-1).join("-");w&&!isNaN(E)&&q.push({annotationId:w,index:E,data:b.body})}}if(!a)throw new Error("Missing screenshot field");return m||(m=""),{screenshot:a,feedback:m,color:v,provider:y,model:B,goal:N,pageUrl:R,viewport:F,planId:h,manifest:W,tasks:H,sourceId:Y,pastedImages:q,pageScreenshots:G}}function us(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 ke(s,t){let n=[];if(s.annotations.length>0){n.push("## Annotations");for(let e of s.annotations){let o=e.elements.map(a=>{let m=[a.selector];return a.reactComponent&&m.push(`(${a.reactComponent})`),m.join(" ")}).join(", "),i=e.instruction||"No text";n.push(`- id=${e.id} [${e.type}] ${i} \u2192 Elements: ${o||"none"}`);let c=t?.[e.id];if(c&&c.length>0)for(let a of c)n.push(` Attached image: use the Read tool to view ${a}`)}}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(`
|
|
199
202
|
### ${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(`
|
|
200
|
-
`)}function
|
|
201
|
-
`)}function
|
|
202
|
-
`)}function
|
|
203
|
-
`)){if(/^\s*`model`:/.test(e)){n=!0;continue}if(n&&/^\s*`[^`]+`:/.test(e))break;if(!n)continue;let o=e.match(
|
|
204
|
-
`));return
|
|
205
|
-
`)[0].trim().slice(0,60):"Running command";break;case"Glob":
|
|
203
|
+
`)}function Nt(s,t,n){let e=[],i=new Set(t.annotations.map(c=>c.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 c=0;for(let a of n.threadHistory)if(a.role==="human")c++,a.replyToQuestion?(e.push(`### Round ${c} (human) \u2014 reply`),e.push(`"${a.replyToQuestion}"`)):(e.push(`### Round ${c} (human)`),a.feedbackSummary&&e.push(`Annotations: ${a.feedbackSummary}`),a.annotationIds&&a.annotationIds.length>0&&e.push(`Annotation IDs: ${a.annotationIds.join(", ")}`));else if(a.question)e.push(`### Round ${c} (assistant) \u2014 question`),e.push(`"${a.question}"`);else{if(e.push(`### Round ${c} (assistant)`),a.responseText&&e.push(`Response: ${a.responseText}`),a.resolutions&&a.resolutions.length>0)for(let m of a.resolutions){let v=m.finalScope??m.inferredScope,y=v?` [${v.breadth} ${v.target}]`:"";e.push(`- ${m.annotationId}: ${m.status}${y} \u2014 ${m.summary}`),m.filesModified&&m.filesModified.length>0&&e.push(` Files: ${m.filesModified.join(", ")}`)}a.toolsUsed&&a.toolsUsed.length>0&&e.push(`Tools used: ${a.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 c=n.designModel.rules;if(Array.isArray(c)&&c.length>0)if(e.push(""),e.push("Rules:"),c.length>0&&typeof c[0]=="object"&&c[0]!==null&&"scope"in c[0]){let y=new Map;for(let B of c)if(typeof B=="object"&&B!==null&&"text"in B){let N=B,R=N.scope||"general";y.has(R)||y.set(R,[]),y.get(R).push(N.text)}for(let[B,N]of y){e.push(`**${B.charAt(0).toUpperCase()+B.slice(1)}**`);for(let R of N)e.push(`- ${R}`)}}else for(let y of c)typeof y=="string"&&e.push(`- ${y}`);let a=n.designModel.tokens;a&&typeof a=="object"&&(e.push(""),e.push("Design tokens:"),e.push("```json"),e.push(JSON.stringify(a,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 c=n?.screenshotPaths??{},a=new Map;for(let m of t.annotations){let v=m.pathname||new URL(t.url).pathname;a.has(v)||a.set(v,[]),a.get(v).push(m)}e.push(""),e.push("The developer annotated multiple pages. Each page section includes its own screenshot where available.");for(let[m,v]of a){e.push(""),e.push(`## Page: ${m}`);let y=c[m];y&&n?.provider!=="codex"?e.push(`IMPORTANT: Use the Read tool to view the screenshot at: ${y}`):!y&&n?.provider!=="codex"&&(Object.keys(c).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:v,styleModifications:t.styleModifications},N=ke(B,n?.imagePaths);N&&(e.push(""),e.push(N))}}else{let c=ke(t,n?.imagePaths);c&&(e.push(""),e.push(c))}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(`
|
|
204
|
+
`)}function Ft(s){return s.match(/<question>\s*([\s\S]*?)\s*<\/question>/)?.[1]??null}function Bt(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(c=>c.role==="human"&&c.feedbackContext);if(i?.feedbackContext&&(o.push(""),o.push(i.feedbackContext)),t.length>0){o.push(""),o.push("## Conversation History");let c=0;for(let a of t)a.role==="human"?(c++,a.replyToQuestion?(o.push(`### Round ${c} (human) \u2014 reply`),o.push(`"${a.replyToQuestion}"`)):(o.push(`### Round ${c} (human)`),a.feedbackSummary&&o.push(`Annotations: ${a.feedbackSummary}`))):a.question?(o.push(`### Round ${c} (assistant) \u2014 question`),o.push(`"${a.question}"`)):(o.push(`### Round ${c} (assistant)`),a.responseText&&o.push(`Response: ${a.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 c of e)o.push(`Attached image: use the Read tool to view the image at: ${c}`)}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(`
|
|
205
|
+
`)}function ps(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 fs(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&&!ps(t[n]))return!1;return!0}function Jt(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(fs):[]}catch{return[]}}function Ut(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 hs}from"child_process";import{readdir as ms,readFile as gs,stat as ys}from"fs/promises";import{homedir as Pe}from"os";import{join as de}from"path";var Ne=[{id:"claude-opus-4-6",label:"Opus 4.6",source:"static"},{id:"claude-sonnet-4-6",label:"Sonn 4.6",source:"static"}],Te=[{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"}],Ee={id:"",label:"Default",source:"cli"},ws=/^\s*-\s+"([^"]+)"/;function vs(s){let t=[],n=!1;for(let e of s.split(`
|
|
206
|
+
`)){if(/^\s*`model`:/.test(e)){n=!0;continue}if(n&&/^\s*`[^`]+`:/.test(e))break;if(!n)continue;let o=e.match(ws);if(!o?.[1])continue;let i=o[1];t.push({id:i,label:nt(i),source:"cli"})}return Fe([Ee,...t])}function Ss(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,c=typeof o.priority=="number"?o.priority:Number.MAX_SAFE_INTEGER;return i-c}).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():nt(o);return[{id:o,label:i,source:"cli"}]})}function xs(s){let t=s.match(/\bclaude-[a-z0-9]+(?:[-.][a-z0-9]+)*\b/gi)??[];return Fe(t.map(n=>({id:n,label:nt(n),source:"cli"})))}function nt(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":Ts(e[1]),c=e[2]&&e[3]?` ${e[2]}.${e[3]}`:"";return`${i}${c}${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 jt(s="copilot"){let t=await De(s,["help","config"]),n=vs(t);return n.length>1?n:[Ee]}async function Lt(s="codex"){let t=await De(s,["debug","models"]).catch(async()=>{let e=process.env.CODEX_HOME||de(Pe(),".codex");return Wt(de(e,"models_cache.json"))}),n=Ss(t);return n.length>0?Fe([...n,...Te]):Te}async function zt(s="claude"){let t=de(Pe(),".claude"),n=[de(Pe(),".claude","stats-cache.json"),de(Pe(),".claude","settings.json"),de(Pe(),".claude.json"),de(t,"cache","changelog.md")],[e,o,i]=await Promise.all([De(s,["--help"]).catch(()=>""),De(s,["--version"]).catch(()=>""),bs(t).catch(()=>[])]),c=Ps([...n,...i]),a=await Promise.all(c.map(v=>Wt(v).catch(()=>""))),m=xs([e,o,...a].join(`
|
|
207
|
+
`));return Fe([...Ne,...m])}function De(s,t,n=5e3){return new Promise((e,o)=>{hs(s,t,{encoding:"utf-8",timeout:n},(i,c)=>{if(i){o(i);return}e(c)})})}async function Wt(s){return gs(s,"utf-8")}async function bs(s){let t=[de(s,"projects"),de(s,"telemetry")],n=[];for(let e of t)await Ht(e,n,4);return n.sort((e,o)=>o.mtimeMs-e.mtimeMs).slice(0,80).map(e=>e.path)}async function Ht(s,t,n){if(n<0)return;let e;try{e=await ms(s,{withFileTypes:!0})}catch{return}await Promise.all(e.map(async o=>{let i=de(s,o.name);if(o.isDirectory()){await Ht(i,t,n-1);return}if(!(!o.isFile()||!ks(o.name)))try{let c=await ys(i);if(c.size>2e6)return;t.push({path:i,mtimeMs:c.mtimeMs})}catch{}}))}function ks(s){return s.endsWith(".json")||s.endsWith(".jsonl")||s.endsWith(".md")||s.endsWith(".txt")}function Fe(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 Ps(s){return[...new Set(s)]}function Ts(s){return s.charAt(0).toUpperCase()+s.slice(1)}var Be=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 c=n<0?e.events:e.events.filter(a=>a.seq>n);return{jobId:t,events:c,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,c=()=>{i||(i=!0,o())};e.on("exit",c),e.on("error",c),setTimeout(()=>{if(!i){try{e.kill("SIGKILL")}catch{}setTimeout(c,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 Es,readFile as Is,writeFile as Ms}from"fs/promises";import{dirname as Rs,join as Cs}from"path";var $s={version:1,threads:{}},Je=class{constructor(t){this.cache=null;this.writeChain=Promise.resolve();this.filePath=Cs(t,".popmelt","threads.json")}async load(){if(this.cache)return this.cache;try{let t=await Is(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={...$s,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(c=>e.has(c)))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 c of n)i.has(c)||o.elementIdentifiers.push(c);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 c=o.messages.find(a=>a.role==="human")?.feedbackSummary||"[thread]";return{id:o.id,createdAt:o.createdAt,updatedAt:o.updatedAt,preview:c,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 Es(Rs(this.filePath),{recursive:!0}),await Ms(this.filePath,JSON.stringify(this.cache,null,2))}catch(t){console.error("[ThreadStore] Failed to persist:",t)}}),await this.writeChain}};var Gt="0.7.0";var zs=1111,Ws=["Read","Edit","Write","Glob","Grep","Bash","WebFetch","WebSearch","Bash(curl:*)"],Hs=1800*1e3,Gs=3600*1e3;function Qt(s){if(!s)return!1;try{let t=new URL(s);return t.hostname==="localhost"||t.hostname==="127.0.0.1"}catch{return!1}}function qs(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,c=o?o.split("/").pop()??o:void 0,a;switch(e){case"Read":a=c?`Reading ${c}`:"Reading file";break;case"Edit":a=c?`Editing ${c}`:"Editing file";break;case"Write":a=c?`Writing ${c}`:"Writing file";break;case"Bash":a=i?i.split(`
|
|
208
|
+
`)[0].trim().slice(0,60):"Running command";break;case"Glob":a="Searching files";break;case"Grep":a="Searching code";break;case"WebFetch":a="Fetching page";break;case"WebSearch":a="Searching web";break;default:a=e?`Using ${e}`:"tool";break}let m=o??i??void 0,v=t[t.length-1];v&&v.kind==="tool_group"&&v.tool===e?v.items.push({label:a,detail:m}):t.push({kind:"tool_group",tool:e,items:[{label:a,detail:m}]})}return t}function Xs(s,t){let n=s.headers.origin;Qt(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 k(s,t,n){s.writeHead(t,{"Content-Type":"application/json"}),s.end(JSON.stringify(n))}function Ys(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 st(s,t){try{s.res.write(`event: ${t.type}
|
|
206
209
|
data: ${JSON.stringify(t)}
|
|
207
210
|
|
|
208
|
-
`)}catch{}}function
|
|
211
|
+
`)}catch{}}function Le(s){return s==="claude"||s==="codex"||s==="copilot"?s:void 0}async function Qs(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 Ks(s,t){return new Promise((n,e)=>{let o=c=>{s.removeListener("listening",i),e(c)},i=()=>{s.removeListener("error",o),n()};s.once("error",o),s.once("listening",i),s.listen(t,"127.0.0.1")})}async function Kt(s={}){let t=s.port??zs,n=s.projectRoot??process.cwd(),e=_s("sha256").update(n).digest("hex").slice(0,12),o=s.devOrigin??(process.env.PORT?`http://localhost:${process.env.PORT}`:null),i=s.tempDir??ue(Js(),"popmelt-bridge"),c=s.maxTurns??40,a=s.maxBudgetUsd??1,m=[...s.allowedTools??Ws],v=s.claudePath??"claude",y=s.copilotPath??"copilot",B=s.provider??"claude",N=s.timeoutMs,R=s.restartMode??"embedded",F=t,h={},W={claude:v,codex:"codex",copilot:y};for(let[r,l]of Object.entries(W))try{let f=l.includes("/")?l:Os("which",[l],{encoding:"utf-8"}).trim();h[r]={available:!0,path:f}}catch{h[r]={available:!1,path:null}}function H(r,l){return new Promise(f=>{let d=qt(l,["--version"],{stdio:["ignore","ignore","ignore"]}),g=!1,p=J=>{g||(g=!0,f(J))},P=setTimeout(()=>{d.kill("SIGTERM"),p(!0)},5e3);d.on("error",()=>{clearTimeout(P),p(!1)}),d.on("close",J=>{clearTimeout(P),p(J===0)})})}await Promise.all([(async()=>{h.claude&&(h.claude.models=h.claude.available&&h.claude.path?await zt(h.claude.path).catch(()=>Ne):Ne)})(),(async()=>{h.codex&&(h.codex.models=h.codex.available&&h.codex.path?await Lt(h.codex.path).catch(()=>Te):Te)})(),(async()=>{h.copilot&&(h.copilot.models=h.copilot.available&&h.copilot.path?await jt(h.copilot.path).catch(()=>[Ee]):[Ee])})()]);let[Y,q,G]=await Promise.all([Qe(n),Ke(n),Ve(n,h.copilot?.path??y)]);h.claude&&(h.claude.mcp=Y),h.codex&&(h.codex.mcp=q),h.copilot&&(h.copilot.mcp=G),Y.found&&Y.name&&m.push(`mcp__${Y.name}__*`),await As(i,{recursive:!0}),Yt(i).catch(()=>{});let I=new Be(1),z=new Set,b=new Je(n),_=new $e(n),u=new Oe(n,_,{claudePath:v,provider:B,timeoutMs:N,onEvent:r=>{for(let l of z)st(l,r)}}),j=20,E=300*1e3,w=[],C=new Set,S=null,X;I.addListener((r,l,f)=>{for(let d of z)(!f||!d.sourceId||d.sourceId===f)&&st(d,r)}),I.setProcessor(async r=>{let l=r._replyPrompt,f=r._replyImagePaths,d=r.provider??B,g;if(r.threadId){let M=await b.getThread(r.threadId);if(M){for(let $=M.messages.length-1;$>=0;$--)if(M.messages[$].sessionId){g=M.messages[$].sessionId;break}}}let p;if(g&&l){let M=(await b.getThread(r.threadId))?.messages.filter(ee=>ee.role==="human").pop();if(p=M?.replyToQuestion||M?.feedbackSummary||"",f&&f.length>0){p+=`
|
|
209
212
|
|
|
210
|
-
The developer attached reference images with their reply:`;for(let
|
|
211
|
-
Attached image: use the Read tool to view the image at: ${
|
|
213
|
+
The developer attached reference images with their reply:`;for(let ee of f)p+=`
|
|
214
|
+
Attached image: use the Read tool to view the image at: ${ee}`}p+=`
|
|
212
215
|
|
|
213
|
-
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(
|
|
216
|
+
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(g)p=ke(r.feedback,r.imagePaths)+`
|
|
214
217
|
|
|
215
218
|
Follow the developer's instructions. If they ask for changes, apply them to the source files.
|
|
216
219
|
|
|
217
220
|
After completing work, output a <resolution> block with declaredScope and inferredScope. If unclear, output a <question> block.`+(d!=="codex"?`
|
|
218
221
|
|
|
219
|
-
IMPORTANT: First, use the Read tool to view the updated screenshot at: ${r.screenshotPath}`:"");else{let M=!
|
|
220
|
-
${
|
|
222
|
+
IMPORTANT: First, use the Read tool to view the updated screenshot at: ${r.screenshotPath}`:"");else{let M=!l&&r.threadId?await b.getThreadHistory(r.threadId):void 0,$=l?null:await u.loadModel();p=l??Nt(r.screenshotPath,r.feedback,{threadHistory:M&&M.length>0?M:void 0,provider:d,imagePaths:r.imagePaths,designModel:$??void 0,screenshotPaths:r.screenshotPaths})}let P=Ys(r.color,`[\u22B9 ${F}:${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(`${P} Reviewing ${J} (provider: ${d})${r.threadId?` (thread: ${r.threadId})`:""}${g?` (resuming: ${g.slice(0,8)})`:""}`);let L=(M,$)=>{M.type==="delta"&&"text"in M?I.accumulateText($,"response",M.text):M.type==="thinking"&&"text"in M&&I.accumulateText($,"thinking",M.text),I.broadcast(M,$,r.sourceId)},{process:ne,result:K}=d==="codex"?Ce(r.id,{prompt:p,projectRoot:n,screenshotPath:r.screenshotPath,resumeSessionId:g,model:r.model,onEvent:L}):d==="copilot"?St(r.id,{prompt:p,projectRoot:n,resumeSessionId:g,model:r.model,timeoutMs:N,copilotPath:h.copilot?.path??y,onEvent:L}):Me(r.id,{prompt:p,projectRoot:n,maxTurns:c,maxBudgetUsd:a,allowedTools:m,claudePath:v,resumeSessionId:g,model:r.model,timeoutMs:N,onEvent:L});I.setActiveProcess(r.id,ne);let x=await K;if(r.result=x.text,x.success){console.log(`${P} Iteration complete`),x.fileEdits&&x.fileEdits.length>0&&console.log(`${P} Captured ${x.fileEdits.length} file edit(s): ${x.fileEdits.map(D=>`${D.tool} ${D.file_path}`).join(", ")}`),x.editEvents&&x.editEvents.length>0&&console.log(`${P} Captured ${x.editEvents.length} edit event(s): ${x.editEvents.map(D=>`${D.operation} ${D.filePath}`).join(", ")}`),r.status="done";let M=Ft(x.text),$=Jt(x.text);if($.length>0&&r.annotationIds&&r.annotationIds.length>0){let D=new Set(r.annotationIds);$.every(U=>D.has(U.annotationId))||($=$.map((U,te)=>({...U,annotationId:r.annotationIds[te%r.annotationIds.length]})))}let ee=x.fileEdits&&x.fileEdits.length>0?x.fileEdits.map(D=>`${D.tool} ${D.file_path.split("/").pop()}`):x.toolsUsed,re=I.getBufferedEvents(r.id),pe=re?qs(re.events):void 0;if(r.threadId&&await b.appendMessage(r.threadId,{role:"assistant",timestamp:Date.now(),jobId:r.id,responseText:x.text,resolutions:$.length>0?$:void 0,question:M??void 0,sessionId:x.sessionId,toolsUsed:ee,editEvents:x.editEvents&&x.editEvents.length>0?x.editEvents:void 0,segments:pe&&pe.length>0?pe:void 0,model:r.model,provider:d}),_.captureGitDiff(n).then(async D=>{let A=Date.now(),U=r.imagePaths?Object.values(r.imagePaths).flat():[],te=[];if(r.imagePaths)for(let[me,O]of Object.entries(r.imagePaths))for(let se=0;se<O.length;se++)te.push(`screenshots/p-${r.id}-${me}-${se}.webp`);await _.persist({version:1,id:r.id,createdAt:r.createdAt,completedAt:A,durationMs:A-r.createdAt,url:r.feedback.url,viewport:r.feedback.viewport,screenshotPath:`screenshots/s-${r.id}.webp`,pastedImagePaths:te,annotations:r.feedback.annotations,styleModifications:r.feedback.styleModifications,inspectedElement:r.feedback.inspectedElement,provider:d,model:r.model,sessionId:x.sessionId,threadId:r.threadId,responseText:x.text,resolutions:$.length>0?$:[],question:M??void 0,fileEdits:x.fileEdits??[],editEvents:x.editEvents??[],toolsUsed:ee,gitDiff:D},r.screenshotPath,U)}).catch(()=>{}),$.length>0){let D=$.some(A=>(A.finalScope??A.inferredScope)?.breadth==="pattern");D&&d!=="copilot"?u.run({provider:d,model:r.model}).catch(()=>{}):D&&console.log(`${P} Skipping materialization for Copilot provider`)}if(r.kind==="synthesize"&&x.text){let D=_e(x.text);if(D){Array.isArray(D.rules)&&(D.rules=xe(D.rules));let A=await u.loadModel();A&&(!D.tokens&&A.tokens&&(D.tokens=A.tokens),!D.components&&A.components&&(D.components=A.components)),await u.writeModel(D),console.log(`${P} Synthesize: model.json updated`)}}M&&(console.log(`${P} \u{1F4AC} Question detected: "${M.slice(0,120)}" \u2192 broadcasting to ${z.size} SSE clients (threadId=${r.threadId??r.id}, annotationIds=${r.annotationIds?.join(",")??"none"})`),I.broadcast({type:"question",jobId:r.id,threadId:r.threadId??r.id,question:M,annotationIds:r.annotationIds},r.id,r.sourceId));let ce=Ut(x.text);ce.length>0&&(console.log(`${P} Novel pattern(s): ${ce.map(D=>`${D.category}/${D.element}`).join(", ")}`),I.broadcast({type:"novel_patterns",jobId:r.id,patterns:ce,threadId:r.threadId},r.id,r.sourceId)),I.broadcast({type:"done",jobId:r.id,success:!0,resolutions:$.length>0?$:void 0,responseText:x.text,threadId:r.threadId},r.id,r.sourceId),w.push({id:r.id,status:"done",completedAt:Date.now(),threadId:r.threadId,annotationIds:r.annotationIds})}else console.error(`${P} Error: ${x.error}`),r.status="error",r.error=x.error,r.threadId&&await b.appendMessage(r.threadId,{role:"assistant",timestamp:Date.now(),jobId:r.id,error:x.error||"Unknown error",model:r.model,provider:d}),I.broadcast({type:"error",jobId:r.id,threadId:r.threadId,message:x.error||"Unknown error",provider:d,model:r.model,recoverable:x.recoverableError,code:x.errorCode},r.id,r.sourceId),w.push({id:r.id,status:"error",completedAt:Date.now(),error:x.error,threadId:r.threadId,annotationIds:r.annotationIds});let V=Date.now()-E;for(;w.length>0&&(w[0].completedAt<V||w.length>j);)w.shift()});let T=Bs(async(r,l)=>{if(Xs(r,l),r.method==="OPTIONS"){l.writeHead(204),l.end();return}let f=r.url||"/",d=f.split("?")[0],g=new URL(f,`http://127.0.0.1:${F}`),p=g.pathname;try{if(r.method==="POST"&&p==="/send")await Z(r,l);else if(r.method==="GET"&&p==="/events")oe(r,l);else if(r.method==="GET"&&p==="/status")sn(l);else if(r.method==="PATCH"&&p==="/config")await on(r,l);else if(r.method==="POST"&&p==="/shutdown"){if(R!=="detached"){k(l,409,{ok:!1,error:"This Popmelt bridge is embedded in the host dev server. Restart the dev server to reload bridge code.",restartMode:R});return}k(l,200,{ok:!0}),setTimeout(()=>process.exit(0),100)}else if(r.method==="POST"&&p==="/restart"){if(R!=="detached"){k(l,409,{ok:!1,error:"This Popmelt bridge is embedded in the host dev server. Restart the dev server to reload bridge code.",restartMode:R});return}k(l,200,{ok:!0,restarting:!0}),setTimeout(()=>{for(let P of z)try{P.res.end()}catch{}T.close(()=>{qt(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"&&p==="/capabilities")k(l,200,{providers:h});else if(r.method==="POST"&&p==="/mcp/install")await ln(r,l);else if(r.method==="POST"&&p==="/reply")await ae(r,l);else if(r.method==="POST"&&p==="/cancel")rn(r,l);else if(r.method==="POST"&&p==="/materialize")await an(r,l);else if(r.method==="POST"&&p==="/model/component")await dn(r,l);else if(r.method==="DELETE"&&p==="/model/component")await un(r,l);else if(r.method==="PATCH"&&p==="/model/token")await pn(r,l);else if(r.method==="DELETE"&&p==="/model/token")await fn(r,l);else if(r.method==="POST"&&p==="/model/consolidate"){let{provider:P,model:J}=await ot(r);if(P==="copilot"){k(l,400,{error:"Model consolidation is not available for Copilot yet."});return}u.consolidate({provider:P,model:J}).catch(L=>console.error("[Bridge] Consolidation error:",L)),k(l,200,{started:!0})}else if(r.method==="POST"&&p==="/model/synthesize")await cn(r,l);else if(r.method==="GET"&&p==="/model"){let P=await u.loadModel();k(l,200,{model:P})}else if(r.method==="GET"&&p.startsWith("/jobs/")&&p.endsWith("/events")){let P=p.slice(6,p.length-7),J=parseInt(g.searchParams.get("afterSeq")??"-1",10),L=I.getBufferedEvents(P,isNaN(J)?-1:J);L?k(l,200,L):k(l,404,{error:"Unknown job"})}else if(r.method==="GET"&&d.startsWith("/files/"))await yn(d.slice(7),l);else if(r.method==="GET"&&p==="/threads/recent"){let P=Math.min(Math.max(parseInt(g.searchParams.get("limit")??"5",10)||5,1),20),J=await b.listRecent(P);k(l,200,J)}else if(r.method==="GET"&&p.startsWith("/thread/")){let P=p.slice(8);await wn(P,l)}else r.method==="GET"&&(p==="/canvas"||p==="/canvas/")?hn(r,l):r.method==="GET"&&p==="/canvas/manifest"?await mn(l):r.method==="GET"&&p==="/canvas/app.mjs"?await gn(l):k(l,404,{error:"Not found"})}catch(P){console.error("[Bridge] Request error:",P),k(l,500,{error:P instanceof Error?P.message:"Internal error"})}});async function Z(r,l){let{screenshot:f,feedback:d,color:g,provider:p,model:P,sourceId:J,pastedImages:L,pageScreenshots:ne}=await tt(r),K;try{K=JSON.parse(d)}catch{k(l,400,{error:"Invalid feedback JSON"});return}let x=Ue().slice(0,8),V={};if(ne.length>0)for(let O of ne){let se=encodeURIComponent(O.pathname),fe=ue(i,`screenshot-${x}-${se}.webp`);await je(fe,O.data),V[O.pathname]=fe}let M=ue(i,`screenshot-${x}.webp`);await je(M,f);let $={};if(L.length>0)for(let O of L){let se=ue(i,`pasted-${x}-${O.annotationId}-${O.index}.webp`);await je(se,O.data),$[O.annotationId]||($[O.annotationId]=[]),$[O.annotationId].push(se)}let ee=K.annotations.map(O=>O.linkedSelector?O.pathname?`${O.pathname}:${O.linkedSelector}`:O.linkedSelector:null).filter(O=>!!O),re;if(ee.length>0){let O=await b.findContinuationThread(ee);O?(re=O.id,await b.addElementIdentifiers(re,ee)):re=(await b.createThread(x,ee)).id}else re=(await b.createThread(x,[])).id;let pe=K.annotations.map(O=>O.id),ce=Object.keys(V).length>0,D={id:x,status:"queued",screenshotPath:M,feedback:K,createdAt:Date.now(),color:g,threadId:re,annotationIds:pe,provider:Le(p),model:P||void 0,...Object.keys($).length>0?{imagePaths:$}:{},sourceId:J||void 0,...ce?{screenshotPaths:V}:{}},A=new Set(K.annotations.map(O=>O.pathname).filter(Boolean)),U;if(A.size>1){let O=new Map;for(let se of K.annotations){let fe=se.pathname||"(unknown)";O.has(fe)||O.set(fe,[]),O.get(fe).push(se.instruction||`[${se.type}]`)}U=[...O.entries()].map(([se,fe])=>`\`${se}\`
|
|
223
|
+
${fe.map(Sn=>`- ${Sn}`).join(`
|
|
221
224
|
`)}`).join(`
|
|
222
|
-
`)}else
|
|
225
|
+
`)}else U=K.annotations.map(O=>O.instruction||`[${O.type}]`).join("; ");let te=ke(K,Object.keys($).length>0?$:void 0);await b.appendMessage(re,{role:"human",timestamp:Date.now(),jobId:x,screenshotPath:M,...ce?{screenshotPaths:V}:{},...Object.keys($).length>0?{imagePaths:$}:{},annotationIds:pe,feedbackSummary:U,feedbackContext:te||void 0});let me=I.enqueue(D);k(l,200,{jobId:x,position:me,threadId:re})}async function ae(r,l){let f=r.headers["content-type"]||"",d,g,p,P,J,L,ne=[];if(f.includes("multipart/form-data")){let A=await tt(r),U=A.feedback?JSON.parse(A.feedback):{};d=U.threadId,g=U.reply,p=U.color,P=U.provider,J=U.model,L=U.sourceId||A.sourceId;for(let te of A.pastedImages)ne.push(te.data)}else{let A=[];for await(let me of r)A.push(typeof me=="string"?Buffer.from(me):me);let U=Buffer.concat(A).toString("utf-8"),te;try{te=JSON.parse(U)}catch{k(l,400,{error:"Invalid JSON"});return}d=te.threadId,g=te.reply,p=te.color,P=te.provider,J=te.model,L=te.sourceId}if(!d||!g){k(l,400,{error:"Missing threadId or reply"});return}if(!await b.getThread(d)){k(l,404,{error:"Thread not found"});return}let x=Ue().slice(0,8),V=[];for(let A=0;A<ne.length;A++){let U=ue(i,`reply-${x}-${A}.webp`);await je(U,ne[A]),V.push(U)}let M="";{let A=await b.getThreadHistory(d);for(let U=A.length-1;U>=0;U--)if(A[U].screenshotPath){M=A[U].screenshotPath;break}}if(!M&&!C.has(d)){k(l,400,{error:"No screenshot available"});return}await b.appendMessage(d,{role:"human",timestamp:Date.now(),jobId:x,replyToQuestion:g,screenshotPath:M,...V.length>0?{replyImagePaths:V}:{}});let $=await b.getThreadHistory(d),ee=[];for(let A of $)if(A.annotationIds)for(let U of A.annotationIds)ee.includes(U)||ee.push(U);let re=Le(P),pe=Bt(M,$,re,V.length>0?V:void 0),ce={id:x,status:"queued",screenshotPath:M,feedback:{timestamp:new Date().toISOString(),url:"",viewport:{width:0,height:0},scrollPosition:{x:0,y:0},annotations:[],styleModifications:[]},createdAt:Date.now(),color:p,threadId:d,annotationIds:ee.length>0?ee:void 0,provider:re,model:J||void 0,sourceId:L||void 0};C.has(d)&&(ce.kind="synthesize"),ce._replyPrompt=pe,V.length>0&&(ce._replyImagePaths=V);let D=I.enqueue(ce);k(l,200,{jobId:x,position:D,threadId:d})}function oe(r,l){l.writeHead(200,{"Content-Type":"text/event-stream","Cache-Control":"no-cache",Connection:"keep-alive"}),l.write(`event: connected
|
|
223
226
|
data: {"status":"connected"}
|
|
224
227
|
|
|
225
|
-
`),r.headers.origin
|
|
226
|
-
`)}async function
|
|
228
|
+
`),r.headers.origin&&Qt(r.headers.origin)&&o!==r.headers.origin&&(o=r.headers.origin);let d=new URL(r.url||"/",`http://127.0.0.1:${F}`).searchParams.get("sourceId")||void 0,g={id:Ue().slice(0,8),res:l,sourceId:d};z.add(g),r.on("close",()=>{z.delete(g)})}function sn(r){let l=I.allActive;k(r,200,{ok:!0,version:Gt,restartMode:R,projectId:e,devOrigin:o,activeJob:l[0]?{id:l[0].id,status:l[0].status}:null,activeJobs:l.map(f=>({id:f.id,status:f.status,threadId:f.threadId,annotationIds:f.annotationIds,color:f.color})),queueDepth:I.depth,recentJobs:w})}async function on(r,l){let f=await new Promise(d=>{let g="";r.on("data",p=>{g+=p.toString()}),r.on("end",()=>d(g))});try{let d=JSON.parse(f);typeof d.devOrigin=="string"&&(o=d.devOrigin||null),k(l,200,{ok:!0,devOrigin:o})}catch{k(l,400,{error:"Invalid JSON"})}}async function rn(r,l){let d=new URL(r.url||"/",`http://127.0.0.1:${F}`).searchParams.get("jobId"),p=(d?I.allActive.filter(J=>J.id===d):I.allActive).map(J=>J.threadId).filter(Boolean),P=d?I.cancelJob(d):I.cancelActive();for(let J of p)await b.appendMessage(J,{role:"assistant",timestamp:Date.now(),jobId:d||"",cancelled:!0});k(l,200,{cancelled:P})}async function an(r,l){if(u.isRunning){k(l,200,{skipped:!0,reason:"Already running"});return}let{provider:f,model:d}=await ot(r);if(f==="copilot"){k(l,400,{error:"Materialization is not available for Copilot yet."});return}let g=await u.getUnmaterializedPatternDecisions();if(g.length===0){k(l,200,{skipped:!0,reason:"No unmaterialized pattern decisions"});return}u.run({provider:f,model:d}).catch(()=>{}),k(l,200,{started:!0,decisionCount:g.length,decisionIds:g.map(p=>p.id)})}async function ot(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 f=JSON.parse(Buffer.concat(l).toString());return{provider:Le(typeof f.provider=="string"?f.provider:void 0),model:typeof f.model=="string"&&f.model?f.model:void 0}}catch{return{}}}async function cn(r,l){let f=await u.loadModel();if(!f){k(l,400,{error:"No model exists yet"});return}let d,g;try{let x=[];for await(let M of r)x.push(typeof M=="string"?Buffer.from(M):M);let V=JSON.parse(Buffer.concat(x).toString());d=V.provider,g=V.model}catch{}if(d==="copilot"){k(l,400,{error:"Rule synthesis is not available for Copilot yet."});return}let p=Pt(f),P=Ue().slice(0,8),L=(await b.createThread(P,[])).id;C.add(L),await b.appendMessage(L,{role:"human",timestamp:Date.now(),jobId:P,feedbackSummary:"Synthesize rules \u2014 review and propose improvements"});let ne={id:P,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:L,kind:"synthesize",provider:Le(d),model:g||void 0};ne._replyPrompt=p;let K=I.enqueue(ne);k(l,200,{jobId:P,position:K,threadId:L})}async function ln(r,l){let f=[];for await(let L of r)f.push(typeof L=="string"?Buffer.from(L):L);let d;if(f.length>0)try{d=JSON.parse(Buffer.concat(f).toString("utf-8")).serverUrl}catch{}let g=[];h.claude?.available&&h.claude.mcp&&!h.claude.mcp.found&&g.push(await _t(d)),h.codex?.available&&h.codex.mcp&&!h.codex.mcp.found&&g.push(await At(d)),h.copilot?.available&&h.copilot.mcp&&!h.copilot.mcp.found&&g.push(await Dt(d,h.copilot.path??y));let[p,P,J]=await Promise.all([Qe(n),Ke(n),Ve(n,h.copilot?.path??y)]);h.claude&&(h.claude.mcp=p),h.codex&&(h.codex.mcp=P),h.copilot&&(h.copilot.mcp=J),k(l,200,{results:g,capabilities:{providers:h}})}async function dn(r,l){let f=[];for await(let p of r)f.push(typeof p=="string"?Buffer.from(p):p);let d;try{d=JSON.parse(Buffer.concat(f).toString("utf-8"))}catch{k(l,400,{error:"Invalid JSON"});return}if(!d.name||typeof d.name!="string"){k(l,400,{error:"Missing or invalid name"});return}let g=await u.addComponent(d.name);k(l,200,g)}async function un(r,l){let f=[];for await(let p of r)f.push(typeof p=="string"?Buffer.from(p):p);let d;try{d=JSON.parse(Buffer.concat(f).toString("utf-8"))}catch{k(l,400,{error:"Invalid JSON"});return}if(!d.name||typeof d.name!="string"){k(l,400,{error:"Missing or invalid name"});return}let g=await u.removeComponent(d.name);k(l,200,g)}async function pn(r,l){let f=[];for await(let p of r)f.push(typeof p=="string"?Buffer.from(p):p);let d;try{d=JSON.parse(Buffer.concat(f).toString("utf-8"))}catch{k(l,400,{error:"Invalid JSON"});return}if(!d.path||typeof d.path!="string"||typeof d.value!="string"){k(l,400,{error:"Missing or invalid path/value"});return}let g=await u.updateToken(d.path,d.value);k(l,200,g)}async function fn(r,l){let f=[];for await(let p of r)f.push(typeof p=="string"?Buffer.from(p):p);let d;try{d=JSON.parse(Buffer.concat(f).toString("utf-8"))}catch{k(l,400,{error:"Invalid JSON"});return}if(!d.path||typeof d.path!="string"){k(l,400,{error:"Missing or invalid path"});return}let g=await u.removeToken(d.path);k(l,200,g)}function hn(r,l){let f=o??"http://localhost:3000";if(!o){let g=r.headers.referer||r.headers.origin;if(g)try{let p=new URL(typeof g=="string"?g:g[0]||"");(p.hostname==="localhost"||p.hostname==="127.0.0.1")&&(f=p.origin)}catch{}}let d=at(F,f);l.writeHead(200,{"Content-Type":"text/html; charset=utf-8"}),l.end(d)}async function mn(r){let l=Date.now();if(S&&l<S.expires){k(r,200,S.data);return}try{let{scanForComponents:f}=await import("./react-scanner-P3VD53VT.mjs"),{generateRenderFiles:d}=await import("./render-generator-HWFLZYC3.mjs"),g=await f(n);S={data:g,expires:l+5e3},d(g,n,X).then(p=>{X=p}).catch(p=>console.warn("[Bridge] Render generation failed:",p)),k(r,200,g)}catch(f){console.error("[Bridge] Scanner error:",f),k(r,500,{error:"Failed to scan components"})}}async function gn(r){let l=[ue(n,"node_modules","@popmelt.com","core","dist","canvas.mjs"),ue(n,"packages","popmelt","dist","canvas.mjs")];try{let f=Ls(import.meta.url);l.unshift(ue(js(f),"canvas.mjs"))}catch{}for(let f of l)try{let d=await Xt(f,"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),k(r,404,{error:"Canvas bundle not found"})}async function yn(r,l){if(!r||r.includes("/")||r.includes("\\")||r.includes("..")){k(l,400,{error:"Invalid filename"});return}try{let f=await Xt(ue(i,r)),d=r.split(".").pop()?.toLowerCase(),g=d==="png"?"image/png":d==="webp"?"image/webp":d==="jpg"||d==="jpeg"?"image/jpeg":"application/octet-stream";l.writeHead(200,{"Content-Type":g,"Cache-Control":"public, max-age=3600"}),l.end(f)}catch{k(l,404,{error:"File not found"})}}function Ie(r){return`/files/${Us(r)}`}async function wn(r,l){let f=await b.getThread(r);if(!f){k(l,404,{error:"Thread not found"});return}let d=f.messages.map(({screenshotPath:g,screenshotPaths:p,imagePaths:P,replyImagePaths:J,...L})=>({...L,...g?{screenshotUrl:Ie(g)}:{},...p?{screenshotUrls:Object.fromEntries(Object.entries(p).map(([ne,K])=>[ne,Ie(K)]))}:{},...P?{imageUrls:Object.fromEntries(Object.entries(P).map(([ne,K])=>[ne,K.map(Ie)]))}:{},...J?{replyImageUrls:J.map(Ie)}:{}}));k(l,200,{id:f.id,createdAt:f.createdAt,messages:d})}let rt=9,it=!1;for(let r=t;r<t+rt;r++)try{await Ks(T,r),F=r,it=!0,console.log(`[\u22B9 is watching :${F}]`);break}catch(l){if(l.code==="EADDRINUSE"){let f=await Qs(r);if(f&&f.projectId===e)return console.log(`[\u22B9 already watching :${r}]`),{port:r,projectId:e,close:async()=>{}};continue}throw l}if(!it)throw new Error(`[Bridge] All ports ${t}\u2013${t+rt-1} in use`);for(let[r,l]of Object.entries(h))!l.available||!l.path||H(r,l.path).then(f=>{if(f)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 z)st(d,{type:"capabilities_changed",data:{}})}});let vn=setInterval(()=>{Yt(i).catch(()=>{})},Hs);return{port:F,projectId:e,close:async()=>{clearInterval(vn),await I.destroyAsync();for(let r of z)try{r.res.end()}catch{}return z.clear(),new Promise(r=>{T.close(()=>r())})}}}async function Yt(s){try{let t=await Ds(s),n=Date.now();for(let e of t){let o=ue(s,e);try{let i=await Ns(o);n-i.mtimeMs>Gs&&await Fs(o)}catch{}}}catch{}}var to=parseInt(process.env.POPMELT_PORT||"1111",10),Zt=process.env.POPMELT_PROJECT_ROOT||process.cwd(),no=process.env.POPMELT_DEV_ORIGIN||void 0,so=process.env.POPMELT_COPILOT_PATH||void 0,en=Vt(Zt,".popmelt"),tn=Vt(en,"bridge.lock");async function oo(s){await Vs(en,{recursive:!0}),await eo(tn,JSON.stringify({pid:process.pid,port:s,startedAt:Date.now()})+`
|
|
229
|
+
`)}async function nn(){try{await Zs(tn)}catch{}}async function ro(){let s=await Kt({port:to,projectRoot:Zt,devOrigin:no,copilotPath:so,restartMode:"detached"});await oo(s.port),await new Promise(t=>{let n=async()=>{await s.close(),await nn(),t()};process.on("SIGTERM",n),process.on("SIGINT",n)})}ro().catch(s=>{console.error("[popmelt bridge-entry] Fatal:",s),nn().finally(()=>process.exit(1))});
|