@busy-app/busy-lib 0.10.0 → 0.11.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.cjs CHANGED
@@ -1 +1 @@
1
- "use strict";var G=Object.defineProperty;var L=(n,t,e)=>t in n?G(n,t,{enumerable:!0,configurable:!0,writable:!0,value:e}):n[t]=e;var c=(n,t,e)=>L(n,typeof t!="symbol"?t+"":t,e);Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const m=require("openapi-fetch"),O=(n,t)=>{if(typeof FormData<"u"&&n instanceof FormData||typeof Buffer<"u"&&typeof Buffer.isBuffer=="function"&&Buffer.isBuffer(n)||typeof File<"u"&&n instanceof File||typeof Blob<"u"&&n instanceof Blob||typeof ArrayBuffer<"u"&&n instanceof ArrayBuffer||typeof ArrayBuffer<"u"&&ArrayBuffer.isView&&ArrayBuffer.isView(n))return n;let e;return t&&(t instanceof Headers?e=t.get("Content-Type")??t.get("content-type")??void 0:typeof t=="object"&&(e=t["Content-Type"]??t["content-type"]),e==="application/x-www-form-urlencoded")?n&&typeof n=="object"&&!(n instanceof URLSearchParams)?new URLSearchParams(n).toString():String(n):JSON.stringify(n)};async function p(n){const i=(n.headers.get("content-type")||"").includes("application/json")?await n.clone().json():await n.clone().text(),r=typeof i=="object"&&i!==null?i.error||i.message:typeof i=="string"?i:void 0;return Object.assign(new Error(r||`HTTP ${n.status} ${n.statusText}`),{status:n.status,statusText:n.statusText,body:i})}async function s(n,t=3e3){if(t<=0)return await n();const e=new AbortController,i=setTimeout(()=>e.abort(),t);try{return await n(e.signal)}catch(r){throw r instanceof DOMException&&r.name==="AbortError"?new Error(`Request timed out after ${t}ms`):r}finally{clearTimeout(i)}}function D(n,t,e){let i,r=e??void 0,o,a=null;const u=async()=>{i||(a||(a=(async()=>{const h=await t();if(!h.api_semver)throw new Error("Empty API version");i=h.api_semver})().finally(()=>{a=null})),await a)},l={async onRequest({request:h,schemaPath:d}){return r&&h.headers.set("Authorization",`Bearer ${r}`),d!=="/version"&&(await u(),i&&h.headers.set("X-API-Sem-Ver",i),o&&h.headers.set("X-API-Token",o)),h},async onResponse({request:h,response:d,options:v,schemaPath:P}){if(d.ok)return d;if(P==="/version")throw await p(d);if(d.status!==405)throw await p(d);i=void 0,await u(),i&&h.headers.set("X-API-Sem-Ver",i),r&&h.headers.set("Authorization",`Bearer ${r}`);const y=await(v.fetch??fetch)(h);if(y.ok)return y;throw await p(y)}},f=m({baseUrl:n,bodySerializer:O});return f.use(l),{client:f,setApiKey:h=>{o=h},setToken:h=>{r=h}}}async function B(n,t){const{data:e,error:i}=await s(r=>n.GET("/version",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function U(n,t){const{data:e,error:i}=await s(r=>n.GET("/status",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function I(n,t){const{data:e,error:i}=await s(r=>n.GET("/status/system",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function $(n,t){const{data:e,error:i}=await s(r=>n.GET("/status/power",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}class K{async SystemVersionGet(t){const e=await B(this.apiClient,t);return this.apiSemver=e.api_semver,e}async SystemStatusGet(t){return await U(this.apiClient,t)}async SystemInfoGet(t){return await I(this.apiClient,t)}async SystemStatusPowerGet(t){return await $(this.apiClient,t)}}async function R(n,t){const{file:e}=t,{data:i,error:r}=await s(o=>n.POST("/update",{headers:{"Content-Type":"application/octet-stream"},body:e,signal:o}),t.timeout);if(r)throw r;return i}async function F(n,t){const{data:e,error:i}=await s(r=>n.POST("/update/check",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function M(n,t){const{data:e,error:i}=await s(r=>n.GET("/update/status",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function W(n,t){const{version:e}=t,{data:i,error:r}=await s(o=>n.GET("/update/changelog",{params:{query:{version:e}},signal:o}),t.timeout);if(r)throw r;return i}async function N(n,t){const{version:e}=t,{data:i,error:r}=await s(o=>n.POST("/update/install",{params:{query:{version:e}},signal:o}),t.timeout);if(r)throw r;return i}async function _(n,t){const{data:e,error:i}=await s(r=>n.POST("/update/abort_download",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}class q{async UpdateFromFile(t){return await R(this.apiClient,t)}async UpdateCheck(t){return await F(this.apiClient,t)}async UpdateStatusGet(t){return await M(this.apiClient,t)}async UpdateChangelogGet(t){return await W(this.apiClient,t)}async UpdateInstall(t){return await N(this.apiClient,t)}async UpdateAbort(t){return await _(this.apiClient,t)}}async function x(n,t){const{data:e,error:i}=await s(r=>n.GET("/time",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function j(n,t){const{data:e,error:i}=await s(r=>n.POST("/time/timestamp",{params:{query:{...t,timeout:void 0}},signal:r}),t.timeout);if(i)throw i;return e}async function V(n,t){const{data:e,error:i}=await s(r=>n.GET("/time/timezone",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function z(n,t){const{data:e,error:i}=await s(r=>n.POST("/time/timezone",{params:{query:{...t,timeout:void 0}},signal:r}),t.timeout);if(i)throw i;return e}async function H(n,t){const{data:e,error:i}=await s(r=>n.GET("/time/tzlist",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}class J{async TimeGet(t){return await x(this.apiClient,t)}async TimeTimestampSet(t){return await j(this.apiClient,t)}async TimeTimezoneGet(t){return await V(this.apiClient,t)}async TimeTimezoneSet(t){return await z(this.apiClient,t)}async TimeTzListGet(t){return await H(this.apiClient,t)}}async function X(n,t){const{data:e,error:i}=await s(r=>n.GET("/account/status",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function Y(n,t){const{data:e,error:i}=await s(r=>n.GET("/account/info",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function Q(n,t){const{data:e,error:i}=await s(r=>n.GET("/account/profile",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function Z(n,t){const{profile:e}=t,{data:i,error:r}=await s(o=>n.POST("/account/profile",{params:{query:{profile:e}},signal:o}),t.timeout);if(r)throw r;return i}async function tt(n,t){const{data:e,error:i}=await s(r=>n.DELETE("/account",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function et(n,t){const{data:e,error:i}=await s(r=>n.POST("/account/link",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}class it{async AccountInfoGet(t){return await Y(this.apiClient,t)}async AccountStateGet(t){return await X(this.apiClient,t)}async AccountProfileGet(t){return await Q(this.apiClient,t)}async AccountProfileSet(t){return await Z(this.apiClient,t)}async AccountUnlink(t){return await tt(this.apiClient,t)}async AccountLink(t){return await et(this.apiClient,t)}}async function nt(n,t){const{appId:e,elements:i,priority:r=6}=t,{data:o,error:a}=await s(u=>n.POST("/display/draw",{body:{app_id:e,priority:r,elements:i},signal:u}),t.timeout);if(a)throw a;return o}async function rt(n,t){const{data:e,error:i}=await s(r=>n.DELETE("/display/draw",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function ot(n,t){const{display:e}=t,{data:i,error:r}=await s(o=>n.GET("/screen",{params:{query:{display:e}},parseAs:"blob",signal:o}),t.timeout);if(r)throw r;return i}async function st(n,t){const{data:e,error:i}=await s(r=>n.GET("/display/brightness",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function at(n,t){const{value:e}=t,r=(u=>{if(typeof u=="number"){if(u<0||u>100)throw new Error("Brightness value must be between 0 and 100 or 'auto'");return String(u)}if(u==="auto")return"auto"})(e),{data:o,error:a}=await s(u=>n.POST("/display/brightness",{params:{query:{value:r}},signal:u}),t.timeout);if(a)throw a;return o}class ct{async DisplayDraw(t){return await nt(this.apiClient,t)}async DisplayClear(t){return await rt(this.apiClient,t)}async DisplayScreenFrameGet(t){return await ot(this.apiClient,t)}async DisplayBrightnessGet(t){return await st(this.apiClient,t)}async DisplayBrightnessSet(t){return await at(this.apiClient,t)}}async function ut(n,t){const{appId:e,path:i}=t,{data:r,error:o}=await s(a=>n.POST("/audio/play",{params:{query:{app_id:e,path:i}},signal:a}),t.timeout);if(o)throw o;return r}async function ht(n,t){const{data:e,error:i}=await s(r=>n.DELETE("/audio/play",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function dt(n,t){const{data:e,error:i}=await s(r=>n.GET("/audio/volume",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function ft(n,t){const{volume:e}=t;if(typeof e!="number"||e<0||e>100)throw new Error("Volume must be a number between 0 and 100");const{data:i,error:r}=await s(o=>n.POST("/audio/volume",{params:{query:{volume:e}},signal:o}),t.timeout);if(r)throw r;return i}class lt{async AudioPlay(t){return await ut(this.apiClient,t)}async AudioStop(t){return await ht(this.apiClient,t)}async AudioVolumeGet(t){return await dt(this.apiClient,t)}async AudioVolumeSet(t){return await ft(this.apiClient,t)}}async function wt(n,t){const{data:e,error:i}=await s(r=>n.GET("/wifi/status",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function yt(n,t){const{data:e,error:i}=await s(r=>n.POST("/wifi/connect",{body:{ssid:t.ssid,password:t.password,security:t.security,ip_config:{ip_method:t.ipConfig.ipMethod,address:t.ipConfig.address,mask:t.ipConfig.mask,gateway:t.ipConfig.gateway}},signal:r}),t.timeout);if(i)throw i;return e}async function pt(n,t){const{data:e,error:i}=await s(r=>n.POST("/wifi/disconnect",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function St(n,t){const{data:e,error:i}=await s(r=>n.GET("/wifi/networks",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}class Tt{async WifiStatusGet(t){return await wt(this.apiClient,t)}async WifiConnect(t){return await yt(this.apiClient,t)}async WifiDisconnect(t){return await pt(this.apiClient,t)}async WifiNetworksGet(t){return await St(this.apiClient,t)}}async function Et(n,t){const{path:e,file:i}=t,{data:r,error:o}=await s(a=>n.POST("/storage/write",{params:{query:{path:e}},headers:{"Content-Type":"application/octet-stream"},body:i,signal:a}),t.timeout);if(o)throw o;return r}async function mt(n,t){const{path:e,asArrayBuffer:i}=t,{data:r,error:o}=await s(a=>n.GET("/storage/read",{params:{query:{path:e}},parseAs:i?"arrayBuffer":"blob",signal:a}),t.timeout);if(o)throw o;return r}async function Ct(n,t){const{path:e}=t,{data:i,error:r}=await s(o=>n.GET("/storage/list",{params:{query:{path:e}},signal:o}),t.timeout);if(r)throw r;return i}async function kt(n,t){const{path:e}=t,{data:i,error:r}=await s(o=>n.DELETE("/storage/remove",{params:{query:{path:e}},signal:o}),t.timeout);if(r)throw r;return i}async function gt(n,t){const{path:e}=t,{data:i,error:r}=await s(o=>n.POST("/storage/mkdir",{params:{query:{path:e}},signal:o}),t.timeout);if(r)throw r;return i}async function bt(n,t){const{data:e,error:i}=await s(r=>n.GET("/storage/status",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}class At{async StorageWrite(t){return await Et(this.apiClient,t)}async StorageRead(t){return await mt(this.apiClient,t)}async StorageListGet(t){return await Ct(this.apiClient,t)}async StorageRemove(t){return await kt(this.apiClient,t)}async StorageMkdir(t){return await gt(this.apiClient,t)}async StorageStatusGet(t){return await bt(this.apiClient,t)}}async function vt(n,t){const{data:e,error:i}=await s(r=>n.GET("/access",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function Pt(n,t){let{mode:e,key:i}=t;if(i=i??"",String(i).trim()&&!/^\d{4,10}$/.test(String(i)))throw new Error("Key must be a string of 4 to 10 digits");const{data:r,error:o}=await s(a=>n.POST("/access",{params:{query:{mode:e,key:i}},signal:a}),t.timeout);if(o)throw o;return r}async function Gt(n,t){const{data:e,error:i}=await s(r=>n.GET("/name",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function Lt(n,t){const{data:e,error:i}=await s(r=>n.POST("/name",{body:t,signal:r}),t.timeout);if(i)throw i;return e}class Ot{async SettingsAccessGet(t){return await vt(this.apiClient,t)}async SettingsAccessSet(t){const e=await Pt(this.apiClient,t);return t.mode==="key"&&t.key&&this.setApiKey(t.key),e}async SettingsNameGet(t){return await Gt(this.apiClient,t)}async SettingsNameSet(t){return await Lt(this.apiClient,t)}}async function Dt(n,t){const{data:e,error:i}=await s(r=>n.POST("/ble/enable",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function Bt(n,t){const{data:e,error:i}=await s(r=>n.POST("/ble/disable",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function Ut(n,t){const{data:e,error:i}=await s(r=>n.DELETE("/ble/pairing",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function It(n,t){const{data:e,error:i}=await s(r=>n.GET("/ble/status",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}class $t{async BleEnable(t){return await Dt(this.apiClient,t)}async BleDisable(t){return await Bt(this.apiClient,t)}async BleUnpair(t){return await Ut(this.apiClient,t)}async BleStatusGet(t){return await It(this.apiClient,t)}}async function Kt(n,t){const{keyName:e}=t,{data:i,error:r}=await s(o=>n.POST("/input",{params:{query:{key:e}},signal:o}),t.timeout);if(r)throw r;return i}class Rt{async InputSend(t){return await Kt(this.apiClient,t)}}async function Ft(n,t){const{data:e,error:i}=await s(r=>n.GET("/matter/commissioning",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function Mt(n,t){const{data:e,error:i}=await s(r=>n.POST("/matter/commissioning",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}async function Wt(n,t){const{data:e,error:i}=await s(r=>n.DELETE("/matter/commissioning",{signal:r}),t==null?void 0:t.timeout);if(i)throw i;return e}class Nt{async MatterStatusGet(t){return await Ft(this.apiClient,t)}async MatterPair(t){return await Mt(this.apiClient,t)}async MatterErase(t){return await Wt(this.apiClient,t)}}async function _t(n,t){const{appId:e,fileName:i,file:r}=t,{data:o,error:a}=await s(u=>n.POST("/assets/upload",{params:{query:{app_id:e,file:i}},headers:{"Content-Type":"application/octet-stream"},body:r,signal:u}),t.timeout);if(a)throw a;return o}async function qt(n,t){const{appId:e}=t,{data:i,error:r}=await s(o=>n.DELETE("/assets/upload",{params:{query:{app_id:e}},signal:o}),t.timeout);if(r)throw r;return i}class xt{async AssetsUpload(t){return await _t(this.apiClient,t)}async AssetsDelete(t){return await qt(this.apiClient,t)}}const S="http://10.0.4.20",jt="https://proxy.busy.app",Vt=/^https?:\/\/proxy(?:\.(?:dev|test|stage))?\.busy\.app$/i;function T(n){const t=n.split(".");if(t.length!==4)return!1;for(const e of t){if(e.length===0||e.length>1&&e[0]==="0"||!/^\d+$/.test(e))return!1;const i=Number(e);if(i<0||i>255)return!1}return!0}function E(n){return/\.local$/i.test(n)}class C{constructor(t){c(this,"addr");c(this,"apiSemver");c(this,"apiClient");c(this,"setApiKeyFn");c(this,"setTokenFn");c(this,"connectionType","unknown");if(!t||!t.addr&&!t.token)this.addr=S;else if(!t.addr)this.addr=jt;else{let o=t.addr.trim();if(/^https?:\/\//i.test(o)||(o=`http://${o}`),Vt.test(o)&&!t.token)throw new Error("Token is required. Please provide it.");this.addr=o}this.apiSemver="";const{client:e,setApiKey:i,setToken:r}=D(`${this.addr}/api/`,this.SystemVersionGet.bind(this),t==null?void 0:t.token);this.apiClient=e,this.setApiKeyFn=i,this.setTokenFn=r,this.detectConnectionType()}async detectConnectionType(){const t=new URL(this.addr).hostname;if(!T(t)&&!E(t)){this.connectionType="wifi";return}const e=m({baseUrl:`${this.addr}/api/`});try{const{response:i}=await e.GET("/name");if(i.status===401||i.status===403)this.connectionType="wifi";else if(i.ok)this.connectionType="usb";else throw new Error(`Failed to detect connection type. Status: ${i.status}`)}catch(i){throw i}}setApiKey(t){this.setApiKeyFn(t)}setToken(t){this.setTokenFn(t)}}function zt(n,t){t.forEach(e=>{Object.getOwnPropertyNames(e.prototype).forEach(i=>{Object.defineProperty(n.prototype,i,Object.getOwnPropertyDescriptor(e.prototype,i)||Object.create(null))})})}zt(C,[K,q,J,it,ct,lt,Tt,At,Ot,$t,Rt,Nt,xt]);var w=(n=>(n[n.FRONT=0]="FRONT",n[n.BACK=1]="BACK",n))(w||{});const k=3e3,g=new Set([1001,1006,1012,1013,1014,3008]);function b(n,t){if(t<0||t>=n.length)throw new Error(`Index ${t} is out of bounds (0…${n.length-1})`);const e=n[t];if(e===void 0)throw new Error(`Unexpected undefined at index ${t}`);return e}function Ht(n,t){let e=0;const i=n.length,r=[];for(;e<i;){const o=b(n,e);if(e+=1,(o&128)!==0){const a=o&127;for(let u=0;u<a*t;u++)r.push(n[e+u]);e+=a*t}else{const a=o,u=n.slice(e,e+t);for(let l=0;l<a;l++)for(let f=0;f<t;f++)r.push(u[f]);e+=t}}return new Uint8Array(r)}function Jt(n){const t=new Uint8Array(n.length*2);let e=0,i=0;for(;e<n.length;){const r=b(n,e),o=r&15,a=r>>4&15;t[i]=o,t[i+1]=a,e+=1,i+=2}return t}const A=()=>typeof window<"u"&&typeof window.document<"u";class Xt{constructor(t){c(this,"addr");c(this,"connected",!1);c(this,"apiKey");c(this,"apiSemver");c(this,"dataListeners",[]);c(this,"stopListeners",[]);c(this,"errorListeners",[]);c(this,"socket",null);if(this.config=t,!A())throw new Error("not browser");if(t.apiKey&&(this.apiKey=t.apiKey),t.apiSemver&&(this.apiSemver=t.apiSemver),!t||!t.addr)this.addr=S;else{let e=t.addr.trim();/^https?:\/\//i.test(e)||(e=`http://${e}`);try{const r=new URL(e).hostname;if(!T(r)&&!E(r))throw new Error(`Invalid address: "${t.addr}". Only IP addresses and mDNS names (ending in .local) are supported.`)}catch(i){throw i instanceof Error&&i.message.startsWith("Invalid address")?i:new Error(`Invalid URL format: "${t.addr}"`)}this.addr=e}}onData(t){this.dataListeners.push(t)}onStop(t){this.stopListeners.push(t)}onError(t){this.errorListeners.push(t)}emitData(t){for(const e of this.dataListeners)e(t)}emitStop(){for(const t of this.stopListeners)t()}emitError(t){for(const e of this.errorListeners)e(t)}async openWebsocket(){this.socket&&await this.closeWebsocket();const t=new URL(`${this.addr}/api/screen/ws`);if(this.apiKey&&t.searchParams.append("x-api-token",this.apiKey),this.apiSemver&&t.searchParams.append("x-api-sem-ver",this.apiSemver),!t)throw new Error("The WebSocket URL is not specified");this.socket=new WebSocket(t),this.socket.onopen=()=>{this.socket&&(this.socket.send(JSON.stringify({display:this.config.deviceScreen})),this.connected=!0)},this.socket.binaryType="arraybuffer",this.socket.onmessage=e=>{try{if(typeof e.data=="string")return;const i=new Uint8Array(e.data);let r;const o=this.config.deviceScreen===w.FRONT?3:2;try{const a=Ht(i,o);this.config.deviceScreen===w.BACK?r=Jt(a):r=a,this.emitData(r)}catch{this.emitData(i)}}catch{this.connected=!1,this.emitStop()}},this.socket.onerror=e=>{this.connected=!1,this.emitError({code:1006,message:"WebSocket error occurred",raw:e}),this.emitStop()},this.socket.onclose=async e=>{if(this.socket=null,this.connected=!1,e.code===k||g.has(e.code)){this.emitError({code:e.code,message:e.reason,raw:e});return}this.emitStop()}}closeWebsocket(){return this.connected=!1,new Promise(t=>{this.socket?(this.socket.onclose=()=>{t()},this.socket.close(),this.socket=null):t(),this.emitStop()})}}class Yt{constructor(t){c(this,"addr");c(this,"connected",!1);c(this,"apiKey");c(this,"apiSemver");c(this,"inputEvent");c(this,"dataListeners",[]);c(this,"stopListeners",[]);c(this,"errorListeners",[]);c(this,"socket",null);if(!A())throw new Error("not browser");if(t!=null&&t.apiKey&&(this.apiKey=t.apiKey),t!=null&&t.apiSemver&&(this.apiSemver=t.apiSemver),!t||!t.addr)this.addr=S;else{let e=t.addr.trim();/^https?:\/\//i.test(e)||(e=`http://${e}`);try{const r=new URL(e).hostname;if(!T(r)&&!E(r))throw new Error(`Invalid address: "${t.addr}". Only IP addresses and mDNS names (ending in .local) are supported.`)}catch(i){throw i instanceof Error&&i.message.startsWith("Invalid address")?i:new Error(`Invalid URL format: "${t.addr}"`)}this.addr=e}this.inputEvent={}}onData(t){this.dataListeners.push(t)}onStop(t){this.stopListeners.push(t)}onError(t){this.errorListeners.push(t)}emitData(t){for(const e of this.dataListeners)e(t)}emitStop(){for(const t of this.stopListeners)t()}emitError(t){for(const e of this.errorListeners)e(t)}async openWebsocket(){this.socket&&await this.closeWebsocket();const t=new URL(`${this.addr}/api/input`);if(this.apiKey&&t.searchParams.append("x-api-token",this.apiKey),this.apiSemver&&t.searchParams.append("x-api-sem-ver",this.apiSemver),!t)throw new Error("The WebSocket URL is not specified");this.socket=new WebSocket(t),this.socket.onopen=()=>{this.socket&&(this.connected=!0)},this.socket.binaryType="arraybuffer",this.socket.onmessage=e=>{try{if(typeof e.data=="string")return;const i=new Uint8Array(e.data);this.emitData(i)}catch{this.connected=!1,this.emitStop()}},this.socket.onerror=e=>{this.connected=!1,this.emitError({code:1006,message:"WebSocket error occurred",raw:e}),this.emitStop()},this.socket.onclose=async e=>{if(this.socket=null,this.connected=!1,e.code===k||g.has(e.code)){this.emitError({code:e.code,message:e.reason,raw:e});return}this.emitStop()}}sendInput({keyName:t,value:e}){if(!this.socket||!this.connected)throw new Error("WebSocket: Not connected");this.inputEvent[t]=e,this.socket.send(JSON.stringify(this.inputEvent)),e===0&&delete this.inputEvent[t]}closeWebsocket(){return this.connected=!1,new Promise(t=>{this.socket?(this.socket.onclose=()=>{t()},this.socket.close(),this.socket=null):t(),this.emitStop()})}}exports.BusyBar=C;exports.DeviceScreen=w;exports.Input=Yt;exports.ScreenStream=Xt;
1
+ "use strict";var P=Object.defineProperty;var O=(n,t,i)=>t in n?P(n,t,{enumerable:!0,configurable:!0,writable:!0,value:i}):n[t]=i;var c=(n,t,i)=>O(n,typeof t!="symbol"?t+"":t,i);Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const m=require("openapi-fetch"),L=(n,t)=>{if(typeof FormData<"u"&&n instanceof FormData||typeof Buffer<"u"&&typeof Buffer.isBuffer=="function"&&Buffer.isBuffer(n)||typeof File<"u"&&n instanceof File||typeof Blob<"u"&&n instanceof Blob||typeof ArrayBuffer<"u"&&n instanceof ArrayBuffer||typeof ArrayBuffer<"u"&&ArrayBuffer.isView&&ArrayBuffer.isView(n))return n;let i;return t&&(t instanceof Headers?i=t.get("Content-Type")??t.get("content-type")??void 0:typeof t=="object"&&(i=t["Content-Type"]??t["content-type"]),i==="application/x-www-form-urlencoded")?n&&typeof n=="object"&&!(n instanceof URLSearchParams)?new URLSearchParams(n).toString():String(n):JSON.stringify(n)};async function p(n){const e=(n.headers.get("content-type")||"").includes("application/json")?await n.clone().json():await n.clone().text(),r=typeof e=="object"&&e!==null?e.error||e.message:typeof e=="string"?e:void 0;return Object.assign(new Error(r||`HTTP ${n.status} ${n.statusText}`),{status:n.status,statusText:n.statusText,body:e})}async function s(n,t=3e3){if(t<=0)return await n();const i=new AbortController,e=setTimeout(()=>i.abort(),t);try{return await n(i.signal)}catch(r){throw r instanceof DOMException&&r.name==="AbortError"?new Error(`Request timed out after ${t}ms`):r}finally{clearTimeout(e)}}function D(n,t,i){let e,r=i??void 0,o,a=null;const u=async()=>{e||(a||(a=(async()=>{const h=await t();if(!h.api_semver)throw new Error("Empty API version");e=h.api_semver})().finally(()=>{a=null})),await a)},l={async onRequest({request:h,schemaPath:d}){return r&&h.headers.set("Authorization",`Bearer ${r}`),d!=="/version"&&(await u(),e&&h.headers.set("X-API-Sem-Ver",e),o&&h.headers.set("X-API-Token",o)),h},async onResponse({request:h,response:d,options:v,schemaPath:G}){if(d.ok)return d;if(G==="/version")throw await p(d);if(d.status!==405)throw await p(d);e=void 0,await u(),e&&h.headers.set("X-API-Sem-Ver",e),r&&h.headers.set("Authorization",`Bearer ${r}`);const y=await(v.fetch??fetch)(h);if(y.ok)return y;throw await p(y)}},f=m({baseUrl:n,bodySerializer:L});return f.use(l),{client:f,setApiKey:h=>{o=h},setToken:h=>{r=h}}}async function U(n,t){const{data:i,error:e}=await s(r=>n.GET("/version",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function B(n,t){const{data:i,error:e}=await s(r=>n.GET("/status",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function I(n,t){const{data:i,error:e}=await s(r=>n.GET("/status/system",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function $(n,t){const{data:i,error:e}=await s(r=>n.GET("/status/power",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function R(n,t){const{data:i,error:e}=await s(r=>n.GET("/status/device",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function K(n,t){const{data:i,error:e}=await s(r=>n.GET("/status/firmware",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}class F{async SystemVersionGet(t){const i=await U(this.apiClient,t);return this.apiSemver=i.api_semver,i}async SystemStatusGet(t){return await B(this.apiClient,t)}async SystemInfoGet(t){return await I(this.apiClient,t)}async SystemStatusPowerGet(t){return await $(this.apiClient,t)}async SystemStatusDeviceGet(t){return await R(this.apiClient,t)}async SystemStatusFirmwareGet(t){return await K(this.apiClient,t)}}async function _(n,t){const{file:i}=t,{data:e,error:r}=await s(o=>n.POST("/update",{headers:{"Content-Type":"application/octet-stream"},body:i,signal:o}),t.timeout);if(r)throw r;return e}async function M(n,t){const{data:i,error:e}=await s(r=>n.POST("/update/check",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function W(n,t){const{data:i,error:e}=await s(r=>n.GET("/update/status",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function N(n,t){const{version:i}=t,{data:e,error:r}=await s(o=>n.GET("/update/changelog",{params:{query:{version:i}},signal:o}),t.timeout);if(r)throw r;return e}async function q(n,t){const{version:i}=t,{data:e,error:r}=await s(o=>n.POST("/update/install",{params:{query:{version:i}},signal:o}),t.timeout);if(r)throw r;return e}async function x(n,t){const{data:i,error:e}=await s(r=>n.POST("/update/abort_download",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function j(n,t){const{data:i,error:e}=await s(r=>n.GET("/update/autoupdate",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function V(n,t){const{is_enabled:i,interval_start:e,interval_end:r}=t,{data:o,error:a}=await s(u=>n.POST("/update/autoupdate",{body:{is_enabled:i,interval_start:e,interval_end:r},signal:u}),t.timeout);if(a)throw a;return o}class z{async UpdateFromFile(t){return await _(this.apiClient,t)}async UpdateCheck(t){return await M(this.apiClient,t)}async UpdateStatusGet(t){return await W(this.apiClient,t)}async UpdateChangelogGet(t){return await N(this.apiClient,t)}async UpdateInstall(t){return await q(this.apiClient,t)}async UpdateAbort(t){return await x(this.apiClient,t)}async UpdateAutoUpdateGet(t){return await j(this.apiClient,t)}async UpdateAutoUpdateSet(t){return await V(this.apiClient,t)}}async function H(n,t){const{data:i,error:e}=await s(r=>n.GET("/time",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function J(n,t){const{data:i,error:e}=await s(r=>n.POST("/time/timestamp",{params:{query:{...t,timeout:void 0}},signal:r}),t.timeout);if(e)throw e;return i}async function X(n,t){const{data:i,error:e}=await s(r=>n.GET("/time/timezone",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function Y(n,t){const{data:i,error:e}=await s(r=>n.POST("/time/timezone",{params:{query:{...t,timeout:void 0}},signal:r}),t.timeout);if(e)throw e;return i}async function Q(n,t){const{data:i,error:e}=await s(r=>n.GET("/time/tzlist",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}class Z{async TimeGet(t){return await H(this.apiClient,t)}async TimeTimestampSet(t){return await J(this.apiClient,t)}async TimeTimezoneGet(t){return await X(this.apiClient,t)}async TimeTimezoneSet(t){return await Y(this.apiClient,t)}async TimeTzListGet(t){return await Q(this.apiClient,t)}}async function tt(n,t){const{data:i,error:e}=await s(r=>n.GET("/account/status",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function et(n,t){const{data:i,error:e}=await s(r=>n.GET("/account/info",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function it(n,t){const{data:i,error:e}=await s(r=>n.GET("/account/profile",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function nt(n,t){const{profile:i}=t,{data:e,error:r}=await s(o=>n.POST("/account/profile",{params:{query:{profile:i}},signal:o}),t.timeout);if(r)throw r;return e}async function rt(n,t){const{data:i,error:e}=await s(r=>n.DELETE("/account",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function ot(n,t){const{data:i,error:e}=await s(r=>n.POST("/account/link",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}class st{async AccountInfoGet(t){return await et(this.apiClient,t)}async AccountStateGet(t){return await tt(this.apiClient,t)}async AccountProfileGet(t){return await it(this.apiClient,t)}async AccountProfileSet(t){return await nt(this.apiClient,t)}async AccountUnlink(t){return await rt(this.apiClient,t)}async AccountLink(t){return await ot(this.apiClient,t)}}async function at(n,t){const{appId:i,elements:e,priority:r=6}=t,{data:o,error:a}=await s(u=>n.POST("/display/draw",{body:{app_id:i,priority:r,elements:e},signal:u}),t.timeout);if(a)throw a;return o}async function ct(n,t){const{data:i,error:e}=await s(r=>n.DELETE("/display/draw",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function ut(n,t){const{display:i}=t,{data:e,error:r}=await s(o=>n.GET("/screen",{params:{query:{display:i}},parseAs:"blob",signal:o}),t.timeout);if(r)throw r;return e}async function ht(n,t){const{data:i,error:e}=await s(r=>n.GET("/display/brightness",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function dt(n,t){const{value:i}=t,r=(u=>{if(typeof u=="number"){if(u<0||u>100)throw new Error("Brightness value must be between 0 and 100 or 'auto'");return String(u)}if(u==="auto")return"auto"})(i),{data:o,error:a}=await s(u=>n.POST("/display/brightness",{params:{query:{value:r}},signal:u}),t.timeout);if(a)throw a;return o}class ft{async DisplayDraw(t){return await at(this.apiClient,t)}async DisplayClear(t){return await ct(this.apiClient,t)}async DisplayScreenFrameGet(t){return await ut(this.apiClient,t)}async DisplayBrightnessGet(t){return await ht(this.apiClient,t)}async DisplayBrightnessSet(t){return await dt(this.apiClient,t)}}async function lt(n,t){const{appId:i,path:e}=t,{data:r,error:o}=await s(a=>n.POST("/audio/play",{params:{query:{app_id:i,path:e}},signal:a}),t.timeout);if(o)throw o;return r}async function wt(n,t){const{data:i,error:e}=await s(r=>n.DELETE("/audio/play",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function yt(n,t){const{data:i,error:e}=await s(r=>n.GET("/audio/volume",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function pt(n,t){const{volume:i}=t;if(typeof i!="number"||i<0||i>100)throw new Error("Volume must be a number between 0 and 100");const{data:e,error:r}=await s(o=>n.POST("/audio/volume",{params:{query:{volume:i}},signal:o}),t.timeout);if(r)throw r;return e}class St{async AudioPlay(t){return await lt(this.apiClient,t)}async AudioStop(t){return await wt(this.apiClient,t)}async AudioVolumeGet(t){return await yt(this.apiClient,t)}async AudioVolumeSet(t){return await pt(this.apiClient,t)}}async function Tt(n,t){const{data:i,error:e}=await s(r=>n.GET("/wifi/status",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function Et(n,t){const{data:i,error:e}=await s(r=>n.POST("/wifi/connect",{body:{ssid:t.ssid,password:t.password,security:t.security,ip_config:{ip_method:t.ipConfig.ipMethod,address:t.ipConfig.address,mask:t.ipConfig.mask,gateway:t.ipConfig.gateway}},signal:r}),t.timeout);if(e)throw e;return i}async function mt(n,t){const{data:i,error:e}=await s(r=>n.POST("/wifi/disconnect",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function Ct(n,t){const{data:i,error:e}=await s(r=>n.GET("/wifi/networks",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}class kt{async WifiStatusGet(t){return await Tt(this.apiClient,t)}async WifiConnect(t){return await Et(this.apiClient,t)}async WifiDisconnect(t){return await mt(this.apiClient,t)}async WifiNetworksGet(t){return await Ct(this.apiClient,t)}}async function gt(n,t){const{path:i,file:e}=t,{data:r,error:o}=await s(a=>n.POST("/storage/write",{params:{query:{path:i}},headers:{"Content-Type":"application/octet-stream"},body:e,signal:a}),t.timeout);if(o)throw o;return r}async function At(n,t){const{path:i,asArrayBuffer:e}=t,{data:r,error:o}=await s(a=>n.GET("/storage/read",{params:{query:{path:i}},parseAs:e?"arrayBuffer":"blob",signal:a}),t.timeout);if(o)throw o;return r}async function bt(n,t){const{path:i}=t,{data:e,error:r}=await s(o=>n.GET("/storage/list",{params:{query:{path:i}},signal:o}),t.timeout);if(r)throw r;return e}async function vt(n,t){const{path:i}=t,{data:e,error:r}=await s(o=>n.DELETE("/storage/remove",{params:{query:{path:i}},signal:o}),t.timeout);if(r)throw r;return e}async function Gt(n,t){const{path:i}=t,{data:e,error:r}=await s(o=>n.POST("/storage/mkdir",{params:{query:{path:i}},signal:o}),t.timeout);if(r)throw r;return e}async function Pt(n,t){const{data:i,error:e}=await s(r=>n.GET("/storage/status",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function Ot(n,t){const{path:i,new_path:e}=t,{data:r,error:o}=await s(a=>n.POST("/storage/rename",{params:{query:{path:i,new_path:e}},signal:a}),t.timeout);if(o)throw o;return r}class Lt{async StorageWrite(t){return await gt(this.apiClient,t)}async StorageRead(t){return await At(this.apiClient,t)}async StorageListGet(t){return await bt(this.apiClient,t)}async StorageRemove(t){return await vt(this.apiClient,t)}async StorageMkdir(t){return await Gt(this.apiClient,t)}async StorageStatusGet(t){return await Pt(this.apiClient,t)}async StorageRename(t){return await Ot(this.apiClient,t)}}async function Dt(n,t){const{data:i,error:e}=await s(r=>n.GET("/access",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function Ut(n,t){let{mode:i,key:e}=t;if(e=e??"",String(e).trim()&&!/^\d{4,10}$/.test(String(e)))throw new Error("Key must be a string of 4 to 10 digits");const{data:r,error:o}=await s(a=>n.POST("/access",{params:{query:{mode:i,key:e}},signal:a}),t.timeout);if(o)throw o;return r}async function Bt(n,t){const{data:i,error:e}=await s(r=>n.GET("/name",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function It(n,t){const{data:i,error:e}=await s(r=>n.POST("/name",{body:t,signal:r}),t.timeout);if(e)throw e;return i}class $t{async SettingsAccessGet(t){return await Dt(this.apiClient,t)}async SettingsAccessSet(t){const i=await Ut(this.apiClient,t);return t.mode==="key"&&t.key&&this.setApiKey(t.key),i}async SettingsNameGet(t){return await Bt(this.apiClient,t)}async SettingsNameSet(t){return await It(this.apiClient,t)}}async function Rt(n,t){const{data:i,error:e}=await s(r=>n.POST("/ble/enable",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function Kt(n,t){const{data:i,error:e}=await s(r=>n.POST("/ble/disable",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function Ft(n,t){const{data:i,error:e}=await s(r=>n.DELETE("/ble/pairing",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function _t(n,t){const{data:i,error:e}=await s(r=>n.GET("/ble/status",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}class Mt{async BleEnable(t){return await Rt(this.apiClient,t)}async BleDisable(t){return await Kt(this.apiClient,t)}async BleUnpair(t){return await Ft(this.apiClient,t)}async BleStatusGet(t){return await _t(this.apiClient,t)}}async function Wt(n,t){const{keyName:i}=t,{data:e,error:r}=await s(o=>n.POST("/input",{params:{query:{key:i}},signal:o}),t.timeout);if(r)throw r;return e}class Nt{async InputSend(t){return await Wt(this.apiClient,t)}}async function qt(n,t){const{data:i,error:e}=await s(r=>n.GET("/matter/commissioning",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function xt(n,t){const{data:i,error:e}=await s(r=>n.POST("/matter/commissioning",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}async function jt(n,t){const{data:i,error:e}=await s(r=>n.DELETE("/matter/commissioning",{signal:r}),t==null?void 0:t.timeout);if(e)throw e;return i}class Vt{async MatterStatusGet(t){return await qt(this.apiClient,t)}async MatterPair(t){return await xt(this.apiClient,t)}async MatterErase(t){return await jt(this.apiClient,t)}}async function zt(n,t){const{appId:i,fileName:e,file:r}=t,{data:o,error:a}=await s(u=>n.POST("/assets/upload",{params:{query:{app_id:i,file:e}},headers:{"Content-Type":"application/octet-stream"},body:r,signal:u}),t.timeout);if(a)throw a;return o}async function Ht(n,t){const{appId:i}=t,{data:e,error:r}=await s(o=>n.DELETE("/assets/upload",{params:{query:{app_id:i}},signal:o}),t.timeout);if(r)throw r;return e}class Jt{async AssetsUpload(t){return await zt(this.apiClient,t)}async AssetsDelete(t){return await Ht(this.apiClient,t)}}const S="http://10.0.4.20",Xt="https://proxy.busy.app",Yt=/^https?:\/\/proxy(?:\.(?:dev|test|stage))?\.busy\.app$/i;function T(n){const t=n.split(".");if(t.length!==4)return!1;for(const i of t){if(i.length===0||i.length>1&&i[0]==="0"||!/^\d+$/.test(i))return!1;const e=Number(i);if(e<0||e>255)return!1}return!0}function E(n){return/\.local$/i.test(n)}class C{constructor(t){c(this,"addr");c(this,"apiSemver");c(this,"apiClient");c(this,"setApiKeyFn");c(this,"setTokenFn");c(this,"connectionType","unknown");if(!t||!t.addr&&!t.token)this.addr=S;else if(!t.addr)this.addr=Xt;else{let o=t.addr.trim();if(/^https?:\/\//i.test(o)||(o=`http://${o}`),Yt.test(o)&&!t.token)throw new Error("Token is required. Please provide it.");this.addr=o}this.apiSemver="";const{client:i,setApiKey:e,setToken:r}=D(`${this.addr}/api/`,this.SystemVersionGet.bind(this),t==null?void 0:t.token);this.apiClient=i,this.setApiKeyFn=e,this.setTokenFn=r,this.detectConnectionType()}async detectConnectionType(){const t=new URL(this.addr).hostname;if(!T(t)&&!E(t)){this.connectionType="wifi";return}const i=m({baseUrl:`${this.addr}/api/`});try{const{response:e}=await i.GET("/name");if(e.status===401||e.status===403)this.connectionType="wifi";else if(e.ok)this.connectionType="usb";else throw new Error(`Failed to detect connection type. Status: ${e.status}`)}catch(e){throw e}}setApiKey(t){this.setApiKeyFn(t)}setToken(t){this.setTokenFn(t)}}function Qt(n,t){t.forEach(i=>{Object.getOwnPropertyNames(i.prototype).forEach(e=>{Object.defineProperty(n.prototype,e,Object.getOwnPropertyDescriptor(i.prototype,e)||Object.create(null))})})}Qt(C,[F,z,Z,st,ft,St,kt,Lt,$t,Mt,Nt,Vt,Jt]);var w=(n=>(n[n.FRONT=0]="FRONT",n[n.BACK=1]="BACK",n))(w||{});const k=3e3,g=new Set([1001,1006,1012,1013,1014,3008]);function A(n,t){if(t<0||t>=n.length)throw new Error(`Index ${t} is out of bounds (0…${n.length-1})`);const i=n[t];if(i===void 0)throw new Error(`Unexpected undefined at index ${t}`);return i}function Zt(n,t){let i=0;const e=n.length,r=[];for(;i<e;){const o=A(n,i);if(i+=1,(o&128)!==0){const a=o&127;for(let u=0;u<a*t;u++)r.push(n[i+u]);i+=a*t}else{const a=o,u=n.slice(i,i+t);for(let l=0;l<a;l++)for(let f=0;f<t;f++)r.push(u[f]);i+=t}}return new Uint8Array(r)}function te(n){const t=new Uint8Array(n.length*2);let i=0,e=0;for(;i<n.length;){const r=A(n,i),o=r&15,a=r>>4&15;t[e]=o,t[e+1]=a,i+=1,e+=2}return t}const b=()=>typeof window<"u"&&typeof window.document<"u";class ee{constructor(t){c(this,"addr");c(this,"connected",!1);c(this,"apiKey");c(this,"apiSemver");c(this,"dataListeners",[]);c(this,"stopListeners",[]);c(this,"errorListeners",[]);c(this,"socket",null);if(this.config=t,!b())throw new Error("not browser");if(t.apiKey&&(this.apiKey=t.apiKey),t.apiSemver&&(this.apiSemver=t.apiSemver),!t||!t.addr)this.addr=S;else{let i=t.addr.trim();/^https?:\/\//i.test(i)||(i=`http://${i}`);try{const r=new URL(i).hostname;if(!T(r)&&!E(r))throw new Error(`Invalid address: "${t.addr}". Only IP addresses and mDNS names (ending in .local) are supported.`)}catch(e){throw e instanceof Error&&e.message.startsWith("Invalid address")?e:new Error(`Invalid URL format: "${t.addr}"`)}this.addr=i}}onData(t){this.dataListeners.push(t)}onStop(t){this.stopListeners.push(t)}onError(t){this.errorListeners.push(t)}emitData(t){for(const i of this.dataListeners)i(t)}emitStop(){for(const t of this.stopListeners)t()}emitError(t){for(const i of this.errorListeners)i(t)}async openWebsocket(){this.socket&&await this.closeWebsocket();const t=new URL(`${this.addr}/api/screen/ws`);if(this.apiKey&&t.searchParams.append("x-api-token",this.apiKey),this.apiSemver&&t.searchParams.append("x-api-sem-ver",this.apiSemver),!t)throw new Error("The WebSocket URL is not specified");this.socket=new WebSocket(t),this.socket.onopen=()=>{this.socket&&(this.socket.send(JSON.stringify({display:this.config.deviceScreen})),this.connected=!0)},this.socket.binaryType="arraybuffer",this.socket.onmessage=i=>{try{if(typeof i.data=="string")return;const e=new Uint8Array(i.data);let r;const o=this.config.deviceScreen===w.FRONT?3:2;try{const a=Zt(e,o);this.config.deviceScreen===w.BACK?r=te(a):r=a,this.emitData(r)}catch{this.emitData(e)}}catch{this.connected=!1,this.emitStop()}},this.socket.onerror=i=>{this.connected=!1,this.emitError({code:1006,message:"WebSocket error occurred",raw:i}),this.emitStop()},this.socket.onclose=async i=>{if(this.socket=null,this.connected=!1,i.code===k||g.has(i.code)){this.emitError({code:i.code,message:i.reason,raw:i});return}this.emitStop()}}closeWebsocket(){return this.connected=!1,new Promise(t=>{this.socket?(this.socket.onclose=()=>{t()},this.socket.close(),this.socket=null):t(),this.emitStop()})}}class ie{constructor(t){c(this,"addr");c(this,"connected",!1);c(this,"apiKey");c(this,"apiSemver");c(this,"inputEvent");c(this,"dataListeners",[]);c(this,"stopListeners",[]);c(this,"errorListeners",[]);c(this,"socket",null);if(!b())throw new Error("not browser");if(t!=null&&t.apiKey&&(this.apiKey=t.apiKey),t!=null&&t.apiSemver&&(this.apiSemver=t.apiSemver),!t||!t.addr)this.addr=S;else{let i=t.addr.trim();/^https?:\/\//i.test(i)||(i=`http://${i}`);try{const r=new URL(i).hostname;if(!T(r)&&!E(r))throw new Error(`Invalid address: "${t.addr}". Only IP addresses and mDNS names (ending in .local) are supported.`)}catch(e){throw e instanceof Error&&e.message.startsWith("Invalid address")?e:new Error(`Invalid URL format: "${t.addr}"`)}this.addr=i}this.inputEvent={}}onData(t){this.dataListeners.push(t)}onStop(t){this.stopListeners.push(t)}onError(t){this.errorListeners.push(t)}emitData(t){for(const i of this.dataListeners)i(t)}emitStop(){for(const t of this.stopListeners)t()}emitError(t){for(const i of this.errorListeners)i(t)}async openWebsocket(){this.socket&&await this.closeWebsocket();const t=new URL(`${this.addr}/api/input`);if(this.apiKey&&t.searchParams.append("x-api-token",this.apiKey),this.apiSemver&&t.searchParams.append("x-api-sem-ver",this.apiSemver),!t)throw new Error("The WebSocket URL is not specified");this.socket=new WebSocket(t),this.socket.onopen=()=>{this.socket&&(this.connected=!0)},this.socket.binaryType="arraybuffer",this.socket.onmessage=i=>{try{if(typeof i.data=="string")return;const e=new Uint8Array(i.data);this.emitData(e)}catch{this.connected=!1,this.emitStop()}},this.socket.onerror=i=>{this.connected=!1,this.emitError({code:1006,message:"WebSocket error occurred",raw:i}),this.emitStop()},this.socket.onclose=async i=>{if(this.socket=null,this.connected=!1,i.code===k||g.has(i.code)){this.emitError({code:i.code,message:i.reason,raw:i});return}this.emitStop()}}sendInput({keyName:t,value:i}){if(!this.socket||!this.connected)throw new Error("WebSocket: Not connected");this.inputEvent[t]=i,this.socket.send(JSON.stringify(this.inputEvent)),i===0&&delete this.inputEvent[t]}closeWebsocket(){return this.connected=!1,new Promise(t=>{this.socket?(this.socket.onclose=()=>{t()},this.socket.close(),this.socket=null):t(),this.emitStop()})}}exports.BusyBar=C;exports.DeviceScreen=w;exports.Input=ie;exports.ScreenStream=ee;