@popmelt.com/core 0.6.7 → 0.6.8

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