sootsim 0.0.1 → 0.0.3

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 (122) hide show
  1. package/README.md +12 -0
  2. package/dist-cli/bin.js +16 -10
  3. package/dist-cli/chunks/agent-D5NBV32O.js +61 -0
  4. package/dist-cli/chunks/agent-wrapper-Y7I5QGHM.js +15 -0
  5. package/dist-cli/chunks/assert-EJ7DQS2H.js +47 -0
  6. package/dist-cli/chunks/auto-bootstrap-Q7GNLISM.js +2 -0
  7. package/dist-cli/chunks/{chunk-7X6OPSRD.js → chunk-2FPPPJE5.js} +2 -2
  8. package/dist-cli/chunks/{chunk-G5MR66EB.js → chunk-3K6VDPVD.js} +2 -2
  9. package/dist-cli/chunks/{chunk-PWXPA745.js → chunk-3SLEIN6B.js} +1 -1
  10. package/dist-cli/chunks/chunk-3WPAEUOO.js +1 -0
  11. package/dist-cli/chunks/chunk-44CBTM22.js +2 -0
  12. package/dist-cli/chunks/chunk-46LRF7PH.js +5 -0
  13. package/dist-cli/chunks/chunk-4RYT6AQV.js +16 -0
  14. package/dist-cli/chunks/chunk-5AG24UFX.js +119 -0
  15. package/dist-cli/chunks/chunk-5IPP4HAW.js +5 -0
  16. package/dist-cli/chunks/{chunk-J2S3OCWA.js → chunk-AFTHIY3L.js} +1 -1
  17. package/dist-cli/chunks/chunk-BU3TZP4Y.js +11 -0
  18. package/dist-cli/chunks/chunk-BYLX2DO4.js +27 -0
  19. package/dist-cli/chunks/chunk-CPMW2QLM.js +1 -0
  20. package/dist-cli/chunks/{chunk-YCETS3B3.js → chunk-CQ6PX2EU.js} +2 -2
  21. package/dist-cli/chunks/chunk-D4JFMCXD.js +2 -0
  22. package/dist-cli/chunks/chunk-EEBR5YP5.js +62 -0
  23. package/dist-cli/chunks/chunk-EQ7G3UHS.js +4 -0
  24. package/dist-cli/chunks/{chunk-64TOMNZX.js → chunk-FTRI7SVV.js} +2 -2
  25. package/dist-cli/chunks/{chunk-GPVPHE2B.js → chunk-H3JVJXOC.js} +2 -2
  26. package/dist-cli/chunks/chunk-LV5U7TI4.js +1 -0
  27. package/dist-cli/chunks/chunk-NKJLTISU.js +4 -0
  28. package/dist-cli/chunks/chunk-O2HBPZW5.js +22 -0
  29. package/dist-cli/chunks/{chunk-KSACMDXK.js → chunk-OG5CKIPC.js} +2 -2
  30. package/dist-cli/chunks/{chunk-E522F5JW.js → chunk-P5C3UASK.js} +1 -1
  31. package/dist-cli/chunks/chunk-REYWQVAH.js +2 -0
  32. package/dist-cli/chunks/chunk-RLS6PHBW.js +4 -0
  33. package/dist-cli/chunks/chunk-SUZR2SZZ.js +34 -0
  34. package/dist-cli/chunks/{chunk-OROM7DZI.js → chunk-USRNDVQ3.js} +1 -1
  35. package/dist-cli/chunks/{chunk-JSF5LPNT.js → chunk-UZL5ZZ4E.js} +5 -5
  36. package/dist-cli/chunks/{chunk-QOBRRY5X.js → chunk-VI3VW5BL.js} +1 -1
  37. package/dist-cli/chunks/chunk-WUYJFYOW.js +2 -0
  38. package/dist-cli/chunks/chunk-X2W4IRXK.js +3 -0
  39. package/dist-cli/chunks/chunk-XJBPH4JR.js +308 -0
  40. package/dist-cli/chunks/chunk-ZSRMXBGK.js +2 -0
  41. package/dist-cli/chunks/{compat-MRN2ORY5.js → compat-5KSMOWLB.js} +4 -4
  42. package/dist-cli/chunks/{config-CO5IYWUY.js → config-NJB6PQHU.js} +5 -5
  43. package/dist-cli/chunks/control-2F3AGZAO.js +2 -0
  44. package/dist-cli/chunks/{daemon-G4XVRFHM.js → daemon-MLG65V4S.js} +2 -2
  45. package/dist-cli/chunks/{debug-ZNSZTWT6.js → debug-QVOBTTLP.js} +4 -4
  46. package/dist-cli/chunks/demo-app-registry-XRYNJ4GC.js +2 -0
  47. package/dist-cli/chunks/{detox-JEGYNTYV.js → detox-ZZSNZL4T.js} +2 -2
  48. package/dist-cli/chunks/{device-BS34FAFM.js → device-PQB3YGHN.js} +2 -2
  49. package/dist-cli/chunks/drivers-GWDQEGWD.js +2 -0
  50. package/dist-cli/chunks/electron-JB26VHOO.js +15 -0
  51. package/dist-cli/chunks/flow-7JRQXMFV.js +2 -0
  52. package/dist-cli/chunks/{hints-7Z656W4H.js → hints-IGYDXXDS.js} +2 -2
  53. package/dist-cli/chunks/home-paths-CEGSGQTD.js +2 -0
  54. package/dist-cli/chunks/{inspect-NAHXP2M5.js → inspect-DSU6ELRM.js} +153 -165
  55. package/dist-cli/chunks/install-K6IJKADG.js +65 -0
  56. package/dist-cli/chunks/{install-desktop-PYIZIH67.js → install-desktop-SC3LNFFF.js} +8 -4
  57. package/dist-cli/chunks/install-dev-desktop-4DP3UY2X.js +100 -0
  58. package/dist-cli/chunks/keys-R5LAPAAL.js +19 -0
  59. package/dist-cli/chunks/launch-K3WJV4QA.js +16 -0
  60. package/dist-cli/chunks/{login-Z5Z54HUJ.js → login-A23PYJAW.js} +5 -5
  61. package/dist-cli/chunks/{logout-T2QDYGCB.js → logout-AJ24PH5O.js} +2 -2
  62. package/dist-cli/chunks/{maestro-4AXTS7OE.js → maestro-YALWKKGU.js} +2 -2
  63. package/dist-cli/chunks/{preview-NMGWHWMX.js → preview-D35EEONY.js} +2 -2
  64. package/dist-cli/chunks/{profile-6RGJA4FR.js → profile-MAF7NM5Q.js} +3 -3
  65. package/dist-cli/chunks/record-ZCPQNGFW.js +37 -0
  66. package/dist-cli/chunks/runtime-Z2WIXYUN.js +25 -0
  67. package/dist-cli/chunks/{screenshot-R3GCCSCI.js → screenshot-NQVZYC3C.js} +3 -3
  68. package/dist-cli/chunks/screenshot-mode-E45D2ZFH.js +17 -0
  69. package/dist-cli/chunks/{screenshots-4UQJE4NC.js → screenshots-I4SQI4DA.js} +2 -2
  70. package/dist-cli/chunks/server-ZUXKJRR5.js +29 -0
  71. package/dist-cli/chunks/{skills-2PPKPL4B.js → skills-N4U63E5W.js} +2 -2
  72. package/dist-cli/chunks/store-4A6X4GBJ.js +2 -0
  73. package/dist-cli/chunks/{test-5LFKOQ4M.js → test-VBD6N3AR.js} +3 -3
  74. package/dist-cli/chunks/upload-Y6FZ5XF2.js +2 -0
  75. package/dist-cli/chunks/{whoami-H6FW34JS.js → whoami-4K6JGMWH.js} +2 -2
  76. package/dist-lib/agent-daemon-client.cjs +414 -0
  77. package/dist-lib/agent-events.cjs +48 -0
  78. package/dist-lib/agent-sessions.cjs +692 -0
  79. package/dist-lib/attached-projects.cjs +448 -0
  80. package/dist-lib/auth/shared-session.cjs +174 -0
  81. package/dist-lib/backend-origin.cjs +70 -0
  82. package/dist-lib/bridge-constants.cjs +32 -0
  83. package/dist-lib/cli-constants.cjs +32 -0
  84. package/dist-lib/config.cjs +88 -0
  85. package/dist-lib/dev-bundle-resolution.cjs +236 -0
  86. package/dist-lib/home-paths.cjs +234 -0
  87. package/dist-lib/host/bridge-host.cjs +3458 -0
  88. package/dist-lib/index.cjs +361 -0
  89. package/dist-lib/metro.cjs +215 -0
  90. package/dist-lib/render-mode.cjs +54 -0
  91. package/dist-lib/vite-base.cjs +4217 -0
  92. package/dist-lib/vite.cjs +178 -0
  93. package/package.json +80 -13
  94. package/scripts/postinstall.cjs +70 -0
  95. package/dist-cli/chunks/bridge-host-2EY7Z4AO.js +0 -2
  96. package/dist-cli/chunks/chunk-3C3ZH7PP.js +0 -4
  97. package/dist-cli/chunks/chunk-3R4ZZESY.js +0 -119
  98. package/dist-cli/chunks/chunk-74XPLOV4.js +0 -2
  99. package/dist-cli/chunks/chunk-7LMDCMSI.js +0 -8
  100. package/dist-cli/chunks/chunk-A2CZQIWO.js +0 -1
  101. package/dist-cli/chunks/chunk-CKZ376AY.js +0 -322
  102. package/dist-cli/chunks/chunk-E5UBZEYR.js +0 -2
  103. package/dist-cli/chunks/chunk-HOIHCO7S.js +0 -3
  104. package/dist-cli/chunks/chunk-KQWZZ56P.js +0 -2
  105. package/dist-cli/chunks/chunk-KSB6MSZ4.js +0 -34
  106. package/dist-cli/chunks/chunk-KXYKAYYB.js +0 -51
  107. package/dist-cli/chunks/chunk-MBFP2LVH.js +0 -3
  108. package/dist-cli/chunks/chunk-MPSZ5EWF.js +0 -16
  109. package/dist-cli/chunks/chunk-X2U72K7X.js +0 -1
  110. package/dist-cli/chunks/control-Y7TKKB6D.js +0 -2
  111. package/dist-cli/chunks/dev-ZUKCZQEX.js +0 -25
  112. package/dist-cli/chunks/dev-checkout-IEZVVTCN.js +0 -2
  113. package/dist-cli/chunks/drivers-46PFFIDF.js +0 -2
  114. package/dist-cli/chunks/electron-P2KOPX2S.js +0 -15
  115. package/dist-cli/chunks/flow-VVOF6UNC.js +0 -2
  116. package/dist-cli/chunks/install-EPUJX4AT.js +0 -67
  117. package/dist-cli/chunks/record-IE27Z2GA.js +0 -37
  118. package/dist-cli/chunks/screenshot-mode-SZQDNGYE.js +0 -17
  119. package/dist-cli/chunks/server-AN2G5KO4.js +0 -21
  120. package/dist-cli/chunks/store-PU5ES4YQ.js +0 -2
  121. package/dist-cli/chunks/upload-BYNPC54C.js +0 -2
  122. package/dist-cli/chunks/vite-plugin-5AEUUBKP.js +0 -9
