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.
Files changed (145) hide show
  1. package/dist-cli/bin.js +3 -3
  2. package/dist-cli/chunks/{agent-PPV725W2.js → agent-R3CYHNIF.js} +2 -2
  3. package/dist-cli/chunks/{agent-wrapper-RWVU7CO2.js → agent-wrapper-TRRE2GH5.js} +2 -2
  4. package/dist-cli/chunks/{assert-JC2FD3TU.js → assert-MRSNTOEC.js} +2 -2
  5. package/dist-cli/chunks/auto-bootstrap-7ABTFLLR.js +2 -0
  6. package/dist-cli/chunks/beta-ZJYYZEFR.js +2 -0
  7. package/dist-cli/chunks/{chunk-7EAARWHH.js → chunk-3PUOG2KU.js} +2 -2
  8. package/dist-cli/chunks/{chunk-D6U6XEI3.js → chunk-47VNJGW7.js} +3 -3
  9. package/dist-cli/chunks/{chunk-NU5KT7WW.js → chunk-5SL6CP7Y.js} +1 -1
  10. package/dist-cli/chunks/{chunk-D2PQBP3C.js → chunk-77VIOTR7.js} +2 -2
  11. package/dist-cli/chunks/{chunk-IOSPY7ZF.js → chunk-AJYQJMAY.js} +2 -2
  12. package/dist-cli/chunks/chunk-BZTJFTU2.js +1 -0
  13. package/dist-cli/chunks/{chunk-56HURVFV.js → chunk-CUHTXNJW.js} +1 -1
  14. package/dist-cli/chunks/{chunk-UKLINY7V.js → chunk-D6PRIVLD.js} +2 -2
  15. package/dist-cli/chunks/{chunk-NQH7JGEZ.js → chunk-DACZHUK2.js} +1 -1
  16. package/dist-cli/chunks/{chunk-7QWLHGAD.js → chunk-E53Y4O7H.js} +1 -1
  17. package/dist-cli/chunks/{chunk-UA5OIC5Y.js → chunk-F46PTRLM.js} +2 -2
  18. package/dist-cli/chunks/chunk-F4LJC2WB.js +1 -0
  19. package/dist-cli/chunks/{chunk-EIRP7XBM.js → chunk-GG5MTRXC.js} +1 -1
  20. package/dist-cli/chunks/{chunk-XPQUENRQ.js → chunk-GUDD5CIA.js} +2 -2
  21. package/dist-cli/chunks/{chunk-HPTCRK5L.js → chunk-GW5TVVRV.js} +2 -2
  22. package/dist-cli/chunks/{chunk-2DW25WOA.js → chunk-HAFZOF63.js} +3 -3
  23. package/dist-cli/chunks/{chunk-PQWCZXCN.js → chunk-HU2K2HVE.js} +2 -2
  24. package/dist-cli/chunks/{chunk-HIQPS53N.js → chunk-HYR63F7Z.js} +1 -1
  25. package/dist-cli/chunks/{chunk-ZVL24GTC.js → chunk-ICZSY2XS.js} +2 -2
  26. package/dist-cli/chunks/{chunk-UBDSTHBH.js → chunk-IWC4SYEG.js} +1 -1
  27. package/dist-cli/chunks/{chunk-TVETGSCD.js → chunk-ML5LV4WM.js} +1 -1
  28. package/dist-cli/chunks/{chunk-6TO3HND7.js → chunk-NXAESORG.js} +2 -2
  29. package/dist-cli/chunks/{chunk-O5WZBC2Y.js → chunk-OK2FCYQN.js} +2 -2
  30. package/dist-cli/chunks/{chunk-VD4CSXIK.js → chunk-PEP5GKW7.js} +2 -2
  31. package/dist-cli/chunks/{chunk-IDUGL3MK.js → chunk-Q3BVLSSD.js} +6 -6
  32. package/dist-cli/chunks/chunk-Q4SUC2SN.js +79 -0
  33. package/dist-cli/chunks/{chunk-WRB6MZBX.js → chunk-SLSNMSD7.js} +21 -4
  34. package/dist-cli/chunks/{chunk-VRS7EUIU.js → chunk-SPVHYFZ3.js} +1 -1
  35. package/dist-cli/chunks/{chunk-FPCP57JZ.js → chunk-SRXOV4BJ.js} +6 -6
  36. package/dist-cli/chunks/{chunk-HKRPDZOI.js → chunk-TF5LTELW.js} +1 -1
  37. package/dist-cli/chunks/{chunk-PFBRPOL6.js → chunk-TMR6MW5W.js} +1 -1
  38. package/dist-cli/chunks/{chunk-V4PAF3TJ.js → chunk-UDQHDFZW.js} +1 -1
  39. package/dist-cli/chunks/{chunk-M6LHGDWE.js → chunk-UNRBTUAA.js} +1 -1
  40. package/dist-cli/chunks/chunk-V2SPOLQA.js +1 -0
  41. package/dist-cli/chunks/chunk-VSKAOUZ5.js +2 -0
  42. package/dist-cli/chunks/{chunk-TSB4XTV3.js → chunk-W646EQFP.js} +2 -2
  43. package/dist-cli/chunks/{chunk-DUSFDC5U.js → chunk-WNIVVJR7.js} +2 -2
  44. package/dist-cli/chunks/{chunk-GR2XFJJF.js → chunk-X35ALXTD.js} +2 -2
  45. package/dist-cli/chunks/{chunk-7NXJ24HB.js → chunk-XMVEHJSM.js} +2 -2
  46. package/dist-cli/chunks/{chunk-MGWQ527R.js → chunk-XO3XTPFR.js} +1 -1
  47. package/dist-cli/chunks/{chunk-BBNJXYUW.js → chunk-XRBI3TG2.js} +2 -2
  48. package/dist-cli/chunks/{chunk-G5IBWOSU.js → chunk-YBAX43B5.js} +1 -1
  49. package/dist-cli/chunks/{chunk-FVPJJQCX.js → chunk-YTJONQGT.js} +3 -3
  50. package/dist-cli/chunks/{chunk-WPYPKVQP.js → chunk-ZPJDADTZ.js} +2 -2
  51. package/dist-cli/chunks/cli-version-RIP4ZUNM.js +2 -0
  52. package/dist-cli/chunks/{compat-BO4GOTHM.js → compat-4ZDISC6K.js} +3 -3
  53. package/dist-cli/chunks/{config-4W2CJ435.js → config-IWM4VJ6H.js} +2 -2
  54. package/dist-cli/chunks/control-FSKX3J4O.js +2 -0
  55. package/dist-cli/chunks/{cpu-profile-5BKUYSEY.js → cpu-profile-RPY4UFNI.js} +2 -2
  56. package/dist-cli/chunks/{daemon-KFYVGHQV.js → daemon-SL3D6Q4P.js} +2 -2
  57. package/dist-cli/chunks/{debug-QBCVIBVX.js → debug-WYNMZ7CT.js} +3 -3
  58. package/dist-cli/chunks/{detox-BKVDVNN3.js → detox-RMLUYFYI.js} +2 -2
  59. package/dist-cli/chunks/{device-YG2NZYU4.js → device-TYY7VA57.js} +2 -2
  60. package/dist-cli/chunks/{diagnose-QBHSWPOE.js → diagnose-7CXKCUT4.js} +2 -2
  61. package/dist-cli/chunks/drivers-OIAGQOCX.js +2 -0
  62. package/dist-cli/chunks/{electron-APIQFWWP.js → electron-IKCSDHAV.js} +3 -3
  63. package/dist-cli/chunks/flow-ZIV2ISC2.js +2 -0
  64. package/dist-cli/chunks/help-RUILKRB4.js +2 -0
  65. package/dist-cli/chunks/{hints-SVV3NUDM.js → hints-YHYKK5HQ.js} +2 -2
  66. package/dist-cli/chunks/{home-paths-IPIEZXKS.js → home-paths-OLHYRKN5.js} +2 -2
  67. package/dist-cli/chunks/{inspect-QA27PCCP.js → inspect-INJGRQD3.js} +3 -3
  68. package/dist-cli/chunks/install-BRAHILXG.js +2 -0
  69. package/dist-cli/chunks/{install-desktop-U4KFKLJI.js → install-desktop-FQS3WZ6W.js} +3 -3
  70. package/dist-cli/chunks/{keys-SWELRMTL.js → keys-JUYFKZD7.js} +2 -2
  71. package/dist-cli/chunks/{launch-KWK7AYPJ.js → launch-XW5OSSNC.js} +3 -3
  72. package/dist-cli/chunks/{login-AFK5TRXO.js → login-QSZQPSEA.js} +4 -4
  73. package/dist-cli/chunks/{logout-KJB2XZ4H.js → logout-2YSS6D65.js} +2 -2
  74. package/dist-cli/chunks/{maestro-IL7IRBBN.js → maestro-UG6C6KQ6.js} +2 -2
  75. package/dist-cli/chunks/{preview-DDOOW574.js → preview-4TH2YLNR.js} +2 -2
  76. package/dist-cli/chunks/{profile-UWDS7T6L.js → profile-63R3YOCR.js} +2 -2
  77. package/dist-cli/chunks/{react-EJOUNOBS.js → react-FMPGSG3A.js} +2 -2
  78. package/dist-cli/chunks/{record-7Q25DB46.js → record-F64J66JH.js} +2 -2
  79. package/dist-cli/chunks/runtime-NE4NMW4R.js +2 -0
  80. package/dist-cli/chunks/{runtime-delivery-Q4CEUD72.js → runtime-delivery-HJPB7EEU.js} +2 -2
  81. package/dist-cli/chunks/{screenshot-3ON5A5EY.js → screenshot-OF6QEH6B.js} +2 -2
  82. package/dist-cli/chunks/{screenshot-mode-MGACLSRL.js → screenshot-mode-BIZP42HF.js} +2 -2
  83. package/dist-cli/chunks/{screenshots-OWSKOBVC.js → screenshots-DKAYDLK5.js} +2 -2
  84. package/dist-cli/chunks/{server-T2NGFWJQ.js → server-ETFFUJ5R.js} +2 -2
  85. package/dist-cli/chunks/setup-repo-JE25CERB.js +2 -0
  86. package/dist-cli/chunks/{skills-SF5HOOMB.js → skills-H2OPZZLC.js} +2 -2
  87. package/dist-cli/chunks/{start-PEFKHFIU.js → start-OI245IG4.js} +4 -4
  88. package/dist-cli/chunks/store-2H5T4NFZ.js +2 -0
  89. package/dist-cli/chunks/telemetry-USTW2NVP.js +2 -0
  90. package/dist-cli/chunks/{test-OBVSN2TZ.js → test-HE6RK2ZX.js} +3 -3
  91. package/dist-cli/chunks/{three-mode-EZ332TID.js → three-mode-K2NTAVQP.js} +2 -2
  92. package/dist-cli/chunks/{timeline-E32QABRP.js → timeline-3DBCJWKF.js} +2 -2
  93. package/dist-cli/chunks/{upgrade-A42OZ5LT.js → upgrade-7YLEP6ED.js} +2 -2
  94. package/dist-cli/chunks/upload-T3AYMH27.js +2 -0
  95. package/dist-cli/chunks/{web-Q22CLCSN.js → web-BWUYH3SK.js} +2 -2
  96. package/dist-cli/chunks/{what-happened-6EHD4QJK.js → what-happened-VDTCMZNP.js} +2 -2
  97. package/dist-cli/chunks/{whoami-ZVBH6ZYC.js → whoami-2ZOUZSB3.js} +2 -2
  98. package/dist-lib/agent-daemon-client.cjs +1 -1
  99. package/dist-lib/agent-events.cjs +1 -1
  100. package/dist-lib/agent-sessions.cjs +1 -1
  101. package/dist-lib/attached-projects.cjs +1 -1
  102. package/dist-lib/auth/shared-session.cjs +1 -1
  103. package/dist-lib/backend-origin.cjs +1 -1
  104. package/dist-lib/beta.cjs +1 -1
  105. package/dist-lib/beta.mjs +1 -1
  106. package/dist-lib/bridge-constants.cjs +1 -1
  107. package/dist-lib/cli-constants.cjs +1 -1
  108. package/dist-lib/config.cjs +1 -1
  109. package/dist-lib/detox/index.cjs +1 -1
  110. package/dist-lib/dev-bundle-resolution.cjs +1 -1
  111. package/dist-lib/home-paths.cjs +1 -1
  112. package/dist-lib/host/bridge-host.cjs +1 -1
  113. package/dist-lib/host/fetch-proxy-handler.cjs +1 -1
  114. package/dist-lib/host/fetch-proxy-overrides.cjs +1 -1
  115. package/dist-lib/host/fetch-proxy-overrides.mjs +1 -1
  116. package/dist-lib/host/websocket-proxy.cjs +1 -1
  117. package/dist-lib/index.cjs +1 -1
  118. package/dist-lib/metro.cjs +1 -1
  119. package/dist-lib/profiles.cjs +1 -1
  120. package/dist-lib/render-mode.cjs +1 -1
  121. package/dist-lib/scripts/demo-app-registry.cjs +1 -1
  122. package/dist-lib/scripts/dev-server-scanner.cjs +1 -1
  123. package/dist-lib/sdk.cjs +22 -5
  124. package/dist-lib/sdk.mjs +22 -5
  125. package/dist-lib/skills.cjs +2 -6
  126. package/dist-lib/vite.cjs +1 -1
  127. package/package.json +1 -1
  128. package/dist-cli/chunks/auto-bootstrap-GJVEKHWE.js +0 -2
  129. package/dist-cli/chunks/beta-HTH4QNXO.js +0 -2
  130. package/dist-cli/chunks/chunk-JV6I42YD.js +0 -79
  131. package/dist-cli/chunks/chunk-MOSWK2MT.js +0 -2
  132. package/dist-cli/chunks/chunk-SJ6DSMZI.js +0 -1
  133. package/dist-cli/chunks/chunk-SMVEVXJN.js +0 -1
  134. package/dist-cli/chunks/chunk-ZL5NFGQ6.js +0 -1
  135. package/dist-cli/chunks/cli-version-O7RDJ2FI.js +0 -2
  136. package/dist-cli/chunks/control-4BKV5LMV.js +0 -2
  137. package/dist-cli/chunks/drivers-JUF5RIRB.js +0 -2
  138. package/dist-cli/chunks/flow-WSLHCPWR.js +0 -2
  139. package/dist-cli/chunks/help-LGGVMOHI.js +0 -2
  140. package/dist-cli/chunks/install-3W3VW6XD.js +0 -2
  141. package/dist-cli/chunks/runtime-M4BIM2UC.js +0 -2
  142. package/dist-cli/chunks/setup-repo-6CZGK337.js +0 -2
  143. package/dist-cli/chunks/store-ZUZGIBZL.js +0 -2
  144. package/dist-cli/chunks/telemetry-DSKEFDAZ.js +0 -2
  145. package/dist-cli/chunks/upload-KZOOYKHT.js +0 -2
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.114 | (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-M6LHGDWE.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
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.114 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{a as k,c as T,e as C}from"./chunk-VRS7EUIU.js";import{E as v,F as P}from"./chunk-M6LHGDWE.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(t,i){super(t),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,t={}){let i=new Set,a=t.port??x(),l=t.commandTimeoutMs??15e3,r,m="none",p=new Set(t.stripBooleanFlags??[]),b=new Set(t.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),r=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),r=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(!r){let n=T();n&&(r=n.trim()||void 0,m="saved")}return{positional:e.filter((n,d)=>!i.has(d)),wsPort:a,simId:r,simIdSource:m,commandTimeoutMs:l}}function E(e,t={}){let i=1,a=t.commandTimeoutMs??15e3,l=new Map,r=new I(`ws://localhost:${e}`),m=t.simId?{key:`sim:${t.simId}`,source:"explicit-sim-id",stable:!0}:k(),p=new Promise((s,o)=>{r.on("open",()=>{try{r.send(JSON.stringify({type:"bridge:hello",id:0,cliIdentityKey:m.key,cliIdentitySource:m.source,cliLabel:t.cliLabel}))}catch{}s()}),r.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?t.simIdSource:"none");process.stderr.write(` \u2192 ${c} (${g})
3
- `)}return r.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(`
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))}),r.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),r.send(JSON.stringify(y))})};try{return await g(t.simId)}catch(u){let f=u instanceof Error?u.message:String(u);throw t.simIdSource==="saved"&&t.simId&&s.simId===void 0&&f===`no sim connected with id ${t.simId}`?(C(),new Error(`saved sim ${t.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{r.readyState===I.OPEN&&r.send(JSON.stringify({type:"bridge:bye",id:0}))}catch{}r.close();let s=setTimeout(()=>{r.readyState!==I.CLOSED&&r.terminate()},250)}}}function V(e){return E(e.wsPort,{commandTimeoutMs:e.commandTimeoutMs,simId:e.simId,simIdSource:e.simIdSource})}function F(e,t){let i=e.filter(r=>r.readyState==="open"&&r.id!==t),a=i.find(r=>r.userFocused)??i.find(r=>r.userVisible===!0&&r.isPrimary)??i.find(r=>r.userVisible===!0)??i.find(r=>r.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 t=await e.send({type:"evaluate",code:`(() => ({
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 t=="object"&&t!==null?t:{hidden:t===!0};if(i.hidden===!0===!0){let l=null;try{l=F(await e.listSims(),i.simId)}catch{}return process.stderr.write(`
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,t,i={}){let a={type:"evaluate",code:t};return i.acquireLock&&(a.acquireLock=!0),e.send(a)}async function D(e,t,...i){return e.send({type:"call",path:t,args:i})}async function q(e,t,...i){return e.send({type:"call",path:t,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};
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.114 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
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 () => await window.__sootsimTest.queryAll({ hasRole: ${JSON.stringify(e.role)}, pruneHidden: true }))()`}:e.type?{mode:"type",code:`(async () => await window.__sootsimTest.queryAll({ type: ${JSON.stringify(e.type)}, pruneHidden: true }))()`}:e.pressable?{mode:"pressable",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 all = await window.__sootsimTest.queryAll({ pruneHidden: true })
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 () => await window.__sootsimTest.findByText(${JSON.stringify(e.text)}))()`}: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 () => {
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.114 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
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.114 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
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 a=t.url.length>96?`${t.url.slice(0,93)}\u2026`:t.url;console.log(` url: ${a}`)}else t.origin&&console.log(` origin: ${t.origin}`);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 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 nt(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 rt(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 st(e){if(!e.connectAckFile)return!1;try{return X.writeFileSync(e.connectAckFile,`${JSON.stringify({connectedAt:Date.now(),pid:e.pid??null})}
4
- `,{flag:"w"}),!0}catch{return!1}}function it(e){if(e.diagnosticLogPath)try{let o=X.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 at(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=rt(r);s&&n.add(s)}for(let r of n){try{process.kill(r,"SIGTERM")}catch{}if(await at(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,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(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 lt(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 () => {
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.114 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
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};