@popmelt.com/core 0.6.5 → 0.6.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/bridge-entry.mjs +45 -45
- package/dist/cli.mjs +42 -42
- package/dist/plugin-astro.mjs +1 -1
- package/dist/plugin-vite.mjs +1 -1
- package/dist/react-scanner-P3VD53VT.mjs +1 -0
- package/dist/react-scanner-ZXYS5M3Y.mjs +2 -0
- package/dist/render-generator-EANIDD2E.mjs +107 -0
- package/dist/render-generator-HWFLZYC3.mjs +106 -0
- package/dist/server-HPHAQXHA.mjs +133 -0
- package/dist/server.mjs +40 -40
- package/package.json +1 -1
- package/dist/chunk-3HWT3PC2.mjs +0 -1
- package/dist/chunk-4LDJX6BW.mjs +0 -2
- package/dist/react-scanner-5NIJ6ZPL.mjs +0 -1
- package/dist/react-scanner-MSMGKCIV.mjs +0 -2
- package/dist/render-generator-QV5BYGPF.mjs +0 -107
- package/dist/render-generator-ZNV3RDU7.mjs +0 -106
- package/dist/server-CXH52R3U.mjs +0 -133
package/dist/bridge-entry.mjs
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import{
|
|
1
|
+
import{mkdir as xn,unlink as Pn,writeFile as In}from"fs/promises";import{join as it}from"path";import{execFileSync as Zt,spawn as en}from"child_process";import{createHash as tn,randomUUID as $e}from"crypto";import{mkdir as nn,readFile as tt,readdir as sn,stat as on,unlink as rn,writeFile as Se}from"fs/promises";import{createServer as an}from"http";import{tmpdir as cn}from"os";import{basename as ln,dirname as dn,join as ne}from"path";import{fileURLToPath as un}from"url";function Ae(i,t){return`<!DOCTYPE html>
|
|
2
2
|
<html>
|
|
3
3
|
<head>
|
|
4
4
|
<meta charset="utf-8">
|
|
@@ -18,28 +18,28 @@ import{a as _,b as te,c as rt,d as ue}from"./chunk-3HWT3PC2.mjs";import{mkdir as
|
|
|
18
18
|
}}
|
|
19
19
|
</script>
|
|
20
20
|
<script type="module">
|
|
21
|
-
import { mountCanvas } from 'http://localhost:${
|
|
21
|
+
import { mountCanvas } from 'http://localhost:${i}/canvas/app.mjs';
|
|
22
22
|
mountCanvas(document.getElementById('root'), {
|
|
23
|
-
devOrigin: '${
|
|
24
|
-
bridgeOrigin: 'http://localhost:${
|
|
23
|
+
devOrigin: '${t}',
|
|
24
|
+
bridgeOrigin: 'http://localhost:${i}',
|
|
25
25
|
});
|
|
26
26
|
</script>
|
|
27
27
|
</body>
|
|
28
|
-
</html>`}import{spawn as
|
|
29
|
-
\u2026[truncated]`:
|
|
30
|
-
`)[0].slice(0,80)}`:"Bash";
|
|
31
|
-
Files modified: ${
|
|
32
|
-
`),
|
|
33
|
-
`),
|
|
28
|
+
</html>`}import{spawn as kt}from"child_process";import{createInterface as Tt}from"readline";var Et=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"]),_e=1e5;function Rt(i){let t=i.input?.file_path||i.input?.path;if(!t)return;let n=t.includes(".")?`.${t.split(".").pop().toLowerCase()}`:"";if(!Et.has(n))return;let e;if(i.name==="Write"&&typeof i.input?.content=="string"?e=i.input.content:i.name==="Edit"&&typeof i.input?.new_string=="string"&&(e=i.input.new_string),!!e)return e.length>_e?e.slice(0,_e)+`
|
|
29
|
+
\u2026[truncated]`:e}function fe(i,t){let{prompt:n,projectRoot:e,maxTurns:s=40,maxBudgetUsd:l=1,allowedTools:c=["Read","Edit","Write","Glob","Grep","Bash"],claudePath:a="claude",resumeSessionId:h,model:f,timeoutMs:N=3e5,onEvent:U}=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(l)),f&&O.push("--model",f);for(let L of c)O.push("--allowedTools",L);let I=kt(a,O,{cwd:e,stdio:["ignore","pipe","pipe"],env:{...process.env,ANTHROPIC_API_KEY:void 0}}),z=new Promise(L=>{let G,T=[],J=[],C=!1,B="",g=!1,F=setTimeout(()=>{g=!0,I.kill("SIGTERM"),setTimeout(()=>{try{I.kill("SIGKILL")}catch{}},5e3)},N),y=Tt({input:I.stdout}),w=new Set;y.on("line",D=>{if(D.trim())try{let A=JSON.parse(D);A.session_id&&!G&&(G=A.session_id);let Z=A.type??(A.event?.type?`event.${A.event.type}`:"unknown");if(w.add(Z),A.type==="result"&&A.result&&T.length===0){let R=typeof A.result=="string"?A.result:"";R&&(T.push(R),U?.({type:"delta",jobId:i,text:R},i))}if(A.type==="assistant"&&Array.isArray(A.message?.content))for(let R of A.message.content){if(R.type==="text"&&R.text&&(T.push(R.text),U?.({type:"delta",jobId:i,text:R.text},i)),R.type==="tool_use"&&R.name){let ae=R.input?.file_path||R.input?.path||void 0,pe=Rt(R);U?.({type:"tool_use",jobId:i,tool:R.name,...ae?{file:ae}:{},...pe?{content:pe}:{}},i),R.name==="Edit"&&R.input?.file_path?J.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&&J.push({tool:"Write",file_path:R.input.file_path,content:R.input.content})}R.type==="thinking"&&R.thinking&&U?.({type:"thinking",jobId:i,text:R.thinking},i)}A.type==="user"&&A.tool_use_result?.file?.filePath&&U?.({type:"tool_use",jobId:i,tool:"Read",file:A.tool_use_result.file.filePath},i)}catch{}});let S=[];I.stderr?.on("data",D=>{S.push(D.toString())}),I.on("close",D=>{if(clearTimeout(F),y.close(),T.length===0&&w.size>0&&console.warn(`[Claude:${i}] No text captured. Event types seen: ${[...w].join(", ")}`),g)C=!0,B=`Timed out after ${Math.round(N/6e4)} minutes`;else if(D!==0&&D!==null){C=!0;let A=S.join("").trim(),Z=T.length===0&&w.size>0?` (no text captured, event types: ${[...w].join(", ")})`:"";B=A||`Claude process exited with code ${D}${Z}`}L({sessionId:G,text:T.join(""),success:!C,error:C?B:void 0,fileEdits:J.length>0?J:void 0})}),I.on("error",D=>{clearTimeout(F),C=!0,B=D.message,L({sessionId:G,text:T.join(""),success:!1,error:B,fileEdits:J.length>0?J:void 0})})});return{process:I,result:z}}import{spawn as $t}from"child_process";import{createInterface as Mt}from"readline";function Ct(i){let t=i.match(/^\/bin\/(?:ba)?sh\s+-\w+\s+"(.*)"$/s);return t?t[1].replace(/\\"/g,'"').replace(/\\\\/g,"\\"):i}function De(i,t){let{prompt:n,projectRoot:e,screenshotPath:s,resumeSessionId:l,model:c,onEvent:a}=t,h=[];l?(h.push("exec","resume",l),c&&h.push("-m",c),h.push("--json","--full-auto",n),s&&h.push("--image",s)):(h.push("exec","--json","--full-auto"),c&&h.push("-m",c),h.push(n),s&&h.push("--image",s));let f=$t("codex",h,{cwd:e,stdio:["ignore","pipe","pipe"],env:{...process.env}}),N=new Promise(U=>{let O,I=[],z=[],L=!1,G="",T=Mt({input:f.stdout}),J=new Set;T.on("line",B=>{if(B.trim())try{let g=JSON.parse(B),F=g.type??"unknown";if(J.add(F),F==="thread.started"&&g.thread_id&&!O&&(O=g.thread_id),(F==="item.agentMessage.delta"||F==="item/agentMessage/delta")&&g.delta?.text&&(I.push(g.delta.text),a?.({type:"delta",jobId:i,text:g.delta.text},i)),(F==="item.reasoning.delta"||F==="item/reasoning/delta")&&g.delta?.text&&a?.({type:"thinking",jobId:i,text:g.delta.text},i),(F==="item.started"||F==="item/started")&&g.item){let y=g.item.type;if(y==="command_execution"){let w=g.item.command,S=w?Ct(w):void 0,D=S?`Bash: ${S.split(`
|
|
30
|
+
`)[0].slice(0,80)}`:"Bash";z.push(D),a?.({type:"tool_use",jobId:i,tool:"Bash",...S?{content:S}:{}},i)}else if(y==="file_change"){let w=g.item.filename||g.item.path;z.push(w?`Edit ${w.split("/").pop()}`:"Edit"),a?.({type:"tool_use",jobId:i,tool:"Edit",...w?{file:w}:{}},i)}else if(y==="file_read"){let w=g.item.filename||g.item.path;z.push(w?`Read ${w.split("/").pop()}`:"Read"),a?.({type:"tool_use",jobId:i,tool:"Read",...w?{file:w}:{}},i)}else if(y==="web_search")z.push("WebSearch"),a?.({type:"tool_use",jobId:i,tool:"WebSearch"},i);else if(y==="mcp_tool_call"){let w=g.item.tool_name||g.item.name||"MCP";z.push(w),a?.({type:"tool_use",jobId:i,tool:w},i)}}if((F==="item.completed"||F==="item/completed")&&g.item){if(g.item.type==="agent_message"){let y=g.item.text;typeof y=="string"&&y&&(I.push(y),a?.({type:"delta",jobId:i,text:y},i))}else if(g.item.type==="reasoning"){let y=g.item.text;typeof y=="string"&&y&&a?.({type:"thinking",jobId:i,text:y},i)}else if(g.item.type==="file_change"&&Array.isArray(g.item.changes))for(let y of g.item.changes){let w=y.path||y.filename,S=y.kind==="add"?"Write":"Edit";w&&(z.push(`${S} ${w.split("/").pop()}`),a?.({type:"tool_use",jobId:i,tool:S,file:w},i))}}F==="turn.failed"&&(L=!0,G=g.error?.message||g.message||"Turn failed")}catch{}});let C=[];f.stderr?.on("data",B=>{C.push(B.toString())}),f.on("close",B=>{T.close(),I.length===0&&J.size>0&&console.warn(`[Codex:${i}] No text captured. Event types seen: ${[...J].join(", ")}`),B!==0&&B!==null&&(L=!0,G=C.join("")||`Codex process exited with code ${B}`),U({sessionId:O,text:I.join(""),success:!L,error:L?G:void 0,toolsUsed:z.length>0?z:void 0})}),f.on("error",B=>{L=!0,G=B.message,U({sessionId:O,text:I.join(""),success:!1,error:G,toolsUsed:z.length>0?z:void 0})})});return{process:f,result:N}}import{execFile as Ot}from"child_process";import{copyFile as Ne,mkdir as Be,readdir as At,readFile as _t,writeFile as Dt}from"fs/promises";import{join as re}from"path";var me=class{constructor(t){this.projectRoot=t;let n=re(t,".popmelt");this.decisionsDir=re(n,"decisions"),this.screenshotsDir=re(n,"screenshots")}async persist(t,n,e){try{await Be(this.decisionsDir,{recursive:!0}),await Be(this.screenshotsDir,{recursive:!0});try{await Ne(n,re(this.screenshotsDir,`s-${t.id}.png`))}catch{}for(let s=0;s<e.length;s++)try{let l=t.pastedImagePaths[s];l&&await Ne(e[s],re(this.screenshotsDir,l.replace("screenshots/","")))}catch{}await Dt(re(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 At(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 _t(re(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=>{Ot("git",["diff","HEAD"],{cwd:t,timeout:5e3,maxBuffer:1024*1024},(e,s)=>{if(e){n(null);return}n(s||null)})})}};import{readFile as Je,writeFile as ce}from"fs/promises";import{join as xe}from"path";var te="[Materializer]",Nt={version:1,materializedIds:[],lastRunAt:null,lastRunDecisionIds:[],lastRunError:null},ge=class{constructor(t,n,e={}){this.projectRoot=t;this.decisionStore=n;this.options=e;this.cachedIndex=null;this.running=!1;let s=xe(t,".popmelt");this.indexPath=xe(s,"materialized.json"),this.modelPath=xe(s,"model.json")}get isRunning(){return this.running}async loadModel(){try{let t=await Je(this.modelPath,"utf-8");return JSON.parse(t)}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 ce(this.modelPath,JSON.stringify(n,null,2)),console.log(`${te} Added component "${t}" to model`),{added:!0,alreadyExists:!1})}async updateToken(t,n){let e=await this.loadModel();e||(e={tokens:{},components:{},rules:[]});let s=t.split("."),l=e;for(let h=0;h<s.length-1;h++){let f=s[h];(!l[f]||typeof l[f]!="object")&&(l[f]={}),l=l[f]}let c=s[s.length-1],a;try{a=JSON.parse(n)}catch{a=null}if(a&&typeof a=="object"&&a!==null&&"value"in a)l[c]=a;else{let h=l[c];h&&typeof h=="object"&&h!==null&&"value"in h?h.value=n:l[c]=n}return await ce(this.modelPath,JSON.stringify(e,null,2)),console.log(`${te} Updated token "${t}" \u2192 "${n.slice(0,80)}"`),{updated:!0}}async removeToken(t){let n=await this.loadModel();if(!n)return{removed:!1};let e=t.split("."),s=n;for(let c=0;c<e.length-1;c++){let a=e[c];if(!s[a]||typeof s[a]!="object")return{removed:!1};s=s[a]}let l=e[e.length-1];return l in s?(delete s[l],await ce(this.modelPath,JSON.stringify(n,null,2)),console.log(`${te} Removed token "${t}" from model`),{removed:!0}):{removed:!1}}async removeComponent(t){let n=await this.loadModel();if(!n)return{removed:!1};let e=n.components;return!e||!e[t]?{removed:!1}:(delete e[t],await ce(this.modelPath,JSON.stringify(n,null,2)),console.log(`${te} Removed component "${t}" from model`),{removed:!0})}async getUnmaterializedPatternDecisions(){let t=await this.loadIndex(),n=new Set(t.materializedIds),s=(await this.decisionStore.listDecisionIds()).filter(c=>!n.has(c));return s.length===0?[]:(await this.decisionStore.loadDecisions(s)).filter(c=>c.resolutions.some(a=>(a.finalScope??a.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(f=>f.id);console.log(`${te} Processing ${n.length} pattern-scoped decision(s): ${n.join(", ")}`),this.options.onEvent?.({type:"materialize_started",decisionIds:n});let e=await this.loadModel(),s=Jt(t,e),l=!0,c;try{let{result:f}=fe(`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"}),N=await f;if(!N.success)l=!1,c=N.error,console.error(`${te} Claude spawn error:`,c);else{let U=Bt(N.text);U?(await ce(this.modelPath,JSON.stringify(U,null,2)),console.log(`${te} Successfully materialized ${n.length} decision(s) \u2192 ${this.modelPath}`)):(l=!1,c="No <model> block found in response",console.error(`${te} ${c}`))}}catch(f){l=!1,c=f instanceof Error?f.message:String(f),console.error(`${te} Error:`,c)}let a=await this.loadIndex(),h=new Set(a.materializedIds);for(let f of n)h.add(f);return a.materializedIds=[...h],a.lastRunAt=Date.now(),a.lastRunDecisionIds=n,a.lastRunError=c??null,await this.persistIndex(a),this.options.onEvent?.({type:"materialize_done",decisionIds:n,success:l,error:c}),{processedIds:n,success:l,error:c}}finally{this.running=!1}}async loadIndex(){if(this.cachedIndex)return this.cachedIndex;try{let t=await Je(this.indexPath,"utf-8"),n=JSON.parse(t);return this.cachedIndex=n,n}catch{return this.cachedIndex={...Nt,materializedIds:[],lastRunDecisionIds:[]},this.cachedIndex}}async persistIndex(t){this.cachedIndex=t;try{await ce(this.indexPath,JSON.stringify(t,null,2))}catch(n){console.error(`${te} Failed to write index:`,n)}}};function Bt(i){let t=i.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 Jt(i,t){let n=i.map(s=>{let c=s.resolutions.filter(f=>(f.finalScope??f.inferredScope)?.breadth==="pattern").map(f=>{let U=(f.finalScope??f.inferredScope)?.target??"unknown",O=f.filesModified?.join(", ")??"none";return`- **${f.summary}** [scope: pattern/${U}]
|
|
31
|
+
Files modified: ${O}`}).join(`
|
|
32
|
+
`),a=s.annotations.map(f=>f.instruction).filter(Boolean).join(`
|
|
33
|
+
`),h=s.gitDiff?`
|
|
34
34
|
\`\`\`diff
|
|
35
|
-
${
|
|
36
|
-
\`\`\``:"";return`### Decision ${
|
|
37
|
-
Page: ${
|
|
38
|
-
${
|
|
39
|
-
${
|
|
40
|
-
${
|
|
35
|
+
${s.gitDiff.slice(0,2e3)}
|
|
36
|
+
\`\`\``:"";return`### Decision ${s.id} (${new Date(s.createdAt).toISOString()})
|
|
37
|
+
Page: ${s.url}
|
|
38
|
+
${c}
|
|
39
|
+
${h}
|
|
40
|
+
${a?`
|
|
41
41
|
Original instructions:
|
|
42
|
-
${
|
|
42
|
+
${a}`:""}`}).join(`
|
|
43
43
|
|
|
44
44
|
`);return`You are extracting a local design model from accumulated design decisions.
|
|
45
45
|
|
|
@@ -48,15 +48,15 @@ ${c}`:""}`}).join(`
|
|
|
48
48
|
2. Determine what design tokens, component patterns, and rules to add or update.
|
|
49
49
|
3. Output the complete updated model as a JSON object inside <model> tags.
|
|
50
50
|
|
|
51
|
-
${
|
|
51
|
+
${t?`## Current Model
|
|
52
52
|
\`\`\`json
|
|
53
|
-
${JSON.stringify(
|
|
53
|
+
${JSON.stringify(t,null,2)}
|
|
54
54
|
\`\`\`
|
|
55
55
|
|
|
56
56
|
Merge the new decisions into the existing model. Preserve existing entries that are not contradicted by new decisions.`:"No model exists yet. Create one from scratch based on the decisions below."}
|
|
57
57
|
|
|
58
58
|
## Design Decisions to Materialize
|
|
59
|
-
${
|
|
59
|
+
${n}
|
|
60
60
|
|
|
61
61
|
## Output Format
|
|
62
62
|
Output the full model inside <model> tags. The model is a JSON object with these sections:
|
|
@@ -95,40 +95,40 @@ Example:
|
|
|
95
95
|
{ "value": "8px", "property": "gap", "bindings": ["gap-2"] }
|
|
96
96
|
- property: "gap", "padding", or "margin" \u2014 which CSS property this token controls
|
|
97
97
|
- bindings: Tailwind class names (without responsive prefixes) that use this token
|
|
98
|
-
Include property and bindings when the decision context has class-level evidence.`}import{readFile as
|
|
99
|
-
`),
|
|
100
|
-
`)}),
|
|
101
|
-
`)}),
|
|
102
|
-
`)}),
|
|
103
|
-
`,"utf-8"),{installed:!0,provider:"claude",scope:"user"}}async function
|
|
98
|
+
Include property and bindings when the decision context has class-level evidence.`}import{readFile as Le}from"fs/promises";import{homedir as ze}from"os";import{join as le}from"path";var Ie=/popmelt/i;function be(){return{found:!1,name:null,scope:null,disabled:!1}}function de(i,t,n=!1){return{found:!0,name:i,scope:t,disabled:n}}function ye(i){for(let t of Object.keys(i))if(Ie.test(t))return t;return null}async function Pe(i){try{let t=await Le(i,"utf-8");return JSON.parse(t)}catch{return null}}async function ke(i){let t=ze(),n=await Pe(le(t,".claude.json"));if(n&&typeof n=="object"){let s=n;if(s.mcpServers&&typeof s.mcpServers=="object"){let l=ye(s.mcpServers);if(l)return de(l,"user")}if(s.projects&&typeof s.projects=="object"){let c=s.projects[i];if(c&&typeof c=="object"){let a=c;if(a.mcpServers&&typeof a.mcpServers=="object"){let h=ye(a.mcpServers);if(h){let f=Array.isArray(a.disabledMcpjsonServers)&&a.disabledMcpjsonServers.some(N=>Ie.test(N));return de(h,"project",f)}}}}}let e=await Pe(le(i,".mcp.json"));if(e&&typeof e=="object"){let s=e;if(s.mcpServers&&typeof s.mcpServers=="object"){let c=ye(s.mcpServers);if(c){let a=await Ue(i,c);return de(c,"mcp.json",a)}}let l=ye(s);if(l&&l!=="mcpServers"){let c=await Ue(i,l);return de(l,"mcp.json",c)}}return be()}async function Ue(i,t){let n=le(i,".claude","settings.local.json"),e=await Pe(n);if(e&&typeof e=="object"){let s=e;if(Array.isArray(s.disabledMcpjsonServers))return s.disabledMcpjsonServers.some(l=>l===t)}return!1}var Ut=/^\[mcp_servers\.([^\]]+)\]/;function Ft(i){let t=i.split(`
|
|
99
|
+
`),n=[],e=null;for(let s of t){let l=s.match(Ut);l?(e&&n.push({name:e.name,body:e.bodyLines.join(`
|
|
100
|
+
`)}),e={name:l[1],bodyLines:[]}):s.startsWith("[")?e&&(n.push({name:e.name,body:e.bodyLines.join(`
|
|
101
|
+
`)}),e=null):e&&e.bodyLines.push(s)}return e&&n.push({name:e.name,body:e.bodyLines.join(`
|
|
102
|
+
`)}),n}function Lt(i){return/enabled\s*=\s*false/i.test(i)}async function Te(i){let t=process.env.CODEX_HOME||le(ze(),".codex"),n=await Fe(le(t,"config.toml"),"user");if(n.found)return n;let e=await Fe(le(i,".codex","config.toml"),"project");return e.found?e:be()}async function Fe(i,t){try{let n=await Le(i,"utf-8"),e=Ft(n);for(let s of e)if(Ie.test(s.name)){let l=Lt(s.body);return de(s.name,t,l)}}catch{}return be()}import{mkdir as zt,readFile as He,writeFile as je}from"fs/promises";import{homedir as We}from"os";import{dirname as Ht,join as Ee}from"path";var Ge="https://mcp.popmelt.com/mcp";async function qe(i=Ge){let t=Ee(We(),".claude.json"),n;try{let s=await He(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:i},await je(t,JSON.stringify(n,null,2)+`
|
|
103
|
+
`,"utf-8"),{installed:!0,provider:"claude",scope:"user"}}async function Qe(i=Ge){let t=process.env.CODEX_HOME||Ee(We(),".codex"),n=Ee(t,"config.toml"),e;try{e=await He(n,"utf-8")}catch{e=""}if(/\[mcp_servers\.[^\]]*popmelt[^\]]*\]/i.test(e))return{installed:!1,provider:"codex",scope:null,reason:"already_configured"};await zt(Ht(n),{recursive:!0});let s=`
|
|
104
104
|
[mcp_servers.popmelt]
|
|
105
|
-
url = "${
|
|
106
|
-
`;return await
|
|
105
|
+
url = "${i}"
|
|
106
|
+
`;return await je(n,e+s,"utf-8"),{installed:!0,provider:"codex",scope:"user"}}async function Re(i){let n=(i.headers["content-type"]||"").match(/boundary=(?:"([^"]+)"|([^\s;]+))/);if(!n)throw new Error("Missing multipart boundary");let e=n[1]||n[2],s=await jt(i),l=Buffer.from(`--${e}`),c=Buffer.from(`--${e}--`),a,h,f,N,U,O,I,z,L,G,T,J,C=[],B=[],g=0,F=[];for(;g<s.length;){let y=s.indexOf(l,g);if(y===-1)break;let w=y+l.length;if(s.slice(y,y+c.length).equals(c))break;let S=w;s[S]===13&&s[S+1]===10&&(S+=2);let D=s.indexOf(`\r
|
|
107
107
|
\r
|
|
108
|
-
`,
|
|
109
|
-
### ${
|
|
110
|
-
`)}function
|
|
111
|
-
`)}function
|
|
112
|
-
`)}function
|
|
113
|
-
`)[0].trim().slice(0,60):"Running command";break;case"Glob":
|
|
114
|
-
data: ${JSON.stringify(
|
|
108
|
+
`,S);if(D===-1)break;let A=s.slice(S,D).toString("utf-8"),Z=D+4,R=s.indexOf(l,Z),ae=R!==-1?R-2:s.length;F.push({headers:A,body:s.slice(Z,ae)}),g=R!==-1?R:s.length}for(let y of F){let w=y.headers.match(/name="([^"]+)"/);if(!w)continue;let S=w[1];if(S==="screenshot")a=y.body;else if(S==="feedback")h=y.body.toString("utf-8");else if(S==="color")f=y.body.toString("utf-8");else if(S==="provider")N=y.body.toString("utf-8");else if(S==="model")U=y.body.toString("utf-8");else if(S==="goal")O=y.body.toString("utf-8");else if(S==="pageUrl")I=y.body.toString("utf-8");else if(S==="viewport")z=y.body.toString("utf-8");else if(S==="planId")L=y.body.toString("utf-8");else if(S==="manifest")G=y.body.toString("utf-8");else if(S==="tasks")T=y.body.toString("utf-8");else if(S==="sourceId")J=y.body.toString("utf-8");else if(S.startsWith("screenshot-")){let D=S.slice(11);try{let A=decodeURIComponent(D);B.push({pathname:A,data:y.body})}catch{}}else if(S.startsWith("image-")){let D=S.split("-"),A=parseInt(D[D.length-1],10),Z=D.slice(1,-1).join("-");Z&&!isNaN(A)&&C.push({annotationId:Z,index:A,data:y.body})}}if(!a)throw new Error("Missing screenshot field");return h||(h=""),{screenshot:a,feedback:h,color:f,provider:N,model:U,goal:O,pageUrl:I,viewport:z,planId:L,manifest:G,tasks:T,sourceId:J,pastedImages:C,pageScreenshots:B}}function jt(i){return new Promise((t,n)=>{let e=[];i.on("data",s=>e.push(s)),i.on("end",()=>t(Buffer.concat(e))),i.on("error",n)})}function ue(i,t){let n=[];if(i.annotations.length>0){n.push("## Annotations");for(let e of i.annotations){let s=e.elements.map(a=>{let h=[a.selector];return a.reactComponent&&h.push(`(${a.reactComponent})`),h.join(" ")}).join(", "),l=e.instruction||"No text";n.push(`- id=${e.id} [${e.type}] ${l} \u2192 Elements: ${s||"none"}`);let c=t?.[e.id];if(c&&c.length>0)for(let a of c)n.push(` Attached image: use the Read tool to view ${a}`)}}if(i.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 i.styleModifications){let s=e.element?.reactComponent?`(${e.element.reactComponent})`:"";for(let l of e.changes)n.push(`- ${e.selector} ${s}: ${l.property} ${l.original} \u2192 ${l.modified}`)}}if(i.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 i.spacingTokenChanges){n.push(`
|
|
109
|
+
### ${e.tokenName}: ${e.originalPx}px \u2192 ${e.newPx}px`);for(let s of e.affectedElements){let l=s.reactComponent?` (${s.reactComponent})`:"";s.matchedClass&&s.suggestedClass?n.push(`- ${s.selector}${l}: \`${s.matchedClass}\` \u2192 \`${s.suggestedClass}\``):n.push(`- ${s.selector}${l}: ${s.property} ${e.originalPx}px \u2192 ${e.newPx}px`),n.push(` class="${s.className}"`)}}}if(i.inspectedElement){let e=i.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(`
|
|
110
|
+
`)}function Xe(i,t,n){let e=[],l=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(""),!l&&n?.provider!=="codex"&&(e.push(`IMPORTANT: First, use the Read tool to view the screenshot at: ${i}`),e.push("")),e.push(`The developer annotated their running app at ${t.url} (${t.viewport.width}x${t.viewport.height}).`),n?.threadHistory&&n.threadHistory.length>0){e.push(""),e.push("## Previous Conversation");let c=0;for(let a of n.threadHistory)if(a.role==="human")c++,a.replyToQuestion?(e.push(`### Round ${c} (human) \u2014 reply`),e.push(`"${a.replyToQuestion}"`)):(e.push(`### Round ${c} (human)`),a.feedbackSummary&&e.push(`Annotations: ${a.feedbackSummary}`),a.annotationIds&&a.annotationIds.length>0&&e.push(`Annotation IDs: ${a.annotationIds.join(", ")}`));else if(a.question)e.push(`### Round ${c} (assistant) \u2014 question`),e.push(`"${a.question}"`);else{if(e.push(`### Round ${c} (assistant)`),a.responseText&&e.push(`Response: ${a.responseText}`),a.resolutions&&a.resolutions.length>0)for(let h of a.resolutions){let f=h.finalScope??h.inferredScope,N=f?` [${f.breadth} ${f.target}]`:"";e.push(`- ${h.annotationId}: ${h.status}${N} \u2014 ${h.summary}`),h.filesModified&&h.filesModified.length>0&&e.push(` Files: ${h.filesModified.join(", ")}`)}a.toolsUsed&&a.toolsUsed.length>0&&e.push(`Tools used: ${a.toolsUsed.join(", ")}`)}e.push(""),e.push("The current round is shown in full below.")}if(n?.designModel){e.push(""),e.push("## Established Design Policies"),e.push("This project has an established design model (stored in .popmelt/model.json), extracted from the developer's previous design decisions. When making changes, follow these patterns unless the developer explicitly overrides them. When asked about design tokens, component patterns, or design decisions, reference this model as the authoritative source.");let c=n.designModel.rules;if(Array.isArray(c)&&c.length>0){e.push(""),e.push("Rules:");for(let f of c)typeof f=="string"&&e.push(`- ${f}`)}let a=n.designModel.tokens;a&&typeof a=="object"&&(e.push(""),e.push("Design tokens:"),e.push("```json"),e.push(JSON.stringify(a,null,2)),e.push("```"));let 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.")),l){let c=n?.screenshotPaths??{},a=new Map;for(let h of t.annotations){let f=h.pathname||new URL(t.url).pathname;a.has(f)||a.set(f,[]),a.get(f).push(h)}e.push(""),e.push("The developer annotated multiple pages. Each page section includes its own screenshot where available.");for(let[h,f]of a){e.push(""),e.push(`## Page: ${h}`);let N=c[h];N&&n?.provider!=="codex"?e.push(`IMPORTANT: Use the Read tool to view the screenshot at: ${N}`):!N&&n?.provider!=="codex"&&(Object.keys(c).length===0?(e.push(`IMPORTANT: Use the Read tool to view the screenshot at: ${i}`),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 U={...t,annotations:f,styleModifications:t.styleModifications},O=ue(U,n?.imagePaths);O&&(e.push(""),e.push(O))}}else{let c=ue(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(`
|
|
111
|
+
`)}function Ye(i){return i.match(/<question>\s*([\s\S]*?)\s*<\/question>/)?.[1]??null}function Ve(i,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: ${i}`);let l=t.find(c=>c.role==="human"&&c.feedbackContext);if(l?.feedbackContext&&(s.push(""),s.push(l.feedbackContext)),t.length>0){s.push(""),s.push("## Conversation History");let c=0;for(let a of t)a.role==="human"?(c++,a.replyToQuestion?(s.push(`### Round ${c} (human) \u2014 reply`),s.push(`"${a.replyToQuestion}"`)):(s.push(`### Round ${c} (human)`),a.feedbackSummary&&s.push(`Annotations: ${a.feedbackSummary}`))):a.question?(s.push(`### Round ${c} (assistant) \u2014 question`),s.push(`"${a.question}"`)):(s.push(`### Round ${c} (assistant)`),a.responseText&&s.push(`Response: ${a.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 c of e)s.push(`Attached image: use the Read tool to view the image at: ${c}`)}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(`
|
|
112
|
+
`)}function Wt(i){if(typeof i!="object"||i===null)return!1;let t=i;return(t.breadth==="instance"||t.breadth==="pattern")&&(t.target==="element"||t.target==="component"||t.target==="token")}function Gt(i){if(typeof i!="object"||i===null||typeof i.annotationId!="string"||i.status!=="resolved"&&i.status!=="needs_review"||typeof i.summary!="string")return!1;let t=i;for(let n of["declaredScope","inferredScope","finalScope"])if(t[n]!==void 0&&t[n]!==null&&!Wt(t[n]))return!1;return!0}function Ke(i){let t=i.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(Gt):[]}catch{return[]}}function Ze(i){let t=i.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 ve=class{constructor(t=5){this.queue=[];this.activeJobs=new Map;this.activeProcesses=new Map;this.listeners=new Set;this.processor=null;this.eventBuffers=new Map;this.accumulators=new Map;this.cleanupTimers=new Map;this.maxConcurrency=t}setProcessor(t){this.processor=t}get active(){let t=this.activeJobs.values().next();return t.done?null:t.value}get allActive(){return Array.from(this.activeJobs.values())}get activeCount(){return this.activeJobs.size}get depth(){return this.queue.length}get isRunning(){return this.activeJobs.size>0}setActiveProcess(t,n){n?this.activeProcesses.set(t,n):this.activeProcesses.delete(t)}enqueue(t){return this.queue.push(t),this.processNext(),this.queue.length+this.activeJobs.size}addListener(t){return this.listeners.add(t),()=>this.listeners.delete(t)}broadcast(t,n,e){let s=this.bufferEvent(n,t);for(let l of this.listeners)l(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:""},l=this.activeJobs.has(t);if(!e)return null;let c=n<0?e.events:e.events.filter(a=>a.seq>n);return{jobId:t,events:c,currentSeq:e.nextSeq-1,accumulated:{...s},jobActive:l}}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 l=!1,c=()=>{l||(l=!0,s())};e.on("exit",c),e.on("error",c),setTimeout(()=>{if(!l){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,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 qt,readFile as Qt,writeFile as Xt}from"fs/promises";import{dirname as Yt,join as Vt}from"path";var Kt={version:1,threads:{}},we=class{constructor(t){this.cache=null;this.writeChain=Promise.resolve();this.filePath=Vt(t,".popmelt","threads.json")}async load(){if(this.cache)return this.cache;try{let t=await Qt(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={...Kt,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(c=>e.has(c)))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 l=new Set(s.elementIdentifiers);for(let c of n)l.has(c)||s.elementIdentifiers.push(c);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,l)=>l.updatedAt-s.updatedAt),e.slice(0,t).map(s=>{let c=s.messages.find(a=>a.role==="human")?.feedbackSummary||"[thread]";return{id:s.id,createdAt:s.createdAt,updatedAt:s.updatedAt,preview:c,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 qt(Yt(this.filePath),{recursive:!0}),await Xt(this.filePath,JSON.stringify(this.cache,null,2))}catch(t){console.error("[ThreadStore] Failed to persist:",t)}}),await this.writeChain}};var et="0.6.6";var pn=1111,hn=["Read","Edit","Write","Glob","Grep","Bash","WebFetch","WebSearch","Bash(curl:*)"],fn=1800*1e3,mn=3600*1e3;function st(i){if(!i)return!1;try{let t=new URL(i);return t.hostname==="localhost"||t.hostname==="127.0.0.1"}catch{return!1}}function gn(i){let t=[];for(let n of i)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,l=n.content??void 0,c=s?s.split("/").pop()??s:void 0,a;switch(e){case"Read":a=c?`Reading ${c}`:"Reading file";break;case"Edit":a=c?`Editing ${c}`:"Editing file";break;case"Write":a=c?`Writing ${c}`:"Writing file";break;case"Bash":a=l?l.split(`
|
|
113
|
+
`)[0].trim().slice(0,60):"Running command";break;case"Glob":a="Searching files";break;case"Grep":a="Searching code";break;case"WebFetch":a="Fetching page";break;case"WebSearch":a="Searching web";break;default:a=e?`Using ${e}`:"tool";break}let h=s??l??void 0,f=t[t.length-1];f&&f.kind==="tool_group"&&f.tool===e?f.items.push({label:a,detail:h}):t.push({kind:"tool_group",tool:e,items:[{label:a,detail:h}]})}return t}function yn(i,t){let n=i.headers.origin;st(n)&&(t.setHeader("Access-Control-Allow-Origin",n),t.setHeader("Access-Control-Allow-Methods","GET, POST, PATCH, DELETE, OPTIONS"),t.setHeader("Access-Control-Allow-Headers","Content-Type"))}function v(i,t,n){i.writeHead(t,{"Content-Type":"application/json"}),i.end(JSON.stringify(n))}function vn(i,t){if(!i)return t;let n=i.match(/^#?([0-9a-f]{2})([0-9a-f]{2})([0-9a-f]{2})$/i);if(!n)return t;let[,e,s,l]=n;return`\x1B[38;2;${parseInt(e,16)};${parseInt(s,16)};${parseInt(l,16)}m${t}\x1B[0m`}function Me(i,t){try{i.res.write(`event: ${t.type}
|
|
114
|
+
data: ${JSON.stringify(t)}
|
|
115
115
|
|
|
116
|
-
`)}catch
|
|
116
|
+
`)}catch{}}async function wn(i){try{let t=new AbortController,n=setTimeout(()=>t.abort(),500),e=await fetch(`http://127.0.0.1:${i}/status`,{signal:t.signal});return clearTimeout(n),e.ok?await e.json():null}catch{return null}}function Sn(i,t){return new Promise((n,e)=>{let s=c=>{i.removeListener("listening",l),e(c)},l=()=>{i.removeListener("error",s),n()};i.once("error",s),i.once("listening",l),i.listen(t,"127.0.0.1")})}async function ot(i={}){let t=i.port??pn,n=i.projectRoot??process.cwd(),e=tn("sha256").update(n).digest("hex").slice(0,12),s=i.devOrigin??(process.env.PORT?`http://localhost:${process.env.PORT}`:null),l=i.tempDir??ne(cn(),"popmelt-bridge"),c=i.maxTurns??40,a=i.maxBudgetUsd??1,h=[...i.allowedTools??hn],f=i.claudePath??"claude",N=i.provider??"claude",U=i.timeoutMs,O=t,I={};for(let o of["claude","codex"])try{let r=Zt("which",[o],{encoding:"utf-8"}).trim();I[o]={available:!0,path:r}}catch{I[o]={available:!1,path:null}}function z(o,r){return new Promise(p=>{let d=en(r,["--version"],{stdio:["ignore","ignore","ignore"]}),m=!1,u=$=>{m||(m=!0,p($))},b=setTimeout(()=>{d.kill("SIGTERM"),u(!0)},5e3);d.on("error",()=>{clearTimeout(b),u(!1)}),d.on("close",$=>{clearTimeout(b),u($===0)})})}let[L,G]=await Promise.all([ke(n),Te(n)]);I.claude&&(I.claude.mcp=L),I.codex&&(I.codex.mcp=G),L.found&&L.name&&h.push(`mcp__${L.name}__*`),await nn(l,{recursive:!0}),nt(l).catch(()=>{});let T=new ve(1),J=new Set,C=new we(n),B=new me(n),g=new ge(n,B,{claudePath:f,onEvent:o=>{for(let r of J)Me(r,o)}}),F=20,y=300*1e3,w=[],S=null,D;T.addListener((o,r,p)=>{for(let d of J)(!p||!d.sourceId||d.sourceId===p)&&Me(d,o)}),T.setProcessor(async o=>{let r=o._replyPrompt,p=o._replyImagePaths,d=o.provider??N,m;if(o.threadId){let E=await C.getThread(o.threadId);if(E){for(let P=E.messages.length-1;P>=0;P--)if(E.messages[P].sessionId){m=E.messages[P].sessionId;break}}}let u;if(m&&r){let E=(await C.getThread(o.threadId))?.messages.filter(j=>j.role==="human").pop();if(u=E?.replyToQuestion||E?.feedbackSummary||"",p&&p.length>0){u+=`
|
|
117
117
|
|
|
118
|
-
The developer attached reference images with their reply:`;for(let
|
|
119
|
-
Attached image: use the Read tool to view the image at: ${
|
|
118
|
+
The developer attached reference images with their reply:`;for(let j of p)u+=`
|
|
119
|
+
Attached image: use the Read tool to view the image at: ${j}`}u+=`
|
|
120
120
|
|
|
121
|
-
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(
|
|
121
|
+
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=ue(o.feedback,o.imagePaths)+`
|
|
122
122
|
|
|
123
123
|
Follow the developer's instructions. If they ask for changes, apply them to the source files.
|
|
124
124
|
|
|
125
|
-
After completing work, output a <resolution> block with declaredScope and inferredScope. If unclear, output a <question> block.`+(
|
|
125
|
+
After completing work, output a <resolution> block with declaredScope and inferredScope. If unclear, output a <question> block.`+(d!=="codex"?`
|
|
126
126
|
|
|
127
|
-
IMPORTANT: First, use the Read tool to view the updated screenshot at: ${
|
|
128
|
-
${
|
|
127
|
+
IMPORTANT: First, use the Read tool to view the updated screenshot at: ${o.screenshotPath}`:"");else{let E=!r&&o.threadId?await C.getThreadHistory(o.threadId):void 0,P=r?null:await g.loadModel();u=r??Xe(o.screenshotPath,o.feedback,{threadHistory:E&&E.length>0?E:void 0,provider:d,imagePaths:o.imagePaths,designModel:P??void 0,screenshotPaths:o.screenshotPaths})}let b=vn(o.color,`[\u22B9 ${O}:${o.id}]`),$=o.screenshotPaths&&Object.keys(o.screenshotPaths).length>0?`${Object.keys(o.screenshotPaths).length} pages [${Object.keys(o.screenshotPaths).join(", ")}]`:o.screenshotPath;console.log(`${b} Reviewing ${$} (provider: ${d})${o.threadId?` (thread: ${o.threadId})`:""}${m?` (resuming: ${m.slice(0,8)})`:""}`);let X=(E,P)=>{E.type==="delta"&&"text"in E?T.accumulateText(P,"response",E.text):E.type==="thinking"&&"text"in E&&T.accumulateText(P,"thinking",E.text),T.broadcast(E,P,o.sourceId)},{process:K,result:q}=d==="codex"?De(o.id,{prompt:u,projectRoot:n,screenshotPath:o.screenshotPath,resumeSessionId:m,model:o.model,onEvent:X}):fe(o.id,{prompt:u,projectRoot:n,maxTurns:c,maxBudgetUsd:a,allowedTools:h,claudePath:f,resumeSessionId:m,model:o.model,timeoutMs:U,onEvent:X});T.setActiveProcess(o.id,K);let x=await q;if(o.result=x.text,x.success){console.log(`${b} Iteration complete`),x.fileEdits&&x.fileEdits.length>0&&console.log(`${b} Captured ${x.fileEdits.length} file edit(s): ${x.fileEdits.map(H=>`${H.tool} ${H.file_path}`).join(", ")}`),o.status="done";let E=Ye(x.text),P=Ke(x.text);if(P.length>0&&o.annotationIds&&o.annotationIds.length>0){let H=new Set(o.annotationIds);P.every(M=>H.has(M.annotationId))||(P=P.map((M,W)=>({...M,annotationId:o.annotationIds[W%o.annotationIds.length]})))}let j=x.fileEdits&&x.fileEdits.length>0?x.fileEdits.map(H=>`${H.tool} ${H.file_path.split("/").pop()}`):x.toolsUsed,V=T.getBufferedEvents(o.id),se=V?gn(V.events):void 0;o.threadId&&await C.appendMessage(o.threadId,{role:"assistant",timestamp:Date.now(),jobId:o.id,responseText:x.text,resolutions:P.length>0?P:void 0,question:E??void 0,sessionId:x.sessionId,toolsUsed:j,segments:se&&se.length>0?se:void 0,model:o.model,provider:o.provider}),B.captureGitDiff(n).then(async H=>{let _=Date.now(),M=o.imagePaths?Object.values(o.imagePaths).flat():[],W=[];if(o.imagePaths)for(let[ie,k]of Object.entries(o.imagePaths))for(let Q=0;Q<k.length;Q++)W.push(`screenshots/p-${o.id}-${ie}-${Q}.png`);await B.persist({version:1,id:o.id,createdAt:o.createdAt,completedAt:_,durationMs:_-o.createdAt,url:o.feedback.url,viewport:o.feedback.viewport,screenshotPath:`screenshots/s-${o.id}.png`,pastedImagePaths:W,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:P.length>0?P:[],question:E??void 0,fileEdits:x.fileEdits??[],toolsUsed:j,gitDiff:H},o.screenshotPath,M)}).catch(()=>{}),P.length>0&&P.some(_=>(_.finalScope??_.inferredScope)?.breadth==="pattern")&&g.run().catch(()=>{}),E&&(console.log(`${b} \u{1F4AC} Question detected: "${E.slice(0,120)}" \u2192 broadcasting to ${J.size} SSE clients (threadId=${o.threadId??o.id}, annotationIds=${o.annotationIds?.join(",")??"none"})`),T.broadcast({type:"question",jobId:o.id,threadId:o.threadId??o.id,question:E,annotationIds:o.annotationIds},o.id,o.sourceId));let ee=Ze(x.text);ee.length>0&&(console.log(`${b} Novel pattern(s): ${ee.map(H=>`${H.category}/${H.element}`).join(", ")}`),T.broadcast({type:"novel_patterns",jobId:o.id,patterns:ee,threadId:o.threadId},o.id,o.sourceId)),T.broadcast({type:"done",jobId:o.id,success:!0,resolutions:P.length>0?P:void 0,responseText:x.text,threadId:o.threadId},o.id,o.sourceId),w.push({id:o.id,status:"done",completedAt:Date.now(),threadId:o.threadId,annotationIds:o.annotationIds})}else console.error(`${b} Error: ${x.error}`),o.status="error",o.error=x.error,o.threadId&&await C.appendMessage(o.threadId,{role:"assistant",timestamp:Date.now(),jobId:o.id,error:x.error||"Unknown error",model:o.model,provider:o.provider}),T.broadcast({type:"error",jobId:o.id,threadId:o.threadId,message:x.error||"Unknown error"},o.id,o.sourceId),w.push({id:o.id,status:"error",completedAt:Date.now(),error:x.error,threadId:o.threadId,annotationIds:o.annotationIds});let Y=Date.now()-y;for(;w.length>0&&(w[0].completedAt<Y||w.length>F);)w.shift()});let A=an(async(o,r)=>{if(yn(o,r),o.method==="OPTIONS"){r.writeHead(204),r.end();return}let p=o.url||"/",d=p.split("?")[0],m=new URL(p,`http://127.0.0.1:${O}`),u=m.pathname;try{if(o.method==="POST"&&u==="/send")await Z(o,r);else if(o.method==="GET"&&u==="/events")ae(o,r);else if(o.method==="GET"&&u==="/status")pe(r);else if(o.method==="PATCH"&&u==="/config")await dt(o,r);else if(o.method==="POST"&&u==="/shutdown")v(r,200,{ok:!0}),setTimeout(()=>process.exit(0),100);else if(o.method==="GET"&&u==="/capabilities")v(r,200,{providers:I});else if(o.method==="POST"&&u==="/mcp/install")await ht(o,r);else if(o.method==="POST"&&u==="/reply")await R(o,r);else if(o.method==="POST"&&u==="/cancel")ut(o,r);else if(o.method==="POST"&&u==="/materialize")await pt(r);else if(o.method==="POST"&&u==="/model/component")await ft(o,r);else if(o.method==="DELETE"&&u==="/model/component")await mt(o,r);else if(o.method==="PATCH"&&u==="/model/token")await gt(o,r);else if(o.method==="DELETE"&&u==="/model/token")await yt(o,r);else if(o.method==="GET"&&u==="/model"){let b=await g.loadModel();v(r,200,{model:b})}else if(o.method==="GET"&&u.startsWith("/jobs/")&&u.endsWith("/events")){let b=u.slice(6,u.length-7),$=parseInt(m.searchParams.get("afterSeq")??"-1",10),X=T.getBufferedEvents(b,isNaN($)?-1:$);X?v(r,200,X):v(r,404,{error:"Unknown job"})}else if(o.method==="GET"&&d.startsWith("/files/"))await xt(d.slice(7),r);else if(o.method==="GET"&&u==="/threads/recent"){let b=Math.min(Math.max(parseInt(m.searchParams.get("limit")??"5",10)||5,1),20),$=await C.listRecent(b);v(r,200,$)}else if(o.method==="GET"&&u.startsWith("/thread/")){let b=u.slice(8);await Pt(b,r)}else o.method==="GET"&&(u==="/canvas"||u==="/canvas/")?vt(o,r):o.method==="GET"&&u==="/canvas/manifest"?await wt(r):o.method==="GET"&&u==="/canvas/app.mjs"?await St(r):v(r,404,{error:"Not found"})}catch(b){console.error("[Bridge] Request error:",b),v(r,500,{error:b instanceof Error?b.message:"Internal error"})}});async function Z(o,r){let{screenshot:p,feedback:d,color:m,provider:u,model:b,sourceId:$,pastedImages:X,pageScreenshots:K}=await Re(o),q;try{q=JSON.parse(d)}catch{v(r,400,{error:"Invalid feedback JSON"});return}let x=$e().slice(0,8),Y={};if(K.length>0)for(let k of K){let Q=encodeURIComponent(k.pathname),oe=ne(l,`screenshot-${x}-${Q}.png`);await Se(oe,k.data),Y[k.pathname]=oe}let E=ne(l,`screenshot-${x}.png`);await Se(E,p);let P={};if(X.length>0)for(let k of X){let Q=ne(l,`pasted-${x}-${k.annotationId}-${k.index}.png`);await Se(Q,k.data),P[k.annotationId]||(P[k.annotationId]=[]),P[k.annotationId].push(Q)}let j=q.annotations.map(k=>k.linkedSelector?k.pathname?`${k.pathname}:${k.linkedSelector}`:k.linkedSelector:null).filter(k=>!!k),V;if(j.length>0){let k=await C.findContinuationThread(j);k?(V=k.id,await C.addElementIdentifiers(V,j)):V=(await C.createThread(x,j)).id}else V=(await C.createThread(x,[])).id;let se=q.annotations.map(k=>k.id),ee=Object.keys(Y).length>0,H={id:x,status:"queued",screenshotPath:E,feedback:q,createdAt:Date.now(),color:m,threadId:V,annotationIds:se,provider:u==="claude"||u==="codex"?u:void 0,model:b||void 0,...Object.keys(P).length>0?{imagePaths:P}:{},sourceId:$||void 0,...ee?{screenshotPaths:Y}:{}},_=new Set(q.annotations.map(k=>k.pathname).filter(Boolean)),M;if(_.size>1){let k=new Map;for(let Q of q.annotations){let oe=Q.pathname||"(unknown)";k.has(oe)||k.set(oe,[]),k.get(oe).push(Q.instruction||`[${Q.type}]`)}M=[...k.entries()].map(([Q,oe])=>`\`${Q}\`
|
|
128
|
+
${oe.map(bt=>`- ${bt}`).join(`
|
|
129
129
|
`)}`).join(`
|
|
130
|
-
`)}else
|
|
130
|
+
`)}else M=q.annotations.map(k=>k.instruction||`[${k.type}]`).join("; ");let W=ue(q,Object.keys(P).length>0?P:void 0);await C.appendMessage(V,{role:"human",timestamp:Date.now(),jobId:x,screenshotPath:E,...ee?{screenshotPaths:Y}:{},...Object.keys(P).length>0?{imagePaths:P}:{},annotationIds:se,feedbackSummary:M,feedbackContext:W||void 0});let ie=T.enqueue(H);v(r,200,{jobId:x,position:ie,threadId:V})}async function R(o,r){let p=o.headers["content-type"]||"",d,m,u,b,$,X,K=[];if(p.includes("multipart/form-data")){let _=await Re(o),M=_.feedback?JSON.parse(_.feedback):{};d=M.threadId,m=M.reply,u=M.color,b=M.provider,$=M.model,X=M.sourceId||_.sourceId;for(let W of _.pastedImages)K.push(W.data)}else{let _=[];for await(let ie of o)_.push(typeof ie=="string"?Buffer.from(ie):ie);let M=Buffer.concat(_).toString("utf-8"),W;try{W=JSON.parse(M)}catch{v(r,400,{error:"Invalid JSON"});return}d=W.threadId,m=W.reply,u=W.color,b=W.provider,$=W.model,X=W.sourceId}if(!d||!m){v(r,400,{error:"Missing threadId or reply"});return}if(!await C.getThread(d)){v(r,404,{error:"Thread not found"});return}let x=$e().slice(0,8),Y=[];for(let _=0;_<K.length;_++){let M=ne(l,`reply-${x}-${_}.png`);await Se(M,K[_]),Y.push(M)}let E="";{let _=await C.getThreadHistory(d);for(let M=_.length-1;M>=0;M--)if(_[M].screenshotPath){E=_[M].screenshotPath;break}}if(!E){v(r,400,{error:"No screenshot available"});return}await C.appendMessage(d,{role:"human",timestamp:Date.now(),jobId:x,replyToQuestion:m,screenshotPath:E,...Y.length>0?{replyImagePaths:Y}:{}});let P=await C.getThreadHistory(d),j=[];for(let _ of P)if(_.annotationIds)for(let M of _.annotationIds)j.includes(M)||j.push(M);let V=b==="claude"||b==="codex"?b:void 0,se=Ve(E,P,V,Y.length>0?Y:void 0),ee={id:x,status:"queued",screenshotPath:E,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:j.length>0?j:void 0,provider:V,model:$||void 0,sourceId:X||void 0};ee._replyPrompt=se,Y.length>0&&(ee._replyImagePaths=Y);let H=T.enqueue(ee);v(r,200,{jobId:x,position:H,threadId:d})}function ae(o,r){r.writeHead(200,{"Content-Type":"text/event-stream","Cache-Control":"no-cache",Connection:"keep-alive"}),r.write(`event: connected
|
|
131
131
|
data: {"status":"connected"}
|
|
132
132
|
|
|
133
|
-
`),
|
|
134
|
-
`)}async function
|
|
133
|
+
`),o.headers.origin&&st(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,m={id:$e().slice(0,8),res:r,sourceId:d};J.add(m),o.on("close",()=>{J.delete(m)})}function pe(o){let r=T.allActive;v(o,200,{ok:!0,version:et,projectId:e,devOrigin:s,activeJob:r[0]?{id:r[0].id,status:r[0].status}:null,activeJobs:r.map(p=>({id:p.id,status:p.status,threadId:p.threadId,annotationIds:p.annotationIds,color:p.color})),queueDepth:T.depth,recentJobs:w})}async function dt(o,r){let p=await new Promise(d=>{let m="";o.on("data",u=>{m+=u.toString()}),o.on("end",()=>d(m))});try{let d=JSON.parse(p);typeof d.devOrigin=="string"&&(s=d.devOrigin||null),v(r,200,{ok:!0,devOrigin:s})}catch{v(r,400,{error:"Invalid JSON"})}}async function ut(o,r){let d=new URL(o.url||"/",`http://127.0.0.1:${O}`).searchParams.get("jobId"),u=(d?T.allActive.filter($=>$.id===d):T.allActive).map($=>$.threadId).filter(Boolean),b=d?T.cancelJob(d):T.cancelActive();for(let $ of u)await C.appendMessage($,{role:"assistant",timestamp:Date.now(),jobId:d||"",cancelled:!0});v(r,200,{cancelled:b})}async function pt(o){if(g.isRunning){v(o,200,{skipped:!0,reason:"Already running"});return}let r=await g.getUnmaterializedPatternDecisions();if(r.length===0){v(o,200,{skipped:!0,reason:"No unmaterialized pattern decisions"});return}g.run().catch(()=>{}),v(o,200,{started:!0,decisionCount:r.length,decisionIds:r.map(p=>p.id)})}async function ht(o,r){let p=[];for await(let $ of o)p.push(typeof $=="string"?Buffer.from($):$);let d;if(p.length>0)try{d=JSON.parse(Buffer.concat(p).toString("utf-8")).serverUrl}catch{}let m=[];I.claude?.available&&I.claude.mcp&&!I.claude.mcp.found&&m.push(await qe(d)),I.codex?.available&&I.codex.mcp&&!I.codex.mcp.found&&m.push(await Qe(d));let[u,b]=await Promise.all([ke(n),Te(n)]);I.claude&&(I.claude.mcp=u),I.codex&&(I.codex.mcp=b),v(r,200,{results:m,capabilities:{providers:I}})}async function ft(o,r){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{v(r,400,{error:"Invalid JSON"});return}if(!d.name||typeof d.name!="string"){v(r,400,{error:"Missing or invalid name"});return}let m=await g.addComponent(d.name);v(r,200,m)}async function mt(o,r){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{v(r,400,{error:"Invalid JSON"});return}if(!d.name||typeof d.name!="string"){v(r,400,{error:"Missing or invalid name"});return}let m=await g.removeComponent(d.name);v(r,200,m)}async function gt(o,r){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{v(r,400,{error:"Invalid JSON"});return}if(!d.path||typeof d.path!="string"||typeof d.value!="string"){v(r,400,{error:"Missing or invalid path/value"});return}let m=await g.updateToken(d.path,d.value);v(r,200,m)}async function yt(o,r){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{v(r,400,{error:"Invalid JSON"});return}if(!d.path||typeof d.path!="string"){v(r,400,{error:"Missing or invalid path"});return}let m=await g.removeToken(d.path);v(r,200,m)}function vt(o,r){let p=s??"http://localhost:3000";if(!s){let m=o.headers.referer||o.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=Ae(O,p);r.writeHead(200,{"Content-Type":"text/html; charset=utf-8"}),r.end(d)}async function wt(o){let r=Date.now();if(S&&r<S.expires){v(o,200,S.data);return}try{let{scanForComponents:p}=await import("./react-scanner-P3VD53VT.mjs"),{generateRenderFiles:d}=await import("./render-generator-HWFLZYC3.mjs"),m=await p(n);S={data:m,expires:r+5e3},d(m,n,D).then(u=>{D=u}).catch(u=>console.warn("[Bridge] Render generation failed:",u)),v(o,200,m)}catch(p){console.error("[Bridge] Scanner error:",p),v(o,500,{error:"Failed to scan components"})}}async function St(o){let r=[ne(n,"node_modules","@popmelt.com","core","dist","canvas.mjs"),ne(n,"packages","popmelt","dist","canvas.mjs")];try{let p=un(import.meta.url);r.unshift(ne(dn(p),"canvas.mjs"))}catch{}for(let p of r)try{let d=await tt(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:",r),v(o,404,{error:"Canvas bundle not found"})}async function xt(o,r){if(!o||o.includes("/")||o.includes("\\")||o.includes("..")){v(r,400,{error:"Invalid filename"});return}try{let p=await tt(ne(l,o)),d=o.split(".").pop()?.toLowerCase(),m=d==="png"?"image/png":d==="jpg"||d==="jpeg"?"image/jpeg":"application/octet-stream";r.writeHead(200,{"Content-Type":m,"Cache-Control":"public, max-age=3600"}),r.end(p)}catch{v(r,404,{error:"File not found"})}}function he(o){return`/files/${ln(o)}`}async function Pt(o,r){let p=await C.getThread(o);if(!p){v(r,404,{error:"Thread not found"});return}let d=p.messages.map(({screenshotPath:m,screenshotPaths:u,imagePaths:b,replyImagePaths:$,...X})=>({...X,...m?{screenshotUrl:he(m)}:{},...u?{screenshotUrls:Object.fromEntries(Object.entries(u).map(([K,q])=>[K,he(q)]))}:{},...b?{imageUrls:Object.fromEntries(Object.entries(b).map(([K,q])=>[K,q.map(he)]))}:{},...$?{replyImageUrls:$.map(he)}:{}}));v(r,200,{id:p.id,createdAt:p.createdAt,messages:d})}let Ce=9,Oe=!1;for(let o=t;o<t+Ce;o++)try{await Sn(A,o),O=o,Oe=!0,console.log(`[\u22B9 is watching :${O}]`);break}catch(r){if(r.code==="EADDRINUSE"){let p=await wn(o);if(p&&p.projectId===e)return console.log(`[\u22B9 already watching :${o}]`),{port:o,projectId:e,close:async()=>{}};continue}throw r}if(!Oe)throw new Error(`[Bridge] All ports ${t}\u2013${t+Ce-1} in use`);for(let[o,r]of Object.entries(I))!r.available||!r.path||z(o,r.path).then(p=>{if(p)console.log(`[Bridge] ${o} warmed up`);else{console.warn(`[Bridge] ${o} warm-up failed \u2014 marking unavailable`),r.available=!1,r.path=null;for(let d of J)Me(d,{type:"capabilities_changed",data:{}})}});let It=setInterval(()=>{nt(l).catch(()=>{})},fn);return{port:O,projectId:e,close:async()=>{clearInterval(It),await T.destroyAsync();for(let o of J)try{o.res.end()}catch{}return J.clear(),new Promise(o=>{A.close(()=>o())})}}}async function nt(i){try{let t=await sn(i),n=Date.now();for(let e of t){let s=ne(i,e);try{let l=await on(s);n-l.mtimeMs>mn&&await rn(s)}catch{}}}catch{}}var bn=parseInt(process.env.POPMELT_PORT||"1111",10),rt=process.env.POPMELT_PROJECT_ROOT||process.cwd(),kn=process.env.POPMELT_DEV_ORIGIN||void 0,at=it(rt,".popmelt"),ct=it(at,"bridge.lock");async function Tn(i){await xn(at,{recursive:!0}),await In(ct,JSON.stringify({pid:process.pid,port:i,startedAt:Date.now()})+`
|
|
134
|
+
`)}async function lt(){try{await Pn(ct)}catch{}}async function En(){let i=await ot({port:bn,projectRoot:rt,devOrigin:kn});await Tn(i.port),await new Promise(t=>{let n=async()=>{await i.close(),await lt(),t()};process.on("SIGTERM",n),process.on("SIGINT",n)})}En().catch(i=>{console.error("[popmelt bridge-entry] Fatal:",i),lt().finally(()=>process.exit(1))});
|