sootsim 0.1.65 → 0.1.67

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 (147) hide show
  1. package/dist-cli/bin.js +3 -3
  2. package/dist-cli/chunks/{agent-44BHCTZA.js → agent-Z45KTLTP.js} +2 -2
  3. package/dist-cli/chunks/{agent-wrapper-BEP5WHZR.js → agent-wrapper-EEKVEJOC.js} +2 -2
  4. package/dist-cli/chunks/{assert-SKDEGGRD.js → assert-MDYZKXZ6.js} +2 -2
  5. package/dist-cli/chunks/auto-bootstrap-YZ6C6VYO.js +2 -0
  6. package/dist-cli/chunks/beta-WXH75XUE.js +2 -0
  7. package/dist-cli/chunks/chunk-2HKCZLY7.js +372 -0
  8. package/dist-cli/chunks/{chunk-46OKLW3Z.js → chunk-2I535GZU.js} +1 -1
  9. package/dist-cli/chunks/{chunk-L2SDB6G5.js → chunk-3KVVIZ6M.js} +2 -2
  10. package/dist-cli/chunks/chunk-42NMKOVE.js +3 -0
  11. package/dist-cli/chunks/{chunk-S46M5JZP.js → chunk-4H6LBA4A.js} +1 -1
  12. package/dist-cli/chunks/{chunk-M4HI2THN.js → chunk-4QSGVTBE.js} +2 -2
  13. package/dist-cli/chunks/{chunk-MLPLYQGK.js → chunk-52RZ6TSZ.js} +3 -3
  14. package/dist-cli/chunks/{chunk-ELZHT3LB.js → chunk-5WRBVYML.js} +2 -2
  15. package/dist-cli/chunks/{chunk-NJ2ACA36.js → chunk-67566AJS.js} +2 -2
  16. package/dist-cli/chunks/chunk-6O64VUC6.js +2 -0
  17. package/dist-cli/chunks/chunk-7PQHQOCI.js +186 -0
  18. package/dist-cli/chunks/{chunk-5ZB5ZN5Z.js → chunk-AQ7YR4MO.js} +14 -14
  19. package/dist-cli/chunks/{chunk-FRANP4PR.js → chunk-ATRGAFJD.js} +1 -1
  20. package/dist-cli/chunks/{chunk-OTWUZRFZ.js → chunk-BCVVANS5.js} +2 -2
  21. package/dist-cli/chunks/chunk-CUNQTNH4.js +35 -0
  22. package/dist-cli/chunks/{chunk-JC5L5ZJX.js → chunk-DOWYG3HG.js} +2 -2
  23. package/dist-cli/chunks/{chunk-MSBMGQJE.js → chunk-EJTXYZEF.js} +128 -36
  24. package/dist-cli/chunks/{chunk-P2C3JX3D.js → chunk-F23Y4HMT.js} +1 -1
  25. package/dist-cli/chunks/{chunk-GHFA4O5B.js → chunk-F4RC4AOY.js} +1 -1
  26. package/dist-cli/chunks/chunk-GFVC6EEB.js +1 -0
  27. package/dist-cli/chunks/{chunk-TWKWXGN5.js → chunk-HW5YHSIE.js} +1 -1
  28. package/dist-cli/chunks/{chunk-VGQXIMEX.js → chunk-IB4BRND4.js} +1 -1
  29. package/dist-cli/chunks/{chunk-X75XFMI4.js → chunk-IBWN764Q.js} +2 -2
  30. package/dist-cli/chunks/chunk-JE4JBSNJ.js +119 -0
  31. package/dist-cli/chunks/{chunk-O56DABNK.js → chunk-K4PKR2EO.js} +2 -2
  32. package/dist-cli/chunks/chunk-KBJ2WHII.js +73 -0
  33. package/dist-cli/chunks/chunk-KR36VJSN.js +1 -0
  34. package/dist-cli/chunks/{chunk-UHYZO26V.js → chunk-LQ5BA5JU.js} +2 -2
  35. package/dist-cli/chunks/chunk-MIAAZSSE.js +12 -0
  36. package/dist-cli/chunks/{chunk-PTUCXL7Y.js → chunk-N4JRDJMN.js} +3 -3
  37. package/dist-cli/chunks/{chunk-U5EQ5RWL.js → chunk-NYR3Z44O.js} +2 -2
  38. package/dist-cli/chunks/{chunk-2EBGZC4A.js → chunk-O624O5MO.js} +2 -2
  39. package/dist-cli/chunks/{chunk-HNXQ2JP6.js → chunk-SH6HMUCY.js} +2 -2
  40. package/dist-cli/chunks/chunk-SVFSMAZN.js +2 -0
  41. package/dist-cli/chunks/chunk-TEMZK6Q2.js +1 -0
  42. package/dist-cli/chunks/{chunk-KKOOZPQP.js → chunk-TLTS533S.js} +1 -1
  43. package/dist-cli/chunks/chunk-U47AHSGE.js +23 -0
  44. package/dist-cli/chunks/{chunk-6GZS5TCL.js → chunk-XCRUPKWB.js} +2 -2
  45. package/dist-cli/chunks/{chunk-5MMRIGC3.js → chunk-YGQV3CUV.js} +2 -2
  46. package/dist-cli/chunks/{chunk-GZCZ4GCS.js → chunk-ZJJKYJMH.js} +2 -2
  47. package/dist-cli/chunks/{chunk-FU33BTPL.js → chunk-ZO7SGE65.js} +1 -1
  48. package/dist-cli/chunks/{chunk-UH4H5HXN.js → chunk-ZP2HMCKS.js} +1 -1
  49. package/dist-cli/chunks/cli-version-5LGQQE5P.js +2 -0
  50. package/dist-cli/chunks/{compat-MDZTV64M.js → compat-5MMZ4R3Q.js} +3 -3
  51. package/dist-cli/chunks/{config-DRGNQOYL.js → config-YWN44WJQ.js} +2 -2
  52. package/dist-cli/chunks/control-LUCFDAZI.js +2 -0
  53. package/dist-cli/chunks/{cpu-profile-JQAGHLDS.js → cpu-profile-VFGUVXU5.js} +2 -2
  54. package/dist-cli/chunks/{daemon-3RT7EABW.js → daemon-ONX5ZBXF.js} +2 -2
  55. package/dist-cli/chunks/{debug-CYINYNWV.js → debug-CPOG32OT.js} +5 -5
  56. package/dist-cli/chunks/demo-app-registry-LWT6JLU6.js +2 -0
  57. package/dist-cli/chunks/{detox-DUABQYC6.js → detox-J6NGSZR5.js} +2 -2
  58. package/dist-cli/chunks/{device-KSSSQOQC.js → device-QYOQXI2P.js} +2 -2
  59. package/dist-cli/chunks/{diagnose-NUDACXIR.js → diagnose-MMW4LJLA.js} +2 -2
  60. package/dist-cli/chunks/drivers-NRWUJJCD.js +2 -0
  61. package/dist-cli/chunks/{electron-LC3Y74FA.js → electron-IK3HIZY4.js} +3 -3
  62. package/dist-cli/chunks/flow-2MB7HINM.js +2 -0
  63. package/dist-cli/chunks/{hints-EOARRVNB.js → hints-KZOD56U5.js} +2 -2
  64. package/dist-cli/chunks/{home-paths-HOBR3K3B.js → home-paths-R2YZZTUM.js} +2 -2
  65. package/dist-cli/chunks/inspect-ZARO3HQ4.js +970 -0
  66. package/dist-cli/chunks/install-O67RTYCX.js +2 -0
  67. package/dist-cli/chunks/{install-desktop-W2OOILJG.js → install-desktop-4GH5I7SC.js} +3 -3
  68. package/dist-cli/chunks/{keys-RR43E2O3.js → keys-AXEFZUUI.js} +2 -2
  69. package/dist-cli/chunks/{launch-GAPSA5HQ.js → launch-QLE56ON7.js} +3 -3
  70. package/dist-cli/chunks/{login-IADXSJEW.js → login-J45NQQM6.js} +4 -4
  71. package/dist-cli/chunks/{logout-AT33MGA7.js → logout-ZE5OUBB7.js} +2 -2
  72. package/dist-cli/chunks/{maestro-G36ATKHQ.js → maestro-FS6CAKDD.js} +2 -2
  73. package/dist-cli/chunks/{preview-BJUMMDBL.js → preview-TSU5IXVR.js} +2 -2
  74. package/dist-cli/chunks/{profile-CEQNS53P.js → profile-ZBJ4TYWA.js} +2 -2
  75. package/dist-cli/chunks/{react-BPEDK3TN.js → react-TGZLTTYN.js} +2 -2
  76. package/dist-cli/chunks/record-JSBFUNOD.js +54 -0
  77. package/dist-cli/chunks/runtime-DGDQ7NVY.js +2 -0
  78. package/dist-cli/chunks/{runtime-delivery-IYK2HNPG.js → runtime-delivery-36UMAXOV.js} +2 -2
  79. package/dist-cli/chunks/screenshot-3CHMALUJ.js +33 -0
  80. package/dist-cli/chunks/{screenshot-mode-VUBCO7T2.js → screenshot-mode-C547UT5H.js} +2 -2
  81. package/dist-cli/chunks/{screenshots-JENURQSV.js → screenshots-XCNGSNYG.js} +2 -2
  82. package/dist-cli/chunks/{server-PVAO3OA2.js → server-W37X3IW7.js} +6 -6
  83. package/dist-cli/chunks/setup-repo-DETS5ZJT.js +2 -0
  84. package/dist-cli/chunks/{skills-4JI327YS.js → skills-OXZLDNDW.js} +2 -2
  85. package/dist-cli/chunks/{start-7YB4I5V5.js → start-N6ZQYUBP.js} +4 -4
  86. package/dist-cli/chunks/store-QOXJKJUZ.js +2 -0
  87. package/dist-cli/chunks/telemetry-74LX7WG4.js +2 -0
  88. package/dist-cli/chunks/{test-5C6NYF62.js → test-DWZ7PRSW.js} +3 -3
  89. package/dist-cli/chunks/{three-mode-C6TE63VR.js → three-mode-P2R7KA62.js} +2 -2
  90. package/dist-cli/chunks/{timeline-NFYS6IPO.js → timeline-VPES3YCN.js} +3 -3
  91. package/dist-cli/chunks/{upgrade-KROEHYG2.js → upgrade-NYQLA4UJ.js} +2 -2
  92. package/dist-cli/chunks/upload-YOMMYESV.js +2 -0
  93. package/dist-cli/chunks/{web-UTOZ7MRF.js → web-OTALJKK5.js} +2 -2
  94. package/dist-cli/chunks/what-happened-GZ73H6KF.js +22 -0
  95. package/dist-cli/chunks/{whoami-O6XVOJBF.js → whoami-UDLQK7IY.js} +2 -2
  96. package/dist-lib/agent-daemon-client.cjs +1 -1
  97. package/dist-lib/agent-events.cjs +1 -1
  98. package/dist-lib/agent-sessions.cjs +1 -1
  99. package/dist-lib/attached-projects.cjs +1 -1
  100. package/dist-lib/auth/shared-session.cjs +1 -1
  101. package/dist-lib/backend-origin.cjs +1 -1
  102. package/dist-lib/bridge-constants.cjs +1 -1
  103. package/dist-lib/cli-constants.cjs +1 -1
  104. package/dist-lib/config.cjs +1 -1
  105. package/dist-lib/dev-bundle-resolution.cjs +9 -2
  106. package/dist-lib/home-paths.cjs +1 -1
  107. package/dist-lib/host/bridge-host.cjs +23 -7
  108. package/dist-lib/host/fetch-proxy-handler.cjs +1 -1
  109. package/dist-lib/host/fetch-proxy-overrides.cjs +1 -1
  110. package/dist-lib/index.cjs +1 -1
  111. package/dist-lib/metro.cjs +1 -1
  112. package/dist-lib/profiles.cjs +1 -1
  113. package/dist-lib/render-mode.cjs +1 -1
  114. package/dist-lib/vite-base.cjs +23 -7
  115. package/dist-lib/vite.cjs +1 -1
  116. package/package.json +1 -1
  117. package/dist-cli/chunks/auto-bootstrap-SMQMN7WS.js +0 -2
  118. package/dist-cli/chunks/beta-DEQZ2J7G.js +0 -2
  119. package/dist-cli/chunks/chunk-3HC3F7CO.js +0 -3
  120. package/dist-cli/chunks/chunk-5MRRLWSD.js +0 -1
  121. package/dist-cli/chunks/chunk-B32KMMQV.js +0 -2
  122. package/dist-cli/chunks/chunk-CBRKFAOD.js +0 -1
  123. package/dist-cli/chunks/chunk-DLSOZVYM.js +0 -1
  124. package/dist-cli/chunks/chunk-E6CRWOWO.js +0 -2
  125. package/dist-cli/chunks/chunk-FPOQPEGK.js +0 -11
  126. package/dist-cli/chunks/chunk-GHTXP5R7.js +0 -348
  127. package/dist-cli/chunks/chunk-NTEZEQSB.js +0 -100
  128. package/dist-cli/chunks/chunk-R2QEABGZ.js +0 -119
  129. package/dist-cli/chunks/chunk-TRHZBTPF.js +0 -34
  130. package/dist-cli/chunks/chunk-UXFF7AEE.js +0 -73
  131. package/dist-cli/chunks/chunk-X2IIAIAP.js +0 -3
  132. package/dist-cli/chunks/chunk-ZW3HKMLQ.js +0 -17
  133. package/dist-cli/chunks/cli-version-HH7DAYBK.js +0 -2
  134. package/dist-cli/chunks/control-NCUWAXPH.js +0 -2
  135. package/dist-cli/chunks/demo-app-registry-JGIZ7NN7.js +0 -2
  136. package/dist-cli/chunks/drivers-CKGMOVM4.js +0 -2
  137. package/dist-cli/chunks/flow-3AHNNHF7.js +0 -2
  138. package/dist-cli/chunks/inspect-QCZVODE6.js +0 -1025
  139. package/dist-cli/chunks/install-747G2O4S.js +0 -2
  140. package/dist-cli/chunks/record-73CGQ3EP.js +0 -50
  141. package/dist-cli/chunks/runtime-QR5IHGKJ.js +0 -2
  142. package/dist-cli/chunks/screenshot-GUM2U2PN.js +0 -28
  143. package/dist-cli/chunks/setup-repo-ECVECJZQ.js +0 -2
  144. package/dist-cli/chunks/store-2MXXL4QE.js +0 -2
  145. package/dist-cli/chunks/telemetry-GIA7FIKW.js +0 -2
  146. package/dist-cli/chunks/upload-OS3J5ZOS.js +0 -2
  147. package/dist-cli/chunks/what-happened-FCP2U43H.js +0 -15
