sootsim 0.1.114 → 0.1.115
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist-cli/bin.js +3 -3
- package/dist-cli/chunks/{agent-PPV725W2.js → agent-R3CYHNIF.js} +2 -2
- package/dist-cli/chunks/{agent-wrapper-RWVU7CO2.js → agent-wrapper-TRRE2GH5.js} +2 -2
- package/dist-cli/chunks/{assert-JC2FD3TU.js → assert-MRSNTOEC.js} +2 -2
- package/dist-cli/chunks/auto-bootstrap-7ABTFLLR.js +2 -0
- package/dist-cli/chunks/beta-ZJYYZEFR.js +2 -0
- package/dist-cli/chunks/{chunk-7EAARWHH.js → chunk-3PUOG2KU.js} +2 -2
- package/dist-cli/chunks/{chunk-D6U6XEI3.js → chunk-47VNJGW7.js} +3 -3
- package/dist-cli/chunks/{chunk-NU5KT7WW.js → chunk-5SL6CP7Y.js} +1 -1
- package/dist-cli/chunks/{chunk-D2PQBP3C.js → chunk-77VIOTR7.js} +2 -2
- package/dist-cli/chunks/{chunk-IOSPY7ZF.js → chunk-AJYQJMAY.js} +2 -2
- package/dist-cli/chunks/chunk-BZTJFTU2.js +1 -0
- package/dist-cli/chunks/{chunk-56HURVFV.js → chunk-CUHTXNJW.js} +1 -1
- package/dist-cli/chunks/{chunk-UKLINY7V.js → chunk-D6PRIVLD.js} +2 -2
- package/dist-cli/chunks/{chunk-NQH7JGEZ.js → chunk-DACZHUK2.js} +1 -1
- package/dist-cli/chunks/{chunk-7QWLHGAD.js → chunk-E53Y4O7H.js} +1 -1
- package/dist-cli/chunks/{chunk-UA5OIC5Y.js → chunk-F46PTRLM.js} +2 -2
- package/dist-cli/chunks/chunk-F4LJC2WB.js +1 -0
- package/dist-cli/chunks/{chunk-EIRP7XBM.js → chunk-GG5MTRXC.js} +1 -1
- package/dist-cli/chunks/{chunk-XPQUENRQ.js → chunk-GUDD5CIA.js} +2 -2
- package/dist-cli/chunks/{chunk-HPTCRK5L.js → chunk-GW5TVVRV.js} +2 -2
- package/dist-cli/chunks/{chunk-2DW25WOA.js → chunk-HAFZOF63.js} +3 -3
- package/dist-cli/chunks/{chunk-PQWCZXCN.js → chunk-HU2K2HVE.js} +2 -2
- package/dist-cli/chunks/{chunk-HIQPS53N.js → chunk-HYR63F7Z.js} +1 -1
- package/dist-cli/chunks/{chunk-ZVL24GTC.js → chunk-ICZSY2XS.js} +2 -2
- package/dist-cli/chunks/{chunk-UBDSTHBH.js → chunk-IWC4SYEG.js} +1 -1
- package/dist-cli/chunks/{chunk-TVETGSCD.js → chunk-ML5LV4WM.js} +1 -1
- package/dist-cli/chunks/{chunk-6TO3HND7.js → chunk-NXAESORG.js} +2 -2
- package/dist-cli/chunks/{chunk-O5WZBC2Y.js → chunk-OK2FCYQN.js} +2 -2
- package/dist-cli/chunks/{chunk-VD4CSXIK.js → chunk-PEP5GKW7.js} +2 -2
- package/dist-cli/chunks/{chunk-IDUGL3MK.js → chunk-Q3BVLSSD.js} +6 -6
- package/dist-cli/chunks/chunk-Q4SUC2SN.js +79 -0
- package/dist-cli/chunks/{chunk-WRB6MZBX.js → chunk-SLSNMSD7.js} +21 -4
- package/dist-cli/chunks/{chunk-VRS7EUIU.js → chunk-SPVHYFZ3.js} +1 -1
- package/dist-cli/chunks/{chunk-FPCP57JZ.js → chunk-SRXOV4BJ.js} +6 -6
- package/dist-cli/chunks/{chunk-HKRPDZOI.js → chunk-TF5LTELW.js} +1 -1
- package/dist-cli/chunks/{chunk-PFBRPOL6.js → chunk-TMR6MW5W.js} +1 -1
- package/dist-cli/chunks/{chunk-V4PAF3TJ.js → chunk-UDQHDFZW.js} +1 -1
- package/dist-cli/chunks/{chunk-M6LHGDWE.js → chunk-UNRBTUAA.js} +1 -1
- package/dist-cli/chunks/chunk-V2SPOLQA.js +1 -0
- package/dist-cli/chunks/chunk-VSKAOUZ5.js +2 -0
- package/dist-cli/chunks/{chunk-TSB4XTV3.js → chunk-W646EQFP.js} +2 -2
- package/dist-cli/chunks/{chunk-DUSFDC5U.js → chunk-WNIVVJR7.js} +2 -2
- package/dist-cli/chunks/{chunk-GR2XFJJF.js → chunk-X35ALXTD.js} +2 -2
- package/dist-cli/chunks/{chunk-7NXJ24HB.js → chunk-XMVEHJSM.js} +2 -2
- package/dist-cli/chunks/{chunk-MGWQ527R.js → chunk-XO3XTPFR.js} +1 -1
- package/dist-cli/chunks/{chunk-BBNJXYUW.js → chunk-XRBI3TG2.js} +2 -2
- package/dist-cli/chunks/{chunk-G5IBWOSU.js → chunk-YBAX43B5.js} +1 -1
- package/dist-cli/chunks/{chunk-FVPJJQCX.js → chunk-YTJONQGT.js} +3 -3
- package/dist-cli/chunks/{chunk-WPYPKVQP.js → chunk-ZPJDADTZ.js} +2 -2
- package/dist-cli/chunks/cli-version-RIP4ZUNM.js +2 -0
- package/dist-cli/chunks/{compat-BO4GOTHM.js → compat-4ZDISC6K.js} +3 -3
- package/dist-cli/chunks/{config-4W2CJ435.js → config-IWM4VJ6H.js} +2 -2
- package/dist-cli/chunks/control-FSKX3J4O.js +2 -0
- package/dist-cli/chunks/{cpu-profile-5BKUYSEY.js → cpu-profile-RPY4UFNI.js} +2 -2
- package/dist-cli/chunks/{daemon-KFYVGHQV.js → daemon-SL3D6Q4P.js} +2 -2
- package/dist-cli/chunks/{debug-QBCVIBVX.js → debug-WYNMZ7CT.js} +3 -3
- package/dist-cli/chunks/{detox-BKVDVNN3.js → detox-RMLUYFYI.js} +2 -2
- package/dist-cli/chunks/{device-YG2NZYU4.js → device-TYY7VA57.js} +2 -2
- package/dist-cli/chunks/{diagnose-QBHSWPOE.js → diagnose-7CXKCUT4.js} +2 -2
- package/dist-cli/chunks/drivers-OIAGQOCX.js +2 -0
- package/dist-cli/chunks/{electron-APIQFWWP.js → electron-IKCSDHAV.js} +3 -3
- package/dist-cli/chunks/flow-ZIV2ISC2.js +2 -0
- package/dist-cli/chunks/help-RUILKRB4.js +2 -0
- package/dist-cli/chunks/{hints-SVV3NUDM.js → hints-YHYKK5HQ.js} +2 -2
- package/dist-cli/chunks/{home-paths-IPIEZXKS.js → home-paths-OLHYRKN5.js} +2 -2
- package/dist-cli/chunks/{inspect-QA27PCCP.js → inspect-INJGRQD3.js} +3 -3
- package/dist-cli/chunks/install-BRAHILXG.js +2 -0
- package/dist-cli/chunks/{install-desktop-U4KFKLJI.js → install-desktop-FQS3WZ6W.js} +3 -3
- package/dist-cli/chunks/{keys-SWELRMTL.js → keys-JUYFKZD7.js} +2 -2
- package/dist-cli/chunks/{launch-KWK7AYPJ.js → launch-XW5OSSNC.js} +3 -3
- package/dist-cli/chunks/{login-AFK5TRXO.js → login-QSZQPSEA.js} +4 -4
- package/dist-cli/chunks/{logout-KJB2XZ4H.js → logout-2YSS6D65.js} +2 -2
- package/dist-cli/chunks/{maestro-IL7IRBBN.js → maestro-UG6C6KQ6.js} +2 -2
- package/dist-cli/chunks/{preview-DDOOW574.js → preview-4TH2YLNR.js} +2 -2
- package/dist-cli/chunks/{profile-UWDS7T6L.js → profile-63R3YOCR.js} +2 -2
- package/dist-cli/chunks/{react-EJOUNOBS.js → react-FMPGSG3A.js} +2 -2
- package/dist-cli/chunks/{record-7Q25DB46.js → record-F64J66JH.js} +2 -2
- package/dist-cli/chunks/runtime-NE4NMW4R.js +2 -0
- package/dist-cli/chunks/{runtime-delivery-Q4CEUD72.js → runtime-delivery-HJPB7EEU.js} +2 -2
- package/dist-cli/chunks/{screenshot-3ON5A5EY.js → screenshot-OF6QEH6B.js} +2 -2
- package/dist-cli/chunks/{screenshot-mode-MGACLSRL.js → screenshot-mode-BIZP42HF.js} +2 -2
- package/dist-cli/chunks/{screenshots-OWSKOBVC.js → screenshots-DKAYDLK5.js} +2 -2
- package/dist-cli/chunks/{server-T2NGFWJQ.js → server-ETFFUJ5R.js} +2 -2
- package/dist-cli/chunks/setup-repo-JE25CERB.js +2 -0
- package/dist-cli/chunks/{skills-SF5HOOMB.js → skills-H2OPZZLC.js} +2 -2
- package/dist-cli/chunks/{start-PEFKHFIU.js → start-OI245IG4.js} +4 -4
- package/dist-cli/chunks/store-2H5T4NFZ.js +2 -0
- package/dist-cli/chunks/telemetry-USTW2NVP.js +2 -0
- package/dist-cli/chunks/{test-OBVSN2TZ.js → test-HE6RK2ZX.js} +3 -3
- package/dist-cli/chunks/{three-mode-EZ332TID.js → three-mode-K2NTAVQP.js} +2 -2
- package/dist-cli/chunks/{timeline-E32QABRP.js → timeline-3DBCJWKF.js} +2 -2
- package/dist-cli/chunks/{upgrade-A42OZ5LT.js → upgrade-7YLEP6ED.js} +2 -2
- package/dist-cli/chunks/upload-T3AYMH27.js +2 -0
- package/dist-cli/chunks/{web-Q22CLCSN.js → web-BWUYH3SK.js} +2 -2
- package/dist-cli/chunks/{what-happened-6EHD4QJK.js → what-happened-VDTCMZNP.js} +2 -2
- package/dist-cli/chunks/{whoami-ZVBH6ZYC.js → whoami-2ZOUZSB3.js} +2 -2
- package/dist-lib/agent-daemon-client.cjs +1 -1
- package/dist-lib/agent-events.cjs +1 -1
- package/dist-lib/agent-sessions.cjs +1 -1
- package/dist-lib/attached-projects.cjs +1 -1
- package/dist-lib/auth/shared-session.cjs +1 -1
- package/dist-lib/backend-origin.cjs +1 -1
- package/dist-lib/beta.cjs +1 -1
- package/dist-lib/beta.mjs +1 -1
- package/dist-lib/bridge-constants.cjs +1 -1
- package/dist-lib/cli-constants.cjs +1 -1
- package/dist-lib/config.cjs +1 -1
- package/dist-lib/detox/index.cjs +1 -1
- package/dist-lib/dev-bundle-resolution.cjs +1 -1
- package/dist-lib/home-paths.cjs +1 -1
- package/dist-lib/host/bridge-host.cjs +1 -1
- package/dist-lib/host/fetch-proxy-handler.cjs +1 -1
- package/dist-lib/host/fetch-proxy-overrides.cjs +1 -1
- package/dist-lib/host/fetch-proxy-overrides.mjs +1 -1
- package/dist-lib/host/websocket-proxy.cjs +1 -1
- package/dist-lib/index.cjs +1 -1
- package/dist-lib/metro.cjs +1 -1
- package/dist-lib/profiles.cjs +1 -1
- package/dist-lib/render-mode.cjs +1 -1
- package/dist-lib/scripts/demo-app-registry.cjs +1 -1
- package/dist-lib/scripts/dev-server-scanner.cjs +1 -1
- package/dist-lib/sdk.cjs +22 -5
- package/dist-lib/sdk.mjs +22 -5
- package/dist-lib/skills.cjs +2 -6
- package/dist-lib/vite.cjs +1 -1
- package/package.json +1 -1
- package/dist-cli/chunks/auto-bootstrap-GJVEKHWE.js +0 -2
- package/dist-cli/chunks/beta-HTH4QNXO.js +0 -2
- package/dist-cli/chunks/chunk-JV6I42YD.js +0 -79
- package/dist-cli/chunks/chunk-MOSWK2MT.js +0 -2
- package/dist-cli/chunks/chunk-SJ6DSMZI.js +0 -1
- package/dist-cli/chunks/chunk-SMVEVXJN.js +0 -1
- package/dist-cli/chunks/chunk-ZL5NFGQ6.js +0 -1
- package/dist-cli/chunks/cli-version-O7RDJ2FI.js +0 -2
- package/dist-cli/chunks/control-4BKV5LMV.js +0 -2
- package/dist-cli/chunks/drivers-JUF5RIRB.js +0 -2
- package/dist-cli/chunks/flow-WSLHCPWR.js +0 -2
- package/dist-cli/chunks/help-LGGVMOHI.js +0 -2
- package/dist-cli/chunks/install-3W3VW6XD.js +0 -2
- package/dist-cli/chunks/runtime-M4BIM2UC.js +0 -2
- package/dist-cli/chunks/setup-repo-6CZGK337.js +0 -2
- package/dist-cli/chunks/store-ZUZGIBZL.js +0 -2
- package/dist-cli/chunks/telemetry-DSKEFDAZ.js +0 -2
- package/dist-cli/chunks/upload-KZOOYKHT.js +0 -2
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
/*! sootsim v0.1.
|
|
2
|
-
import{A as R,C as w,i as p,r as S,t as O,y as v,z as f}from"./chunk-
|
|
1
|
+
/*! sootsim v0.1.115 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
|
|
2
|
+
import{A as R,C as w,i as p,r as S,t as O,y as v,z as f}from"./chunk-UNRBTUAA.js";import{spawn as I}from"child_process";import C from"crypto";import s from"fs";import l from"path";import{Readable as k}from"stream";import{pipeline as M}from"stream/promises";var N="https://sootbean.com",D="SOOTSIM_CDN_ORIGIN",E="SOOTSIM_RUNTIME_CHANNEL";function $(){try{let e=JSON.parse(s.readFileSync(O(),"utf8"));return e&&typeof e=="object"?e:{}}catch{return{}}}function d(e){let t=$();return(e||process.env[D]||t.cdnOrigin||N).replace(/\/+$/,"")}function x(e){let t=$();return e||process.env[E]||t.runtimeChannel||"stable"}function T(e){let t=new URL(`${d(e)}/runtimes/manifest.json`);return t.searchParams.set("t",String(Date.now())),t.toString()}function P(e,t){return`${d(t)}/runtimes/sootsim-runtime-${e}.tar.gz`}async function y(e){let t=T(e),r=await fetch(t,{headers:{Accept:"application/json"}});if(!r.ok)throw new Error(`manifest fetch failed: ${r.status} ${r.statusText} (${t})`);return await r.json()}function U(e,t={}){let r=x(t.channel),n=t.version||e.channels[r]?.latest;if(!n)throw new Error(`no version specified and channel '${r}' has no latest entry in the manifest`);let i=e.versions[n];if(!i)throw new Error(`version ${n} not found in manifest; available: ${Object.keys(e.versions).slice(-10).join(", ")||"(none)"}`);return{version:n,channel:r,entry:i}}async function V(e={}){v();let t=d(e.cdnOrigin),r=await y(t),{version:n,channel:i,entry:o}=U(r,e),a=p(n),c=e.setActive!==!1;if(!e.force&&s.existsSync(l.join(a,"index.html")))return c&&R(n),{version:n,channel:i,cdnOrigin:t,runtimeDir:a,installed:!1,activated:c,manifest:r};let g=o.tarball||P(n,t),m=l.join(S(),`sootsim-runtime-${n}.tar.gz`);process.stderr.write(`sootsim: downloading runtime ${n}\u2026
|
|
3
3
|
`),await _(g,m),process.stderr.write(`sootsim: extracting runtime ${n}\u2026
|
|
4
4
|
`);let h=await j(m);if(h!==o.sha256)throw s.rmSync(m,{force:!0}),new Error(`sha256 mismatch for runtime ${n}: expected ${o.sha256}, actual ${h}`);let u=l.join(l.dirname(a),`.installing-${n}-${process.pid}`);s.rmSync(u,{recursive:!0,force:!0}),s.mkdirSync(u,{recursive:!0});try{if(await A(m,u),!s.existsSync(l.join(u,"index.html")))throw new Error(`extracted tarball for runtime ${n} is missing index.html`);s.rmSync(a,{recursive:!0,force:!0}),s.renameSync(u,a)}catch(b){throw s.rmSync(u,{recursive:!0,force:!0}),b}return c&&R(n),{version:n,channel:i,cdnOrigin:t,runtimeDir:a,installed:!0,activated:c,manifest:r}}async function q(e={}){v();let t=d(e.cdnOrigin),r=x(e.channel),n=await y(t),i=n.channels[r]?.latest;if(!i)return{checked:!0,updated:!1,reason:`channel '${r}' has no latest runtime`,activeVersion:f()};if(!n.versions[i])return{checked:!0,updated:!1,reason:`manifest is missing version ${i}`,activeVersion:f(),latestVersion:i};let a=f(),c=a?p(a):null,g=c?s.existsSync(l.join(c,"index.html")):!1;if(!(!a||!g||w(i,a)>0))return{checked:!0,updated:!1,reason:"active runtime is current",activeVersion:a,latestVersion:i};let h=await V({version:i,channel:r,cdnOrigin:t,setActive:!1});return{checked:!0,updated:!0,activeVersion:i,latestVersion:i,install:h}}async function B(e={}){let t=f();try{let r=await y(d(e.cdnOrigin)),n=x(e.channel),i=r.channels[n]?.latest??null,o=!!(i&&(!t||w(i,t)>0));return{active:t,latest:i,outdated:o}}catch{return{active:t,latest:null,outdated:!1}}}async function _(e,t){let r=await fetch(e);if(!r.ok||!r.body)throw new Error(`download failed: ${r.status} ${r.statusText} (${e})`);s.mkdirSync(l.dirname(t),{recursive:!0});let n=`${t}.partial`;try{await M(k.fromWeb(r.body),s.createWriteStream(n)),s.renameSync(n,t)}catch(i){try{s.unlinkSync(n)}catch{}throw i}}function j(e){return new Promise((t,r)=>{let n=C.createHash("sha256"),i=s.createReadStream(e);i.on("data",o=>n.update(o)),i.on("error",r),i.on("end",()=>t(n.digest("hex")))})}function A(e,t){return new Promise((r,n)=>{let i=I("tar",["-xzf",e,"-C",t],{stdio:["ignore","inherit","inherit"]});i.on("error",n),i.on("exit",o=>{o===0?r():n(new Error(`tar exited with code ${o}`))})})}export{N as a,D as b,E as c,d,x as e,T as f,P as g,y as h,U as i,V as j,q as k,B as l};
|
|
@@ -1,18 +1,18 @@
|
|
|
1
|
-
/*! sootsim v0.1.
|
|
2
|
-
import{a as k,c as T,e as C}from"./chunk-
|
|
3
|
-
`)}return
|
|
1
|
+
/*! sootsim v0.1.115 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
|
|
2
|
+
import{a as k,c as T,e as C}from"./chunk-SPVHYFZ3.js";import{E as v,F as P}from"./chunk-UNRBTUAA.js";import{WebSocket as I}from"ws";function x(){let e=v();return e&&P(e)?e.bridgePort:7668}var w=class extends Error{lock;constructor(r,i){super(r),this.name="BridgeSimLockedError",this.lock=i}};function $(e){return e==="flag"?"via --sim":e==="saved"?"saved via `sootsim use`":"primary fallback \u2014 no sim pinned"}function R(e,r={}){let i=new Set,a=r.port??x(),l=r.commandTimeoutMs??15e3,t,m="none",p=new Set(r.stripBooleanFlags??[]),b=new Set(r.stripValueFlags??[]);for(let n=0;n<e.length;n++){let d=e[n];if(d==="--port"||d==="-p"){i.add(n),n+1<e.length&&(i.add(n+1),a=Number(e[n+1])),n++;continue}if(d.startsWith("--port=")){i.add(n),a=Number(d.slice(7));continue}if(d==="--timeout"){i.add(n),n+1<e.length&&(i.add(n+1),l=Number(e[n+1])),n++;continue}if(d==="--sim"||d==="--session"||d==="--tab"){i.add(n),n+1<e.length&&(i.add(n+1),t=e[n+1]?.trim()||void 0,m="flag"),n++;continue}let s=d.startsWith("--sim=")?d.slice(6):d.startsWith("--session=")?d.slice(10):d.startsWith("--tab=")?d.slice(6):void 0;if(s!==void 0){i.add(n),t=s.trim()||void 0,m="flag";continue}if(p.has(d)){i.add(n);continue}b.has(d)&&(i.add(n),n+1<e.length&&i.add(n+1),n++)}if(!t){let n=T();n&&(t=n.trim()||void 0,m="saved")}return{positional:e.filter((n,d)=>!i.has(d)),wsPort:a,simId:t,simIdSource:m,commandTimeoutMs:l}}function E(e,r={}){let i=1,a=r.commandTimeoutMs??15e3,l=new Map,t=new I(`ws://localhost:${e}`),m=k(),p=new Promise((s,o)=>{t.on("open",()=>{try{t.send(JSON.stringify({type:"bridge:hello",id:0,cliIdentityKey:m.key,cliIdentitySource:m.source,cliLabel:r.cliLabel}))}catch{}s()}),t.on("error",c=>{let g=c.message?`: ${c.message}`:"";o(new Error(`could not connect to ws://localhost:${e}${g}`))})}),b=!1,n=!1;function d(s,o){if(n||process.env.SOOTSIM_QUIET_TARGET_NOTICE==="1"||s.startsWith("bridge:")||s==="focus"||s==="close")return;n=!0;let c=o??"primary",g=$(o?r.simIdSource:"none");process.stderr.write(` \u2192 ${c} (${g})
|
|
3
|
+
`)}return t.on("message",s=>{let o;try{o=JSON.parse(s.toString())}catch{return}if(o.id===0)return;let c=l.get(o.id);c&&(l.delete(o.id),o.i>0&&!b&&(b=!0,process.stderr.write(`
|
|
4
4
|
\u26A0 ${o.i} other CLI identity/identities are driving this sim
|
|
5
5
|
taps, scrolls, and keyboard input from multiple agents will collide on
|
|
6
6
|
the same screen state (this is not a bridge throughput limit).
|
|
7
7
|
use \`sootsim open --new\` for an isolated sim per agent.
|
|
8
8
|
|
|
9
|
-
`)),o.error?o.o?c.reject(new w(o.error,o.o)):c.reject(new Error(o.error)):c.resolve(o.result))}),
|
|
9
|
+
`)),o.error?o.o?c.reject(new w(o.error,o.o)):c.reject(new Error(o.error)):c.resolve(o.result))}),t.on("close",(s,o)=>{let c=o?.toString()||"connection closed";for(let[g,u]of l)l.delete(g),u.reject(new Error(`sim disconnected: ${c} (code ${s})`))}),{async send(s,o){let c=o?.timeoutMs??a,g=async u=>{await p;let f=i++;return new Promise((A,S)=>{let B=setTimeout(()=>{l.delete(f),S(new Error(`command timed out after ${Math.round(c/1e3)}s`))},c);l.set(f,{resolve:h=>{clearTimeout(B),A(h)},reject:h=>{clearTimeout(B),S(h)}});let y={...s,id:f};y.simId===void 0&&u&&(y.simId=u),d(s.type??"",y.simId),t.send(JSON.stringify(y))})};try{return await g(r.simId)}catch(u){let f=u instanceof Error?u.message:String(u);throw r.simIdSource==="saved"&&r.simId&&s.simId===void 0&&f===`no sim connected with id ${r.simId}`?(C(),new Error(`saved sim ${r.simId} is gone; run \`sootsim list\` and \`sootsim use <sim>\`, or open a fresh sim`)):u}},async listSims(){let s=await this.send({type:"bridge:list-sims"});return Array.isArray(s)?s:[]},async openUrl(s,o={}){return this.send({type:"bridge:open",url:s,newWindow:o.newWindow===!0})},async focusSim(s){return this.send({type:"focus",simId:s})},async closeSim(s){return this.send({type:"close",simId:s})},async claim(s,o={}){return await this.send({type:"bridge:claim",simId:s,force:o.force===!0})},close(){try{t.readyState===I.OPEN&&t.send(JSON.stringify({type:"bridge:bye",id:0}))}catch{}t.close();let s=setTimeout(()=>{t.readyState!==I.CLOSED&&t.terminate()},250)}}}function V(e){return E(e.wsPort,{commandTimeoutMs:e.commandTimeoutMs,simId:e.simId,simIdSource:e.simIdSource})}function F(e,r){let i=e.filter(t=>t.readyState==="open"&&t.id!==r),a=i.find(t=>t.userFocused)??i.find(t=>t.userVisible===!0&&t.isPrimary)??i.find(t=>t.userVisible===!0)??i.find(t=>t.isPrimary);if(!a)return null;let l=[a.isPrimary?"primary":null,a.userVisible===!1?"hidden":a.userVisible===!0?"visible":null,a.userFocused?"focused":null].filter(Boolean);return`${a.id}${l.length?` [${l.join(", ")}]`:""}`}async function _(e){try{let r=await e.send({type:"evaluate",code:`(() => ({
|
|
10
10
|
hidden: document.hidden,
|
|
11
11
|
visibilityState: document.visibilityState,
|
|
12
12
|
hasFocus: typeof document.hasFocus === 'function' ? document.hasFocus() : null,
|
|
13
13
|
simId: window.__sootsimBridge?.id ?? window.SootSim?.state?.simId ?? null,
|
|
14
14
|
url: location.href
|
|
15
|
-
}))()`}),i=typeof
|
|
15
|
+
}))()`}),i=typeof r=="object"&&r!==null?r:{hidden:r===!0};if(i.hidden===!0===!0){let l=null;try{l=F(await e.listSims(),i.simId)}catch{}return process.stderr.write(`
|
|
16
16
|
\u26A0 target sim${i.simId?` ${i.simId}`:""} is hidden (document.hidden = true)
|
|
17
17
|
`+(i.visibilityState?` visibility: ${i.visibilityState}
|
|
18
18
|
`:"")+` animations and rAF callbacks are throttled \u2014 coordinates may be wrong
|
|
@@ -20,4 +20,4 @@ import{a as k,c as T,e as C}from"./chunk-VRS7EUIU.js";import{E as v,F as P}from"
|
|
|
20
20
|
`+(l?` another open sim looks usable: ${l}; target it with --sim.
|
|
21
21
|
`:" run `sootsim list` to find the visible sim and target it with --sim.\n")+` page-level window.focus() cannot unhide a browser tab.
|
|
22
22
|
|
|
23
|
-
`),{hidden:!0,warned:!0,simId:i.simId}}return{hidden:!1,warned:!1,simId:i.simId}}catch{return{hidden:!1,warned:!1}}}async function j(e,
|
|
23
|
+
`),{hidden:!0,warned:!0,simId:i.simId}}return{hidden:!1,warned:!1,simId:i.simId}}catch{return{hidden:!1,warned:!1}}}async function j(e,r,i={}){let a={type:"evaluate",code:r};return i.acquireLock&&(a.acquireLock=!0),e.send(a)}async function D(e,r,...i){return e.send({type:"call",path:r,args:i})}async function q(e,r,...i){return e.send({type:"call",path:r,args:i,acquireLock:!0})}export{x as a,w as b,R as c,E as d,V as e,_ as f,j as g,D as h,q as i};
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
/*! sootsim v0.1.115 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
|
|
2
|
+
import{d as de,e as pe,f as fe}from"./chunk-D6PRIVLD.js";import{c as ue}from"./chunk-TF5LTELW.js";import{c as le,e as ce,g as I}from"./chunk-Q3BVLSSD.js";import{d as ae}from"./chunk-IWC4SYEG.js";import{a as ie,b as P}from"./chunk-NXAESORG.js";import{a as se}from"./chunk-ZPJDADTZ.js";import{createHash as ye}from"crypto";import{readFileSync as K}from"fs";import{brotliCompressSync as Ae,constants as ze,gzipSync as Me}from"zlib";function ge(e){let t=(e||"").toLowerCase().split(";")[0].trim();return t?!(t.startsWith("image/")||t.startsWith("video/")||t.startsWith("audio/")||t==="font/woff2"||t==="application/font-woff2"||t==="application/zip"||t==="application/gzip"):!0}function G(e,t){let r=new AbortController,o=setTimeout(()=>r.abort(new Error(`${t} timed out after ${e/1e3}s`)),e);return{signal:r.signal,cancel:()=>clearTimeout(o)}}async function Le(e){let t=G(12e4,`PUT ${e.url}`),r;try{r=await fetch(e.url,{method:"PUT",headers:{"content-type":e.contentType},body:e.bytes,signal:t.signal})}finally{t.cancel()}if(!r.ok){let o=await r.text().catch(()=>"");throw new Error(`PUT ${e.url} failed ${r.status}: ${o.slice(0,200)}`)}}async function ke(e,t=8,r){let o=0,s=0,a=e.length;await Promise.all(Array.from({length:Math.min(t,a)},async()=>{for(;o<a;){let c=e[o++];await Le(c),s++,r?.(s,a)}}))}function Ee(e,t,r,o){let s=[];s.push({url:e.bundle.putUrl,contentType:e.bundle.contentType,bytes:t});for(let a of e.files){let c=r.get(a.urlhash);c&&s.push({url:a.putUrl,contentType:a.contentType,bytes:c})}e.events&&o?.eventsBytes&&s.push({url:e.events.putUrl,contentType:e.events.contentType,bytes:o.eventsBytes}),e.flowVideo&&o?.videoBytes&&s.push({url:e.flowVideo.putUrl,contentType:e.flowVideo.contentType,bytes:o.videoBytes}),e.storage&&o?.storageBytes&&s.push({url:e.storage.putUrl,contentType:e.storage.contentType,bytes:o.storageBytes});for(let a of e.snapshots??[]){let c=o?.snapshotBytes?.get(a.id);c&&s.push({url:a.putUrl,contentType:a.contentType,bytes:c})}return s}function Fe(e){return e.replace(/\/+$/,"")}var A=class extends Error{constructor(r,o,s,a=""){super(r);this.status=o;this.phase=s;this.responseText=a;this.name="PresignedUploadError"}};async function he(e){let t=Fe(e.originBase),r={"content-type":"application/json"};e.authHeader&&(r.authorization=e.authHeader),e.adminBypassToken&&(r["x-soot-admin-upload-bypass"]=e.adminBypassToken);let o=G(3e4,"init"),s;try{s=await fetch(`${t}/api/preview/upload/init`,{method:"POST",headers:r,body:JSON.stringify(e.initBody),signal:o.signal})}finally{o.cancel()}if(!s.ok){let f=await s.text().catch(()=>"");throw new A(`init failed: ${s.status} ${f.slice(0,200)}`,s.status,"init",f)}let a=await s.json(),c=Ee(a,e.bundleBytes,e.filesByHash,e.extras);await ke(c,e.concurrency??8,e.onPutProgress);let y=G(3e4,"finalize"),d;try{d=await fetch(`${t}/api/preview/upload/finalize`,{method:"POST",headers:r,body:JSON.stringify({token:a.token}),signal:y.signal})}finally{y.cancel()}if(!d.ok){let f=await d.text().catch(()=>"");throw new A(`finalize failed: ${d.status} ${f.slice(0,200)}`,d.status,"finalize",f)}let $=await d.json();return{init:a,finalize:$}}function He(e){if(!e)return null;try{let t=JSON.parse(e);return typeof t.message=="string"&&t.message?t.message:null}catch{return null}}var be=process.env.SOOTSIM_UPLOAD_ORIGIN||"https://sootbean.com",ve=process.env.SOOTSIM_PREVIEW_ORIGIN||"https://sootsim.com",Y="http://localhost:3000",Ce=2e3;async function Ne(e){if(e)return e;if(process.env.SOOTSIM_UPLOAD_ORIGIN)return process.env.SOOTSIM_UPLOAD_ORIGIN;try{let t=await fetch(`${Y}/api/preview/upload/init`,{method:"OPTIONS",signal:AbortSignal.timeout(Ce)});if(t.ok||t.status===204||t.status===405)return Y}catch{}return be}function S(e){return e.replace(/\/$/,"")}function We(e){try{let r=new URL(e).hostname.replace(/^\[|\]$/g,"").toLowerCase();return r==="localhost"||r.endsWith(".localhost")||r==="0.0.0.0"||r==="::1"||/^127(?:\.\d{1,3}){3}$/.test(r)}catch{return!1}}function qe(e){try{let t=new URL(e).hostname.toLowerCase();return t==="sootbean.com"||t.endsWith(".sootbean.com")}catch{return!1}}function Ve(e,t){return t?S(t):process.env.SOOTSIM_PREVIEW_ORIGIN?S(process.env.SOOTSIM_PREVIEW_ORIGIN):We(e)?S(e):qe(e)?S(ve):S(e)}function De(){console.log(`
|
|
3
|
+
sootsim upload \u2014 publish the current recorded bundle as a /preview/<id> link
|
|
4
|
+
|
|
5
|
+
usage:
|
|
6
|
+
sootsim upload [--origin <url>] [--public-origin <url>] (--events <path> | --video <path>) [--screenshot <path>] [--owner <org> --repo <repo>] [--sim <sim>] [--open] [--assets-only]
|
|
7
|
+
|
|
8
|
+
options:
|
|
9
|
+
--origin <url> upload target (default: auto)
|
|
10
|
+
prefers ${Y} when available, otherwise
|
|
11
|
+
falls back to ${be}
|
|
12
|
+
override with SOOTSIM_UPLOAD_ORIGIN env var
|
|
13
|
+
--public-origin <url>
|
|
14
|
+
public /preview link origin (default: ${ve}
|
|
15
|
+
for prod uploads, upload origin for localhost/custom origins).
|
|
16
|
+
override with SOOTSIM_PREVIEW_ORIGIN env var
|
|
17
|
+
--events <path> path to a gzipped events .jsonl.gz file to attach
|
|
18
|
+
required unless --video is present; previews must
|
|
19
|
+
have replay or recording playback data
|
|
20
|
+
--screenshot <path>
|
|
21
|
+
path to a PNG thumbnail to attach to the share
|
|
22
|
+
--video <path> path to a webm/mp4/gif flow recording. embedded inline
|
|
23
|
+
in the pr sticky comment served at
|
|
24
|
+
/api/preview/flow-video?id=<share-id>
|
|
25
|
+
--video-duration-ms <ms>
|
|
26
|
+
duration hint for attached flow video
|
|
27
|
+
--owner <org> associate a session upload with a linked org repo
|
|
28
|
+
--repo <repo> repo name for --owner; the signed-in user must belong
|
|
29
|
+
to the local/org team that owns the link
|
|
30
|
+
--sim <sim> target a specific sim (see: sootsim list)
|
|
31
|
+
--open open the resulting /preview/<id> url in the browser
|
|
32
|
+
--assets-only drop API/JSON/HTML records before upload; keep images,
|
|
33
|
+
fonts, css, js, and binary blobs. live-data demos hit
|
|
34
|
+
the real network at replay time instead of serving
|
|
35
|
+
recorded API snapshots
|
|
36
|
+
-h, --help
|
|
37
|
+
|
|
38
|
+
examples:
|
|
39
|
+
sootsim upload --events ./my-session.jsonl.gz
|
|
40
|
+
sootsim upload --origin http://localhost:3000 --events ./my-session.jsonl.gz --open
|
|
41
|
+
sootsim upload --video /tmp/soot-flow.webm --video-duration-ms 12000
|
|
42
|
+
`)}function b(e,t){let r=e.findIndex(s=>s===t);if(r<0)return;let o=e[r+1];return e.splice(r,2),o}function me(e,t){let r=e.findIndex(o=>o===t);return r<0?!1:(e.splice(r,1),!0)}function we(e){let t=(e||"").toLowerCase().split(";")[0].trim();return t?!!(t.startsWith("image/")||t.startsWith("font/")||t.startsWith("video/")||t.startsWith("audio/")||t.startsWith("model/")||t==="text/css"||t==="application/javascript"||t==="text/javascript"||t==="application/wasm"||t==="application/font-woff"||t==="application/font-woff2"||t==="application/octet-stream"):!1}async function j(e){let t=await fetch(e);if(!t.ok)throw new Error(`fetch ${e} -> ${t.status} ${t.statusText}`);return new Uint8Array(await t.arrayBuffer())}function Ge(e){return/\.bundle($|\?)/.test(e)||/\.js($|\?)/.test(e)?"application/javascript":/\.zip($|\?)/.test(e)?"application/zip":"application/javascript"}function Ke(e){try{return new URL(e).searchParams.get("appFonts")||""}catch{return""}}function je(e){let t=(e.split("?")[0].split(".").pop()||"").toLowerCase();return t==="ttf"?"font/ttf":t==="otf"?"font/otf":t==="woff"?"font/woff":t==="woff2"?"font/woff2":"application/octet-stream"}function Je(e){try{let t=new URL(e);return t.searchParams.get("lazy")==="true"?(t.searchParams.set("lazy","false"),t.toString()):e}catch{return e}}async function Ye(e,t){let o=[];for(let s=0;s<t;s+=15e5){let a=Math.min(s+15e5,t),c=await I(e,`(window.__sootsimLastTransformedBundle?.text || "").slice(${s}, ${a})`);if(typeof c!="string")return null;o.push(c)}return o.join("")}function J(e){let t=new Map;for(let r of e)t.set(r.urlhash||r.url,r);return Array.from(t.values())}async function dt(e,t){(e.includes("--help")||e.includes("-h"))&&(De(),process.exit(0));let r=[...e],o=await Ne(b(r,"--origin")),s=Ve(o,b(r,"--public-origin")),a=b(r,"--events"),c=b(r,"--screenshot"),y=b(r,"--video"),d=b(r,"--video-duration-ms"),$=b(r,"--owner"),f=b(r,"--repo"),Ue=me(r,"--open");($&&!f||!$&&f)&&(console.error(" --owner and --repo must be provided together"),process.exit(1)),!a&&!y&&(console.error(" preview uploads require playback data: pass --events or --video.\n for a normal local preview, use `sootsim record --mode combined --open`\n or `sootsim flow --preview` instead of uploading a bundle-only snapshot."),process.exit(1));let Q=me(r,"--assets-only"),Te=le(r,{stripBooleanFlags:[],stripValueFlags:[]}),v=ce(Te),l,X=[],m=[],z=null;try{if(l=await I(v,'(typeof window.__sootsimCaptureBundle === "function") ? window.__sootsimCaptureBundle() : null'),l?.bundleUrl){let n=new URL(l.bundleUrl).origin,[i,T]=await Promise.all([I(v,`(window.__sootsimPreviewRecorder?.list?.(${JSON.stringify(n)}) || [])`),I(v,`(async () => {
|
|
43
|
+
const list = window.__sootsimListWorkerFetchRecorder
|
|
44
|
+
return typeof list === 'function'
|
|
45
|
+
? await list(${JSON.stringify(n)})
|
|
46
|
+
: []
|
|
47
|
+
})()`)]);X=J([...i,...T]).filter(p=>{try{return!new URL(p.url).pathname.startsWith("/api/")}catch{return!0}});let u=await I(v,`(async () => {
|
|
48
|
+
const rec = window.__sootsimPreviewRecorder
|
|
49
|
+
const workerList = window.__sootsimListWorkerFetchRecorder
|
|
50
|
+
return {
|
|
51
|
+
main: rec?.list ? rec.list() : [],
|
|
52
|
+
worker: typeof workerList === 'function' ? await workerList() : [],
|
|
53
|
+
}
|
|
54
|
+
})()`),x=(u?J([...u.main,...u.worker]):[]).filter(p=>{try{return new URL(p.url).origin!==n}catch{return!1}}),B=x.length>0?{count:x.length,totalBytes:x.reduce((p,O)=>p+(O.size||0),0)}:null;if(B&&B.count>0){console.log(` ${B.count} recorded cross-origin responses (${(B.totalBytes/1024).toFixed(1)} KiB)`);let p=await I(v,`(async () => {
|
|
55
|
+
const rec = window.__sootsimPreviewRecorder
|
|
56
|
+
const workerDump = window.__sootsimDumpWorkerFetchRecorder
|
|
57
|
+
const bundleUrl = ${JSON.stringify(l.bundleUrl)}
|
|
58
|
+
const keep = (r) => {
|
|
59
|
+
try {
|
|
60
|
+
const u = new URL(r.url)
|
|
61
|
+
if (u.origin !== ${JSON.stringify(n)}) return true
|
|
62
|
+
// bundle-origin api responses came from the rewritten
|
|
63
|
+
// /__app-api proxy at record time (re-keyed by the
|
|
64
|
+
// recorder's unwrap step). they need bodies attached the
|
|
65
|
+
// same way as cross-origin records \u2014 direct re-fetch from
|
|
66
|
+
// the dev server lacks the bundle's session cookies.
|
|
67
|
+
if (r.url !== bundleUrl && u.pathname.startsWith('/api/')) return true
|
|
68
|
+
return false
|
|
69
|
+
} catch { return false }
|
|
70
|
+
}
|
|
71
|
+
return {
|
|
72
|
+
main: rec?.dump ? rec.dump().filter(keep) : [],
|
|
73
|
+
worker: typeof workerDump === 'function'
|
|
74
|
+
? (await workerDump()).filter(keep)
|
|
75
|
+
: [],
|
|
76
|
+
}
|
|
77
|
+
})()`);if(m=J([...p.main,...p.worker]),Q){let O=m.length;m=m.filter(D=>we(D.contentType));let V=B.totalBytes-m.reduce((D,$e)=>D+($e.size||0),0);console.log(` --assets-only: kept ${m.length}/${O} records (dropped ${(V/1024).toFixed(1)} KiB of API responses)`)}}}l?.transformedBundle&&(z=await Ye(v,l.transformedBundle.byteLength))}finally{v.close()}l||(console.error(" could not read bundle snapshot \u2014 is sootsim running and is the bundle loaded?"),process.exit(2)),l.bundleUrl||(console.error(` no ?bundle= URL on the current sootsim tab.
|
|
78
|
+
open the app you want to share first (e.g. sootsim open 8082), then run upload.`),process.exit(2));let g,M=!1;if(z!==null)g=new TextEncoder().encode(z),M=!0,console.log(` using post-transform bundle: ${(g.byteLength/1024).toFixed(1)} KiB`);else{let n=Je(l.bundleUrl);n!==l.bundleUrl&&console.log(" forcing lazy=false for self-contained bundle"),console.log(` capturing: ${n}`),g=await j(n),console.log(` main bundle: ${(g.byteLength/1024).toFixed(1)} KiB`)}let k=X.filter(n=>n.url!==l.bundleUrl);Q&&(k=k.filter(n=>we(n.contentType))),console.log(` fetching ${k.length} extra files\u2026`);let E=(await Promise.all(k.map(async n=>{try{let i=await j(n.url);return{...n,bytes:i}}catch(i){return console.error(` warning: failed to re-fetch ${n.url}: ${i instanceof Error?i.message:i}`),null}}))).filter(n=>!!n),Be=E.reduce((n,i)=>n+i.bytes.byteLength,0);console.log(` ${E.length} extra files: ${(Be/1024).toFixed(1)} KiB`);let U;a&&(U=K(a));let h;c&&(h=K(c),console.log(` attaching screenshot: ${(h.byteLength/1024).toFixed(1)} KiB`));let w,F,Z;if(y){if(w=K(y),F=y.endsWith(".mp4")?"video/mp4":y.endsWith(".gif")?"image/gif":"video/webm",d){let n=Number(d);(!Number.isFinite(n)||n<=0)&&(console.error(` invalid --video-duration-ms: ${d}`),await P(),process.exit(1)),Z=Math.round(n)}console.log(` attaching flow video: ${(w.byteLength/1024).toFixed(1)} KiB (${F})`)}let R=de(),_=fe(R),Pe=se();ie({event:"preview_upload_started",identity:{userId:R?.kind==="session"?Pe?.user?.id??null:null,repoId:_?.repoId??null,installationId:_?.installationId??null},properties:{origin:o,hasAuth:!!R,authMode:R?.kind??"none",bundleBytes:g.byteLength,isTransformed:M,extraFiles:E.length,recordedFetches:m.length,hasEvents:!!U,hasScreenshot:!!h,hasVideo:!!w}}),R||(console.error(" preview uploads need auth."),console.error(" set SOOTSIM_API_KEY=sk_sootsim_..., run `sootsim login`, or use the soot github runner."),await P(),process.exit(1));let Ie=pe(R),Re=(()=>{try{return new URL(l.bundleUrl).origin}catch{return null}})(),H=(n,i,T,u,oe,x,B,p)=>{let O=ge(x),V=O?Me(n):n;return{url:i,urlhash:T,method:u,bodyHash:oe,requestVaryHash:p,contentType:x,responseHeaders:B,bytes:V,encoding:O?"gzip":void 0}},ee=[],C=Ke(l.pageHref)||"";if(C){let n=new Set;for(let i of ue(C)){let T=ye("sha256").update(i).digest("hex");if(!n.has(T)){n.add(T);try{let u=await j(i);ee.push(H(u,i,T,"GET","-",je(i))),console.log(` staged app font: ${i} (${(u.byteLength/1024).toFixed(1)} KiB)`)}catch(u){console.error(` warning: failed to fetch app font ${i}: ${u instanceof Error?u.message:u}`)}}}}let N=[...E.map(n=>H(n.bytes,n.url,n.urlhash,"GET","-",n.contentType,n.responseHeaders,n.requestVaryHash)),...m.map(n=>H(Buffer.from(n.bodyBase64,"base64"),n.url,n.urlhash,n.method||"GET",n.bodyHash||"-",n.contentType,n.responseHeaders,n.requestVaryHash)),...ee],W=Ae(g,{params:{[ze.BROTLI_PARAM_QUALITY]:9}}),_e=ye("sha256").update(g).digest("hex");console.log(` bundle raw=${(g.byteLength/1024).toFixed(1)} KiB br=${(W.byteLength/1024).toFixed(1)} KiB`);let xe={contentHash:_e,bundleSizeBytes:g.byteLength,bundleContentType:Ge(l.bundleUrl),bundleEncoding:"br",bundleOrigin:Re,appFonts:C||void 0,entry:l.entry,isTransformed:M,deviceSpec:l.deviceSpec,installationId:_?.installationId??void 0,repoId:_?.repoId,owner:$??_?.owner,repo:f??_?.repo,files:N.map(n=>({url:n.url,urlhash:n.urlhash,method:n.method,bodyHash:n.bodyHash,contentType:n.contentType,responseHeaders:n.responseHeaders,requestVaryHash:n.requestVaryHash,encoding:n.encoding,sizeBytes:n.bytes.byteLength})),events:U?{sizeBytes:U.byteLength}:void 0,snapshots:h?[{id:"landing",label:"Landing",kind:"landing",t:0,contentType:"image/png",sizeBytes:h.byteLength}]:void 0,flowVideo:w&&F?{sizeBytes:w.byteLength,contentType:F,durationMs:Z}:void 0},te=new Map;for(let n of N)te.set(n.urlhash,n.bytes);let ne=new Map;h&&ne.set("landing",h);let Oe=Date.now(),L=W.byteLength;for(let n of N)L+=n.bytes.byteLength;U&&(L+=U.byteLength),h&&(L+=h.byteLength),w&&(L+=w.byteLength);let Se=`${S(o)}/api/preview/upload/init`;console.log(` init: ${Se}`);let q;try{let n=await he({originBase:o,initBody:xe,bundleBytes:W,filesByHash:te,extras:{eventsBytes:U,videoBytes:w,snapshotBytes:ne},authHeader:Ie,concurrency:16});q=n.finalize;let i=1+n.init.files.length+(n.init.events?1:0)+n.init.snapshots.length+(n.init.flowVideo?1:0);console.log(` PUT ${i} objects in ${Date.now()-Oe}ms (${(L/1024).toFixed(1)} KiB)`)}catch(n){let i=n;i.phase==="init"&&i.status===401&&(console.error(" preview upload requires a valid login."),console.error(" run `sootsim login` and retry."),await P(),process.exit(1)),i.status===402&&(console.error(` ${He(i.responseText)??"preview uploads require Personal, Team, or an active trial \u2014 upgrade from the billing dialog."}`),await P(),process.exit(1)),console.error(` ${i.message??`${i.phase??"upload"} failed: ${String(n)}`}`),await P(),process.exit(1)}let re=`${s}${q.url}`;console.log(`
|
|
79
|
+
stored ${q.filesStored??0} extra files`),console.log(` preview: ${re}`),Ue&&await ae(re),await P()}export{Ne as a,Ve as b,dt as c};
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
/*! sootsim v0.1.
|
|
1
|
+
/*! sootsim v0.1.115 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
|
|
2
2
|
function A(e,t){let n=["--max-ms","--maxMs","--maxms","--max_ms"];for(let r of n){let s=e.indexOf(r);if(s>=0&&e[s+1]){let a=Number(e[s+1]);if(Number.isFinite(a))return Math.max(100,a)}}return t}var m=new Set(["tap","double-tap","tap-text","tap-id","long-press","touch"]);function f(e){return typeof e=="string"&&m.has(e)}async function w(e){return await e.send({type:"evaluate",code:"window.__sootsimEngineState?.inspectActive === true"})===!0}async function E(e,t){if(!f(t))return!1;try{return await w(e)}catch{return!1}}async function L(e){let t=await e.send({type:"evaluate",code:"(async () => await window.__sootsimTest.getNodeCount())()"});return{nodes:typeof t=="number"?t:0}}async function D(e,t=5){let n=await e.send({type:"evaluate",code:`(async () => await window.__sootsimTest.dumpTree(${t}))()`});return{depth:t,tree:n}}async function N(e){let t=await e.send({type:"evaluate",code:"window.location.href"});return{url:typeof t=="string"?t:""}}async function H(e,t){let n=`(async () => {
|
|
3
3
|
const t = window.__sootsimTest
|
|
4
4
|
const mainShell = window.SootSim?.bridges?.mainShell
|
|
@@ -161,9 +161,19 @@ function A(e,t){let n=["--max-ms","--maxMs","--maxms","--max_ms"];for(let r of n
|
|
|
161
161
|
}
|
|
162
162
|
`;function x(e){return e.testId?{mode:"testid",code:`(async () => {
|
|
163
163
|
const t = window.__sootsimTest
|
|
164
|
+
if (!t) return null
|
|
164
165
|
return (await t.findByTestId(${JSON.stringify(e.testId)})) || (await t.findById(${JSON.stringify(e.testId)}))
|
|
165
|
-
})()`}:e.role?{mode:"role",code:`(async () =>
|
|
166
|
+
})()`}:e.role?{mode:"role",code:`(async () => {
|
|
166
167
|
const t = window.__sootsimTest
|
|
168
|
+
if (!t) return []
|
|
169
|
+
return await t.queryAll({ hasRole: ${JSON.stringify(e.role)}, pruneHidden: true })
|
|
170
|
+
})()`}:e.type?{mode:"type",code:`(async () => {
|
|
171
|
+
const t = window.__sootsimTest
|
|
172
|
+
if (!t) return []
|
|
173
|
+
return await t.queryAll({ type: ${JSON.stringify(e.type)}, pruneHidden: true })
|
|
174
|
+
})()`}:e.pressable?{mode:"pressable",code:`(async () => {
|
|
175
|
+
const t = window.__sootsimTest
|
|
176
|
+
if (!t) return []
|
|
167
177
|
${y}
|
|
168
178
|
const inspectable = await fromInspectable()
|
|
169
179
|
if (inspectable) return inspectable.filter(n => n.pressable && isVisibleTarget(n))
|
|
@@ -171,6 +181,7 @@ ${y}
|
|
|
171
181
|
return all.filter(n => n.pressable && isVisibleTarget(n))
|
|
172
182
|
})()`}:e.interactive?{mode:"interactive-targets",code:`(async () => {
|
|
173
183
|
const t = window.__sootsimTest
|
|
184
|
+
if (!t) return []
|
|
174
185
|
${y}
|
|
175
186
|
const inspectable = await fromInspectable()
|
|
176
187
|
if (inspectable) {
|
|
@@ -179,9 +190,15 @@ ${y}
|
|
|
179
190
|
const all = await t.queryAll({ pruneHidden: true })
|
|
180
191
|
return all.filter(n => n.pressable && isVisibleTarget(n))
|
|
181
192
|
})()`}:e.visible?{mode:"visible",code:`(async () => {
|
|
182
|
-
const
|
|
193
|
+
const t = window.__sootsimTest
|
|
194
|
+
if (!t) return []
|
|
195
|
+
const all = await t.queryAll({ pruneHidden: true })
|
|
183
196
|
return all.filter(n => n.layout && n.layout.width > 0 && n.layout.height > 0)
|
|
184
|
-
})()`}:e.text?{mode:"text",code:`(async () =>
|
|
197
|
+
})()`}:e.text?{mode:"text",code:`(async () => {
|
|
198
|
+
const t = window.__sootsimTest
|
|
199
|
+
if (!t) return null
|
|
200
|
+
return await t.findByText(${JSON.stringify(e.text)})
|
|
201
|
+
})()`}:null}async function C(e,t){let n=x(t);if(!n)return null;let r=await e.send({type:"evaluate",code:n.code});return{mode:n.mode,result:r}}function M(e){return[...e].sort((t,n)=>p(n)-p(t))}function p(e){let t=0;e.testID&&(t+=100),typeof e.text=="string"&&e.text.trim().length>0&&(t+=60),typeof e.accessibilityLabel=="string"&&e.accessibilityLabel.trim().length>0&&(t+=30),e.accessibilityRole&&(t+=15);let n=e.layout?.width??0,r=e.layout?.height??0,s=n*r;return s>=400&&s<=6e4?t+=25:s>6e4&&(t-=20),(e.absolutePosition?.y??0)<0&&(t-=30),t}function $(e){if(e.testID)return`sootsim do tap-id ${g(e.testID)}`;let t=typeof e.text=="string"?e.text.trim():"";if(t.length>0&&t.length<=80)return`sootsim do tap-text ${g(t)}`;let n=Math.round(((e.absolutePosition?.x??0)+(e.layout?.width??0)/2)*10)/10,r=Math.round(((e.absolutePosition?.y??0)+(e.layout?.height??0)/2)*10)/10;return`sootsim do tap ${n} ${r}`}function g(e){return/^[A-Za-z0-9_./@:-]+$/.test(e)?e:`'${e.replace(/'/g,"'\\''")}'`}var S=`(async () => {
|
|
185
202
|
const t = window.__sootsimTest
|
|
186
203
|
let nodes = 0
|
|
187
204
|
try { nodes = (await t?.getNodeCount?.()) || 0 } catch {}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
/*! sootsim v0.1.
|
|
1
|
+
/*! sootsim v0.1.115 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
|
|
2
2
|
import{existsSync as C,mkdirSync as D,readFileSync as O,rmSync as s,writeFileSync as g}from"fs";import{tmpdir as P}from"os";import{dirname as R,join as A,resolve as x}from"path";import{execFileSync as a}from"child_process";import{readFileSync as I}from"fs";import{createRequire as S}from"module";var o=null;function u(r){o||(o=_());try{return o(r)}catch{return null}}function _(){let r=Number(process.env.PPID);if(Number.isFinite(r)&&r>1&&r!==process.ppid)return()=>r;if(process.platform==="linux")return y();if(process.platform==="darwin"){let n=E();return n||T()}return()=>null}function y(){return r=>{try{let n=I(`/proc/${r}/stat`,"utf8"),t=n.lastIndexOf(")");if(t<0)return null;let i=n.slice(t+1).trim().split(/\s+/),e=Number(i[1]);return Number.isFinite(e)&&e>0?e:null}catch{return null}}}var N=3,d=216,b=16;function E(){if(!process.versions?.bun)return null;try{let r=S(import.meta.url),{dlopen:n,FFIType:t}=r("bun:ffi"),i=n("/usr/lib/libproc.dylib",{proc_pidinfo:{args:[t.i32,t.i32,t.u64,t.ptr,t.i32],returns:t.i32}}),e=Buffer.alloc(d);return m=>{if(Number(i.symbols.proc_pidinfo(m,N,0n,e,d))<=0)return null;let p=e.readUInt32LE(b);return p>0?p:null}}catch{return null}}function T(){return r=>{try{let t=a("lsof",["-R","-p",String(r),"-d","cwd","-a"],{encoding:"utf8",timeout:2e3,stdio:["ignore","pipe","ignore"]}).split(`
|
|
3
3
|
`)[1];if(!t)return null;let i=t.trim().split(/\s+/),e=Number(i[2]);return Number.isFinite(e)&&e>1?e:null}catch{return null}}}var c=1,f="SOOTSIM_CLI_CURRENT_SIM_PATH",v=["SOOTSIM_CLI_IDENTITY","CLAUDE_CODE_SESSION_ID","CODEX_THREAD_ID","TERM_SESSION_ID","ITERM_SESSION_ID","TMUX_PANE","STY","KITTY_WINDOW_ID","WEZTERM_PANE","ALACRITTY_WINDOW_ID","WINDOWID","VSCODE_INJECTION"];function F(r,n=20){let t=u(r);if(!t||t<=1)return null;for(let i=0;i<n;i++){let e=u(t);if(!e||e<=1)return t;t=e}return t}function k(){for(let n of v){let t=process.env[n];if(t&&t.trim())return{key:`${n}:${t.trim()}`,source:n,stable:!0}}let r=F(process.ppid);return r&&r>1?{key:`gppid-${r}`,source:"grand-ppid",stable:!0}:{key:`pid-${process.ppid}`,source:"ppid",stable:!1}}function L(){return k().key}function l(){if(process.env[f])return x(process.env[f]);let r=L();return A(P(),`sootsim-current-sim-${r}.json`)}function h(){let r=l();if(!C(r))return null;try{let n=JSON.parse(O(r,"utf8"));return n.version!==c||typeof n.simId!="string"||!n.simId.trim()||typeof n.updatedAt!="string"?(s(r,{force:!0}),null):{version:c,simId:n.simId.trim(),updatedAt:n.updatedAt}}catch{return s(r,{force:!0}),null}}function J(){return h()?.simId||null}function j(r){let n=r.trim();if(!n)return;let t=l();D(R(t),{recursive:!0}),g(t,JSON.stringify({version:c,simId:n,updatedAt:new Date().toISOString()},null,2)+`
|
|
4
4
|
`)}function K(){s(l(),{force:!0})}export{k as a,L as b,J as c,j as d,K as e};
|
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
/*! sootsim v0.1.
|
|
2
|
-
import{a as D,b as q,c as de,e as me}from"./chunk-GR2XFJJF.js";import{m as O}from"./chunk-ZVL24GTC.js";import{k as le,m as ce}from"./chunk-WRB6MZBX.js";import{a as pe,b as fe}from"./chunk-HKRPDZOI.js";import{b as ae,c as I,d as v,e as x}from"./chunk-IDUGL3MK.js";import{c as k,d as w,e as T}from"./chunk-VRS7EUIU.js";import{g as G}from"./chunk-O5WZBC2Y.js";import{a as M,b as V}from"./chunk-MGWQ527R.js";import{c as ue}from"./chunk-XPQUENRQ.js";import{d as j}from"./chunk-UBDSTHBH.js";import{a as ie}from"./chunk-MOSWK2MT.js";import{E as re,F as se}from"./chunk-M6LHGDWE.js";import X from"node:fs";import Le from"node:os";import P from"node:path";function Ee(e){return e.replace(/^\[|\]$/g,"").toLowerCase()}function J(e){let o=Ee(e);return o==="localhost"||o.endsWith(".localhost")||o==="0.0.0.0"||o==="::1"||/^127(?:\.\d{1,3}){3}$/.test(o)}function ge(e){return new Error(`could not resolve a native bundle for ${e}. pass an explicit bundle URL or open Connect and choose the app there.`)}function Re(e,o){return e?.includes("/one/metro-entry.bundle")?"one":typeof o=="string"&&o?"expo":"unknown"}function be(e,o,t){return`${e}//${o}:${t}`}function Ue(e){if(typeof window>"u")return!1;try{let o=new URL(e);return J(o.hostname)&&J(window.location.hostname)&&o.origin!==window.location.origin}catch{return!1}}async function he(e,o){let t={...o,cache:o?.cache??"no-store"};return Ue(e)?fetch(`/__fetch-proxy?url=${encodeURIComponent(e)}`,t):fetch(e,t)}function Be(e){return e==="https:"?443:80}function Oe(e){let o=e.pathname||"/";return(o==="/"||o==="")&&!e.search&&!e.hash}async function ye(e,o){let t=e.replace(/\/+$/,"");try{let n=await he(`${t}/`,{headers:{"expo-platform":"ios"}});if(n.ok){let r=await n.json(),s=r?.extra?.expoClient||r?.extra||{},a=typeof r?.launchAsset?.url=="string"?r.launchAsset.url:void 0;if(a||s.name)return{bundleUrl:de(a||`${t}${q}`),port:o,framework:Re(a,s.sdkVersion),projectName:s.name}}}catch{}try{let n=await he(`${t}/status`);if(n.ok&&(await n.text()).includes("packager-status:running"))return{bundleUrl:`${t}${q}`,port:o,framework:"metro"}}catch{}return null}async function we(e){return ye(be("http:","localhost",e),e)}async function ve(e){let o=e.trim();if(/^\d+$/.test(o)){let i=parseInt(o,10),l=await we(i);if(l)return l;let c=3;for(let d=1;d<=c;d++){let m=await we(i+d);if(m)return m}throw ge(`localhost:${i} (also scanned +1..+${c})`)}let t=o.startsWith("http")?o:`http://${o}`,n;try{n=new URL(t)}catch{throw new Error(`could not parse "${e}". pass a dev-server port, a dev-server base URL, or a full bundle URL.`)}let r=n.protocol||"http:",s=n.port?parseInt(n.port,10):Be(r),a=be(r,n.hostname,s);if(Oe(n)){let i=await ye(a,s);if(i)return i;throw ge(a)}return{bundleUrl:n.toString(),port:s,framework:"unknown"}}function wt(){console.error(" no sootsim desktop companion found."),console.error(""),console.error(" install the desktop app:"),console.error(" sootsim install-desktop"),console.error(""),console.error(" or wire sootsim into your own project so your dev server hosts the bridge:"),console.error(" sootsim setup-repo"),console.error("")}function Se(e){let o=M();if(console.log(` note: no sootsim bridge detected on port ${e}`),o){console.log(" launch the installed companion with:"),console.log(" sootsim electron");return}console.log(""),console.log(" to get a bridge running, either:"),console.log(" sootsim install-desktop # download the electron app"),console.log(" sootsim setup-repo # wire sootsim into your own project")}function De(e){console.error(""),console.error(` no sim is connected to the sootsim bridge on port ${e}.`),console.error(""),console.error(" start your app dev server, then open it in sootsim:"),console.error(" npx expo start --localhost --port 8081"),console.error(" sootsim open 8081"),console.error(""),console.error(" then inspect the live app:"),console.error(" sootsim describe")}async function bt(e,o,t){let n=[];try{n=await e.listSims()}catch{}if(console.error(""),n.length===0){De(o);return}console.error(` no sim with id "${t}" is connected to the bridge on port ${o}.`),console.error(""),console.error(" connected sims:");for(let r of n){let s=[r.isPrimary?"primary":null,r.readyState,r.userVisible===!1?"hidden":r.userVisible===!0?"visible":null,r.userFocused?"focused":null].filter(Boolean).join(", ");console.error(` ${r.id} (${s})`)}console.error(""),console.error(" pass a valid --sim id, or run `sootsim list` to see all sessions.")}var Ae=12e4,$e=500,Ne=5e3,Fe={rn:"/rn",connectrn:"/rn","connect-rn":"/rn",clock:"/app/clock","native-ui":"/app/native-ui",tamagui:"/app/tamagui",settings:"/app/settings",photos:"/app/photos",camera:"/app/camera"};function S(e){return new Promise(o=>setTimeout(o,e))}function Ie(e){let o=process.env[e]?.trim();if(!o)return null;let t=Number(o);return Number.isInteger(t)&&t>0?t:null}function He(){let e=Ie("SOOTSIM_DRIVER_CONNECT_TIMEOUT_MS")??Ie("SOOTSIM_PW_CONNECT_TIMEOUT_MS")??Ae;return{timeoutMs:e,hostTimeoutMs:e+Ne,intervalMs:$e,attempts:Math.max(1,Math.ceil(e/$e))}}function Ke(e){return e==="chromium"||e==="electron"||e==="playwright"||e==="system"}function N(e){return e.trim()}function Y(e){try{let o=new URL(e),t=o.pathname.replace(/\/+$/,"")||"/";return o.searchParams.has("open")||o.searchParams.has("port")||o.searchParams.has("bundle")||o.searchParams.has("demo")||o.pathname.includes("/sootsim/index.html")||t==="/sootsim"||o.pathname==="/__soot"||o.pathname==="/__soot/"||t==="/rn"||/^\/rn\/[^/]+$/i.test(t)||/^\/app\/[^/]+$/i.test(t)||t==="/__soot/rn"||/^\/__soot\/rn\/[^/]+$/i.test(t)||/^\/__soot\/app\/[^/]+$/i.test(t)}catch{return!1}}function We(e){try{let o=new URL(e).pathname.replace(/\/+$/,"")||"/";return/^\/preview\/[^/]+$/.test(o)||/^\/build\/.+/.test(o)}catch{return!1}}async function At(e){let o=N(e);return(await ke(o)).bundleUrl}function ze(e){try{let o=new URL(e.startsWith("http")?e:`http://${e}`),t=o.pathname||"/";if(t!=="/"&&t!==""||o.search||o.hash||!/^(localhost|127\.0\.0\.1|\[::1\]|::1)$/i.test(o.hostname))return null;let n=o.port?Number(o.port):o.protocol==="https:"?443:80;return Number.isFinite(n)&&n>0?n:null}catch{return null}}async function ke(e){let o=N(e),t=/^\d+$/.test(o)?Number(o):ze(o);if(t&&t>0){let n=await me(t);if(n)return{bundleUrl:n.bundleUrl,port:n.port,framework:n.framework,projectName:n.projectName}}return ve(o)}function Z(e){let o=e.replace(/\/+$/,"")||"/";return o==="/__soot"||o.startsWith("/__soot/")?"/__soot":""}function je(e,o){let t=N(e),n=new URL(o),r=Z(n.pathname);return n.pathname=`${r}/rn`,n.searchParams.delete("bundle"),n.searchParams.delete("demo"),n.searchParams.delete("app"),n.searchParams.delete("open"),n.searchParams.delete("port"),/^\d+$/.test(t)?n.pathname=`${r}/rn/${t}`:(n.pathname=`${r}/rn`,n.searchParams.set("open",t)),n.toString()}function qe(e){let o=N(e);return/^\d+$/.test(o)||/^https?:\/\//i.test(o)?!0:/^(localhost|127\.0\.0\.1|\[::1\]|[^/]+\.localhost):\d+(?:\/.*)?$/i.test(o)}async function Ve(e,o){let t=await ke(e),n=new URL(o),r=Z(n.pathname);return n.pathname=`${r}/rn`,n.searchParams.delete("open"),n.searchParams.delete("port"),n.searchParams.delete("demo"),n.searchParams.delete("app"),n.searchParams.set("bundle",t.bundleUrl),n.toString()}function Ge(e){return e.startsWith("~/")?P.join(Le.homedir(),e.slice(2)):P.isAbsolute(e)?e:P.resolve(process.cwd(),e)}function Je(e){let o={};for(let t=0;t<e.length;t++){if(e[t]!=="--replace")continue;let n=e[t+1];n||(console.error(" sootsim open: --replace expects <module>=<file>"),process.exit(1));let r=n.indexOf("=");(r<=0||r===n.length-1)&&(console.error(" sootsim open: --replace expects <module>=<file>"),process.exit(1));let s=n.slice(0,r).trim(),a=Ge(n.slice(r+1).trim());X.existsSync(a)||(console.error(` sootsim open: replacement file not found: ${a}`),process.exit(1)),o[s]={file:a},t++}return Object.keys(o).length>0?{modules:o}:void 0}function Te(){let e=re();return se(e)&&e.runtimePort>0?`http://localhost:${e.runtimePort}/`:ie}async function Qe(e){try{let o=new URL(e),t=o.searchParams.get("bundle")||"",n=pe({bundleUrl:t});if(n.length===0)return e;let r=await Ye(n);return r.length===0?e:(o.searchParams.set("appFonts",fe(r)),o.toString())}catch{return e}}async function Ye(e){let o=await Promise.all(e.map(async n=>{let r=new AbortController,s=setTimeout(()=>r.abort(),1500);try{let a=await fetch(n.url,{method:"HEAD",signal:r.signal});return a.status>=200&&a.status<400?n:null}catch{return null}finally{clearTimeout(s)}})),t=[];for(let n of o)n!==null&&t.push(n);return t}async function Xe(e,o){if(!e)return new URL(o).toString();if(Y(e))return new URL(e).toString();let t=Fe[e.toLowerCase()];if(t){let n=new URL(o),r=Z(n.pathname);return n.pathname=`${r}${t}`,n.toString()}return qe(e)?Ve(e,o):je(e,o)}async function A(e,o=Te()){let t=await Xe(e,o);return Qe(t)}function xe(e,o){let t=e?.url||e?.origin||o;try{let n=new URL(t);return n.searchParams.delete("bundle"),n.searchParams.delete("demo"),n.searchParams.delete("app"),n.searchParams.delete("open"),n.searchParams.delete("port"),n.searchParams.delete("inspectOpen"),n.searchParams.delete("appFonts"),n.toString()}catch{return o}}async function Ze(e,o,t){let n=new URL(await A(e,o));return n.searchParams.set("inspectOpen",t),n.toString()}async function et(e,o,t,n={}){let r=n.attempts??30,s=n.intervalMs??500,a=n.minNodeCount??10;for(let i=0;i<r;i++){let l=v(e,{commandTimeoutMs:o,simId:t,simIdSource:"flag"});try{let c=await l.send({type:"evaluate",code:"(async () => (await window.__sootsimTest?.getNodeCount()) || 0)()"});if(typeof c=="number"&&c>a)return{bridge:l,count:c}}catch{}l.close(),await S(s)}return null}function tt(e){if(!e)return null;try{let o=new URL(e);if(o.searchParams.has("bundle")){let t=o.searchParams.get("bundle")||"";try{let n=new URL(t),r=n.pathname.length>36?`...${n.pathname.slice(-36)}`:n.pathname;return`bundle ${n.host}${r}`}catch{return"bundle"}}return o.searchParams.has("port")?`connect :${o.searchParams.get("port")||""}`:o.searchParams.has("open")?`connect ${o.searchParams.get("open")||""}`:o.searchParams.has("demo")?`demo ${o.searchParams.get("demo")||"default"}`:o.pathname.includes("/sootsim/index.html")||o.pathname==="/sootsim/"||o.pathname==="/sootsim"?"embedded sootsim":null}catch{return null}}function ot(e,o){if(e.length===0){console.log(" no sims connected");return}console.log(` connected sims (${e.length}):
|
|
3
|
-
`);for(let t of e){let n=t.lockedBy&&t.lockExpiresAt?`locked by ${t.lockedBy} (${Math.max(0,Math.round((t.lockExpiresAt-Date.now())/1e3))}s)`:"",r=[t.isPrimary?"primary":"",t.id===o?"selected":"",t.readyState,t.userVisible===!1?"hidden":t.userVisible===!0?"visible":"",t.attachedCliCount&&t.attachedCliCount>0?"in use":"",t.userFocused?"focused":"",n].filter(Boolean);console.log(` ${t.id}${r.length?` [${r.join(", ")}]`:""}`);let s=tt(t.url);if(s)console.log(` loaded: ${s}`);else if(t.url){let
|
|
4
|
-
`,{flag:"w"}),!0}catch{return!1}}function
|
|
5
|
-
${o.slice(-4e3)}`:o;console.error(` ${P.basename(e.diagnosticLogPath)}:`),console.error(t)}catch(o){let t=o instanceof Error?o.message:String(o);console.error(` failed to read ${P.basename(e.diagnosticLogPath)}: ${t}`)}}function Ce(e){try{return process.kill(e,0),!0}catch(o){return o?.code==="EPERM"}}async function
|
|
1
|
+
/*! sootsim v0.1.115 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
|
|
2
|
+
import{a as D,b as q,c as de,e as me}from"./chunk-X35ALXTD.js";import{m as B}from"./chunk-ICZSY2XS.js";import{k as le,m as ce}from"./chunk-SLSNMSD7.js";import{a as pe,b as fe}from"./chunk-TF5LTELW.js";import{b as ae,c as I,d as v,e as x}from"./chunk-Q3BVLSSD.js";import{c as k,d as w,e as T}from"./chunk-SPVHYFZ3.js";import{g as G}from"./chunk-OK2FCYQN.js";import{a as M,b as V}from"./chunk-XO3XTPFR.js";import{c as ue}from"./chunk-GUDD5CIA.js";import{d as j}from"./chunk-IWC4SYEG.js";import{a as ie}from"./chunk-VSKAOUZ5.js";import{E as re,F as se}from"./chunk-UNRBTUAA.js";import Y from"node:fs";import Le from"node:os";import P from"node:path";function _e(e){return e.replace(/^\[|\]$/g,"").toLowerCase()}function J(e){let o=_e(e);return o==="localhost"||o.endsWith(".localhost")||o==="0.0.0.0"||o==="::1"||/^127(?:\.\d{1,3}){3}$/.test(o)}function ge(e){return new Error(`could not resolve a native bundle for ${e}. pass an explicit bundle URL or open Connect and choose the app there.`)}function Re(e,o){return e?.includes("/one/metro-entry.bundle")?"one":typeof o=="string"&&o?"expo":"unknown"}function be(e,o,t){return`${e}//${o}:${t}`}function Ue(e){if(typeof window>"u")return!1;try{let o=new URL(e);return J(o.hostname)&&J(window.location.hostname)&&o.origin!==window.location.origin}catch{return!1}}async function he(e,o){let t={...o,cache:o?.cache??"no-store"};return Ue(e)?fetch(`/__fetch-proxy?url=${encodeURIComponent(e)}`,t):fetch(e,t)}function Oe(e){return e==="https:"?443:80}function Be(e){let o=e.pathname||"/";return(o==="/"||o==="")&&!e.search&&!e.hash}async function ye(e,o){let t=e.replace(/\/+$/,"");try{let n=await he(`${t}/`,{headers:{"expo-platform":"ios"}});if(n.ok){let r=await n.json(),s=r?.extra?.expoClient||r?.extra||{},a=typeof r?.launchAsset?.url=="string"?r.launchAsset.url:void 0;if(a||s.name)return{bundleUrl:de(a||`${t}${q}`),port:o,framework:Re(a,s.sdkVersion),projectName:s.name}}}catch{}try{let n=await he(`${t}/status`);if(n.ok&&(await n.text()).includes("packager-status:running"))return{bundleUrl:`${t}${q}`,port:o,framework:"metro"}}catch{}return null}async function we(e){return ye(be("http:","localhost",e),e)}async function ve(e){let o=e.trim();if(/^\d+$/.test(o)){let i=parseInt(o,10),l=await we(i);if(l)return l;let c=3;for(let d=1;d<=c;d++){let m=await we(i+d);if(m)return m}throw ge(`localhost:${i} (also scanned +1..+${c})`)}let t=o.startsWith("http")?o:`http://${o}`,n;try{n=new URL(t)}catch{throw new Error(`could not parse "${e}". pass a dev-server port, a dev-server base URL, or a full bundle URL.`)}let r=n.protocol||"http:",s=n.port?parseInt(n.port,10):Oe(r),a=be(r,n.hostname,s);if(Be(n)){let i=await ye(a,s);if(i)return i;throw ge(a)}return{bundleUrl:n.toString(),port:s,framework:"unknown"}}function yt(){console.error(" no sootsim desktop companion found."),console.error(""),console.error(" install the desktop app:"),console.error(" sootsim install-desktop"),console.error(""),console.error(" or wire sootsim into your own project so your dev server hosts the bridge:"),console.error(" sootsim setup-repo"),console.error("")}function Se(e){let o=M();if(console.log(` note: no sootsim bridge detected on port ${e}`),o){console.log(" launch the installed companion with:"),console.log(" sootsim electron");return}console.log(""),console.log(" to get a bridge running, either:"),console.log(" sootsim install-desktop # download the electron app"),console.log(" sootsim setup-repo # wire sootsim into your own project")}function De(e){console.error(""),console.error(` no sim is connected to the sootsim bridge on port ${e}.`),console.error(""),console.error(" start your app dev server, then open it in sootsim:"),console.error(" npx expo start --localhost --port 8081"),console.error(" sootsim open 8081"),console.error(""),console.error(" then inspect the live app:"),console.error(" sootsim describe")}async function vt(e,o,t){let n=[];try{n=await e.listSims()}catch{}if(console.error(""),n.length===0){De(o);return}console.error(` no sim with id "${t}" is connected to the bridge on port ${o}.`),console.error(""),console.error(" connected sims:");for(let r of n){let s=[r.isPrimary?"primary":null,r.readyState,r.userVisible===!1?"hidden":r.userVisible===!0?"visible":null,r.userFocused?"focused":null].filter(Boolean).join(", ");console.error(` ${r.id} (${s})`)}console.error(""),console.error(" pass a valid --sim id, or run `sootsim list` to see all sessions.")}var Ae=12e4,$e=500,Ne=5e3,Fe={rn:"/rn",connectrn:"/rn","connect-rn":"/rn",clock:"/app/clock","native-ui":"/app/native-ui",tamagui:"/app/tamagui",settings:"/app/settings",photos:"/app/photos",camera:"/app/camera"};function S(e){return new Promise(o=>setTimeout(o,e))}function Ie(e){let o=process.env[e]?.trim();if(!o)return null;let t=Number(o);return Number.isInteger(t)&&t>0?t:null}function He(){let e=Ie("SOOTSIM_DRIVER_CONNECT_TIMEOUT_MS")??Ie("SOOTSIM_PW_CONNECT_TIMEOUT_MS")??Ae;return{timeoutMs:e,hostTimeoutMs:e+Ne,intervalMs:$e,attempts:Math.max(1,Math.ceil(e/$e))}}function Ke(e){return e==="chromium"||e==="electron"||e==="playwright"||e==="system"}function N(e){return e.trim()}function X(e){try{let o=new URL(e),t=o.pathname.replace(/\/+$/,"")||"/";return o.searchParams.has("open")||o.searchParams.has("port")||o.searchParams.has("bundle")||o.searchParams.has("demo")||o.pathname.includes("/sootsim/index.html")||t==="/sootsim"||o.pathname==="/__soot"||o.pathname==="/__soot/"||t==="/rn"||/^\/rn\/[^/]+$/i.test(t)||/^\/app\/[^/]+$/i.test(t)||t==="/__soot/rn"||/^\/__soot\/rn\/[^/]+$/i.test(t)||/^\/__soot\/app\/[^/]+$/i.test(t)}catch{return!1}}function We(e){try{let o=new URL(e).pathname.replace(/\/+$/,"")||"/";return/^\/preview\/[^/]+$/.test(o)||/^\/build\/.+/.test(o)}catch{return!1}}async function Ft(e){let o=N(e);return(await ke(o)).bundleUrl}function ze(e){try{let o=new URL(e.startsWith("http")?e:`http://${e}`),t=o.pathname||"/";if(t!=="/"&&t!==""||o.search||o.hash||!/^(localhost|127\.0\.0\.1|\[::1\]|::1)$/i.test(o.hostname))return null;let n=o.port?Number(o.port):o.protocol==="https:"?443:80;return Number.isFinite(n)&&n>0?n:null}catch{return null}}async function ke(e){let o=N(e),t=/^\d+$/.test(o)?Number(o):ze(o);if(t&&t>0){let n=await me(t);if(n)return{bundleUrl:n.bundleUrl,port:n.port,framework:n.framework,projectName:n.projectName}}return ve(o)}function Z(e){let o=e.replace(/\/+$/,"")||"/";return o==="/__soot"||o.startsWith("/__soot/")?"/__soot":""}function je(e,o){let t=N(e),n=new URL(o),r=Z(n.pathname);return n.pathname=`${r}/rn`,n.searchParams.delete("bundle"),n.searchParams.delete("demo"),n.searchParams.delete("app"),n.searchParams.delete("open"),n.searchParams.delete("port"),/^\d+$/.test(t)?n.pathname=`${r}/rn/${t}`:(n.pathname=`${r}/rn`,n.searchParams.set("open",t)),n.toString()}function qe(e){let o=N(e);return/^\d+$/.test(o)||/^https?:\/\//i.test(o)?!0:/^(localhost|127\.0\.0\.1|\[::1\]|[^/]+\.localhost):\d+(?:\/.*)?$/i.test(o)}async function Ve(e,o){let t=await ke(e),n=new URL(o),r=Z(n.pathname);return n.pathname=`${r}/rn`,n.searchParams.delete("open"),n.searchParams.delete("port"),n.searchParams.delete("demo"),n.searchParams.delete("app"),n.searchParams.set("bundle",t.bundleUrl),n.toString()}function Ge(e){return e.startsWith("~/")?P.join(Le.homedir(),e.slice(2)):P.isAbsolute(e)?e:P.resolve(process.cwd(),e)}function Je(e){let o={};for(let t=0;t<e.length;t++){if(e[t]!=="--replace")continue;let n=e[t+1];n||(console.error(" sootsim open: --replace expects <module>=<file>"),process.exit(1));let r=n.indexOf("=");(r<=0||r===n.length-1)&&(console.error(" sootsim open: --replace expects <module>=<file>"),process.exit(1));let s=n.slice(0,r).trim(),a=Ge(n.slice(r+1).trim());Y.existsSync(a)||(console.error(` sootsim open: replacement file not found: ${a}`),process.exit(1)),o[s]={file:a},t++}return Object.keys(o).length>0?{modules:o}:void 0}function Te(){let e=re();return se(e)&&e.runtimePort>0?`http://localhost:${e.runtimePort}/`:ie}async function Qe(e){try{let o=new URL(e),t=o.searchParams.get("bundle")||"",n=pe({bundleUrl:t});if(n.length===0)return e;let r=await Xe(n);return r.length===0?e:(o.searchParams.set("appFonts",fe(r)),o.toString())}catch{return e}}async function Xe(e){let o=await Promise.all(e.map(async n=>{let r=new AbortController,s=setTimeout(()=>r.abort(),1500);try{let a=await fetch(n.url,{method:"HEAD",signal:r.signal});return a.status>=200&&a.status<400?n:null}catch{return null}finally{clearTimeout(s)}})),t=[];for(let n of o)n!==null&&t.push(n);return t}async function Ye(e,o){if(!e)return new URL(o).toString();if(X(e))return new URL(e).toString();let t=Fe[e.toLowerCase()];if(t){let n=new URL(o),r=Z(n.pathname);return n.pathname=`${r}${t}`,n.toString()}return qe(e)?Ve(e,o):je(e,o)}async function A(e,o=Te()){let t=await Ye(e,o);return Qe(t)}function xe(e,o){let t=e?.url||e?.origin||o;try{let n=new URL(t);return n.searchParams.delete("bundle"),n.searchParams.delete("demo"),n.searchParams.delete("app"),n.searchParams.delete("open"),n.searchParams.delete("port"),n.searchParams.delete("inspectOpen"),n.searchParams.delete("appFonts"),n.toString()}catch{return o}}async function Ze(e,o,t){let n=new URL(await A(e,o));return n.searchParams.set("inspectOpen",t),n.toString()}async function et(e,o,t,n={}){let r=n.attempts??30,s=n.intervalMs??500,a=n.minNodeCount??10;for(let i=0;i<r;i++){let l=v(e,{commandTimeoutMs:o,simId:t,simIdSource:"flag"});try{let c=await l.send({type:"evaluate",code:"(async () => (await window.__sootsimTest?.getNodeCount()) || 0)()"});if(typeof c=="number"&&c>a)return{bridge:l,count:c}}catch{}l.close(),await S(s)}return null}function tt(e){if(!e)return null;try{let o=new URL(e);if(o.searchParams.has("bundle")){let t=o.searchParams.get("bundle")||"";try{let n=new URL(t),r=n.pathname.length>36?`...${n.pathname.slice(-36)}`:n.pathname;return`bundle ${n.host}${r}`}catch{return"bundle"}}return o.searchParams.has("port")?`connect :${o.searchParams.get("port")||""}`:o.searchParams.has("open")?`connect ${o.searchParams.get("open")||""}`:o.searchParams.has("demo")?`demo ${o.searchParams.get("demo")||"default"}`:o.pathname.includes("/sootsim/index.html")||o.pathname==="/sootsim/"||o.pathname==="/sootsim"?"embedded sootsim":null}catch{return null}}function ot(e){let o=e.indexOf(":");if(o<=0)return e.length>24?`${e.slice(0,21)}\u2026`:e;let t=e.slice(0,o),n=e.slice(o+1),r=n.length>12?`${n.slice(0,8)}\u2026`:n;return t==="CLAUDE_CODE_SESSION_ID"||t==="CODEX_THREAD_ID"?`agent ${r}`:`${t} ${r}`}function nt(e,o){if(e.length===0){console.log(" no sims connected");return}console.log(` connected sims (${e.length}):
|
|
3
|
+
`);for(let t of e){let n=t.lockedBy&&t.lockExpiresAt?`locked by ${ot(t.lockedBy)} (${Math.max(0,Math.round((t.lockExpiresAt-Date.now())/1e3))}s)`:"",r=[t.isPrimary?"primary":"",t.id===o?"selected":"",t.readyState,t.userVisible===!1?"hidden":t.userVisible===!0?"visible":"",t.attachedCliCount&&t.attachedCliCount>0?"in use":"",t.userFocused?"focused":"",n].filter(Boolean);console.log(` ${t.id}${r.length?` [${r.join(", ")}]`:""}`);let s=tt(t.url);if(s)console.log(` loaded: ${s}`);else if(t.url){let i=t.url.length>96?`${t.url.slice(0,93)}\u2026`:t.url;console.log(` url: ${i}`)}else t.origin&&console.log(` origin: ${t.origin}`);let a=rt(t);a&&console.log(` engine: ${a}`),t.title&&console.log(` title: ${t.title}`),t.visibilityState&&t.visibilityState!=="visible"&&console.log(` visibility: ${t.visibilityState}`),console.log(` connected: ${Pe(Date.now()-t.connectedAt)}`),t.lastActiveAt&&t.lastActiveAt>0&&console.log(` last active: ${Pe(Date.now()-t.lastActiveAt)}`)}}function rt(e){let o=e.url||e.origin||"";if(!o)return null;try{let t=new URL(o).port;return t==="5173"?"dev \u2014 local source (:5173)":t==="3000"?"soot app embedded (:3000)":`runtime \u2014 published engine (:${t||"?"})`}catch{return null}}function Pe(e){let o=Math.max(0,Math.round(e/1e3));return o<60?`${o}s ago`:o<3600?`${Math.round(o/60)}m ago`:o<86400?`${Math.round(o/3600)}h ago`:`${Math.round(o/86400)}d ago`}async function L(e,o,t,n={}){let r=n.attempts??30,s=n.intervalMs??500;for(let a=0;a<r;a++){let i=v(e,{commandTimeoutMs:o});try{let c=(await i.listSims()).find(t);if(c)return c}catch{}finally{i.close()}await S(s)}return null}async function st(e,o,t,n={}){let r=n.attempts??20,s=n.intervalMs??250;for(let a=0;a<r;a++){let i=v(e,{commandTimeoutMs:o});try{let c=(await i.listSims()).find(d=>d.id===t);if(!c||c.readyState!=="open")return!0}catch{return!0}finally{i.close()}await S(s)}return!1}function it(e){let o=e.meta;if(!o||o.sootsimHostDriver!=="playwright")return null;let t=Number(o.sootsimHostPid);return!Number.isInteger(t)||t<=1||t===process.pid?null:t}function at(e){if(!e.connectAckFile)return!1;try{return Y.writeFileSync(e.connectAckFile,`${JSON.stringify({connectedAt:Date.now(),pid:e.pid??null})}
|
|
4
|
+
`,{flag:"w"}),!0}catch{return!1}}function lt(e){if(e.diagnosticLogPath)try{let o=Y.readFileSync(e.diagnosticLogPath,"utf8").trim();if(!o){console.error(` ${P.basename(e.diagnosticLogPath)} was empty`);return}let t=o.length>4e3?`\u2026
|
|
5
|
+
${o.slice(-4e3)}`:o;console.error(` ${P.basename(e.diagnosticLogPath)}:`),console.error(t)}catch(o){let t=o instanceof Error?o.message:String(o);console.error(` failed to read ${P.basename(e.diagnosticLogPath)}: ${t}`)}}function Ce(e){try{return process.kill(e,0),!0}catch(o){return o?.code==="EPERM"}}async function ct(e,o=2500){let t=Date.now()+o;for(;Date.now()<t;){if(!Ce(e))return!0;await S(100)}return!Ce(e)}async function Q(e,o){let t=new Set(o),n=new Set;for(let r of e){if(!t.has(r.id))continue;let s=it(r);s&&n.add(s)}for(let r of n){try{process.kill(r,"SIGTERM")}catch{}if(await ct(r)){console.log(` closed playwright host process ${r}`);continue}try{process.kill(r,"SIGKILL"),console.log(` force-closed playwright host process ${r}`)}catch{}}}function F(e,o){if(o){let r=o.trim(),s=e.find(a=>a.id===r);if(!s)throw new Error(`no sim connected with id ${r}`);return s}let t=e.find(r=>r.isPrimary&&r.readyState==="open");if(t)return t;let n=e.find(r=>r.readyState==="open");if(n)return n;throw new Error("no sim connected")}function E(e,o,t){console.log(` ${t==="current sim"?"loaded":"opened"}: ${e} [${t}]`),console.log(` current sim: ${o.id}`),console.log(JSON.stringify({simId:o.id,url:o.url},null,2))}async function _(e,o,t){if(t.includes("--no-describe"))return;let n=process.env.SOOTSIM_QUIET_TARGET_NOTICE;process.env.SOOTSIM_QUIET_TARGET_NOTICE="1";try{await dt(e,o,{stableMs:150,maxMs:400});let r=v(e,{commandTimeoutMs:3e3,simId:o,cliLabel:"open --describe",simIdSource:"flag"});try{let s=null;try{s=await r.send({type:"evaluate",code:le})}catch{}if(s&&!(s.flag===!0&&!s.loadingText&&ce(s))){let d=s.loadingText?`still showing "${s.loadingText}"`:s.flag!==!0?"guest ready event has not fired":s.targets<=0?"no visible app content is inspectable yet":"app tree is still stabilizing";console.log(` app still loading: ${d} (nodes: ${s.nodes}, targets: ${s.targets})`),console.log(" before interacting, run: sootsim wait ready --max-ms 120000")}let i=`(async () => {
|
|
6
6
|
const t = window.__sootsimTest
|
|
7
7
|
const ms = window.SootSim?.bridges?.mainShell
|
|
8
8
|
if (!t) return null
|
|
@@ -10,4 +10,4 @@ ${o.slice(-4e3)}`:o;console.error(` ${P.basename(e.diagnosticLogPath)}:`),conso
|
|
|
10
10
|
try { shell = ms?.getState ? await ms.getState() : null } catch {}
|
|
11
11
|
const tree = await t.dumpTree(12, ${JSON.stringify({describe:!0,verbose:!1,filter:""})})
|
|
12
12
|
return { tree, shell }
|
|
13
|
-
})()`,l=await r.send({type:"evaluate",code:i});if(!l?.tree)return;if(console.log(""),l.shell?.state){let c=[`state=${l.shell.state}`,l.shell.activeApp?`app=${l.shell.activeApp}`:null].filter(Boolean);console.log(` shell: ${c.join(" ")}`)}console.log(l.tree)}finally{r.close()}}catch{}finally{n===void 0?delete process.env.SOOTSIM_QUIET_TARGET_NOTICE:process.env.SOOTSIM_QUIET_TARGET_NOTICE=n}}async function lt(e,o,t){let n=Date.now()+t.maxMs,r="(async () => (await window.__sootsimTest?.getNodeCount?.()) || 0)()",s=v(e,{commandTimeoutMs:2e3,simId:o,simIdSource:"flag"});try{let a=-1,i=0;for(;Date.now()<n;){let l=-1;try{let c=await s.send({type:"evaluate",code:r});typeof c=="number"&&(l=c)}catch{}if(l>=0&&l===a){if(Date.now()-i>=t.stableMs)return}else a=l,i=Date.now();await S(50)}}finally{s.close()}}async function Nt(e,o={}){let t=I(e,{port:o.port,commandTimeoutMs:o.timeoutMs}),n=x(t);try{let r=await n.listSims();ot(r,t.simId)}finally{n.close()}}async function Ft(e,o={}){let t=I(e,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--new","--headless","--ephemeral"],stripValueFlags:["--base-url","--replace","--driver","--profile","--cdp-port"]}),n=e.find((u,p)=>e[p-1]==="--profile"),r=e.includes("--ephemeral"),s=e.find((u,p)=>e[p-1]==="--cdp-port"),a=s?Number(s):void 0;s&&(!Number.isFinite(a)||a<=0)&&(console.error(` sootsim open: --cdp-port must be a positive port number, got "${s}"`),process.exit(1)),n&&r&&(console.error(" sootsim open: --profile cannot be combined with --ephemeral"),process.exit(1));let i=n?ue(n).id:void 0,l=!!i||r,c=e.includes("--new")||l,d=Je(e);c&&t.simIdSource==="flag"&&(console.error(" sootsim open: --new, --profile, and --ephemeral cannot be combined with --sim"),process.exit(1));let m=t.positional[0]||"";if(We(m)){let u=(()=>{try{return/^(localhost|127\.0\.0\.1|\[::1\]|::1)$/i.test(new URL(m).hostname)}catch{return!1}})();console.log(" that\u2019s a SootSim preview/build link \u2014 a full player page, not a driveable sim."),console.log(" opening it in your browser for viewing\u2026");try{await j(m,{background:!1})}catch(p){console.error(` could not launch a browser: ${p instanceof Error?p.message:String(p)}`),console.error(` open it yourself: ${m}`),process.exit(1)}console.log(u?" to drive it under the CLI, point `sootsim open` at the app\u2019s dev port instead (e.g. `sootsim open 8081`).":" the CLI can\u2019t drive a remote preview (the engine only attaches to the local bridge from localhost). to drive the app, run it locally and `sootsim open <port>`.");return}let h=e.find((u,p)=>e[p-1]==="--driver")||"",H=e.includes("--headless");a&&h!=="playwright"&&(console.error(" sootsim open: --cdp-port is only honored by the playwright driver \u2014 add `--driver playwright`"),process.exit(1));let C=e.find((u,p)=>e[p-1]==="--base-url")||Te(),ee=e.includes("--base-url"),te=k();if(!h&&!c&&(t.simIdSource==="flag"||t.simIdSource==="saved"&&!!te)){let u=x(t),p=t.simId?` --sim ${t.simId}`:"",f=!1;try{let g=null;try{let y=await u.listSims();if(t.simIdSource==="saved"?(g=y.find($=>$.id===te&&$.readyState==="open")??null,g||T()):g=F(y,t.simId),!g)if(t.simIdSource==="saved")f=!0;else throw new Error("no sim connected");if(!f&&g){let $=ee||Y(m)?C:xe(g,C),B=await A(m,D($,d));u.send({type:"evaluate",simId:g.id,code:`window.location.href = ${JSON.stringify(B)}`}).catch(()=>{})}}catch(y){console.error(` open failed: ${y instanceof Error?y.message:String(y)}`),await O(u,{errorsCommand:`sootsim get errors 5${p}`,warningsCommand:`sootsim get warnings 5${p}`,requestsCommand:`sootsim get requests 5${p}`}),process.exit(1)}if(!f&&g){await S(1500);let y=await et(t.wsPort,t.commandTimeoutMs,g.id);y||(console.error(" timed out waiting for current sim to load target"),process.exit(1)),y.bridge.close(),w(g.id);let $=ee||Y(m)?C:xe(g,C),B=await A(m,D($,d));_(B,{...g,url:B},"current sim"),await E(t.wsPort,g.id,e);return}}finally{u.close()}}let oe=D(C,d),_e=await A(m,oe),ne=`cli-${Date.now().toString(36)}-${Math.random().toString(36).slice(2,8)}`,b=await Ze(m,oe,ne),K={newWindow:!0},R=u=>u.url?.includes(`inspectOpen=${ne}`)??!1;if(h){Ke(h)||(console.error(` unknown driver "${h}" \u2014 run \`sootsim list --drivers\``),process.exit(1));let u=G(h);u||(console.error(` unknown driver "${h}" \u2014 run \`sootsim list --drivers\``),process.exit(1));let p=He(),f=await u.launch({url:b,headless:H,newWindow:K.newWindow,profileId:i,ephemeralProfile:r,cdpPort:a,connectTimeoutMs:p.hostTimeoutMs});f.launched||(console.error(` ${u.name} driver: ${f.message}`),process.exit(1));let g=await L(t.wsPort,t.commandTimeoutMs,R,{attempts:p.attempts,intervalMs:p.intervalMs});if(!g){if(f.pid)try{process.kill(f.pid,"SIGTERM"),console.error(` closed ${u.name} host process ${f.pid}`)}catch{}it(f),console.error(` timed out after ${p.timeoutMs}ms waiting for opened sim to connect`),process.exit(1)}w(g.id),st(f),_(b,g,`${u.name} driver`),await E(t.wsPort,g.id,e);return}if(l){let u=M();u||(console.error(" profiles require electron or playwright; install the desktop companion or use `--driver playwright`"),process.exit(1));try{(await V(b,u,{profileId:i,ephemeralProfile:r})).launched||(console.error(" desktop companion failed to start"),process.exit(1))}catch(f){console.error(` desktop companion failed: ${f instanceof Error?f.message:String(f)}`),process.exit(1)}let p=await L(t.wsPort,t.commandTimeoutMs,R,{attempts:40,intervalMs:500});p||(console.error(" timed out waiting for profiled sim to connect"),process.exit(1)),w(p.id),_(b,p,"desktop companion"),await E(t.wsPort,p.id,e);return}let W=!1,z=!1;try{let u=v(t.wsPort,{commandTimeoutMs:2e3});try{await u.listSims(),W=!0}finally{u.close()}}catch{}if(W)try{let u=v(t.wsPort,{commandTimeoutMs:3e3});try{await u.openUrl(b,K),z=!0}finally{u.close()}}catch{}if(!z){let u=M();if(u)try{if((await V(b,u)).launched){let f=await L(t.wsPort,t.commandTimeoutMs,R,{attempts:20,intervalMs:500});if(f){w(f.id),_(b,f,"desktop companion"),await E(t.wsPort,f.id,e);return}}}catch{}try{await j(b,K)}catch(p){console.error(` open failed: ${p instanceof Error?p.message:String(p)}`),G("playwright")?.availability().available?console.error(" no system browser found \u2014 retry with `sootsim open \u2026 --driver playwright` for a headless sim."):console.error(" run `sootsim list --drivers` to see available drivers."),process.exit(1)}if(!W){console.log(` opened: ${_e}`),Se(t.wsPort);return}}let U=await L(t.wsPort,t.commandTimeoutMs,R,{attempts:30,intervalMs:500});U||(console.error(" timed out waiting for opened sim to connect"),process.exit(1)),w(U.id),_(b,U,z?"bridge":"direct shell open"),await E(t.wsPort,U.id,e)}async function Me(e,o={}){let t=I(e,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--force"]}),n=e.includes("--force"),r=x(t),s=t.simId?` --sim ${t.simId}`:"";try{try{let a=await r.listSims(),i=F(a,t.positional[0]||t.simId);if(i.lockedBy&&i.lockExpiresAt&&i.lockExpiresAt>Date.now()){let l=Math.max(0,Math.round((i.lockExpiresAt-Date.now())/1e3));i.lockedByKind==="user-active"&&(console.error(` refused: ${i.id} is locked by the active user (${l}s) \u2014 that's a live human tab`),console.error(" run `sootsim open --new` for a fresh investigation sim"),process.exit(1)),n||(console.error(` refused: ${i.id} is leased by ${i.lockedBy} (${l}s)`),console.error(` \`sootsim use ${i.id} --force\` to take it, or \`sootsim open --new\` for a fresh sim`),process.exit(1))}await r.focusSim(i.id),w(i.id),console.log(` using: ${i.id}`)}catch(a){console.error(` use failed: ${a instanceof Error?a.message:String(a)}`),await O(r,{errorsCommand:`sootsim get errors 5${s}`,warningsCommand:`sootsim get warnings 5${s}`,requestsCommand:`sootsim get requests 5${s}`}),process.exit(1)}}finally{r.close()}}async function Ht(e,o={}){await Me(e,o)}async function Kt(e,o={}){await Me(e,o)}async function Wt(e,o={}){let t=I(e,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--force"]}),n=e.includes("--force"),r=x(t);try{try{let s=await r.listSims(),a=F(s,t.positional[0]||t.simId),i=n&&a.lockedBy&&a.lockedByKind!=="user-active"?a.lockedBy:null,l=await r.claim(a.id,{force:n});w(a.id);let c=Math.max(0,Math.round((l.lockExpiresAt-Date.now())/1e3)),d=l.bootedCount>0?` (booted ${l.bootedCount})`:"";console.log(` claimed: ${l.simId} [${c}s]${d}`),i&&console.log(` took over from: ${i}`)}catch(s){if(s instanceof ae){let a=Math.max(0,Math.round(s.lock.expiresInMs/1e3));console.error(` claim failed: locked by ${s.lock.by} for ${a}s more`),console.error(" use --force to take it, or `sootsim open --new` for a fresh sim"),process.exit(1)}console.error(` claim failed: ${s instanceof Error?s.message:String(s)}`),process.exit(1)}}finally{r.close()}}async function ct(e,o,t,n){if(n.length===0)return{closed:[],remaining:[]};await Promise.all(n.map(s=>e.closeSim(s).catch(()=>{})));let r=new Set(n);for(let s=0;s<40;s++){let a=[];try{let i=v(o,{commandTimeoutMs:t});try{a=(await i.listSims()).filter(c=>r.has(c.id)&&c.readyState==="open").map(c=>c.id)}finally{i.close()}}catch{a=[]}if(a.length===0)return{closed:[...r],remaining:[]};if(s===39)return{closed:[...r].filter(i=>!a.includes(i)),remaining:a};await S(250)}return{closed:[...r],remaining:[]}}function dt(e,o){let t=e.filter(m=>m.readyState==="open"),n=t.map(m=>m.id),r=o.explicitKeepId?.trim()||"",s=o.savedKeepId?.trim()||"",a=s?t.some(m=>m.id===s):!1,i=s&&!a?s:null,l=null,c=null,d=null;if(o.closeOthers)if(r){let m=t.find(h=>h.id===r);m?l=m.id:c=r}else if(s&&a)l=s;else{let m=t.find(h=>h.isPrimary)??t[0]??null;m&&(l=m.id,s&&!a&&(d=m.id))}return{openIds:n,keepId:l,targets:c?[]:n.filter(m=>m!==l),staleSavedId:i,missingExplicitKeepId:c,fallbackKeepId:d}}async function zt(e,o={}){let t=I(e,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--all","--others"]}),n=x(t),r=t.simId?` --sim ${t.simId}`:"",s=e.includes("--all"),a=e.includes("--others");if(s||a){try{let i=await n.listSims(),l=k(),c=t.positional[0]||(t.simIdSource==="flag"?t.simId:void 0),d=dt(i,{closeOthers:a,explicitKeepId:c,savedKeepId:l});if(d.staleSavedId&&T(),d.fallbackKeepId&&(w(d.fallbackKeepId),console.log(` saved sim ${d.staleSavedId} is gone \u2014 keeping primary ${d.fallbackKeepId}`)),d.missingExplicitKeepId&&(console.error(` close failed: keep sim ${d.missingExplicitKeepId} is not connected; not closing other sims`),process.exit(1)),d.targets.length===0){console.log(d.keepId?` nothing to close \u2014 only the kept sim ${d.keepId} is connected`:d.staleSavedId?` nothing to close \u2014 no sims connected (cleared stale current sim ${d.staleSavedId})`:" nothing to close \u2014 no sims connected");return}let m=await ct(n,t.wsPort,t.commandTimeoutMs,d.targets);await Q(i,d.targets);let h=k();h&&m.closed.includes(h)&&(d.keepId?w(d.keepId):T());let H=` closed ${m.closed.length} sim(s)${d.keepId?` (kept ${d.keepId})`:""}`;console.log(H),m.remaining.length>0&&(console.error(` close failed: still connected: ${m.remaining.join(", ")}`),process.exit(1))}catch(i){console.error(` close failed: ${i instanceof Error?i.message:String(i)}`),process.exit(1)}finally{n.close()}return}try{try{let i=await n.listSims(),l=F(i,t.positional[0]||t.simId),c=i.find(m=>m.id!==l.id&&m.readyState==="open");await n.closeSim(l.id),await nt(t.wsPort,t.commandTimeoutMs,l.id)||(await Q(i,[l.id]),console.error(` close failed: ${l.id} is still connected`),process.exit(1)),await Q(i,[l.id]),k()===l.id&&(c?w(c.id):T()),console.log(` closed: ${l.id}`)}catch(i){console.error(` close failed: ${i instanceof Error?i.message:String(i)}`),await O(n,{errorsCommand:`sootsim get errors 5${r}`,warningsCommand:`sootsim get warnings 5${r}`,requestsCommand:`sootsim get requests 5${r}`}),process.exit(1)}}finally{n.close()}}export{wt as a,De as b,bt as c,He as d,At as e,Te as f,A as g,Ze as h,tt as i,ot as j,L as k,rt as l,st as m,Q as n,Nt as o,Ft as p,Ht as q,Kt as r,Wt as s,ct as t,dt as u,zt as v};
|
|
13
|
+
})()`,l=await r.send({type:"evaluate",code:i});if(!l?.tree)return;if(console.log(""),l.shell?.state){let c=[`state=${l.shell.state}`,l.shell.activeApp?`app=${l.shell.activeApp}`:null].filter(Boolean);console.log(` shell: ${c.join(" ")}`)}console.log(l.tree)}finally{r.close()}}catch{}finally{n===void 0?delete process.env.SOOTSIM_QUIET_TARGET_NOTICE:process.env.SOOTSIM_QUIET_TARGET_NOTICE=n}}async function dt(e,o,t){let n=Date.now()+t.maxMs,r="(async () => (await window.__sootsimTest?.getNodeCount?.()) || 0)()",s=v(e,{commandTimeoutMs:2e3,simId:o,simIdSource:"flag"});try{let a=-1,i=0;for(;Date.now()<n;){let l=-1;try{let c=await s.send({type:"evaluate",code:r});typeof c=="number"&&(l=c)}catch{}if(l>=0&&l===a){if(Date.now()-i>=t.stableMs)return}else a=l,i=Date.now();await S(50)}}finally{s.close()}}async function Ht(e,o={}){let t=I(e,{port:o.port,commandTimeoutMs:o.timeoutMs}),n=x(t);try{let r=await n.listSims();nt(r,t.simId)}finally{n.close()}}async function Kt(e,o={}){let t=I(e,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--new","--headless","--ephemeral"],stripValueFlags:["--base-url","--replace","--driver","--profile","--cdp-port"]}),n=e.find((u,p)=>e[p-1]==="--profile"),r=e.includes("--ephemeral"),s=e.find((u,p)=>e[p-1]==="--cdp-port"),a=s?Number(s):void 0;s&&(!Number.isFinite(a)||a<=0)&&(console.error(` sootsim open: --cdp-port must be a positive port number, got "${s}"`),process.exit(1)),n&&r&&(console.error(" sootsim open: --profile cannot be combined with --ephemeral"),process.exit(1));let i=n?ue(n).id:void 0,l=!!i||r,c=e.includes("--new")||l,d=Je(e);c&&t.simIdSource==="flag"&&(console.error(" sootsim open: --new, --profile, and --ephemeral cannot be combined with --sim"),process.exit(1));let m=t.positional[0]||"";if(We(m)){let u=(()=>{try{return/^(localhost|127\.0\.0\.1|\[::1\]|::1)$/i.test(new URL(m).hostname)}catch{return!1}})();console.log(" that\u2019s a SootSim preview/build link \u2014 a full player page, not a driveable sim."),console.log(" opening it in your browser for viewing\u2026");try{await j(m,{background:!1})}catch(p){console.error(` could not launch a browser: ${p instanceof Error?p.message:String(p)}`),console.error(` open it yourself: ${m}`),process.exit(1)}console.log(u?" to drive it under the CLI, point `sootsim open` at the app\u2019s dev port instead (e.g. `sootsim open 8081`).":" the CLI can\u2019t drive a remote preview (the engine only attaches to the local bridge from localhost). to drive the app, run it locally and `sootsim open <port>`.");return}let h=e.find((u,p)=>e[p-1]==="--driver")||"",H=e.includes("--headless");a&&h!=="playwright"&&(console.error(" sootsim open: --cdp-port is only honored by the playwright driver \u2014 add `--driver playwright`"),process.exit(1));let C=e.find((u,p)=>e[p-1]==="--base-url")||Te(),ee=e.includes("--base-url"),te=k();if(!h&&!c&&(t.simIdSource==="flag"||t.simIdSource==="saved"&&!!te)){let u=x(t),p=t.simId?` --sim ${t.simId}`:"",f=!1;try{let g=null;try{let y=await u.listSims();if(t.simIdSource==="saved"?(g=y.find($=>$.id===te&&$.readyState==="open")??null,g||T()):g=F(y,t.simId),!g)if(t.simIdSource==="saved")f=!0;else throw new Error("no sim connected");if(!f&&g){let $=ee||X(m)?C:xe(g,C),O=await A(m,D($,d));u.send({type:"evaluate",simId:g.id,code:`window.location.href = ${JSON.stringify(O)}`}).catch(()=>{})}}catch(y){console.error(` open failed: ${y instanceof Error?y.message:String(y)}`),await B(u,{errorsCommand:`sootsim get errors 5${p}`,warningsCommand:`sootsim get warnings 5${p}`,requestsCommand:`sootsim get requests 5${p}`}),process.exit(1)}if(!f&&g){await S(1500);let y=await et(t.wsPort,t.commandTimeoutMs,g.id);y||(console.error(" timed out waiting for current sim to load target"),process.exit(1)),y.bridge.close(),w(g.id);let $=ee||X(m)?C:xe(g,C),O=await A(m,D($,d));E(O,{...g,url:O},"current sim"),await _(t.wsPort,g.id,e);return}}finally{u.close()}}let oe=D(C,d),Ee=await A(m,oe),ne=`cli-${Date.now().toString(36)}-${Math.random().toString(36).slice(2,8)}`,b=await Ze(m,oe,ne),K={newWindow:!0},R=u=>u.url?.includes(`inspectOpen=${ne}`)??!1;if(h){Ke(h)||(console.error(` unknown driver "${h}" \u2014 run \`sootsim list --drivers\``),process.exit(1));let u=G(h);u||(console.error(` unknown driver "${h}" \u2014 run \`sootsim list --drivers\``),process.exit(1));let p=He(),f=await u.launch({url:b,headless:H,newWindow:K.newWindow,profileId:i,ephemeralProfile:r,cdpPort:a,connectTimeoutMs:p.hostTimeoutMs});f.launched||(console.error(` ${u.name} driver: ${f.message}`),process.exit(1));let g=await L(t.wsPort,t.commandTimeoutMs,R,{attempts:p.attempts,intervalMs:p.intervalMs});if(!g){if(f.pid)try{process.kill(f.pid,"SIGTERM"),console.error(` closed ${u.name} host process ${f.pid}`)}catch{}lt(f),console.error(` timed out after ${p.timeoutMs}ms waiting for opened sim to connect`),process.exit(1)}w(g.id),at(f),E(b,g,`${u.name} driver`),await _(t.wsPort,g.id,e);return}if(l){let u=M();u||(console.error(" profiles require electron or playwright; install the desktop companion or use `--driver playwright`"),process.exit(1));try{(await V(b,u,{profileId:i,ephemeralProfile:r})).launched||(console.error(" desktop companion failed to start"),process.exit(1))}catch(f){console.error(` desktop companion failed: ${f instanceof Error?f.message:String(f)}`),process.exit(1)}let p=await L(t.wsPort,t.commandTimeoutMs,R,{attempts:40,intervalMs:500});p||(console.error(" timed out waiting for profiled sim to connect"),process.exit(1)),w(p.id),E(b,p,"desktop companion"),await _(t.wsPort,p.id,e);return}let W=!1,z=!1;try{let u=v(t.wsPort,{commandTimeoutMs:2e3});try{await u.listSims(),W=!0}finally{u.close()}}catch{}if(W)try{let u=v(t.wsPort,{commandTimeoutMs:3e3});try{await u.openUrl(b,K),z=!0}finally{u.close()}}catch{}if(!z){let u=M();if(u)try{if((await V(b,u)).launched){let f=await L(t.wsPort,t.commandTimeoutMs,R,{attempts:20,intervalMs:500});if(f){w(f.id),E(b,f,"desktop companion"),await _(t.wsPort,f.id,e);return}}}catch{}try{await j(b,K)}catch(p){console.error(` open failed: ${p instanceof Error?p.message:String(p)}`),G("playwright")?.availability().available?console.error(" no system browser found \u2014 retry with `sootsim open \u2026 --driver playwright` for a headless sim."):console.error(" run `sootsim list --drivers` to see available drivers."),process.exit(1)}if(!W){console.log(` opened: ${Ee}`),Se(t.wsPort);return}}let U=await L(t.wsPort,t.commandTimeoutMs,R,{attempts:30,intervalMs:500});U||(console.error(" timed out waiting for opened sim to connect"),process.exit(1)),w(U.id),E(b,U,z?"bridge":"direct shell open"),await _(t.wsPort,U.id,e)}async function Me(e,o={}){let t=I(e,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--force"]}),n=e.includes("--force"),r=x(t),s=t.simId?` --sim ${t.simId}`:"";try{try{let a=await r.listSims(),i=F(a,t.positional[0]||t.simId);if(i.lockedBy&&i.lockExpiresAt&&i.lockExpiresAt>Date.now()){let l=Math.max(0,Math.round((i.lockExpiresAt-Date.now())/1e3));i.lockedByKind==="user-active"&&(console.error(` refused: ${i.id} is locked by the active user (${l}s) \u2014 that's a live human tab`),console.error(" run `sootsim open --new` for a fresh investigation sim"),process.exit(1)),n||(console.error(` refused: ${i.id} is leased by ${i.lockedBy} (${l}s)`),console.error(` \`sootsim use ${i.id} --force\` to take it, or \`sootsim open --new\` for a fresh sim`),process.exit(1))}await r.focusSim(i.id),w(i.id),console.log(` using: ${i.id}`)}catch(a){console.error(` use failed: ${a instanceof Error?a.message:String(a)}`),await B(r,{errorsCommand:`sootsim get errors 5${s}`,warningsCommand:`sootsim get warnings 5${s}`,requestsCommand:`sootsim get requests 5${s}`}),process.exit(1)}}finally{r.close()}}async function Wt(e,o={}){await Me(e,o)}async function zt(e,o={}){await Me(e,o)}async function jt(e,o={}){let t=I(e,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--force"]}),n=e.includes("--force"),r=x(t);try{try{let s=await r.listSims(),a=F(s,t.positional[0]||t.simId),i=n&&a.lockedBy&&a.lockedByKind!=="user-active"?a.lockedBy:null,l=await r.claim(a.id,{force:n});w(a.id);let c=Math.max(0,Math.round((l.lockExpiresAt-Date.now())/1e3)),d=l.bootedCount>0?` (booted ${l.bootedCount})`:"";console.log(` claimed: ${l.simId} [${c}s]${d}`),i&&console.log(` took over from: ${i}`)}catch(s){if(s instanceof ae){let a=Math.max(0,Math.round(s.lock.expiresInMs/1e3));console.error(` claim failed: locked by ${s.lock.by} for ${a}s more`),console.error(" use --force to take it, or `sootsim open --new` for a fresh sim"),process.exit(1)}console.error(` claim failed: ${s instanceof Error?s.message:String(s)}`),process.exit(1)}}finally{r.close()}}async function mt(e,o,t,n){if(n.length===0)return{closed:[],remaining:[]};await Promise.all(n.map(s=>e.closeSim(s).catch(()=>{})));let r=new Set(n);for(let s=0;s<40;s++){let a=[];try{let i=v(o,{commandTimeoutMs:t});try{a=(await i.listSims()).filter(c=>r.has(c.id)&&c.readyState==="open").map(c=>c.id)}finally{i.close()}}catch{a=[]}if(a.length===0)return{closed:[...r],remaining:[]};if(s===39)return{closed:[...r].filter(i=>!a.includes(i)),remaining:a};await S(250)}return{closed:[...r],remaining:[]}}function ut(e,o){let t=e.filter(m=>m.readyState==="open"),n=t.map(m=>m.id),r=o.explicitKeepId?.trim()||"",s=o.savedKeepId?.trim()||"",a=s?t.some(m=>m.id===s):!1,i=s&&!a?s:null,l=null,c=null,d=null;if(o.closeOthers)if(r){let m=t.find(h=>h.id===r);m?l=m.id:c=r}else if(s&&a)l=s;else{let m=t.find(h=>h.isPrimary)??t[0]??null;m&&(l=m.id,s&&!a&&(d=m.id))}return{openIds:n,keepId:l,targets:c?[]:n.filter(m=>m!==l),staleSavedId:i,missingExplicitKeepId:c,fallbackKeepId:d}}async function qt(e,o={}){let t=I(e,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--all","--others"]}),n=x(t),r=t.simId?` --sim ${t.simId}`:"",s=e.includes("--all"),a=e.includes("--others");if(s||a){try{let i=await n.listSims(),l=k(),c=t.positional[0]||(t.simIdSource==="flag"?t.simId:void 0),d=ut(i,{closeOthers:a,explicitKeepId:c,savedKeepId:l});if(d.staleSavedId&&T(),d.fallbackKeepId&&(w(d.fallbackKeepId),console.log(` saved sim ${d.staleSavedId} is gone \u2014 keeping primary ${d.fallbackKeepId}`)),d.missingExplicitKeepId&&(console.error(` close failed: keep sim ${d.missingExplicitKeepId} is not connected; not closing other sims`),process.exit(1)),d.targets.length===0){console.log(d.keepId?` nothing to close \u2014 only the kept sim ${d.keepId} is connected`:d.staleSavedId?` nothing to close \u2014 no sims connected (cleared stale current sim ${d.staleSavedId})`:" nothing to close \u2014 no sims connected");return}let m=await mt(n,t.wsPort,t.commandTimeoutMs,d.targets);await Q(i,d.targets);let h=k();h&&m.closed.includes(h)&&(d.keepId?w(d.keepId):T());let H=` closed ${m.closed.length} sim(s)${d.keepId?` (kept ${d.keepId})`:""}`;console.log(H),m.remaining.length>0&&(console.error(` close failed: still connected: ${m.remaining.join(", ")}`),process.exit(1))}catch(i){console.error(` close failed: ${i instanceof Error?i.message:String(i)}`),process.exit(1)}finally{n.close()}return}try{try{let i=await n.listSims(),l=F(i,t.positional[0]||t.simId),c=i.find(m=>m.id!==l.id&&m.readyState==="open");await n.closeSim(l.id),await st(t.wsPort,t.commandTimeoutMs,l.id)||(await Q(i,[l.id]),console.error(` close failed: ${l.id} is still connected`),process.exit(1)),await Q(i,[l.id]),k()===l.id&&(c?w(c.id):T()),console.log(` closed: ${l.id}`)}catch(i){console.error(` close failed: ${i instanceof Error?i.message:String(i)}`),await B(n,{errorsCommand:`sootsim get errors 5${r}`,warningsCommand:`sootsim get warnings 5${r}`,requestsCommand:`sootsim get requests 5${r}`}),process.exit(1)}}finally{n.close()}}export{yt as a,De as b,vt as c,He as d,Ft as e,Te as f,A as g,Ze as h,tt as i,nt as j,L as k,it as l,at as m,Q as n,Ht as o,Kt as p,Wt as q,zt as r,jt as s,mt as t,ut as u,qt as v};
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
/*! sootsim v0.1.
|
|
1
|
+
/*! sootsim v0.1.115 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
|
|
2
2
|
import y from"node:fs";import{createRequire as C}from"node:module";import l from"node:path";var A=C(import.meta.url);function b(n){try{let t=new URL(n),e=t.hostname.replace(/^\[|\]$/g,"").toLowerCase();if(e==="localhost"||e.endsWith(".localhost")||e==="0.0.0.0"||e==="::1"||/^127(?:\.\d{1,3}){3}$/.test(e))return t.origin}catch{}return null}var j=["app.config.js","app.config.ts","app.config.cjs","app.config.mjs","app.json"];function w(n){for(let t of j)try{if(y.statSync(l.join(n,t)).isFile())return!0}catch{}return!1}var k=new Set(["node_modules","artifacts","dist","build","out","coverage","tmp","public","ios","android","macos","windows","__fixtures__","__tests__","examples","example"]);function x(n,t){w(n)&&t.push(n);let e;try{e=y.readdirSync(n,{withFileTypes:!0})}catch{return}for(let o of e)o.isDirectory()&&(o.name.startsWith(".")||k.has(o.name)||x(l.join(n,o.name),t))}function D(n){try{return y.statSync(l.join(n,"package.json")).isFile()}catch{return!1}}function _(n,t,e){if(!t||typeof t!="object")return;let o=t[e];if(!o||typeof o!="object")return;let i=o.fonts;if(Array.isArray(i))for(let c of i){if(typeof c=="string"){n.push(c);continue}if(!c||typeof c!="object")continue;let p=c.fontDefinitions;if(Array.isArray(p))for(let f of p){if(!f||typeof f!="object")continue;let u=f.path;typeof u=="string"&&n.push(u)}}}function v(n){let t=b(n.bundleUrl);if(!t)return[];let e=n.repoDir||process.env.SOOT_REPO_DIR,o=e?l.resolve(e):process.cwd();if(!e&&!D(o))return[];let i=[];if(e)x(o,i),i.includes(o)||i.push(o);else{if(!w(o))return[];i.push(o)}if(i.length===0)return[];let c=(n.platform||process.env.SOOT_PLATFORM||"ios").toLowerCase(),p=null;try{let s=A.resolve("@expo/config",{paths:[...i,o]}),r=A(s);typeof r.getConfig=="function"&&(p=r.getConfig)}catch{}if(!p)return[];let f;for(let s of i)try{let r=p(s,{skipSDKVersionRequirement:!0});if(r?.exp&&typeof r.exp=="object"){let a=r.exp.plugins;if(Array.isArray(a)){f=r.exp;break}}}catch{}if(!f||typeof f!="object")return[];let u=f.plugins;if(!Array.isArray(u))return[];let g=[];for(let s of u){let r=Array.isArray(s)?s[0]:s;if(typeof r!="string"||!/(^|\/)expo-font$/.test(r))continue;let a=Array.isArray(s)?s[1]||{}:{};if(!a||typeof a!="object")continue;let h=a.fonts;Array.isArray(h)&&g.push(...h.filter(R=>typeof R=="string")),_(g,a,c)}let d=new Set,m=[];for(let s of g){let r=s.replace(/^\.\//,"").replace(/^\/+/,"");!r||d.has(r)||(d.add(r),/\.(ttf|otf|woff|woff2)$/i.test(r)&&m.push({family:l.basename(r).replace(/\.(ttf|otf|woff|woff2)$/i,""),url:`${t}/assets/${r}`}))}return m}function O(n){return n.map(t=>`${encodeURIComponent(t.url)}::${encodeURIComponent(t.family)}`).join(",")}function $(n){let t=[];for(let e of n.split(",")){let o=e.trim();if(!o)continue;let i=o.indexOf("::"),c=i===-1?o:o.slice(0,i);try{t.push(decodeURIComponent(c))}catch{t.push(c)}}return t.filter(Boolean)}export{v as a,O as b,$ as c};
|