@@ -1,25 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{g as k,h as S}from"./chunk-7X6OPSRD.js";import"./chunk-MBFP2LVH.js";import"./chunk-E522F5JW.js";import{spawn as D}from"child_process";import{existsSync as w,readFileSync as P}from"fs";import R from"http";import{resolve as c,dirname as _}from"path";import{fileURLToPath as E}from"url";var u=_(E(import.meta.resolve("sootsim-engine/package.json"))),j=new Set(["turbo","nx","lerna","pnpm","npm","yarn","bun","npx","bunx"]);function A(o){let s=o[0]?.replace(/^.*[\\/]/,"");return!!(j.has(s)||s==="pnpm"||s==="npm"||s==="yarn"||s==="bun")}async function J(o){o.command.length===0&&(console.log(`
3
- sootsim dev \u2014 iOS simulator proxy
4
-
5
- wraps any bundler. proxies everything through sootsim.
6
-
7
- usage:
8
- sootsim pnpm dev
9
- sootsim npx expo start
10
- sootsim --port 3000 npm start
11
- sootsim dev -- one dev
12
-
13
- add to package.json:
14
- "dev:soot": "sootsim dev -- bun metro:dev"
15
- `),process.exit(1));let s=o.port||I(),e=s+100,t=[...o.command],m=A(t);if(!m){let n=t.findIndex(b=>b==="--port"||b==="-p");n>=0&&t[n+1]?t[n+1]=String(e):t.push("--port",String(e))}console.log(`
16
- \u250C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2510
17
- \u2502 sootsim \u2502
18
- \u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518
19
-
20
- port: ${s} \u2192 ${e} (internal)
21
- command: ${t.join(" ")}
22
- mode: ${m?"env (task runner detected)":"flag (direct bundler)"}
23
- `);let r=process.cwd(),p=[...[c(r,"node_modules/.bin"),c(r,"../../node_modules/.bin")].filter(n=>w(n)),process.env.PATH].join(":"),i=D(t[0],t.slice(1),{cwd:r,stdio:"inherit",env:{...process.env,PATH:p,PORT:String(e),RCT_METRO_PORT:String(e),ONE_PORT:String(e),EXPO_METRO_PORT:String(e)}});i.on("error",n=>{console.error(` failed to start: ${n.message}`),process.exit(1)}),console.log(" waiting for bundler..."),await H(e)||(console.error(` bundler didn't start on port ${e} within 120s`),i.kill(),process.exit(1));let a=W(),l=a[0];for(let n of a)try{if((await fetch(`http://localhost:${e}${n}`,{method:"HEAD",signal:AbortSignal.timeout(5e3)})).ok){l=n;break}}catch{}console.log(" bundler ready"),console.log(` bundle: ${l.split("?")[0]}`);let{createServer:$}=await import("vite"),{sootsim:T}=await import("./vite-plugin-5AEUUBKP.js"),O=`http://localhost:${s}${l}`,f=`http://localhost:${s}/?bundle=${encodeURIComponent(O)}`,g=await $({root:u,configFile:!1,plugins:[{name:"sootsim-proxy",configureServer(n){n.middlewares.use(C(e,f))}},T()],server:{port:s,strictPort:!0,fs:{allow:[u,c(u,".."),c(u,"../../node_modules")]}},optimizeDeps:{include:["react-reconciler","react-reconciler/constants","canvaskit-wasm","yoga-layout"]},resolve:{alias:[{find:"@tamagui/config/v5-css",replacement:c(u,"../../node_modules/@tamagui/config/dist/esm/v5-css.mjs")}]}});await g.listen(),console.log(`
24
- sootsim: ${f}
25
- `);let y=o.driver?k(o.driver):null;o.driver&&!y&&console.error(` unknown driver "${o.driver}" \u2014 try \`sootsim list --drivers\``);let v=y??S(null,["electron","system","chromium"])??null;if(v){let n=await v.launch({url:f,headless:o.headless});n.launched?console.log(` ${n.message}`):console.log(` ${v.id}: ${n.message}`)}let x=()=>{i.kill(),g.close()};process.on("SIGINT",()=>{x(),process.exit(0)}),process.on("SIGTERM",x),i.on("exit",n=>{g.close(),process.exit(n||0)})}function I(){let o=c(process.cwd(),"package.json");if(w(o))try{let s=JSON.parse(P(o,"utf8")),e={...s.dependencies,...s.devDependencies};if(e.one||e.expo||e["react-native"])return 8081}catch{}return 8081}function W(){let o=c(process.cwd(),"package.json");if(w(o))try{let s=JSON.parse(P(o,"utf8"));if({...s.dependencies,...s.devDependencies}.one)return["/node_modules/one/metro-entry.bundle?platform=ios&dev=true&minify=false","/index.bundle?platform=ios&dev=true&hot=true&minify=false"]}catch{}return["/index.bundle?platform=ios&dev=true&hot=true&minify=false"]}async function H(o,s=12e4){let e=Date.now();for(;Date.now()-e<s;){try{let t=await fetch(`http://localhost:${o}/status`,{method:"GET",signal:AbortSignal.timeout(2e3)});if(t.ok||t.status<500)return!0}catch{}try{let t=await fetch(`http://localhost:${o}/`,{method:"HEAD",signal:AbortSignal.timeout(2e3)});if(t.ok||t.status<500)return!0}catch{}await new Promise(t=>setTimeout(t,2e3))}return!1}function C(o,s){return(e,t,m)=>{let r=e.url||"/",h=r.split("?")[0];if(h==="/"&&s&&!r.includes("bundle=")){let i=s.replace(/^https?:\/\/[^/]+/,"");t.writeHead(302,{Location:i}),t.end();return}if(h==="/"||r.startsWith("/src/")||r.startsWith("/@vite/")||r.startsWith("/@id/")||r.startsWith("/@fs/")||r.startsWith("/node_modules/.vite/")||r.startsWith("/canvaskit.wasm")||r.startsWith("/fonts/")||r.startsWith("/icons/")||r.startsWith("/sounds/"))return m();let p=R.request({hostname:"localhost",port:o,path:r,method:e.method||"GET",headers:{...e.headers,host:`localhost:${o}`}},i=>{let d={...i.headers},a=i.headers["set-cookie"];if(a){d["x-sootsim-set-cookie"]=Array.isArray(a)?a.join(", "):a;let l=String(d["access-control-expose-headers"]||"");d["access-control-expose-headers"]=(l?l+", ":"")+"x-sootsim-set-cookie"}t.writeHead(i.statusCode||502,d),i.pipe(t)});p.on("error",i=>{t.writeHead(502,{"Content-Type":"text/plain"}),t.end(`sootsim proxy error: ${i.message}`)}),e.pipe(p)}}export{J as runDev};
@@ -1,2 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import"./chunk-E522F5JW.js";import{spawn as d,execSync as m}from"child_process";import{openSync as f,closeSync as v}from"fs";import{resolve as i}from"path";function p(e){try{return m(`lsof -ti:${e}`,{encoding:"utf8",timeout:3e3}).trim().length>0}catch{return!1}}async function h(e,t=3e4){let o=Date.now();for(;Date.now()-o<t;){if(p(e))return!0;await new Promise(n=>setTimeout(n,500))}return!1}function b(){try{return m('pgrep -f "electron.*packages/sootsim"',{encoding:"utf8",timeout:3e3}).trim().length>0}catch{return!1}}async function P(e,t){let{port:o,verbose:n,device:l}=t,r=i(process.env.TMPDIR||"/tmp","sootsim-dev-vite.log"),a=i(process.env.TMPDIR||"/tmp","sootsim-dev-electron.log"),u=i(e,"..","sootsim-engine");if(await p(o))console.log(` vite already on :${o}`);else{console.log(` starting vite on :${o}`);let c=f(r,"a");d("bun",["run","dev"],{cwd:u,stdio:["ignore",c,c],detached:!0}).unref(),v(c),await h(o)||(console.error(` vite failed to start on :${o}`),console.error(` log: ${r}`),process.exit(1)),n&&console.log(" vite ready")}if(b()){console.log(" sootsim electron already running");return}console.log(" starting electron");let s=f(a,"a"),g={...process.env};l&&(g.SOOTSIM_INITIAL_DEVICE=l),d("bun",["run","dev:electron"],{cwd:u,stdio:["ignore",s,s],detached:!0,env:g}).unref(),v(s),console.log(` vite log: ${r}`),console.log(` electron log: ${a}`)}export{P as runDevCheckout};
@@ -1,2 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{a,b,c,d,e,f,g,h,i}from"./chunk-7X6OPSRD.js";import"./chunk-MBFP2LVH.js";import"./chunk-E522F5JW.js";export{e as ALL_DRIVERS,i as buildDriverListRows,a as chromiumDriver,b as electronDriver,f as getAllDrivers,g as getDriver,c as playwrightDriver,h as resolveDriver,d as systemDriver};
@@ -1,15 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{b as l,g as p}from"./chunk-7X6OPSRD.js";import{a as d}from"./chunk-QOBRRY5X.js";import{a}from"./chunk-74XPLOV4.js";import{a as s}from"./chunk-MBFP2LVH.js";import"./chunk-E522F5JW.js";async function I(o,r){(o.includes("--help")||o.includes("-h"))&&(console.log(`
3
- sootsim electron \u2014 launch the desktop companion
4
-
5
- usage:
6
- sootsim electron [options]
7
-
8
- options:
9
- --port <number> connect to running dev server on this port
10
- --driver <id> override the launch driver (default: electron)
11
-
12
- examples:
13
- sootsim electron
14
- sootsim electron --port 5173
15
- `),process.exit(0));let c=o.indexOf("--driver"),n=(c>=0?o[c+1]:void 0)||r.driver,i=l;if(n){let e=p(n);e||(console.error(` unknown driver "${n}" \u2014 run \`sootsim list --drivers\``),process.exit(1)),i=e}if(i.id==="electron"){let e=s();if(!e){if(a(),process.stdin.isTTY&&process.env.CI!=="1"&&process.env.SOOTSIM_NO_PROMPT!=="1"&&await d("run sootsim install-desktop now?",!0)){console.log();let{runInstallDesktop:v}=await import("./install-desktop-PYIZIH67.js");await v(["--yes"]),e=s()}e||process.exit(1)}console.log(` launching ${e.path}`)}let m=r.port?`http://localhost:${r.port}`:void 0,t=await i.launch({url:m,device:r.device});t.launched||(console.error(` ${t.message}`),process.exit(1)),console.log(` ${t.message}`)}export{I as runElectron};
@@ -1,2 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{a,b,c,d}from"./chunk-CKZ376AY.js";import"./chunk-G5MR66EB.js";import"./chunk-A2CZQIWO.js";import"./chunk-KXYKAYYB.js";import"./chunk-PWXPA745.js";import"./chunk-KSACMDXK.js";import"./chunk-KQWZZ56P.js";import"./chunk-YCETS3B3.js";import"./chunk-JSF5LPNT.js";import"./chunk-HOIHCO7S.js";import"./chunk-KSB6MSZ4.js";import"./chunk-OROM7DZI.js";import"./chunk-MPSZ5EWF.js";import"./chunk-3C3ZH7PP.js";import"./chunk-E5UBZEYR.js";import"./chunk-X2U72K7X.js";import"./chunk-74XPLOV4.js";import"./chunk-MBFP2LVH.js";import"./chunk-E522F5JW.js";export{b as discoverSootsimUrl,a as parseFlowFile,d as runFlow,c as runFlowPlayback};
@@ -1,67 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{a as v,b as P}from"./chunk-QOBRRY5X.js";import"./chunk-E522F5JW.js";import{execSync as R}from"child_process";import{existsSync as w,readFileSync as j,writeFileSync as y}from"fs";import{resolve as u,relative as V}from"path";import{existsSync as a,readFileSync as h,readdirSync as E,statSync as F}from"fs";import{basename as W,dirname as J,join as c}from"path";function M(e){let n=e;for(;;){if(a(c(n,"turbo.json"))||a(c(n,"nx.json"))||a(c(n,"lerna.json"))||a(c(n,"pnpm-workspace.yaml")))return n;let r=c(n,"package.json");if(a(r))try{if(JSON.parse(h(r,"utf8")).workspaces)return n}catch{}let s=J(n);if(s===n)break;n=s}return e}function L(e){return a(c(e,"bun.lock"))||a(c(e,"bun.lockb"))?"bun":a(c(e,"pnpm-lock.yaml"))?"pnpm":a(c(e,"yarn.lock"))?"yarn":"npm"}function N(e){if(a(c(e,"turbo.json")))return"turbo";if(a(c(e,"nx.json")))return"nx";if(a(c(e,"pnpm-workspace.yaml")))return"pnpm";let n=c(e,"package.json");if(a(n))try{if(JSON.parse(h(n,"utf8")).workspaces)return"npm-workspaces"}catch{}return"single"}function T(e){return e.one?"one":e.expo?"expo":e["react-native"]?"bare":"unknown"}function b(e){let n=c(e,"package.json");if(!a(n))return null;let r;try{r=JSON.parse(h(n,"utf8"))}catch{return null}let s={...r.dependencies,...r.devDependencies},i=T(s);return i==="unknown"?null:{dir:e,name:r.name||W(e),framework:i,hasViteConfig:a(c(e,"vite.config.ts"))||a(c(e,"vite.config.js")),hasMetroConfig:a(c(e,"metro.config.js"))||a(c(e,"metro.config.ts")),devCommand:r.scripts?.dev||null}}function _(e){let n=[],r=c(e,"pnpm-workspace.yaml");if(a(r)){let o=h(r,"utf8").match(/packages:\s*\n((?:\s+-\s+.+\n?)+)/);if(o){let t=o[1].split(`
3
- `).filter(Boolean);for(let f of t){let p=f.replace(/^\s*-\s*['"]?/,"").replace(/['"]?\s*$/,"");p&&n.push(...S(e,p))}}return n}let s=c(e,"package.json");if(a(s))try{let i=JSON.parse(h(s,"utf8")),o=Array.isArray(i.workspaces)?i.workspaces:i.workspaces?.packages||[];for(let t of o)n.push(...S(e,t))}catch{}return n}function S(e,n){let r=n.replace(/\/\*\*?$/,"").replace(/\*$/,""),s=c(e,r);if(!a(s))return[];try{return E(s).map(o=>c(s,o)).filter(o=>{try{return F(o).isDirectory()&&a(c(o,"package.json"))}catch{return!1}})}catch{return[]}}function D(e){let n=N(e),r=L(e);if(n==="single"){let t=b(e);return{root:e,type:n,packageManager:r,apps:t?[t]:[]}}let s=_(e),i=[],o=b(e);o&&i.push(o);for(let t of s){if(t===e)continue;let f=b(t);f&&i.push(f)}return{root:e,type:n,packageManager:r,apps:i}}function C(e){switch(e){case"bun":return"bun add -d sootsim";case"pnpm":return"pnpm add -D sootsim";case"yarn":return"yarn add -D sootsim";case"npm":return"npm install -D sootsim"}}async function re(e){(e.includes("--help")||e.includes("-h"))&&(console.log(`
4
- sootsim install \u2014 set up sootsim in your project
5
-
6
- detects your bundler (one, expo, metro), adds sootsim as a
7
- dependency, configures the bundler plugin, and adds dev scripts.
8
-
9
- usage:
10
- sootsim install interactive setup
11
- sootsim install --yes skip confirmations
12
- sootsim install --dry-run show what would be done
13
-
14
- works with monorepos (turbo, pnpm, nx) \u2014 scans for app packages.
15
- `),process.exit(0));let n=e.includes("--yes")||e.includes("-y"),r=e.includes("--dry-run"),s=process.cwd(),i=M(s),o=D(i);console.log(`
16
- sootsim \u2014 project setup
17
- `),o.apps.length===0&&(console.log(" no react native apps found in this project."),console.log(` sootsim works with one, expo, and react-native projects.
18
- `),process.exit(1));let t;if(o.type!=="single"&&o.apps.length>1){console.log(` monorepo detected (${o.type} + ${o.packageManager})
19
- `);let g=await P("which app?",o.apps.map(l=>`${(V(i,l.dir)||".").padEnd(30)} (${l.framework})`));t=o.apps[g]}else t=o.apps[0];let f=t.framework==="one"?"metro (via one)":t.framework==="expo"?"metro (via expo)":"metro";console.log(" detected:"),console.log(` framework: ${t.framework}`),console.log(` bundler: ${f}`),console.log(` package: ${t.name}`),console.log(` manager: ${o.packageManager}`),console.log();let p=[],d=C(o.packageManager);p.push({label:"add sootsim as devDependency",run:()=>{R(d,{cwd:t.dir,stdio:"pipe"})}}),t.framework==="one"&&t.hasViteConfig?p.push({label:"add sootsimPlugin() to vite.config.ts",run:()=>$(t.dir)}):(t.framework==="expo"||t.framework==="bare")&&t.hasMetroConfig?p.push({label:"add withSootsim() to metro.config.js",run:()=>A(t.dir)}):t.framework==="one"?p.push({label:"add sootsimPlugin() to vite.config.ts",run:()=>$(t.dir)}):p.push({label:"create metro.config.js with withSootsim()",run:()=>B(t.dir,t.framework)});let m=t.framework==="one"?"one dev":t.devCommand||"expo start";p.push({label:'add "dev:sootsim" script to package.json',run:()=>G(t.dir,m)}),console.log(" will:"),p.forEach((g,l)=>console.log(` ${l+1}. ${g.label}`)),console.log(),r&&(console.log(` (dry run \u2014 no changes made)
20
- `),process.exit(0)),n||(await v("continue?")||(console.log(`
21
- cancelled.
22
- `),process.exit(0)),console.log());for(let g=0;g<p.length;g++){let l=p[g];process.stdout.write(` [${g+1}/${p.length}] ${l.label}...`);try{l.run(),console.log(" done")}catch(x){console.log(` failed: ${x.message}`),(l.label.includes("vite.config")||l.label.includes("metro.config"))&&U(t)}}let k=o.packageManager==="bun"?"bun":o.packageManager;console.log(`
23
- done! run:
24
- ${k} run dev:sootsim
25
-
26
- then open:
27
- http://localhost:8081/__soot/
28
- `)}function $(e){let n=u(e,"vite.config.ts")||u(e,"vite.config.js"),r=u(e,"vite.config.ts"),s=u(e,"vite.config.js"),i=w(r)?r:s;if(!w(i))throw new Error(`no vite config found at ${e}`);let o=j(i,"utf8");if(o.includes("sootsimPlugin")){console.log(" (already configured)");return}let t="import { sootsimPlugin } from 'sootsim/vite'",f=/^(import\s+.+)$/gm,p=0,d;for(;(d=f.exec(o))!==null;)p=d.index+d[0].length;p>0?o=o.slice(0,p)+`
29
- `+t+o.slice(p):o=t+`
30
- `+o;let m=o.match(/plugins\s*:\s*\[/);if(m&&m.index!==void 0){let k=m.index+m[0].length,g=1,l=k;for(;l<o.length&&g>0;)o[l]==="["&&g++,o[l]==="]"&&g--,g>0&&l++;let x=o.slice(0,l),O=o.slice(l),q=o.slice(k).match(/\n(\s+)\S/)?.[1]||" ",I=o.slice(o.lastIndexOf(`
31
- `,l),l+1).match(/\n(\s*)/)?.[1]||" ";o=x.trimEnd()+`
32
- `+q+`sootsimPlugin(),
33
- `+I+O}else throw new Error("could not find plugins array in vite config");y(i,o)}function A(e){let n=u(e,"metro.config.js"),r=u(e,"metro.config.ts"),s=w(n)?n:r;if(!w(s))throw new Error(`no metro config found at ${e}`);let i=j(s,"utf8");if(i.includes("withSootsim")){console.log(" (already configured)");return}i="const { withSootsim } = require('sootsim/metro')"+`
34
- `+i,i=i.replace(/(module\.exports\s*=\s*)([^;]+)/,(t,f,p)=>`${f}withSootsim(${p.trim()})`),y(s,i)}function B(e,n){let r=u(e,"metro.config.js"),s;n==="expo"?s=`const { getDefaultConfig } = require('expo/metro-config')
35
- const { withSootsim } = require('sootsim/metro')
36
-
37
- const config = getDefaultConfig(__dirname)
38
-
39
- module.exports = withSootsim(config)
40
- `:s=`const { getDefaultConfig, mergeConfig } = require('@react-native/metro-config')
41
- const { withSootsim } = require('sootsim/metro')
42
-
43
- const config = {}
44
-
45
- module.exports = withSootsim(mergeConfig(getDefaultConfig(__dirname), config))
46
- `,y(r,s)}function G(e,n){let r=u(e,"package.json"),s=JSON.parse(j(r,"utf8"));if(s.scripts||(s.scripts={}),s.scripts["dev:sootsim"]){console.log(" (already exists)");return}s.scripts["dev:sootsim"]=n,y(r,JSON.stringify(s,null,2)+`
47
- `)}function U(e){e.framework==="one"?console.log(`
48
- manual setup \u2014 add to your vite.config.ts:
49
-
50
- import { sootsimPlugin } from 'sootsim/vite'
51
-
52
- export default {
53
- plugins: [
54
- one({ /* ... */ }),
55
- sootsimPlugin(),
56
- ],
57
- }
58
- `):console.log(`
59
- manual setup \u2014 add to your metro.config.js:
60
-
61
- const { withSootsim } = require('sootsim/metro')
62
- module.exports = withSootsim(config)
63
- `)}async function ie(){console.log(`
64
- sootsim uninstall is no longer needed.
65
- to remove sootsim, just remove the plugin from your config
66
- and remove sootsim from your dependencies.
67
- `)}export{re as runInstall,ie as runUninstall};
@@ -1,37 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{a as W}from"./chunk-OROM7DZI.js";import{b as R,d as $}from"./chunk-MPSZ5EWF.js";import{c as N}from"./chunk-3C3ZH7PP.js";import"./chunk-X2U72K7X.js";import"./chunk-E522F5JW.js";import{existsSync as Y,mkdirSync as _,readFileSync as Z,rmSync as C,writeFileSync as U}from"fs";import{tmpdir as ee}from"os";import{dirname as z,extname as oe,join as re,resolve as O}from"path";var D=6e4;async function we(e,r){if((e.includes("--help")||e.includes("-h"))&&(console.log(`
3
- sootsim record \u2014 capture the running session
4
-
5
- usage:
6
- sootsim record [options] one-shot capture (duration-bounded)
7
- sootsim record start [options] begin stateful recording
8
- sootsim record stop [--output <path>] finalize stateful recording
9
-
10
- options:
11
- --mode <kind> video | live | combined (default: video)
12
- video \u2014 local webm/mp4 file
13
- live \u2014 pointer events only, uploads /preview/<id>
14
- combined \u2014 events + video, uploads /preview/<id>
15
- --duration <sec> recording duration (default: 10, atomic mode only)
16
- --fps <n> frame rate for video/gif (default: 30)
17
- --format <fmt> webm | mp4 | gif | png (video mode; inferred from --output ext)
18
- --output <path> output file (video/gif) or directory (png frames)
19
- --frames <n> sample N png frames evenly across --duration
20
- --max-width <px> downscale gif frames to this width
21
- --no-shell exclude the simulated iOS chrome (status bar, keyboard,
22
- toasts) \u2014 records the tenant surfaces only
23
- --shell-only record only the shell chrome
24
- --open open the preview URL in the browser after stop (live/combined)
25
- --session <tab-id> target a specific bridge tab
26
-
27
- examples:
28
- sootsim record # webm, 10s, ./sootsim-<ts>.webm
29
- sootsim record --format mp4 --duration 5
30
- sootsim record --output demo.gif --duration 3
31
- sootsim record --frames 10 --output ./frames/
32
- sootsim record --mode combined --duration 8 --open
33
- sootsim record start --format mp4 # start video; interact freely
34
- sootsim record start --mode combined # start preview-share recording
35
- sootsim record stop --output flow.mp4 # finalize video
36
- sootsim record stop --open # finalize live/combined + open URL
37
- `),process.exit(0)),e[0]==="start"){await ie(e.slice(1),r);return}if(e[0]==="stop"){await ce(e.slice(1),r);return}if(e[0]==="cancel"){await de(e.slice(1),r);return}if(e[0]==="status"){ae();return}let o=R(e,{port:r.port,stripBooleanFlags:["--no-shell","--shell-only","--open"],stripValueFlags:["--mode","--duration","--fps","--format","--output","--frames","--max-width"]}),t=q(c(e,"--mode")),n=e.includes("--shell-only")?"shell":e.includes("--no-shell")?"tenant":void 0,i=c(e,"--format"),s=c(e,"--output"),a=Number(c(e,"--duration")??"10"),l=Number(c(e,"--fps")??"30"),d=c(e,"--frames"),w=e.includes("--open"),h=c(e,"--max-width")?Number(c(e,"--max-width")):void 0,v=Math.max(100,Math.round(a*1e3));if(t==="live"||t==="combined"){let u=$({...o,commandTimeoutMs:6e4});try{await V(u),await j(u,t)||(console.error(` start failed: recording store refused to start (${t})`),process.exit(1)),console.log(` recording ${t} for ${a}s`),await new Promise(y=>setTimeout(y,v)),await K(u);let k=await G(u);H(k,w)}finally{u.close()}return}let M=d?Number(d):null,S=ne(i,s,M),p=$({...o,commandTimeoutMs:6e4});try{if(await L(p),S==="png"){let b=M??10,f=O(process.cwd(),s??`./sootsim-frames-${B()}`);_(f,{recursive:!0}),console.log(` sampling ${b} frames over ${a}s \u2192 ${f}`);let g=await p.send({type:"evaluate",code:`window.__sootsimRecorder.startFrameCapture({ count: ${b}, durationMs: ${v} })`});(!g.ok||!g.requestId)&&(console.error(` frame capture start failed: ${g.error??"unknown error"}`),process.exit(1)),await new Promise(m=>setTimeout(m,v));let Q=Date.now()+Math.max(5e3,v),A=null;for(;;){let m=await p.send({type:"evaluate",code:`window.__sootsimRecorder.getFrameCaptureResult(${g.requestId})`});if(m||(console.error(" frame capture result missing"),process.exit(1)),m.done){m.ok||(console.error(` frame capture failed: ${m.error??"unknown error"}`),process.exit(1)),A=m.frames??[];break}Date.now()>=Q&&(console.error(" frame capture timed out"),process.exit(1)),await new Promise(E=>setTimeout(E,100))}A.forEach((m,E)=>{let X=`${f}/frame-${String(E+1).padStart(3,"0")}.png`;U(X,Buffer.from(m.data,"base64"))}),console.log(` saved ${A.length} frames`);return}if(S==="gif"){let b=M??Math.max(10,Math.round(a*l/3)),f=O(process.cwd(),s??`./sootsim-${B()}.gif`);_(z(f),{recursive:!0}),console.log(` encoding gif: ${b} frames over ${a}s \u2192 ${f}`);let g=await p.send({type:"evaluate",code:`window.__sootsimRecorder.captureGif({ frames: ${b}, durationMs: ${v}${h?`, maxWidth: ${h}`:""} })`});g||(console.error(" gif capture returned no frames"),process.exit(1)),U(f,Buffer.from(g.data,"base64")),console.log(` saved: ${f} (${T(g.size)})`);return}let u=O(process.cwd(),s??`./sootsim-${B()}.${S}`);_(z(u),{recursive:!0});let I={format:S,fps:l};n&&(I.layers=n);let k=await p.send({type:"evaluate",code:`window.__sootsimRecorder.start(${JSON.stringify(I)})`});k.ok||(console.error(` start failed: ${k.error??"unknown error"}`),process.exit(1)),console.log(` recording ${S} for ${a}s \u2192 ${u}`),await new Promise(b=>setTimeout(b,v));let y=await p.send({type:"evaluate",code:"window.__sootsimRecorder.stop()"});y.ok||(console.error(` stop failed: ${y.error??"unknown error"}`),process.exit(1)),y.size||(console.error(" recorder returned an empty blob \u2014 nothing written"),process.exit(1)),await J(p,u),console.log(` saved: ${u} (${T(y.size)})`)}finally{p.close()}}async function L(e){if(!await e.send({type:"evaluate",code:'typeof window.__sootsimRecorder !== "undefined"'}))throw new Error("window.__sootsimRecorder missing \u2014 is sootsim engine running in this tab?")}async function J(e,r){let o=[],t=0;for(;;){let n=await e.send({type:"evaluate",code:`window.__sootsimRecorder.getBlobBase64({ offset: ${t}, chunk: 2097152 })`});if(!n)throw new Error("no blob available on recorder");if(o.push(Buffer.from(n.data,"base64")),t=n.offset,n.done)break}U(r,Buffer.concat(o))}function c(e,r){let o=e.indexOf(r);if(!(o<0||o===e.length-1))return e[o+1]}function te(e){if(!e)return;let r=oe(e).toLowerCase().replace(/^\./,"");if(r==="webm"||r==="mp4"||r==="gif")return r;if(r==="png")return"png"}function ne(e,r,o){return e||(o!=null?"png":te(r)??"webm")}function B(){return new Date().toISOString().replace(/[:T]/g,"-").replace(/\..+/,"")}function T(e){return e<1024?`${e}B`:e<1024*1024?`${(e/1024).toFixed(1)}KB`:`${(e/(1024*1024)).toFixed(2)}MB`}function q(e){if(!e)return"video";if(e==="video"||e==="live"||e==="combined")return e;console.error(` invalid --mode "${e}" \u2014 expected video | live | combined`),process.exit(1)}function P(){return re(ee(),`sootsim-recording-${N()}.json`)}function F(){let e=P();if(!Y(e))return null;try{return{mode:"video",...JSON.parse(Z(e,"utf8"))}}catch{return C(e,{force:!0}),null}}function se(e){U(P(),JSON.stringify(e,null,2))}function x(){C(P(),{force:!0})}async function ie(e,r){let o=F();o&&(console.error(` recording already in progress (started ${o.startedAt}, tab ${o.browserId??"?"}). run \`sootsim record stop\` first, or \`sootsim record cancel\` to discard.`),process.exit(1));let t=R(e,{port:r.port,stripBooleanFlags:["--no-shell","--shell-only"],stripValueFlags:["--mode","--fps","--format","--max-width"]}),n=q(c(e,"--mode")),i=e.includes("--shell-only")?"shell":e.includes("--no-shell")?"tenant":void 0,s=c(e,"--format"),a=s==="mp4"?"mp4":"webm";s&&a!==s&&(console.error(` record start only supports webm or mp4 (got: ${s}). for gif/png use atomic mode: sootsim record --format ${s} --duration <s>`),process.exit(1));let l=Number(c(e,"--fps")??"30"),d=$({...t,commandTimeoutMs:15e3});try{if(n==="live"||n==="combined")await V(d),await j(d,n)||(console.error(` start failed: recording store refused to start (${n})`),process.exit(1));else{await L(d);let w={format:a,fps:l};i&&(w.layers=i);let h=await d.send({type:"evaluate",code:`window.__sootsimRecorder.start(${JSON.stringify(w)})`});h.ok||(console.error(` start failed: ${h.error??"unknown error"}`),process.exit(1))}se({browserId:t.browserId??null,mode:n,format:a,fps:l,layers:i,startedAt:new Date().toISOString()}),console.log(n==="video"?` recording ${a} @ ${l}fps${i?` (${i})`:""} \u2014 run \`sootsim record stop --output <path>\` when done`:` recording ${n} \u2014 run \`sootsim record stop\` when done (add --open to launch the preview URL)`)}finally{d.close()}}function ae(){let e=F();if(!e){console.log(" no recording in progress");return}e.mode==="video"?console.log(` recording ${e.mode} (${e.format} @ ${e.fps}fps) on tab ${e.browserId??"?"} since ${e.startedAt}`):console.log(` recording ${e.mode} on tab ${e.browserId??"?"} since ${e.startedAt}`)}async function de(e,r){let o=F();if(!o){console.log(" no recording in progress");return}let t=R(e,{port:r.port}),n=t.browserId??o.browserId??void 0,i=$({...t,browserId:n,commandTimeoutMs:15e3});try{o.mode==="live"||o.mode==="combined"?await i.send({type:"evaluate",code:"void window.SootSim?.bridges?.cancelRecording?.()"}):await i.send({type:"evaluate",code:"window.__sootsimRecorder.stop()"})}catch{}finally{x(),i.close()}console.log(" recording cancelled")}async function ce(e,r){let o=F();o||(console.error(" no recording in progress. start one with `sootsim record start`."),process.exit(1));let t=R(e,{port:r.port,stripBooleanFlags:["--open"],stripValueFlags:["--output"]}),n=t.browserId??o.browserId??void 0,i=e.includes("--open"),s=$({...t,browserId:n,commandTimeoutMs:6e4});try{if(o.mode==="live"||o.mode==="combined"){await K(s);let w=await G(s);x(),H(w,i);return}let a=c(e,"--output"),l=O(process.cwd(),a??`./sootsim-${B()}.${o.format}`);_(z(l),{recursive:!0});let d=await s.send({type:"evaluate",code:"window.__sootsimRecorder.stop()"});d.ok||(console.error(` stop failed: ${d.error??"unknown error"}`),x(),process.exit(1)),d.size||(console.error(" recorder returned an empty blob \u2014 nothing written"),x(),process.exit(1)),await J(s,l),x(),console.log(` saved: ${l} (${T(d.size)})`)}finally{s.close()}}async function V(e){if(!await e.send({type:"evaluate",code:'typeof window.SootSim?.bridges?.startRecording === "function" && typeof window.SootSim?.bridges?.stopRecording === "function"'}))throw new Error("SootSim.bridges.startRecording missing \u2014 is sootsim engine running in this tab?")}async function j(e,r){return await e.send({type:"evaluate",code:`window.SootSim.bridges.startRecording(${JSON.stringify(r)})`})===!0}async function K(e){await e.send({type:"evaluate",code:"void window.SootSim.bridges.stopRecording()"})}async function G(e){let r=Date.now()+D;for(;Date.now()<r;){let o=await e.send({type:"evaluate",code:"(() => { const s = window.SootSim?.bridges?.getRecordingState?.(); return s ? { state: s.state, lastUpload: s.lastUpload, uploadError: s.uploadError } : null })()"});if(o&&o.state==="idle"){if(o.uploadError)return{uploadError:o.uploadError};if(o.lastUpload?.previewUrl)return{previewUrl:o.lastUpload.previewUrl}}await new Promise(t=>setTimeout(t,300))}return{uploadError:`upload did not settle within ${D/1e3}s`}}function H(e,r){e.uploadError&&(console.error(` upload failed: ${e.uploadError}`),process.exit(1)),e.previewUrl||(console.error(" upload returned no preview URL"),process.exit(1)),console.log(` preview: ${e.previewUrl}`),r&&W(e.previewUrl)}export{te as extToFormat,ne as resolveFormat,we as runRecord,c as valueOf};
@@ -1,17 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{b as c,d as i,g as d}from"./chunk-MPSZ5EWF.js";import"./chunk-3C3ZH7PP.js";import"./chunk-X2U72K7X.js";import"./chunk-E522F5JW.js";async function g(o,l){(o.includes("--help")||o.includes("-h"))&&(console.log(`
3
- sootsim screenshot-mode \u2014 toggle the screenshot-mode chrome overlay
4
-
5
- usage:
6
- sootsim screenshot-mode [on|off|toggle]
7
-
8
- arguments:
9
- on enable screenshot mode (enables showFrame if needed)
10
- off disable screenshot mode
11
- toggle flip the current state (default)
12
-
13
- examples:
14
- sootsim screenshot-mode
15
- sootsim screenshot-mode on
16
- sootsim screenshot-mode off
17
- `),process.exit(0));let r=c(o,{port:l.port}),e=r.commandArgs[0]?.toLowerCase()??"toggle";e!=="on"&&e!=="off"&&e!=="toggle"&&(console.error(` unknown argument: "${e}" (expected on | off | toggle)`),process.exit(1));let s=i({...r,commandTimeoutMs:5e3});try{let t=!!(await d(s,"SootSim.bridges.settings.get"))?.screenshotMode,n=e==="on"?!0:e==="off"?!1:!t;if(t===n){console.log(` screenshot-mode: already ${n?"on":"off"}`);return}await s.send({type:"evaluate",code:"window.dispatchEvent(new CustomEvent('sootsim:shell-command', { detail: { type: 'fire-action', id: 'toggle-screenshot-mode' } }))"}),console.log(` screenshot-mode: ${t?"on":"off"} \u2192 ${n?"on":"off"}`)}finally{s.close()}}export{g as runScreenshotMode};
@@ -1,21 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{a as m}from"./chunk-7LMDCMSI.js";import{c as u}from"./chunk-MPSZ5EWF.js";import"./chunk-3C3ZH7PP.js";import"./chunk-X2U72K7X.js";import"./chunk-E522F5JW.js";async function $(e,i={}){(e.includes("--help")||e.includes("-h"))&&(console.log(`
3
- sootsim server \u2014 run the sootsim bridge daemon in the foreground
4
-
5
- hosts the WS bridge that CLI commands talk to. once running, any sootsim
6
- tab (browser, electron, headless playwright) that connects to port 7668
7
- becomes drivable from 'sootsim describe', 'sootsim tap', etc.
8
-
9
- usage:
10
- sootsim server [options]
11
-
12
- options:
13
- --port <n> bridge port (defaults to ${7668})
14
- --quiet suppress per-connection logging
15
-
16
- examples:
17
- sootsim server
18
- sootsim server --port 7668 --quiet
19
- `),process.exit(0));let s=e.indexOf("--port"),n=s>=0&&e[s+1]?Number(e[s+1]):i.port??7668;Number.isNaN(n)&&(console.error(` invalid --port value: ${e[s+1]}`),process.exit(1));let w=e.includes("--quiet")||e.includes("-q"),a=await S(n);a&&(console.error(` port ${n} is already hosting a sootsim bridge`),a.browsers>0&&console.error(` ${a.browsers} browser tab(s) connected`),console.error(" stop the other process first, or pass --port <other>"),process.exit(1));let l=new m({port:n});l.start();let f=Date.now(),t=r=>{w||process.stdout.write(`${r}
20
- `)},c=new Set,h=setInterval(()=>{let r=l.listBrowsers(),p=new Set(r.map(o=>o.id));for(let o of r)if(!c.has(o.id)){let g=o.title||o.url||o.origin||"(unknown)";t(` + ${o.id} ${g}`)}for(let o of c)p.has(o)||t(` - ${o}`);c.clear();for(let o of p)c.add(o)},500);t(`sootsim bridge listening on ws://localhost:${n}`),t(" ready for browser tabs, electron, or headless playwright to connect"),t(" (ctrl-c to stop)");let d=async r=>{clearInterval(h),t(`
21
- ${r} received \u2014 shutting down after ${Math.round((Date.now()-f)/1e3)}s`);try{await l.close()}catch{}process.exit(0)};process.on("SIGINT",()=>d("SIGINT")),process.on("SIGTERM",()=>d("SIGTERM")),await new Promise(()=>{})}async function S(e){let i=u(e,{commandTimeoutMs:1e3});try{return{browsers:(await i.listBrowsers()).length}}catch{return null}finally{i.close()}}export{$ as runServer};
@@ -1,2 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{a}from"./chunk-KQWZZ56P.js";import"./chunk-YCETS3B3.js";import"./chunk-E522F5JW.js";export{a as settingsStore};
@@ -1,2 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{c as a,d as b}from"./chunk-KXYKAYYB.js";import"./chunk-PWXPA745.js";import"./chunk-OROM7DZI.js";import"./chunk-MPSZ5EWF.js";import"./chunk-3C3ZH7PP.js";import"./chunk-X2U72K7X.js";import"./chunk-E522F5JW.js";export{a as resolveDefaultUploadOrigin,b as runUpload};
@@ -1,9 +0,0 @@
1
- /*! sootsim v0.0.1 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import"./chunk-X2U72K7X.js";import"./chunk-E522F5JW.js";import g from"fs";import a from"path";import{fileURLToPath as E}from"url";import{transformWithEsbuild as j}from"vite";var x=a.resolve(a.dirname(E(import.meta.url)),".."),y=a.resolve(x,"..",".."),W=a.resolve(y,"node_modules"),$=a.resolve(W,"@tamagui"),_=a.resolve(y,"packages/sootsim-engine"),v=a.resolve(_,"src/react-native/index.ts"),R=a.resolve(_,"src/react-bridge.ts"),I=a.resolve(x,"..","compat"),d=a.resolve(I,"src/stubs"),O=a.resolve(d,"native-auto-stub.ts"),w=a.resolve(d,"react-native-internals.ts");function A(){try{return["tamagui",...g.readdirSync($,{withFileTypes:!0}).filter(t=>t.isDirectory()).map(t=>`@tamagui/${t.name}`)].sort()}catch{return["tamagui"]}}var h=A(),P={"react-native/Libraries/Pressability/usePressability":a.resolve(d,"rn-libraries/usePressability.ts"),"react-native/Libraries/Pressability/Pressability":a.resolve(d,"rn-libraries/Pressability.ts"),"react-native/Libraries/Renderer/shims/ReactFabric":a.resolve(d,"rn-libraries/ReactFabric.ts"),"react-native/Libraries/Renderer/shims/ReactNative":a.resolve(d,"rn-libraries/ReactNative.ts")},T={"react-native-safe-area-context":"react-native-safe-area-context.ts","react-native-gesture-handler":"react-native-gesture-handler.ts","react-native-gesture-handler/ReanimatedSwipeable":"reanimated-swipeable.ts","react-native-reanimated":"react-native-reanimated.ts","react-native-screens":"react-native-screens.ts","@react-native-masked-view/masked-view":"masked-view.ts","react-native-svg":"react-native-svg.ts","react-native-keyboard-controller":"react-native-keyboard-controller.ts","react-native-teleport":"react-native-teleport.tsx","react-native-ios-context-menu":"react-native-ios-context-menu.ts","react-native-ios-utilities":"react-native-ios-utilities.ts","@react-native-menu/menu":"react-native-menu.ts","expo-splash-screen":"expo-splash-screen.ts","expo-font":"expo-font.ts","react-native-launch-arguments":"react-native-launch-arguments.ts",expo:"expo.ts","@react-native-async-storage/async-storage":"async-storage.ts","expo-constants":"expo-constants.ts","expo-image":"expo-image.ts","expo-linear-gradient":"expo-linear-gradient.ts","expo-blur":"expo-blur.ts","expo-liquid-glass-view":"expo-liquid-glass-view.ts","expo-glass-effect":"expo-glass-effect.ts","react-native-bottom-tabs":"react-native-bottom-tabs.ts","@callstack/react-native-bottom-tabs":"react-native-bottom-tabs.ts","@expo/ui/swift-ui":"expo-ui.ts","@expo/ui":"expo-ui.ts","react-native-worklets":"react-native-worklets.ts","react-native-webview":"react-native-webview.ts","expo-haptics":"expo-haptics.ts","expo-crypto":"expo-crypto.ts","expo-clipboard":"expo-clipboard.ts","expo-file-system":"expo-file-system.ts","expo-linking":"expo-linking.ts","expo-web-browser":"expo-web-browser.ts","expo-secure-store":"expo-secure-store.ts","expo-status-bar":"expo-status-bar.ts","expo-device":"expo-device.ts","expo-image-picker":"expo-image-picker.ts","expo-document-picker":"expo-document-picker.ts","expo-application":"expo-application.ts","expo-notifications":"expo-notifications.ts","expo-location":"expo-location.ts","@shopify/flash-list":"flash-list.ts","@react-native-community/netinfo":"netinfo.ts","@react-native-community/blur":"react-native-community-blur.ts","@react-native-community/checkbox":"react-native-community-checkbox.ts","@react-native-picker/picker":"react-native-picker.ts","@react-native-community/datetimepicker":"react-native-datetimepicker.ts"},q=["react","react-dom","react-reconciler","react/jsx-runtime","react/jsx-dev-runtime","canvaskit-wasm","canvaskit-wasm/full","yoga-layout"],F=[/^expo-/,/^react-native-/,/^@react-native\//,/^@react-native-community\//,/^@expo\//,/^expo$/];function k(e){return e.startsWith("@")?e.split("/").slice(0,2).join("/"):e.split("/")[0]}function L(e){return F.some(t=>t.test(e))}function b(e,t){try{return g.existsSync(a.join(t,"node_modules",e))}catch{return!1}}function te(e={}){let t=e.app?a.resolve(e.app):"",i=(e.sources||[]).map(r=>a.resolve(r)),n=new Set([...q,...e.ownedPackages||[]]);function s(r){return r.includes("/node_modules/")?!1:t&&r.startsWith(t)?!0:i.some(l=>r.startsWith(l))}return[M(),B(),J(t,i),C(),D(),V(s),G(s),z(t,s,n),U(s),Z(t),H(s)]}function M(){let e=null;return{name:"sootsim-ws-bridge",configureServer(t){e||(import("./bridge-host-2EY7Z4AO.js").then(({SootSimBridgeHost:i})=>{e=new i({port:7668}),e.start?.({silent:!0})}).catch(i=>{let n=i instanceof Error?i.message:String(i);console.warn("[sootsim] ws bridge failed to start:",n)}),t.httpServer?.on("close",()=>{e&&(e.close?.(),e=null)}))}}}function J(e,t){return{name:"sootsim-config",config(i,{mode:n}){let s=["."];for(let o of t)s.push(o);if(e){s.push(e);let o=a.join(e,"node_modules");g.existsSync(o)&&s.push(o)}let r=Object.entries(T).sort((o,c)=>c[0].length-o[0].length).map(([o,c])=>({find:o,replacement:a.resolve(d,c)})),l=["react","react-dom","react/jsx-runtime","react/jsx-dev-runtime"];e||l.push(...h);let m=["@tamagui/config","@tamagui/demos","react-native",...e?[]:h,"react-native-actions-sheet","@gorhom/bottom-sheet","@gorhom/portal"];return{define:{"process.env.NODE_ENV":JSON.stringify(n),"process.env.TEST_NATIVE_PLATFORM":JSON.stringify(""),"process.env.TAMAGUI_TARGET":JSON.stringify("native"),global:"globalThis"},optimizeDeps:{include:["react","react-dom","react-dom/client","react-reconciler","react-reconciler/constants","react/jsx-runtime","react/jsx-dev-runtime","@react-native/normalize-color","invariant","nanoid/non-secure"],exclude:m,rolldownOptions:{resolve:{conditionNames:["react-native","import","require"],mainFields:["react-native","module","jsnext:main","jsnext"],extensions:[".ios.tsx",".ios.ts",".ios.jsx",".ios.js",".native.tsx",".native.ts",".native.jsx",".native.js",".native.mjs",".mjs",".js",".mts",".ts",".jsx",".tsx",".json"]},plugins:[{name:"sootsim-stub-images",resolveId(o,c){if(/\.(jpg|png|gif)$/.test(o)&&c?.includes("node_modules"))return{id:"\0stub-image"}},load(o){if(o==="\0stub-image")return'export default ""'}}]}},build:{target:"esnext"},oxc:{target:"esnext"},server:{fs:{allow:s}},resolve:{dedupe:l,conditions:["react-native","import","require"],mainFields:["react-native","module","jsnext:main","jsnext"],extensions:[".ios.tsx",".ios.ts",".ios.jsx",".ios.js",".native.tsx",".native.ts",".native.jsx",".native.js",".native.mjs",".mjs",".js",".mts",".ts",".jsx",".tsx",".json"],alias:[...Object.entries(P).map(([o,c])=>({find:o,replacement:c})),{find:/^react-native\/Libraries\/.*/,replacement:w},{find:"react-native",replacement:v},{find:"react-native-web",replacement:v},...r]}}}}}function B(){let e=!1,t=0,i=0;return{name:"sootsim-react-bridge",enforce:"pre",configResolved(n){e=n.command==="build",console.log("[react-bridge-plugin] isBuild:",e)},async resolveId(n,s,r){if(n!=="react"||!s)return null;if((s.includes("/compat/")||s.includes("/sootsim/src/")||s.includes("/sootsim-engine/src/"))&&(i++,i<=5)){let m=r?.scan||r?.ssr||e;m||console.log("[react-bridge-plugin] react import from:",s.split("/").slice(-3).join("/"),"skip:",m)}return r?.scan||r?.ssr||e||!(s.includes("/sootsim/src/")||s.includes("/sootsim-engine/src/")||s.includes("/compat/src/")||s.includes("react-reconciler"))||s.includes("react-bridge")||s.includes("stub-registry")?null:(t++,t<=10?console.log("[react-bridge-plugin] REDIRECTING:",s.split("/").slice(-3).join("/")):t===11&&console.log("[react-bridge-plugin] ... more redirects"),{id:R,external:!1})}}}function se(){return{name:"sootsim-worker-react-bridge",enforce:"pre",async resolveId(e,t,i){return e!=="react"||!t||i?.scan||i?.ssr||!(t.includes("/sootsim/src/")||t.includes("/sootsim-engine/src/")||t.includes("/compat/src/")||t.includes("react-reconciler"))||t.includes("react-bridge")||t.includes("stub-registry")?null:{id:R,external:!1}}}}function C(){return{name:"sootsim-fix-jsx-runtime",enforce:"pre",async resolveId(e,t,i){if(e==="react/jsx-runtime"||e==="react/jsx-dev-runtime"){let n=await this.resolve(e,t,{...i,skipSelf:!0});return n?(e==="react/jsx-runtime"?"\0sootsim:jsx-runtime:":"\0sootsim:jsx-dev-runtime:")+n.id:null}return null},load(e){if(e.startsWith("\0sootsim:jsx-runtime:")){let t=e.slice(21);return[`import * as _mod from ${JSON.stringify(t)};`,"const _m = _mod.default || _mod;","export const jsx = _m.jsx || _mod.jsx;","export const jsxs = _m.jsxs || _mod.jsxs;","export const Fragment = _m.Fragment || _mod.Fragment;"].join(`
3
- `)}if(e.startsWith("\0sootsim:jsx-dev-runtime:")){let t=e.slice(25);return[`import * as _mod from ${JSON.stringify(t)};`,'import * as _runtime from "react/jsx-runtime";',"const _m = _mod.default || _mod;","export const jsxDEV = _m.jsxDEV || _mod.jsxDEV || _m.jsx || _mod.jsx || _runtime.jsx;","export const Fragment = _m.Fragment || _mod.Fragment || _runtime.Fragment;"].join(`
4
- `)}return null}}}function D(){return{name:"sootsim-node-modules-jsx",enforce:"pre",async transform(e,t){if(!t.includes("/node_modules/")||!t.endsWith(".js")||!e.includes("<")||!/[=(\s,]<[A-Z]/.test(e)&&!/<\/[A-Z]/.test(e))return null;let n=!e.includes("import React")&&!e.includes("import * as React")?`import React from 'react';
5
- ${e}`:e,s=await j(n,t,{loader:"jsx"});return{code:s.code,map:s.map}}}}function V(e){let t=[".ios.tsx",".ios.ts",".ios.jsx",".ios.js",".native.tsx",".native.ts",".native.jsx",".native.js",".native.mjs"],i=[...t,".tsx",".ts",".jsx",".js",".mjs",".cjs"];function n(s){for(let r of t){let l=s+r;try{if(g.existsSync(l))return l}catch{}}for(let r of t){let l=a.join(s,"index"+r);try{if(g.existsSync(l))return l}catch{}}return null}return{name:"sootsim-metro-native-resolve",enforce:"pre",resolveId(s,r){if(!r||!r.includes("node_modules")&&!e(r)||!s.startsWith(".")||t.some(o=>s.endsWith(o)))return null;let l=i.some(o=>s.endsWith(o)),m=a.dirname(r);if(l){let o=a.resolve(m,s);for(let c of i)if(s.endsWith(c)){let f=n(o.slice(0,-c.length));if(f)return f;break}}else{let o=n(a.resolve(m,s));if(o)return o}return null}}}function G(e){return{name:"sootsim-react-native-require",enforce:"pre",transform(t,i){if(!i.includes("node_modules")&&!e(i)||!t.includes("react-native"))return null;let n=!1,s="",r=t,l=0;(r.includes('require("react-native")')||r.includes("require('react-native')"))&&(s+=`import * as __soot_rn_shim__ from ${JSON.stringify(v)};
6
- `,r=r.replace(/require\(["']react-native["']\)/g,"__soot_rn_shim__"),n=!0);let m=/require\(["'](react-native\/Libraries\/[^"']+)["']\)/g,o=[],c=new Map,f;for(m.lastIndex=0;(f=m.exec(r))!==null;){let u=f[1];if(!c.has(u)){let p=`__soot_rn_lib_${l++}__`,S=P[u]||w;s+=`import * as ${p} from ${JSON.stringify(S)};
7
- `,c.set(u,p)}o.push({full:f[0],path:u,varName:c.get(u)})}if(o.length>0){for(let u of o)r=r.replace(u.full,u.varName);n=!0}return n?{code:s+r,map:null}:null}}}function z(e,t,i){if(!e)return{name:"sootsim-external-app-resolve"};let n=a.join(e,"_virtual_.js");return{name:"sootsim-external-app-resolve",enforce:"pre",async resolveId(s,r,l){return!r||s.startsWith(".")||s.startsWith("/")||s.startsWith("\0")||i.has(s)||i.has(k(s))||!t(r)&&!r.includes(a.join(e,"node_modules"))||s==="react-native"||s.startsWith("react-native/")?null:await this.resolve(s,n,{...l,skipSelf:!0})||null}}}function U(e){return{name:"sootsim-external-app-transform",enforce:"pre",async transform(t,i){if(!e(i))return null;let n=t,s="",r=0,l=!1;if(t.includes("require(")){let m=/require\(["']([^"']+)["']\)/g,o=[],c=new Map,f;for(m.lastIndex=0;(f=m.exec(n))!==null;){let u=f[1];if(!(u==="react-native"||u.startsWith("react-native/"))){if(!c.has(u)){let p=`__soot_cjs_${r++}__`;s+=`import * as ${p} from ${JSON.stringify(u)};
8
- `,c.set(u,p)}o.push({full:f[0],varName:c.get(u)})}}if(o.length>0){for(let u of o)n=n.replace(u.full,u.varName);l=!0}}if(s&&(n=s+n),i.endsWith(".js")&&n.includes("<")){let o=!n.includes("import React")&&!n.includes("import * as React")?`import React from 'react';
9
- ${n}`:n,c=await j(o,i,{loader:"jsx"});return{code:c.code,map:c.map}}return l?{code:n,map:null}:null}}}function Z(e){let t=new Map;function i(n){let s=t.get(n);if(s!==void 0)return s;let r=b(n,x);return!r&&e&&(r=b(n,e)),t.set(n,r),r}return{name:"sootsim-stub-missing-native-deps",resolveId(n){if(!L(n))return null;let s=k(n);return i(s)?null:(console.log(`[sootsim] auto-stubbing missing native dep: ${n}`),O)}}}function H(e){return{name:"sootsim-stub-images",enforce:"pre",resolveId(t,i){return(t.endsWith(".jpg")||t.endsWith(".png")||t.endsWith(".gif"))&&(i?.includes("node_modules")||i&&e(i))?"\0stub-image":null},load(t){return t==="\0stub-image"?'export default ""':null}}}export{C as fixJsxRuntimeExports,te as sootsim,se as workerReactBridgePlugin,M as wsBridgePlugin};