@@ -0,0 +1,22 @@
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
+ import{A as y,L as x,y as F,z as T}from"./chunk-AQ7YR4MO.js";import{c as S,e as N}from"./chunk-U47AHSGE.js";import{b as k}from"./chunk-IB4BRND4.js";import"./chunk-6O64VUC6.js";import"./chunk-HW5YHSIE.js";import"./chunk-TLTS533S.js";function _(n){let e=[];for(let t=0;t<n.length;t++)if(n[t]==="--since"&&t+1<n.length){e.push(t,t+1);let s=n[t+1].trim(),l=/^(\d+(?:\.\d+)?)(ms|s|m)?$/.exec(s);if(l){let a=Number(l[1]),m=l[2]??"ms",$=m==="s"?a*1e3:m==="m"?a*6e4:a;return{since:Date.now()-$,consumed:e}}let o=Number(s);if(Number.isFinite(o)&&o>1e12)return{since:o,consumed:e}}return{consumed:e}}function K(n){let e=[];for(let t=0;t<n.length;t++)if(n[t]==="--kinds"&&t+1<n.length)return e.push(t,t+1),{kinds:n[t+1].split(",").map(s=>s.trim()).filter(Boolean),consumed:e};return{consumed:e}}function O(n){let e=[];for(let t=0;t<n.length;t++)if(n[t]==="--limit"&&t+1<n.length){e.push(t,t+1);let s=Number(n[t+1]);if(Number.isFinite(s)&&s>0)return{limit:s,consumed:e}}return{consumed:e}}function j(n,e){if(e===null)return new Date(n).toLocaleTimeString();let t=(n-e)/1e3;return`${t>=0?"+":""}${t.toFixed(2)}s`}function B(n,e){switch(n){case"app-launch":return e.phase==="launch"?`launch ${e.appName??e.toAppId??""}`:`dismiss ${e.appName??e.fromAppId??""} \u2192 ${e.toAppId??""}`;case"toast":return`"${e.text??""}"${e.durationMs?` (${e.durationMs}ms)`:""}`;case"keyboard":return`${e.phase??"?"}${e.heightPx?` h=${e.heightPx}`:""}${e.mode?` ${e.mode}`:""}`;case"screen":return`${e.phase??"?"} ${e.name??e.activeName??""}`;case"route":return`${e.phase??"?"} ${e.path??e.pathname??""}`;case"alert":case"actionsheet":case"picker":return`${e.phase??"?"} ${e.title??e.message??""}`;case"notification":return`${e.title??""}${e.body?` \u2014 ${e.body}`:""}`;case"fetch":return`${e.method??"GET"} ${e.url??""}${e.status?` -> ${e.status}`:""}`;case"console":return`${e.level??"log"}: ${(e.message??"").toString().slice(0,120)}`;case"shell":return`${e.event??e.type??e.phase??""}`;case"scroll":return`${e.phase??"?"} ${e.target??""}`;case"gesture":return`${e.phase??"?"} ${e.type??""}`;case"text-input":return`${e.phase??"?"}${e.value!==void 0?` "${String(e.value).slice(0,40)}"`:""}`;case"layout":return`${e.kind??"?"} ${e.testID??e.type??""}${e.skipped?` skipped:${e.reason??"unknown"}`:""}`;case"react-commit":{let t=e.slowest;return`${e.fiberCount??"?"} fibers ${e.durationMs??"?"}ms${t?.displayName?` \xB7 ${t.displayName} ${t.durationMs??"?"}ms`:""}`}case"reanimated":case"animation":return`${e.kind??""} ${e.target??""}${e.durationMs?` ${e.durationMs}ms`:""}`}}function A(n,e){let t=j(n.t,e).padStart(8),s=n.context.padEnd(6),l=`[${n.kind}]`.padEnd(15),o="",a=n.data;return a&&typeof a=="object"&&(o=B(n.kind,a)),` ${t} ${s} ${l} ${o}`}function M(n){let e=[],t={label:"initial state",events:[],startedAt:n[0]?.t??null};e.push(t);for(let s of n)if(t.events.push(s),s.kind==="screen"||s.kind==="route"){let l=s.data,o=l?.phase;if(!o||o==="enter"||o==="appear"||o==="active"){let a=l?.name||l?.activeName||l?.path||l?.pathname||s.kind;e.length===1&&t.events.length===1?t.label=`${s.kind}: ${a}`:(t={label:`${s.kind}: ${a}`,events:[],startedAt:s.t},e.push(t))}}return e}async function J(n,e){let t=S(n,{port:e.port,stripBooleanFlags:["--summary","--all","--json","--no-advance","--help","-h","--flow","--noisy"],stripValueFlags:["--since","--kinds","--limit"]});(n.includes("--help")||n.includes("-h"))&&(console.log(`
3
+ sootsim what-happened \u2014 show recent events from the semantic timeline
4
+
5
+ usage:
6
+ sootsim what-happened # since last CLI call
7
+ sootsim what-happened --summary # one-line counts
8
+ sootsim what-happened --all # full ring (ignore cursor)
9
+ sootsim what-happened --since 5s # absolute window
10
+ sootsim what-happened --kinds toast,fetch
11
+ sootsim what-happened --noisy # include react-commit/layout/scroll
12
+ sootsim what-happened --limit 50
13
+ sootsim what-happened --json
14
+ sootsim what-happened --no-advance # don't advance cursor after read
15
+
16
+ note: by default react-commit, layout, and scroll events are hidden as
17
+ noise \u2014 pass --noisy or include them in --kinds to see them.
18
+ `),process.exit(0));let s=n.includes("--summary"),l=n.includes("--flow"),o=n.includes("--all"),a=n.includes("--json"),m=n.includes("--no-advance"),$=n.includes("--noisy"),{since:c}=_(n),{kinds:p}=K(n),{limit:E}=O(n),g=k(),w={limit:E??200,...p&&p.length?{kinds:p}:{},...c!==void 0?{since:c}:o?{}:{sinceCursor:g}},d=N(t);try{if(s){let r=await F(d,w);if(a)console.log(JSON.stringify(r));else{let u=o?"all time":c!==void 0?`last ${((Date.now()-c)/1e3).toFixed(1)}s`:"since last call";console.log(` ${u}: ${x(r)}`)}!m&&!o&&r.lastAt&&await y(d,g,r.lastAt);return}let i=await T(d,w),b=new Set(["react-commit","layout","scroll"]),I=Array.isArray(p)&&p.some(r=>b.has(r)),C=!$&&!I,h=0;if(C){let r=i.events.filter(u=>b.has(u.kind)?(h+=1,!1):!0);i.events=r}if(a)console.log(JSON.stringify(l?M(i.events):i,null,2));else{if(i.events.length===0)h>0?console.log(` no non-noise events (${h} react-commit/layout/scroll hidden)`):console.log(o?" no events recorded":c!==void 0?" no events in window":" no new events since last call");else if(l){let r=i.events[0]?.t??null,u=M(i.events),v=o?`\u2500\u2500\u2500 ${i.events.length} event(s) total \u2014 flow view \u2500\u2500\u2500`:c!==void 0?`\u2500\u2500\u2500 ${i.events.length} event(s) in last ${((Date.now()-c)/1e3).toFixed(1)}s \u2014 flow view \u2500\u2500\u2500`:`\u2500\u2500\u2500 ${i.events.length} event(s) since last call \u2014 flow view \u2500\u2500\u2500`;console.log(` ${v}`);for(let f of u){console.log(`
19
+ \u2500\u2500 ${f.label} (${f.events.length} event${f.events.length===1?"":"s"}) \u2500\u2500`);for(let D of f.events)console.log(A(D,r))}}else{let r=i.events[0]?.t??null,u=o?`\u2500\u2500\u2500 ${i.events.length} event(s) total \u2500\u2500\u2500`:c!==void 0?`\u2500\u2500\u2500 ${i.events.length} event(s) in last ${((Date.now()-c)/1e3).toFixed(1)}s \u2500\u2500\u2500`:`\u2500\u2500\u2500 ${i.events.length} event(s) since last call \u2500\u2500\u2500`;console.log(` ${u}`);for(let v of i.events)console.log(A(v,r))}h>0&&i.events.length>0&&process.stderr.write(`
20
+ ${h} high-frequency event(s) hidden (react-commit/layout/scroll)
21
+ rerun with --noisy or --kinds react-commit,layout,scroll to include them
22
+ `)}!m&&!o&&i.watermark>0&&await y(d,g,i.watermark)}finally{d.close()}}export{J as runWhatHappened};
@@ -1,2 +1,2 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{d as i}from"./chunk-UHYZO26V.js";import{d as n}from"./chunk-2EBGZC4A.js";import"./chunk-TWKWXGN5.js";import"./chunk-KKOOZPQP.js";async function u(){let o=i();if(o||(console.log(" not signed in"),console.log(" set SOOTSIM_API_KEY=sk_sootsim_\u2026 or run `sootsim login`"),process.exit(1)),o.kind==="api-key"){let r=`${o.secret.slice(0,14)}\u2026`,t=o.source==="env"?"SOOTSIM_API_KEY env var":"saved key (~/.config/sootsim/credentials.json)";console.log(` api key ${r}`),console.log(` source: ${t}`);return}if(o.kind==="github"){console.log(` github token (${o.source})`),console.log(` repo: ${o.repoId}`);return}let e=await n(),s=e?.user;console.log(` ${s?.email||s?.name||s?.id||"signed in"}`),console.log(` origin: ${e?.origin??o.origin}`),e?.updatedAt&&console.log(` updated: ${e.updatedAt}`)}export{u as runWhoami};
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
+ import{d as i}from"./chunk-LQ5BA5JU.js";import{d as n}from"./chunk-O624O5MO.js";import"./chunk-HW5YHSIE.js";import"./chunk-TLTS533S.js";async function u(){let o=i();if(o||(console.log(" not signed in"),console.log(" set SOOTSIM_API_KEY=sk_sootsim_\u2026 or run `sootsim login`"),process.exit(1)),o.kind==="api-key"){let r=`${o.secret.slice(0,14)}\u2026`,t=o.source==="env"?"SOOTSIM_API_KEY env var":"saved key (~/.config/sootsim/credentials.json)";console.log(` api key ${r}`),console.log(` source: ${t}`);return}if(o.kind==="github"){console.log(` github token (${o.source})`),console.log(` repo: ${o.repoId}`);return}let e=await n(),s=e?.user;console.log(` ${s?.email||s?.name||s?.id||"signed in"}`),console.log(` origin: ${e?.origin??o.origin}`),e?.updatedAt&&console.log(` updated: ${e.updatedAt}`)}export{u as runWhoami};
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __defProp = Object.defineProperty;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __defProp = Object.defineProperty;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __defProp = Object.defineProperty;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __defProp = Object.defineProperty;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __defProp = Object.defineProperty;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __defProp = Object.defineProperty;
@@ -156,7 +156,14 @@ async function resolveConnectionInput(input) {
156
156
  const port2 = parseInt(trimmed, 10);
157
157
  const resolved = await probePortBundle(port2);
158
158
  if (resolved) return resolved;
159
- throw buildUnresolvedTargetError(`localhost:${port2}`);
159
+ const SHIFT_WINDOW = 3;
160
+ for (let offset = 1; offset <= SHIFT_WINDOW; offset++) {
161
+ const shifted = await probePortBundle(port2 + offset);
162
+ if (shifted) return shifted;
163
+ }
164
+ throw buildUnresolvedTargetError(
165
+ `localhost:${port2} (also scanned +1..+${SHIFT_WINDOW})`
166
+ );
160
167
  }
161
168
  const bundleUrl = trimmed.startsWith("http") ? trimmed : `http://${trimmed}`;
162
169
  let parsed;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
@@ -3572,7 +3572,7 @@ var SootSimBridgeHost = class _SootSimBridgeHost {
3572
3572
  }
3573
3573
  if (msg.type === "bridge:user-focus-state" && sim) {
3574
3574
  const focusState = msg;
3575
- this.updateUserFocusLease(sim, focusState.focused === true);
3575
+ this.updateUserFocusLease(sim, focusState);
3576
3576
  return;
3577
3577
  }
3578
3578
  if (msg.type === "bridge:user-interact" && sim) {
@@ -4188,6 +4188,8 @@ var SootSimBridgeHost = class _SootSimBridgeHost {
4188
4188
  let rel = url.pathname;
4189
4189
  if (rel === "/runtime" || rel === "/runtime/") rel = "/";
4190
4190
  else if (rel.startsWith("/runtime/")) rel = rel.slice("/runtime".length);
4191
+ else if (rel === "/sootsim" || rel === "/sootsim/") rel = "/";
4192
+ else if (rel.startsWith("/sootsim/")) rel = rel.slice("/sootsim".length);
4191
4193
  if (rel === "" || rel === "/") rel = "/index.html";
4192
4194
  if (rel.includes("\0")) {
4193
4195
  res.writeHead(400);
@@ -4557,6 +4559,9 @@ var SootSimBridgeHost = class _SootSimBridgeHost {
4557
4559
  lockedByKind: lease ? lease.kind : void 0,
4558
4560
  lockExpiresAt: lease ? lease.expiresAt : void 0,
4559
4561
  userFocused: sim.userFocused || void 0,
4562
+ userVisible: sim.userVisible,
4563
+ visibilityState: sim.visibilityState,
4564
+ documentFocused: sim.documentFocused,
4560
4565
  kind: sim.kind,
4561
4566
  meta: sim.meta
4562
4567
  };
@@ -4621,10 +4626,18 @@ var SootSimBridgeHost = class _SootSimBridgeHost {
4621
4626
  // locked out agent inspect calls for 15s — the opposite of what you want
4622
4627
  // when debugging something the user is actively looking at. use
4623
4628
  // updateUserActivity() to lock on real interaction instead.
4624
- updateUserFocusLease(sim, focused) {
4625
- const next = focused;
4626
- if (sim.userFocused === next) return;
4627
- sim.userFocused = next;
4629
+ updateUserFocusLease(sim, focusState) {
4630
+ const nextFocused = focusState.focused === true;
4631
+ const nextVisible = typeof focusState.visible === "boolean" ? focusState.visible : void 0;
4632
+ const nextVisibilityState = typeof focusState.visibilityState === "string" ? focusState.visibilityState : void 0;
4633
+ const nextDocumentFocused = typeof focusState.documentFocused === "boolean" ? focusState.documentFocused : void 0;
4634
+ if (sim.userFocused === nextFocused && sim.userVisible === nextVisible && sim.visibilityState === nextVisibilityState && sim.documentFocused === nextDocumentFocused) {
4635
+ return;
4636
+ }
4637
+ sim.userFocused = nextFocused;
4638
+ sim.userVisible = nextVisible;
4639
+ sim.visibilityState = nextVisibilityState;
4640
+ sim.documentFocused = nextDocumentFocused;
4628
4641
  this.broadcastSimClientStates();
4629
4642
  }
4630
4643
  // called when the sim reports a real user interaction (pointerdown,
@@ -4714,7 +4727,10 @@ var SootSimBridgeHost = class _SootSimBridgeHost {
4714
4727
  lockedBy: lease ? lease.cliLabel || lease.cliIdentityKey : void 0,
4715
4728
  lockedByKind: lease ? lease.kind : void 0,
4716
4729
  lockExpiresAt: lease ? lease.expiresAt : void 0,
4717
- userFocused: sim.userFocused || void 0
4730
+ userFocused: sim.userFocused || void 0,
4731
+ userVisible: sim.userVisible,
4732
+ visibilityState: sim.visibilityState,
4733
+ documentFocused: sim.documentFocused
4718
4734
  };
4719
4735
  sim.ws.send(JSON.stringify(message));
4720
4736
  }
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __defProp = Object.defineProperty;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __defProp = Object.defineProperty;
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
@@ -3682,7 +3682,7 @@ var init_bridge_host = __esm({
3682
3682
  }
3683
3683
  if (msg.type === "bridge:user-focus-state" && sim) {
3684
3684
  const focusState = msg;
3685
- this.updateUserFocusLease(sim, focusState.focused === true);
3685
+ this.updateUserFocusLease(sim, focusState);
3686
3686
  return;
3687
3687
  }
3688
3688
  if (msg.type === "bridge:user-interact" && sim) {
@@ -4298,6 +4298,8 @@ var init_bridge_host = __esm({
4298
4298
  let rel = url.pathname;
4299
4299
  if (rel === "/runtime" || rel === "/runtime/") rel = "/";
4300
4300
  else if (rel.startsWith("/runtime/")) rel = rel.slice("/runtime".length);
4301
+ else if (rel === "/sootsim" || rel === "/sootsim/") rel = "/";
4302
+ else if (rel.startsWith("/sootsim/")) rel = rel.slice("/sootsim".length);
4301
4303
  if (rel === "" || rel === "/") rel = "/index.html";
4302
4304
  if (rel.includes("\0")) {
4303
4305
  res.writeHead(400);
@@ -4667,6 +4669,9 @@ var init_bridge_host = __esm({
4667
4669
  lockedByKind: lease ? lease.kind : void 0,
4668
4670
  lockExpiresAt: lease ? lease.expiresAt : void 0,
4669
4671
  userFocused: sim.userFocused || void 0,
4672
+ userVisible: sim.userVisible,
4673
+ visibilityState: sim.visibilityState,
4674
+ documentFocused: sim.documentFocused,
4670
4675
  kind: sim.kind,
4671
4676
  meta: sim.meta
4672
4677
  };
@@ -4731,10 +4736,18 @@ var init_bridge_host = __esm({
4731
4736
  // locked out agent inspect calls for 15s — the opposite of what you want
4732
4737
  // when debugging something the user is actively looking at. use
4733
4738
  // updateUserActivity() to lock on real interaction instead.
4734
- updateUserFocusLease(sim, focused) {
4735
- const next = focused;
4736
- if (sim.userFocused === next) return;
4737
- sim.userFocused = next;
4739
+ updateUserFocusLease(sim, focusState) {
4740
+ const nextFocused = focusState.focused === true;
4741
+ const nextVisible = typeof focusState.visible === "boolean" ? focusState.visible : void 0;
4742
+ const nextVisibilityState = typeof focusState.visibilityState === "string" ? focusState.visibilityState : void 0;
4743
+ const nextDocumentFocused = typeof focusState.documentFocused === "boolean" ? focusState.documentFocused : void 0;
4744
+ if (sim.userFocused === nextFocused && sim.userVisible === nextVisible && sim.visibilityState === nextVisibilityState && sim.documentFocused === nextDocumentFocused) {
4745
+ return;
4746
+ }
4747
+ sim.userFocused = nextFocused;
4748
+ sim.userVisible = nextVisible;
4749
+ sim.visibilityState = nextVisibilityState;
4750
+ sim.documentFocused = nextDocumentFocused;
4738
4751
  this.broadcastSimClientStates();
4739
4752
  }
4740
4753
  // called when the sim reports a real user interaction (pointerdown,
@@ -4824,7 +4837,10 @@ var init_bridge_host = __esm({
4824
4837
  lockedBy: lease ? lease.cliLabel || lease.cliIdentityKey : void 0,
4825
4838
  lockedByKind: lease ? lease.kind : void 0,
4826
4839
  lockExpiresAt: lease ? lease.expiresAt : void 0,
4827
- userFocused: sim.userFocused || void 0
4840
+ userFocused: sim.userFocused || void 0,
4841
+ userVisible: sim.userVisible,
4842
+ visibilityState: sim.visibilityState,
4843
+ documentFocused: sim.documentFocused
4828
4844
  };
4829
4845
  sim.ws.send(JSON.stringify(message));
4830
4846
  }
package/dist-lib/vite.cjs CHANGED
@@ -1,4 +1,4 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
1
+ /*! sootsim v0.1.67 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
2
  let __sootsim_import_meta_url = ''; try { __sootsim_import_meta_url = require('url').pathToFileURL(__filename).href; } catch {}
3
3
  "use strict";
4
4
  var __create = Object.create;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "sootsim",
3
- "version": "0.1.65",
3
+ "version": "0.1.67",
4
4
  "description": "sootsim CLI + vite/metro plugins + skills registry. bridge client for driving the proprietary sootsim-engine over WebSocket.",
5
5
  "author": "Tamagui LLC",
6
6
  "license": "MIT",
@@ -1,2 +0,0 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{a,b,c,d}from"./chunk-PTUCXL7Y.js";import"./chunk-ZW3HKMLQ.js";import"./chunk-VGQXIMEX.js";import"./chunk-E6CRWOWO.js";import"./chunk-TWKWXGN5.js";import"./chunk-KKOOZPQP.js";export{c as ensureDaemonRunning,a as ensureRuntimeInstalled,d as ensureSootsimReady,b as resolveBootstrapPort};
@@ -1,2 +0,0 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{a,b,c,d,e}from"./chunk-46OKLW3Z.js";import"./chunk-KKOOZPQP.js";export{e as BETA_ASK_HEADLINE,c as BETA_LABEL,d as BETA_TAGLINE,b as BETA_VERSION_TARGET,a as IS_BETA};
@@ -1,3 +0,0 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{B as E,j as B,k as b,l as I,m as S}from"./chunk-5ZB5ZN5Z.js";import{d as y,h as T,i as v}from"./chunk-ZW3HKMLQ.js";var C={timeoutMs:1800,settleMs:48,startWindowMs:64};function p(e){return new Promise(t=>setTimeout(t,e))}function A(e){return e.includes("--json")}function k(e){process.stdout.write(`${JSON.stringify(e??null,null,2)}
3
- `)}async function P(e,t,...r){return T(e,`__sootsimTest.${t}`,...r)}async function W(e,t,...r){return v(e,`SootSim.bridges.mainShell.${t}`,...r)}async function F(e,t={}){try{let r=await P(e,"waitForScreenTransitions",C);if(!t.verbose||!r?.started)return;if(r.timedOut){console.log(` screen transition still active after ${r.waitedMs}ms; continuing`);return}r.waitedMs>0&&console.log(` waited ${r.waitedMs}ms for screen transition settle`)}catch{}}async function O(e,t,r,...n){let a=Date.now()+Math.max(0,r);for(;;)try{return await W(e,t,...n)}catch(o){if(!E(o)||Date.now()>=a)throw o;await p(50)}}async function $(e,t,r,n={}){let a=n.attempts??30,o=n.intervalMs??500,m=n.minNodeCount??10;for(let s=0;s<a;s++){let u=y(e,{commandTimeoutMs:t,simId:r,simIdSource:n.simIdSource});try{let c=await u.send({type:"evaluate",code:"(async () => (await window.__sootsimTest?.getNodeCount()) || 0)()"});if(typeof c=="number"&&c>m)return{bridge:u,count:c}}catch{}u.close(),await p(o)}return null}async function G(e,t,r,n={}){let a=n.timeoutMs??8e3,o=n.intervalMs??250,m=Date.now()+a;for(;Date.now()<m;){let s=y(e,{commandTimeoutMs:t,simId:r,simIdSource:n.simIdSource});try{return await s.send({type:"evaluate",code:"1"}),s}catch{s.close()}await p(o)}return null}async function H(e,t={}){let r=t.timeoutMs??1e4,n=t.errorGraceMs??3e3,a=t.pollIntervalMs??200,o=Date.now(),m=o+r,s=-1,u=o,c=null,M=!1,w={ready:!1,source:"timeout",elapsedMs:0,nodes:0,targets:0,errors:0};for(;Date.now()<m;){try{let i=await e.send({type:"evaluate",code:B}),d=Date.now()-o,l=Number(i?.nodes)||0,g=Number(i?.targets)||0,f=Number(i?.errors)||0,R=i?.flag===!0,h=i?.externalReady,x=typeof i?.externalError=="string"&&i.externalError.trim().length>0;if(c=h,M=x,l!==s&&(s=l,u=d),w={ready:!1,source:"timeout",elapsedMs:d,nodes:l,targets:g,errors:f},R&&h!==!1&&!x&&l>=b&&S({nodes:l,targets:g})&&d-u>=I)return{ready:!0,source:"flag",elapsedMs:d,nodes:l,targets:g,errors:f};if(d>=n&&f>0&&!R)return{ready:!1,source:"error-bail",elapsedMs:d,nodes:l,targets:g,errors:f}}catch{}await p(a)}return c!==!1&&!M&&w.nodes>=b&&S(w)?{...w,ready:!0,source:"nodes-fallback"}:{...w,ready:!1,source:"timeout",elapsedMs:Date.now()-o}}export{p as a,A as b,k as c,P as d,F as e,O as f,$ as g,G as h,H as i};
@@ -1 +0,0 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
@@ -1,2 +0,0 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- var t="http://localhost:5173/";export{t as a};
@@ -1 +0,0 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
@@ -1 +0,0 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
@@ -1,2 +0,0 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- var O="sootsim close";export{O as a};
@@ -1,11 +0,0 @@
1
- /*! sootsim v0.1.65 | (c) 2026 Tamagui LLC | Proprietary — see LICENSE */
2
- import{a as _,b as z,c as le,d as ce}from"./chunk-O56DABNK.js";import{b as O}from"./chunk-TRHZBTPF.js";import{j as se,k as ie,m as ae}from"./chunk-5ZB5ZN5Z.js";import{g as q}from"./chunk-NTEZEQSB.js";import{a as C,b as j}from"./chunk-FU33BTPL.js";import{c as de}from"./chunk-ELZHT3LB.js";import{d as ne}from"./chunk-UH4H5HXN.js";import{a as ee}from"./chunk-B32KMMQV.js";import{b as re,c as I,d as S,e as x}from"./chunk-ZW3HKMLQ.js";import{c as P,d as w,e as k}from"./chunk-VGQXIMEX.js";import{E as te,F as oe}from"./chunk-TWKWXGN5.js";import Me from"node:fs";import Ue from"node:os";import G from"node:path";function Ie(t){return t.replace(/^\[|\]$/g,"").toLowerCase()}function W(t){let o=Ie(t);return o==="localhost"||o.endsWith(".localhost")||o==="0.0.0.0"||o==="::1"||/^127(?:\.\d{1,3}){3}$/.test(o)}function me(t){return new Error(`could not resolve a native bundle for ${t}. pass an explicit bundle URL or open Connect and choose the app there.`)}function xe(t,o){return t?.includes("/one/metro-entry.bundle")?"one":typeof o=="string"&&o?"expo":"unknown"}function pe(t,o,e){return`${t}//${o}:${e}`}function Pe(t){if(typeof window>"u")return!1;try{let o=new URL(t);return W(o.hostname)&&W(window.location.hostname)&&o.origin!==window.location.origin}catch{return!1}}async function ue(t,o){let e={...o,cache:o?.cache??"no-store"};return Pe(t)?fetch(`/__fetch-proxy?url=${encodeURIComponent(t)}`,e):fetch(t,e)}function ke(t){return t==="https:"?443:80}function Ce(t){let o=t.pathname||"/";return(o==="/"||o==="")&&!t.search&&!t.hash}async function fe(t,o){let e=t.replace(/\/+$/,"");try{let n=await ue(`${e}/`,{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:le(a||`${e}${z}`),port:o,framework:xe(a,s.sdkVersion),projectName:s.name}}}catch{}try{let n=await ue(`${e}/status`);if(n.ok&&(await n.text()).includes("packager-status:running"))return{bundleUrl:`${e}${z}`,port:o,framework:"metro"}}catch{}return null}async function Be(t){return fe(pe("http:","localhost",t),t)}async function ge(t){let o=t.trim();if(/^\d+$/.test(o)){let i=parseInt(o,10),l=await Be(i);if(l)return l;throw me(`localhost:${i}`)}let e=o.startsWith("http")?o:`http://${o}`,n;try{n=new URL(e)}catch{throw new Error(`could not parse "${t}". 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):ke(r),a=pe(r,n.hostname,s);if(Ce(n)){let i=await fe(a,s);if(i)return i;throw me(a)}return{bundleUrl:n.toString(),port:s,framework:"unknown"}}function tt(){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 he(t){let o=C();if(console.log(` note: no sootsim bridge detected on port ${t}`),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 Te(t){console.error(""),console.error(` no sim is connected to the sootsim bridge on port ${t}.`),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 ot(t,o,e){let n=[];try{n=await t.listSims()}catch{}if(console.error(""),n.length===0){Te(o);return}console.error(` no sim with id "${e}" 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].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 Re={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 $(t){return new Promise(o=>setTimeout(o,t))}function A(t){return t.trim()}function J(t){try{let o=new URL(t),e=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")||e==="/sootsim"||o.pathname==="/__soot"||o.pathname==="/__soot/"||e==="/rn"||/^\/rn\/[^/]+$/i.test(e)||/^\/app\/[^/]+$/i.test(e)||e==="/__soot/rn"||/^\/__soot\/rn\/[^/]+$/i.test(e)||/^\/__soot\/app\/[^/]+$/i.test(e)}catch{return!1}}async function $t(t){let o=A(t);return(await be(o)).bundleUrl}function Ee(t){try{let o=new URL(t.startsWith("http")?t:`http://${t}`),e=o.pathname||"/";if(e!=="/"&&e!==""||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 be(t){let o=A(t),e=/^\d+$/.test(o)?Number(o):Ee(o);if(e&&e>0){let n=await ce(e);if(n)return{bundleUrl:n.bundleUrl,port:n.port,framework:n.framework,projectName:n.projectName}}return ge(o)}function V(t){let o=t.replace(/\/+$/,"")||"/";return o==="/__soot"||o.startsWith("/__soot/")?"/__soot":""}function Oe(t,o){let e=A(t),n=new URL(o),r=V(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(e)?n.pathname=`${r}/rn/${e}`:(n.pathname=`${r}/rn`,n.searchParams.set("open",e)),n.toString()}function _e(t){let o=A(t);return/^\d+$/.test(o)||/^https?:\/\//i.test(o)?!0:/^(localhost|127\.0\.0\.1|\[::1\]|[^/]+\.localhost):\d+(?:\/.*)?$/i.test(o)}async function Le(t,o){let e=await be(t),n=new URL(o),r=V(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",e.bundleUrl),n.toString()}function De(t){return t.startsWith("~/")?G.join(Ue.homedir(),t.slice(2)):G.isAbsolute(t)?t:G.resolve(process.cwd(),t)}function Ae(t){let o={};for(let e=0;e<t.length;e++){if(t[e]!=="--replace")continue;let n=t[e+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=De(n.slice(r+1).trim());Me.existsSync(a)||(console.error(` sootsim open: replacement file not found: ${a}`),process.exit(1)),o[s]={file:a},e++}return Object.keys(o).length>0?{modules:o}:void 0}function Se(){let t=te();return oe(t)&&t.runtimePort>0?`http://localhost:${t.runtimePort}/`:ee}async function D(t,o=Se()){if(!t)return new URL(o).toString();if(J(t))return new URL(t).toString();let e=Re[t.toLowerCase()];if(e){let n=new URL(o),r=V(n.pathname);return n.pathname=`${r}${e}`,n.toString()}return _e(t)?Le(t,o):Oe(t,o)}function we(t,o){let e=t?.url||t?.origin||o;try{let n=new URL(e);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.toString()}catch{return o}}async function Ne(t,o,e){let n=new URL(await D(t,o));return n.searchParams.set("inspectOpen",e),n.toString()}async function Fe(t,o,e,n={}){let r=n.attempts??30,s=n.intervalMs??500,a=n.minNodeCount??10;for(let i=0;i<r;i++){let l=S(t,{commandTimeoutMs:o,simId:e,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)}return null}function He(t){if(!t)return null;try{let o=new URL(t);if(o.searchParams.has("bundle")){let e=o.searchParams.get("bundle")||"";try{let n=new URL(e),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 Ke(t,o){if(t.length===0){console.log(" no sims connected");return}console.log(` connected sims (${t.length}):
3
- `);for(let e of t){let n=e.lockedBy&&e.lockExpiresAt?`locked by ${e.lockedBy} (${Math.max(0,Math.round((e.lockExpiresAt-Date.now())/1e3))}s)`:"",r=[e.isPrimary?"primary":"",e.id===o?"selected":"",e.readyState,e.attachedCliCount&&e.attachedCliCount>0?"in use":"",e.userFocused?"focused":"",n].filter(Boolean);console.log(` ${e.id}${r.length?` [${r.join(", ")}]`:""}`);let s=He(e.url);if(s&&console.log(` loaded: ${s}`),e.url?console.log(` url: ${e.url}`):e.origin&&console.log(` origin: ${e.origin}`),e.title&&console.log(` title: ${e.title}`),console.log(` connected: ${new Date(e.connectedAt).toISOString()}`),e.lastActiveAt&&e.lastActiveAt>0){let a=Math.round((Date.now()-e.lastActiveAt)/1e3),i=a<60?`${a}s ago`:a<3600?`${Math.round(a/60)}m ago`:`${Math.round(a/3600)}h ago`;console.log(` last active: ${i}`)}}}async function L(t,o,e,n={}){let r=n.attempts??30,s=n.intervalMs??500;for(let a=0;a<r;a++){let i=S(t,{commandTimeoutMs:o});try{let c=(await i.listSims()).find(e);if(c)return c}catch{}finally{i.close()}await $(s)}return null}async function ze(t,o,e,n={}){let r=n.attempts??20,s=n.intervalMs??250;for(let a=0;a<r;a++){let i=S(t,{commandTimeoutMs:o});try{let c=(await i.listSims()).find(d=>d.id===e);if(!c||c.readyState!=="open")return!0}catch{return!0}finally{i.close()}await $(s)}return!1}function je(t){let o=t.meta;if(!o||o.sootsimHostDriver!=="playwright")return null;let e=Number(o.sootsimHostPid);return!Number.isInteger(e)||e<=1||e===process.pid?null:e}function ye(t){try{return process.kill(t,0),!0}catch(o){return o?.code==="EPERM"}}async function qe(t,o=2500){let e=Date.now()+o;for(;Date.now()<e;){if(!ye(t))return!0;await $(100)}return!ye(t)}async function Q(t,o){let e=new Set(o),n=new Set;for(let r of t){if(!e.has(r.id))continue;let s=je(r);s&&n.add(s)}for(let r of n){try{process.kill(r,"SIGTERM")}catch{}if(await qe(r)){console.log(` closed playwright host process ${r}`);continue}try{process.kill(r,"SIGKILL"),console.log(` force-closed playwright host process ${r}`)}catch{}}}function N(t,o){if(o){let r=o.trim(),s=t.find(a=>a.id===r);if(!s)throw new Error(`no sim connected with id ${r}`);return s}let e=t.find(r=>r.isPrimary&&r.readyState==="open");if(e)return e;let n=t.find(r=>r.readyState==="open");if(n)return n;throw new Error("no sim connected")}function B(t,o,e){console.log(` ${e==="current sim"?"loaded":"opened"}: ${t} [${e}]`),console.log(` current sim: ${o.id}`),console.log(JSON.stringify({simId:o.id,url:o.url},null,2))}async function T(t,o,e){if(e.includes("--no-describe"))return;let n=process.env.SOOTSIM_QUIET_TARGET_NOTICE;process.env.SOOTSIM_QUIET_TARGET_NOTICE="1";try{await We(t,o,{stableMs:150,maxMs:400});let r=S(t,{commandTimeoutMs:3e3,simId:o,cliLabel:"open --describe",simIdSource:"flag"});try{let s=null;try{s=await r.send({type:"evaluate",code:se})}catch{}if(s&&!(s.flag===!0&&!s.loadingText&&s.nodes>=ie&&ae(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 () => {
4
- const t = window.__sootsimTest
5
- const ms = window.SootSim?.bridges?.mainShell
6
- if (!t) return null
7
- let shell = null
8
- try { shell = ms?.getState ? await ms.getState() : null } catch {}
9
- const tree = await t.dumpTree(12, ${JSON.stringify({describe:!0,verbose:!1,filter:""})})
10
- return { tree, shell }
11
- })()`,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 We(t,o,e){let n=Date.now()+e.maxMs,r="(async () => (await window.__sootsimTest?.getNodeCount?.()) || 0)()",s=S(t,{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>=e.stableMs)return}else a=l,i=Date.now();await $(50)}}finally{s.close()}}async function vt(t,o={}){let e=I(t,{port:o.port,commandTimeoutMs:o.timeoutMs}),n=x(e);try{let r=await n.listSims();Ke(r,e.simId)}finally{n.close()}}async function It(t,o={}){let e=I(t,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--new","--headless","--ephemeral"],stripValueFlags:["--base-url","--replace","--driver","--profile"]}),n=t.find((m,u)=>t[u-1]==="--profile"),r=t.includes("--ephemeral");n&&r&&(console.error(" sootsim open: --profile cannot be combined with --ephemeral"),process.exit(1));let s=n?de(n).id:void 0,a=!!s||r,i=t.includes("--new")||a,l=Ae(t);i&&e.simIdSource==="flag"&&(console.error(" sootsim open: --new, --profile, and --ephemeral cannot be combined with --sim"),process.exit(1));let c=e.positional[0]||"",d=t.find((m,u)=>t[u-1]==="--driver")||"",p=t.includes("--headless"),h=t.find((m,u)=>t[u-1]==="--base-url")||Se(),M=t.includes("--base-url"),Y=P();if(!d&&!i&&(e.simIdSource==="flag"||e.simIdSource==="saved"&&!!Y)){let m=x(e),u=e.simId?` --sim ${e.simId}`:"",f=!1;try{let g=null;try{let b=await m.listSims();if(e.simIdSource==="saved"?(g=b.find(v=>v.id===Y&&v.readyState==="open")??null,g||k()):g=N(b,e.simId),!g)if(e.simIdSource==="saved")f=!0;else throw new Error("no sim connected");if(!f&&g){let v=M||J(c)?h:we(g,h),E=await D(c,_(v,l));m.send({type:"evaluate",simId:g.id,code:`window.location.href = ${JSON.stringify(E)}`}).catch(()=>{})}}catch(b){console.error(` open failed: ${b instanceof Error?b.message:String(b)}`),await O(m,{errorsCommand:`sootsim get errors 5${u}`,warningsCommand:`sootsim get warnings 5${u}`,requestsCommand:`sootsim get requests 5${u}`}),process.exit(1)}if(!f&&g){await $(1500);let b=await Fe(e.wsPort,e.commandTimeoutMs,g.id);b||(console.error(" timed out waiting for current sim to load target"),process.exit(1)),b.bridge.close(),w(g.id);let v=M||J(c)?h:we(g,h),E=await D(c,_(v,l));B(E,{...g,url:E},"current sim"),await T(e.wsPort,g.id,t);return}}finally{m.close()}}let X=_(h,l),ve=await D(c,X),Z=`cli-${Date.now().toString(36)}-${Math.random().toString(36).slice(2,8)}`,y=await Ne(c,X,Z),F={newWindow:!0},U=m=>m.url?.includes(`inspectOpen=${Z}`)??!1;if(d){let m=q(d);m||(console.error(` unknown driver "${d}" \u2014 run \`sootsim list --drivers\``),process.exit(1));let u=await m.launch({url:y,headless:p,newWindow:F.newWindow,profileId:s,ephemeralProfile:r});u.launched||(console.error(` ${m.name} driver: ${u.message}`),process.exit(1));let f=await L(e.wsPort,e.commandTimeoutMs,U,{attempts:60,intervalMs:500});if(!f){if(u.pid)try{process.kill(u.pid,"SIGTERM"),console.error(` closed ${m.name} host process ${u.pid}`)}catch{}console.error(" timed out waiting for opened sim to connect"),process.exit(1)}w(f.id),B(y,f,`${m.name} driver`),await T(e.wsPort,f.id,t);return}if(a){let m=C();m||(console.error(" profiles require electron or playwright; install the desktop companion or use `--driver playwright`"),process.exit(1));try{(await j(y,m,{profileId:s,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 u=await L(e.wsPort,e.commandTimeoutMs,U,{attempts:40,intervalMs:500});u||(console.error(" timed out waiting for profiled sim to connect"),process.exit(1)),w(u.id),B(y,u,"desktop companion"),await T(e.wsPort,u.id,t);return}let H=!1,K=!1;try{let m=S(e.wsPort,{commandTimeoutMs:2e3});try{await m.listSims(),H=!0}finally{m.close()}}catch{}if(H)try{let m=S(e.wsPort,{commandTimeoutMs:3e3});try{await m.openUrl(y,F),K=!0}finally{m.close()}}catch{}if(!K){let m=C();if(m)try{if((await j(y,m)).launched){let f=await L(e.wsPort,e.commandTimeoutMs,U,{attempts:20,intervalMs:500});if(f){w(f.id),B(y,f,"desktop companion"),await T(e.wsPort,f.id,t);return}}}catch{}try{await ne(y,F)}catch(u){console.error(` open failed: ${u instanceof Error?u.message:String(u)}`),q("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(!H){console.log(` opened: ${ve}`),he(e.wsPort);return}}let R=await L(e.wsPort,e.commandTimeoutMs,U,{attempts:30,intervalMs:500});R||(console.error(" timed out waiting for opened sim to connect"),process.exit(1)),w(R.id),B(y,R,K?"bridge":"direct shell open"),await T(e.wsPort,R.id,t)}async function $e(t,o={}){let e=I(t,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--force"]}),n=t.includes("--force"),r=x(e),s=e.simId?` --sim ${e.simId}`:"";try{try{let a=await r.listSims(),i=N(a,e.positional[0]||e.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 xt(t,o={}){await $e(t,o)}async function Pt(t,o={}){await $e(t,o)}async function kt(t,o={}){let e=I(t,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--force"]}),n=t.includes("--force"),r=x(e);try{try{let s=await r.listSims(),a=N(s,e.positional[0]||e.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 re){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 Ge(t,o,e,n){if(n.length===0)return{closed:[],remaining:[]};await Promise.all(n.map(s=>t.closeSim(s).catch(()=>{})));let r=new Set(n);for(let s=0;s<40;s++){let a=[];try{let i=S(o,{commandTimeoutMs:e});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 $(250)}return{closed:[...r],remaining:[]}}function Qe(t,o){let e=t.filter(p=>p.readyState==="open"),n=e.map(p=>p.id),r=o.explicitKeepId?.trim()||"",s=o.savedKeepId?.trim()||"",a=s?e.some(p=>p.id===s):!1,i=s&&!a?s:null,l=null,c=null,d=null;if(o.closeOthers)if(r){let p=e.find(h=>h.id===r);p?l=p.id:c=r}else if(s&&a)l=s;else{let p=e.find(h=>h.isPrimary)??e[0]??null;p&&(l=p.id,s&&!a&&(d=p.id))}return{openIds:n,keepId:l,targets:c?[]:n.filter(p=>p!==l),staleSavedId:i,missingExplicitKeepId:c,fallbackKeepId:d}}async function Ct(t,o={}){let e=I(t,{port:o.port,commandTimeoutMs:o.timeoutMs,stripBooleanFlags:["--all","--others"]}),n=x(e),r=e.simId?` --sim ${e.simId}`:"",s=t.includes("--all"),a=t.includes("--others");if(s||a){try{let i=await n.listSims(),l=P(),c=e.positional[0]||(e.simIdSource==="flag"?e.simId:void 0),d=Qe(i,{closeOthers:a,explicitKeepId:c,savedKeepId:l});if(d.staleSavedId&&k(),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 p=await Ge(n,e.wsPort,e.commandTimeoutMs,d.targets);await Q(i,d.targets);let h=P();h&&p.closed.includes(h)&&(d.keepId?w(d.keepId):k());let M=` closed ${p.closed.length} sim(s)${d.keepId?` (kept ${d.keepId})`:""}`;console.log(M),p.remaining.length>0&&(console.error(` close failed: still connected: ${p.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=N(i,e.positional[0]||e.simId),c=i.find(p=>p.id!==l.id&&p.readyState==="open");await n.closeSim(l.id),await ze(e.wsPort,e.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]),P()===l.id&&(c?w(c.id):k()),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{tt as a,Te as b,ot as c,$t as d,Se as e,D as f,Ne as g,He as h,Ke as i,L as j,je as k,vt as l,It as m,xt as n,Pt as o,kt as p,Qe as q,Ct as r};