@urun-sh/core 0.1.32 → 0.1.34

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.mjs CHANGED
@@ -1 +1 @@
1
- import{a as A,b as x,c as O}from"./chunk-HAKPBBUN.mjs";import*as g from"yjs";var P=Symbol("urun:doc:remote"),B=Symbol("urun:doc:local");function v(n){return n!==null&&typeof n=="object"&&!Array.isArray(n)}function R(n,e){let t={...n};for(let r of Object.keys(e)){let s=e[r],o=t[r];v(s)&&v(o)?t[r]=R(o,s):t[r]=s}return t}function F(n,e){let t=e.split("."),r=n;for(let s of t){if(r==null||typeof r!="object")return;r=r[s]}return r}function U(n){return n===void 0?n:JSON.parse(JSON.stringify(n))}var k=class{_doc=new g.Doc;_root=this._doc.getMap("session");_listeners=new Set;_key;_sessionId;_transport;_transportUnsubs=[];_synced;_pendingPatches=[];_observer=()=>this.notify();_updateHandler=(e,t)=>{t!==P&&this._transport?.sendDocSync(e)};constructor(e,t,r){this._key=e,this._sessionId=t,this._transport=r??null,this._synced=this._transport===null,this._root.observeDeep(this._observer),this._doc.on("update",this._updateHandler),this._transport&&(this._transportUnsubs.push(this._transport.onDocSync(s=>this.applyRemote(s)),this._transport.onDocSyncReady(()=>this.sendHello())),this._transport.isOpen&&this.sendHello())}setSessionId(e){this._sessionId=e}set(e){if(!this._synced){this._pendingPatches.push(U(e)),this.notify();return}this.applyPatch(e)}get(e,t){let r=this.snapshot();return e?F(r,e)??t:r}on(e,t){return this._listeners.add(t),()=>this._listeners.delete(t)}dispose(){this._root.unobserveDeep(this._observer),this._doc.off("update",this._updateHandler);for(let e of this._transportUnsubs)e();this._transportUnsubs=[],this._transport=null,this._listeners.clear(),this._doc.destroy()}sendHello(){this._transport&&this._transport.sendDocSync(g.encodeStateAsUpdate(this._doc))}applyRemote(e){if(g.applyUpdate(this._doc,e,P),!this._synced){this._synced=!0;let t=this._pendingPatches.splice(0);for(let r of t)this.applyPatch(r)}}applyPatch(e){this._doc.transact(()=>{let t=this._root.toJSON();for(let[r,s]of Object.entries(e)){let o=t[r],a=v(s)&&v(o)?R(o,s):U(s);JSON.stringify(a)!==JSON.stringify(o)&&this._root.set(r,a)}},B)}snapshot(){let e=this._root.toJSON();for(let t of this._pendingPatches)e=R(e,t);return e}notify(){let e=this.snapshot();for(let t of this._listeners)t(e)}};var G=new Set(["control"]),I=class{constructor(e,t){this._name=e;this._transport=t;this._track=this._transport.getTrackByName(this._name)??null,this._unsubscribeTransport=this._transport.on("track",()=>{let r=this._transport.getTrackByName(this._name)??null;r!==this._track&&this._setTrack(r)})}_name;_transport;_track=null;_handlers=new Map;_unsubscribeTransport=null;get track(){let e=this._transport.getTrackByName(this._name)??this._track;return e!==this._track&&this._setTrack(e),this._track}async attach(e){await this._transport.addTrack(e)}async detach(){this._transport.removeTrack()}async seek(e){if(e!=="live"&&(!Number.isFinite(e)||e<0))throw new Error('[urun] stream.seek target must be "live" or a non-negative number of seconds');this._transport.seekStream(this._name,e)}on(e,t){let r=this._handlers.get(e);return r||(r=new Set,this._handlers.set(e,r)),r.add(t),()=>{r?.delete(t),r?.size===0&&this._handlers.delete(e)}}dispose(){this._unsubscribeTransport?.(),this._unsubscribeTransport=null,this._handlers.clear()}_setTrack(e){this._track=e,e&&e.addEventListener("ended",()=>{this._track===e&&this._setTrack(null)},{once:!0});let t=this._handlers.get("track");if(t)for(let r of t)r(e)}},w=class{_sessionId;_multiplexer;_transport;_docs=new Map;_streams=new Map;constructor(e,t,r){this._sessionId=e,this._multiplexer=t,this._transport=r}get id(){return this._sessionId}setSessionId(e){if(e===this._sessionId)return;this._multiplexer.rewriteSessionChannels?.call(this._multiplexer,this._sessionId,e),this._sessionId=e;for(let r of this._docs.values())r.setSessionId(e)}stream(e){let t=this._streams.get(e);return t||(t=new I(e,this._transport),this._streams.set(e,t)),t}doc(e){let t=this._docs.get(e);return t||(t=new k(e,this._sessionId,G.has(e)?this._transport:null),this._docs.set(e,t)),t}disconnect(){for(let e of this._docs.values())e.dispose();this._docs.clear();for(let e of this._streams.values())e.dispose();this._streams.clear(),this._transport.disconnect()}};var J="session-api.prod.cloud.urun.sh",H=["https://session-api.use2.prod.cloud.urun.sh","https://session-api.usw2.prod.cloud.urun.sh"];function L(n){return n.trim().replace(/\/+$/,"")}function K(n,e){let t=[],r=s=>{if(!s)return;let o=L(s);o&&!t.includes(o)&&t.push(o)};r(n);for(let s of e??[])r(s);for(let s of re(n))r(s);return t}var q=120,z=1e3,W=3;function Y(n){if(!n)return null;try{let e=new URL(n),t=e.pathname.replace(/\/api\/sessions\/create\/?$/,"");return L(`${e.origin}${t}`)}catch{return null}}async function D(n){let e=K(n.baseUrl,n.fallbackUrls),t=null,r=e.length>1;for(let s of e)try{return await V(s,n,{pollAttempts:q,retryAfterLimitMs:r?z:void 0})}catch(o){if(t=o instanceof Error?o:new Error(String(o)),t instanceof S)throw t}throw t??new Error("[urun] session allocation failed")}async function V(n,e,t){let r,s,o,a=n,p=0,h=new Set;for(let l=0;l<t.pollAttempts;l++){s=o??await C(e),o=void 0;let d=r?await fetch(`${a}/api/session-requests/${encodeURIComponent(r)}`,{headers:M(e,s)}):await fetch(`${a}/api/sessions/create`,{method:"POST",redirect:"manual",headers:{...M(e,s),"Content-Type":"application/json","Idempotency-Key":e.idempotencyKey},body:JSON.stringify(X(e))});if(!r&&(d.status===307||d.status===308)){let u=Y(d.headers.get("location"));if(u&&u!==a&&p<W){p+=1,a=u;continue}throw new Error(`[urun] session create redirected from ${a} without a usable target`)}let i=await Q(d);if(d.ok&&(i.status==="allocated"||!i.status&&!!i.session_id)&&!!i.session_id){let u=i.session_id;if(!i.ws_url)throw new S(`[urun] gateway allocated session ${u} without ws_url`);if(!te(i.ws_url))throw new S(`[urun] gateway allocated session ${u} with non-absolute ws_url`);return{baseUrl:a,sessionId:u,wsUrl:i.ws_url,response:i}}if(d.status===202||i.status==="pending"||i.status==="queued"||i.queued){if(r=i.request_id||r,!r)throw new Error("[urun] gateway queued session without request_id");await ee(Z(d,i,t.retryAfterLimitMs));continue}if((d.status===401||d.status===403)&&e.getAccessToken){let u=s??"";h.add(u);let _=await C(e,{forceRefresh:!0,reason:"unauthorized"});if(_&&!h.has(_)){o=_,r=void 0;continue}}let m=i.error_message||i.error_code||`HTTP ${d.status}`;throw new Error(`[urun] session allocation failed at ${a}: ${m}`)}throw new Error(`[urun] timed out waiting for session allocation at ${a}`)}function M(n,e){let t={"X-Tenant-Id":n.orgId,"X-Auth-Provider":n.authProvider||"default","X-User-Id":"anonymous"};return e&&(t.Authorization=`Bearer ${e}`),t}async function C(n,e){return n.getAccessToken?await n.getAccessToken(e)??void 0:n.jwt}function X(n){let e=Number(n.args?.gpus??1),t={app:n.app,function:n.functionName,gpus:Number.isFinite(e)&&e>0?e:1,idempotency_key:n.idempotencyKey},r=n.args?.preferred_zone;return typeof r=="string"&&r&&(t.preferred_zone=r),t}async function Q(n){try{let e=await n.json();return e&&typeof e=="object"?e:{}}catch{return{}}}function Z(n,e,t){let r=Number(n.headers.get("Retry-After")||""),s=Number(e.retry_after_seconds??r),o=t??1e4;return Number.isFinite(s)&&s>0?Math.min(s*1e3,o):1e3}function ee(n){return new Promise(e=>setTimeout(e,n))}var S=class extends Error{};function te(n){try{let e=new URL(n);return e.protocol==="ws:"||e.protocol==="wss:"}catch{return!1}}function re(n){try{return new URL(n).hostname!==J?[]:H}catch{return[]}}var ne=new Set(["then","catch","finally","toJSON",Symbol.toPrimitive]);function se(n,e){return new Proxy({},{get(t,r){if(!(ne.has(r)||typeof r!="string"))return s=>{let o=s??{},a=()=>`s_${Date.now()}_${Math.random().toString(36).slice(2)}`,p=a(),h=new x,l=new O({url:"",orgId:e.orgId,jwt:e.jwt,authProvider:e.authProvider,sessionId:p}),d=new w(p,h,l),i=async c=>{let m=await D({baseUrl:e.baseUrl,fallbackUrls:e.fallbackUrls,app:n,functionName:r,args:o,orgId:e.orgId,jwt:e.jwt,getAccessToken:e.getAccessToken,authProvider:e.authProvider,idempotencyKey:c}),u=e.getAccessToken?await e.getAccessToken()??void 0:e.jwt;return d.setSessionId(m.sessionId),{url:m.wsUrl,sessionId:m.sessionId,jwt:u}};return l.setConnectionResolver(()=>i(a())),l.on("connected",()=>{l.multiplexer&&h.setTarget(l.multiplexer)}),(async()=>{let c=await i(p);l.setAuth({jwt:c.jwt,authProvider:e.authProvider,orgId:e.orgId}),l.setConnection({url:c.url,sessionId:c.sessionId}),await l.connect({app:n,functionName:r,args:o})})().catch(c=>{console.error(`[urun] Failed to connect session for ${r}:`,c)}),d}}})}var N=3e4,$="stream:",T=class{_ws;_multiplexer;_pendingRpc=new Map;_rpcId=0;constructor(e,t){this._ws=e,this._multiplexer=t,this._ws.addEventListener("message",r=>{let s;try{s=JSON.parse(typeof r.data=="string"?r.data:"")}catch{return}if(s.rpcId&&this._pendingRpc.has(s.rpcId)){let o=this._pendingRpc.get(s.rpcId);this._pendingRpc.delete(s.rpcId),clearTimeout(o.timer),s.error?o.reject(new Error(s.error)):o.resolve(s.result)}})}async get(e){return this._sendRpc("store.get",{key:e})}async has(e){return await this._sendRpc("store.has",{key:e})}on(e,t){let r=`${$}${e}`;return this._multiplexer.on(r,s=>{t(s)})}emit(e,t){let r=`${$}${e}`,s=typeof t=="object"&&t!==null&&!Array.isArray(t)?t:{data:t};this._multiplexer.emit(r,s)}_getInternals(){return{multiplexer:this._multiplexer,ws:this._ws}}_sendRpc(e,t){return new Promise((r,s)=>{if(!this._ws||this._ws.readyState!==WebSocket.OPEN){s(new Error("WebSocket not open"));return}let o=String(++this._rpcId),a=setTimeout(()=>{this._pendingRpc.has(o)&&(this._pendingRpc.delete(o),s(new Error(`RPC request "${e}" timed out after ${N}ms`)))},N);this._pendingRpc.set(o,{resolve:r,reject:s,timer:a}),this._ws.send(JSON.stringify({rpcId:o,type:e,...t}))})}};function oe(n){let r=`${n.baseUrl.replace(/\/$/,"").replace(/^http/,"ws")}/store/${encodeURIComponent(n.orgId)}`,s=new WebSocket(r);(n.jwt||n.apiKey)&&s.addEventListener("open",()=>{s.send(JSON.stringify({type:"auth",orgId:n.orgId,jwt:n.jwt,authProvider:n.authProvider,apiKey:n.apiKey}))});let o=new A(s);return new T(s,o)}var ie='video/mp4; codecs="avc1.42E01E"';function ae(n,e){let t=e?.codec??ie,r=document.createElement("video");r.muted=!0,r.playsInline=!0;let s=new MediaSource,o=URL.createObjectURL(s);r.src=o,s.addEventListener("sourceopen",()=>{let c;try{c=s.addSourceBuffer(t)}catch(f){throw f instanceof DOMException&&f.name==="NotSupportedError"?new Error(`Codec not supported: "${t}". Ensure the codec string matches the fMP4 container format. Common values: video/mp4; codecs="avc1.42E01E" (H.264 Baseline), video/mp4; codecs="avc1.4D401F" (H.264 Main).`):f}let m=n.getReader();function u(){return c.updating?new Promise(f=>{c.addEventListener("updateend",()=>f(),{once:!0})}):Promise.resolve()}async function _(){try{for(;;){let{done:f,value:E}=await m.read();if(f){await u(),s.readyState==="open"&&s.endOfStream();return}await u();try{c.appendBuffer(E)}catch(y){if(y instanceof DOMException&&y.name==="QuotaExceededError"){let b=c.buffered;if(b.length>0){let j=b.start(b.length-1);c.remove(0,j),await u(),c.appendBuffer(E)}else throw y}else throw y}}}catch{if(s.readyState==="open")try{s.endOfStream("decode")}catch{}}}_()}),r.play().catch(()=>{});let a=r,p=a.captureStream??a.mozCaptureStream;if(!p)throw new Error("captureStream() is not supported in this browser. videoStream() requires HTMLMediaElement.captureStream() support.");let h=p.call(r),l=h.getTracks(),d=0,i=l.length;if(i===0)r.addEventListener("emptied",()=>{URL.revokeObjectURL(o)},{once:!0});else for(let c of l)c.addEventListener("ended",()=>{d++,d>=i&&URL.revokeObjectURL(o)});return h}export{se as App,w as Session,oe as createStore,ae as videoStream};
1
+ import{a as M,b as v,c as k}from"./chunk-TSVDZX3O.mjs";var H="session-api.prod.cloud.urun.sh",K=["https://session-api.use2.prod.cloud.urun.sh","https://session-api.usw2.prod.cloud.urun.sh"];function b(n){return n.trim().replace(/\/+$/,"")}function C(n,e){let t=[],r=s=>{if(!s)return;let o=b(s);o&&!t.includes(o)&&t.push(o)};r(n);for(let s of e??[])r(s);for(let s of ee(n))r(s);return t}var q=120,V=1e3,W=3;function Y(n){if(!n)return null;try{let e=new URL(n),t=e.pathname.replace(/\/api\/sessions\/create\/?$/,"");return b(`${e.origin}${t}`)}catch{return null}}async function L(n){let e=C(n.baseUrl,n.fallbackUrls),t=null,r=e.length>1;for(let s of e)try{return await z(s,n,{pollAttempts:q,retryAfterLimitMs:r?V:void 0})}catch(o){if(t=o instanceof Error?o:new Error(String(o)),t instanceof w)throw t}throw t??new Error("[urun] session allocation failed")}async function z(n,e,t){let r,s,o,i=n,l=0,h=new Set;for(let p=0;p<t.pollAttempts;p++){s=o??await E(e),o=void 0;let d=r?await fetch(`${i}/api/session-requests/${encodeURIComponent(r)}`,{headers:A(e,s)}):await fetch(`${i}/api/sessions/create`,{method:"POST",redirect:"manual",headers:{...A(e,s),"Content-Type":"application/json","Idempotency-Key":e.idempotencyKey},body:JSON.stringify(X(e))});if(!r&&(d.status===307||d.status===308)){let u=Y(d.headers.get("location"));if(u&&u!==i&&l<W){l+=1,i=u;continue}throw new Error(`[urun] session create redirected from ${i} without a usable target`)}let a=await $(d);if(d.ok&&(a.status==="allocated"||!a.status&&!!a.session_id)&&!!a.session_id){let u=a.session_id;if(!a.ws_url)throw new w(`[urun] gateway allocated session ${u} without ws_url`);if(!D(a.ws_url))throw new w(`[urun] gateway allocated session ${u} with non-absolute ws_url`);return{baseUrl:i,sessionId:u,wsUrl:a.ws_url,response:a}}if(d.status===202||a.status==="pending"||a.status==="queued"||a.queued){if(r=a.request_id||r,!r)throw new Error("[urun] gateway queued session without request_id");await Z(Q(d,a,t.retryAfterLimitMs));continue}if((d.status===401||d.status===403)&&e.getAccessToken){let u=s??"";h.add(u);let _=await E(e,{forceRefresh:!0,reason:"unauthorized"});if(_&&!h.has(_)){o=_,r=void 0;continue}}let m=a.error_message||a.error_code||`HTTP ${d.status}`;throw new Error(`[urun] session allocation failed at ${i}: ${m}`)}throw new Error(`[urun] timed out waiting for session allocation at ${i}`)}function A(n,e){let t={"X-Tenant-Id":n.orgId,"X-Auth-Provider":n.authProvider||"default","X-User-Id":"anonymous"};return e&&(t.Authorization=`Bearer ${e}`),t}async function E(n,e){return n.getAccessToken?await n.getAccessToken(e)??void 0:n.jwt}function X(n){return{app:n.app,function:n.functionName,gpus:1,idempotency_key:n.idempotencyKey}}async function j(n){let e=C(n.baseUrl,n.fallbackUrls),t=null;for(let r of e)try{let s={baseUrl:r,app:"",functionName:"",orgId:n.orgId,jwt:n.jwt,getAccessToken:n.getAccessToken,authProvider:n.authProvider,idempotencyKey:""},o=await E(s),i=await fetch(`${b(r)}/api/sessions/${encodeURIComponent(n.sessionId)}/viewer-connect`,{headers:A(s,o)});if(!i.ok)throw new Error(`[urun] viewer-connect for ${n.sessionId} failed: HTTP ${i.status}`);let l=await $(i);if(!l.ws_url||!D(l.ws_url))throw new w(`[urun] viewer-connect returned no usable ws_url for ${n.sessionId}`);return{baseUrl:b(r),sessionId:n.sessionId,wsUrl:l.ws_url,response:l}}catch(s){if(t=s instanceof Error?s:new Error(String(s)),t instanceof w)throw t}throw t??new Error("[urun] viewer-connect failed")}async function $(n){try{let e=await n.json();return e&&typeof e=="object"?e:{}}catch{return{}}}function Q(n,e,t){let r=Number(n.headers.get("Retry-After")||""),s=Number(e.retry_after_seconds??r),o=t??1e4;return Number.isFinite(s)&&s>0?Math.min(s*1e3,o):1e3}function Z(n){return new Promise(e=>setTimeout(e,n))}var w=class extends Error{};function D(n){try{let e=new URL(n);return e.protocol==="ws:"||e.protocol==="wss:"}catch{return!1}}function ee(n){try{return new URL(n).hostname!==H?[]:K}catch{return[]}}import*as g from"yjs";var N=Symbol("urun:doc:remote"),te=Symbol("urun:doc:local");function R(n){return n!==null&&typeof n=="object"&&!Array.isArray(n)}function O(n,e){let t={...n};for(let r of Object.keys(e)){let s=e[r],o=t[r];R(s)&&R(o)?t[r]=O(o,s):t[r]=s}return t}function re(n,e){let t=e.split("."),r=n;for(let s of t){if(r==null||typeof r!="object")return;r=r[s]}return r}function B(n){return n===void 0?n:JSON.parse(JSON.stringify(n))}var I=class{_doc=new g.Doc;_root=this._doc.getMap("session");_listeners=new Set;_key;_sessionId;_transport;_transportUnsubs=[];_synced;_pendingPatches=[];_observer=()=>this.notify();_updateHandler=(e,t)=>{t!==N&&this._transport?.sendDocSync(e)};constructor(e,t,r){this._key=e,this._sessionId=t,this._transport=r??null,this._synced=this._transport===null,this._root.observeDeep(this._observer),this._doc.on("update",this._updateHandler),this._transport&&(this._transportUnsubs.push(this._transport.onDocSync(s=>this.applyRemote(s)),this._transport.onDocSyncReady(()=>this.sendHello())),this._transport.isOpen&&this.sendHello())}setSessionId(e){this._sessionId=e}set(e){if(!this._synced){this._pendingPatches.push(B(e)),this.notify();return}this.applyPatch(e)}get(e,t){let r=this.snapshot();return e?re(r,e)??t:r}on(e,t){return this._listeners.add(t),()=>this._listeners.delete(t)}dispose(){this._root.unobserveDeep(this._observer),this._doc.off("update",this._updateHandler);for(let e of this._transportUnsubs)e();this._transportUnsubs=[],this._transport=null,this._listeners.clear(),this._doc.destroy()}sendHello(){this._transport&&this._transport.sendDocSync(g.encodeStateAsUpdate(this._doc))}applyRemote(e){if(g.applyUpdate(this._doc,e,N),!this._synced){this._synced=!0;let t=this._pendingPatches.splice(0);for(let r of t)this.applyPatch(r)}}applyPatch(e){this._doc.transact(()=>{let t=this._root.toJSON();for(let[r,s]of Object.entries(e)){let o=t[r],i=R(s)&&R(o)?O(o,s):B(s);JSON.stringify(i)!==JSON.stringify(o)&&this._root.set(r,i)}},te)}snapshot(){let e=this._root.toJSON();for(let t of this._pendingPatches)e=O(e,t);return e}notify(){let e=this.snapshot();for(let t of this._listeners)t(e)}};var ne=new Set(["control"]),P=class{constructor(e,t){this._name=e;this._transport=t;this._track=this._transport.getTrackByName(this._name)??null,this._unsubscribeTransport=this._transport.on("track",()=>{let r=this._transport.getTrackByName(this._name)??null;r!==this._track&&this._setTrack(r)})}_name;_transport;_track=null;_handlers=new Map;_unsubscribeTransport=null;get track(){let e=this._transport.getTrackByName(this._name)??this._track;return e!==this._track&&this._setTrack(e),this._track}async attach(e){await this._transport.addTrack(e)}async detach(){this._transport.removeTrack()}async seek(e){if(e!=="live"&&(!Number.isFinite(e)||e<0))throw new Error('[urun] stream.seek target must be "live" or a non-negative number of seconds');this._transport.seekStream(this._name,e)}on(e,t){let r=this._handlers.get(e);return r||(r=new Set,this._handlers.set(e,r)),r.add(t),()=>{r?.delete(t),r?.size===0&&this._handlers.delete(e)}}dispose(){this._unsubscribeTransport?.(),this._unsubscribeTransport=null,this._handlers.clear()}_setTrack(e){this._track=e,e&&e.addEventListener("ended",()=>{this._track===e&&this._setTrack(null)},{once:!0});let t=this._handlers.get("track");if(t)for(let r of t)r(e)}},S=class n{_sessionId;_multiplexer;_transport;_docs=new Map;_streams=new Map;constructor(e,t,r){this._sessionId=e,this._multiplexer=t,this._transport=r}static async attach(e,t){let r=await j({baseUrl:t.baseUrl,fallbackUrls:t.fallbackUrls,sessionId:e,orgId:t.orgId,jwt:t.jwt,getAccessToken:t.getAccessToken,authProvider:t.authProvider}),s=t.getAccessToken?await t.getAccessToken()??void 0:t.jwt,o=new v,i=new k({url:r.wsUrl,orgId:t.orgId,jwt:s,authProvider:t.authProvider,sessionId:e}),l=new n(e,o,i);return i.on("connected",()=>{i.multiplexer&&o.setTarget(i.multiplexer)}),i.setConnection({url:r.wsUrl,sessionId:e}),await i.connect({role:"viewer"}),l}get id(){return this._sessionId}setSessionId(e){if(e===this._sessionId)return;this._multiplexer.rewriteSessionChannels?.call(this._multiplexer,this._sessionId,e),this._sessionId=e;for(let r of this._docs.values())r.setSessionId(e)}stream(e){let t=this._streams.get(e);return t||(t=new P(e,this._transport),this._streams.set(e,t)),t}doc(e){let t=this._docs.get(e);return t||(t=new I(e,this._sessionId,ne.has(e)?this._transport:null),this._docs.set(e,t)),t}disconnect(){for(let e of this._docs.values())e.dispose();this._docs.clear();for(let e of this._streams.values())e.dispose();this._streams.clear(),this._transport.disconnect()}};var se=new Set(["then","catch","finally","toJSON",Symbol.toPrimitive]);function oe(n,e){return new Proxy({},{get(t,r){if(!(se.has(r)||typeof r!="string"))return s=>{let o=s??{},i=()=>`s_${Date.now()}_${Math.random().toString(36).slice(2)}`,l=i(),h=new v,p=new k({url:"",orgId:e.orgId,jwt:e.jwt,authProvider:e.authProvider,sessionId:l}),d=new S(l,h,p),a=async c=>{let m=await L({baseUrl:e.baseUrl,fallbackUrls:e.fallbackUrls,app:n,functionName:r,orgId:e.orgId,jwt:e.jwt,getAccessToken:e.getAccessToken,authProvider:e.authProvider,idempotencyKey:c}),u=e.getAccessToken?await e.getAccessToken()??void 0:e.jwt;return d.setSessionId(m.sessionId),{url:m.wsUrl,sessionId:m.sessionId,jwt:u}};return p.setConnectionResolver(()=>a(i())),p.on("connected",()=>{p.multiplexer&&h.setTarget(p.multiplexer)}),(async()=>{let c=await a(l);p.setAuth({jwt:c.jwt,authProvider:e.authProvider,orgId:e.orgId}),p.setConnection({url:c.url,sessionId:c.sessionId}),await p.connect({app:n,functionName:r,args:o})})().catch(c=>{console.error(`[urun] Failed to connect session for ${r}:`,c)}),d}}})}var G=3e4,F="stream:",x=class{_ws;_multiplexer;_pendingRpc=new Map;_rpcId=0;constructor(e,t){this._ws=e,this._multiplexer=t,this._ws.addEventListener("message",r=>{let s;try{s=JSON.parse(typeof r.data=="string"?r.data:"")}catch{return}if(s.rpcId&&this._pendingRpc.has(s.rpcId)){let o=this._pendingRpc.get(s.rpcId);this._pendingRpc.delete(s.rpcId),clearTimeout(o.timer),s.error?o.reject(new Error(s.error)):o.resolve(s.result)}})}async get(e){return this._sendRpc("store.get",{key:e})}async has(e){return await this._sendRpc("store.has",{key:e})}on(e,t){let r=`${F}${e}`;return this._multiplexer.on(r,s=>{t(s)})}emit(e,t){let r=`${F}${e}`,s=typeof t=="object"&&t!==null&&!Array.isArray(t)?t:{data:t};this._multiplexer.emit(r,s)}_getInternals(){return{multiplexer:this._multiplexer,ws:this._ws}}_sendRpc(e,t){return new Promise((r,s)=>{if(!this._ws||this._ws.readyState!==WebSocket.OPEN){s(new Error("WebSocket not open"));return}let o=String(++this._rpcId),i=setTimeout(()=>{this._pendingRpc.has(o)&&(this._pendingRpc.delete(o),s(new Error(`RPC request "${e}" timed out after ${G}ms`)))},G);this._pendingRpc.set(o,{resolve:r,reject:s,timer:i}),this._ws.send(JSON.stringify({rpcId:o,type:e,...t}))})}};function ie(n){let r=`${n.baseUrl.replace(/\/$/,"").replace(/^http/,"ws")}/store/${encodeURIComponent(n.orgId)}`,s=new WebSocket(r);(n.jwt||n.apiKey)&&s.addEventListener("open",()=>{s.send(JSON.stringify({type:"auth",orgId:n.orgId,jwt:n.jwt,authProvider:n.authProvider,apiKey:n.apiKey}))});let o=new M(s);return new x(s,o)}var ae='video/mp4; codecs="avc1.42E01E"';function ce(n,e){let t=e?.codec??ae,r=document.createElement("video");r.muted=!0,r.playsInline=!0;let s=new MediaSource,o=URL.createObjectURL(s);r.src=o,s.addEventListener("sourceopen",()=>{let c;try{c=s.addSourceBuffer(t)}catch(f){throw f instanceof DOMException&&f.name==="NotSupportedError"?new Error(`Codec not supported: "${t}". Ensure the codec string matches the fMP4 container format. Common values: video/mp4; codecs="avc1.42E01E" (H.264 Baseline), video/mp4; codecs="avc1.4D401F" (H.264 Main).`):f}let m=n.getReader();function u(){return c.updating?new Promise(f=>{c.addEventListener("updateend",()=>f(),{once:!0})}):Promise.resolve()}async function _(){try{for(;;){let{done:f,value:U}=await m.read();if(f){await u(),s.readyState==="open"&&s.endOfStream();return}await u();try{c.appendBuffer(U)}catch(y){if(y instanceof DOMException&&y.name==="QuotaExceededError"){let T=c.buffered;if(T.length>0){let J=T.start(T.length-1);c.remove(0,J),await u(),c.appendBuffer(U)}else throw y}else throw y}}}catch{if(s.readyState==="open")try{s.endOfStream("decode")}catch{}}}_()}),r.play().catch(()=>{});let i=r,l=i.captureStream??i.mozCaptureStream;if(!l)throw new Error("captureStream() is not supported in this browser. videoStream() requires HTMLMediaElement.captureStream() support.");let h=l.call(r),p=h.getTracks(),d=0,a=p.length;if(a===0)r.addEventListener("emptied",()=>{URL.revokeObjectURL(o)},{once:!0});else for(let c of p)c.addEventListener("ended",()=>{d++,d>=a&&URL.revokeObjectURL(o)});return h}export{oe as App,S as Session,ie as createStore,ce as videoStream};
@@ -1,6 +1,6 @@
1
1
 
2
2
 
3
- type TransportState = 'connecting' | 'connected' | 'disconnected' | 'failed';
3
+ type TransportState = 'connecting' | 'connected' | 'reconnecting' | 'disconnected' | 'failed';
4
4
 
5
5
  interface TransportSessionOptions {
6
6
 
@@ -170,6 +170,21 @@ interface AccessTokenOptions {
170
170
  reason?: 'initial' | 'unauthorized' | 'background';
171
171
  }
172
172
 
173
+ interface AttachOptions {
174
+
175
+ baseUrl: string;
176
+
177
+ fallbackUrls?: readonly string[];
178
+
179
+ orgId: string;
180
+
181
+ jwt?: string;
182
+
183
+ getAccessToken?: (options?: AccessTokenOptions) => string | null | undefined | Promise<string | null | undefined>;
184
+
185
+ authProvider?: string;
186
+ }
187
+
173
188
  interface App {
174
189
  [fnName: string]: (args?: Record<string, unknown>) => Session;
175
190
  }
@@ -213,6 +228,8 @@ interface TransportEvents {
213
228
  docSync: (payload: Uint8Array) => void;
214
229
 
215
230
  docSyncReady: () => void;
231
+
232
+ stateChange: (state: TransportState) => void;
216
233
  }
217
234
  interface SessionStartOptions {
218
235
  app?: string;
@@ -225,6 +242,8 @@ interface SessionStartOptions {
225
242
  height?: number;
226
243
  frameRate?: number;
227
244
  };
245
+
246
+ role?: 'viewer';
228
247
  }
229
248
  declare class TransportSession {
230
249
  private _state;
@@ -245,6 +264,8 @@ declare class TransportSession {
245
264
  private _reconnectAttempts;
246
265
  private _intentionalDisconnect;
247
266
  private _reconnectTimer;
267
+
268
+ private _stableResetTimer;
248
269
  private _generation;
249
270
  private _lastStartOptions;
250
271
  private _resolveConnection;
@@ -292,9 +313,12 @@ declare class TransportSession {
292
313
  private _connectOnce;
293
314
  private _startMessage;
294
315
  private _cleanupSocket;
316
+
317
+ private _scheduleReconnectBudgetReset;
318
+ private _clearReconnectBudgetReset;
295
319
  private _cleanupMedia;
296
320
  private _send;
297
321
  private _setState;
298
322
  }
299
323
 
300
- export { type App as A, type ChannelEndpoint as C, type Layer as L, type SceneContext as S, TransportSession as T, type AppOptions as a, type ChannelMessage as b, ChannelMultiplexer as c, type ChannelName as d, type LayoutConfig as e, type LayoutContext as f, type SceneGraph as g, type Session as h, type SessionDocument as i, type SessionStartOptions as j, type SessionStream as k, type Store as l, type StoreOptions as m, type TransportSessionOptions as n, type TransportState as o };
324
+ export { type App as A, type ChannelEndpoint as C, type Layer as L, type SceneContext as S, TransportSession as T, type AppOptions as a, type AttachOptions as b, type ChannelMessage as c, ChannelMultiplexer as d, type ChannelName as e, type LayoutConfig as f, type LayoutContext as g, type SceneGraph as h, type Session as i, type SessionDocument as j, type SessionStartOptions as k, type SessionStream as l, type Store as m, type StoreOptions as n, type TransportSessionOptions as o, type TransportState as p };
@@ -1,6 +1,6 @@
1
1
 
2
2
 
3
- type TransportState = 'connecting' | 'connected' | 'disconnected' | 'failed';
3
+ type TransportState = 'connecting' | 'connected' | 'reconnecting' | 'disconnected' | 'failed';
4
4
 
5
5
  interface TransportSessionOptions {
6
6
 
@@ -170,6 +170,21 @@ interface AccessTokenOptions {
170
170
  reason?: 'initial' | 'unauthorized' | 'background';
171
171
  }
172
172
 
173
+ interface AttachOptions {
174
+
175
+ baseUrl: string;
176
+
177
+ fallbackUrls?: readonly string[];
178
+
179
+ orgId: string;
180
+
181
+ jwt?: string;
182
+
183
+ getAccessToken?: (options?: AccessTokenOptions) => string | null | undefined | Promise<string | null | undefined>;
184
+
185
+ authProvider?: string;
186
+ }
187
+
173
188
  interface App {
174
189
  [fnName: string]: (args?: Record<string, unknown>) => Session;
175
190
  }
@@ -213,6 +228,8 @@ interface TransportEvents {
213
228
  docSync: (payload: Uint8Array) => void;
214
229
 
215
230
  docSyncReady: () => void;
231
+
232
+ stateChange: (state: TransportState) => void;
216
233
  }
217
234
  interface SessionStartOptions {
218
235
  app?: string;
@@ -225,6 +242,8 @@ interface SessionStartOptions {
225
242
  height?: number;
226
243
  frameRate?: number;
227
244
  };
245
+
246
+ role?: 'viewer';
228
247
  }
229
248
  declare class TransportSession {
230
249
  private _state;
@@ -245,6 +264,8 @@ declare class TransportSession {
245
264
  private _reconnectAttempts;
246
265
  private _intentionalDisconnect;
247
266
  private _reconnectTimer;
267
+
268
+ private _stableResetTimer;
248
269
  private _generation;
249
270
  private _lastStartOptions;
250
271
  private _resolveConnection;
@@ -292,9 +313,12 @@ declare class TransportSession {
292
313
  private _connectOnce;
293
314
  private _startMessage;
294
315
  private _cleanupSocket;
316
+
317
+ private _scheduleReconnectBudgetReset;
318
+ private _clearReconnectBudgetReset;
295
319
  private _cleanupMedia;
296
320
  private _send;
297
321
  private _setState;
298
322
  }
299
323
 
300
- export { type App as A, type ChannelEndpoint as C, type Layer as L, type SceneContext as S, TransportSession as T, type AppOptions as a, type ChannelMessage as b, ChannelMultiplexer as c, type ChannelName as d, type LayoutConfig as e, type LayoutContext as f, type SceneGraph as g, type Session as h, type SessionDocument as i, type SessionStartOptions as j, type SessionStream as k, type Store as l, type StoreOptions as m, type TransportSessionOptions as n, type TransportState as o };
324
+ export { type App as A, type ChannelEndpoint as C, type Layer as L, type SceneContext as S, TransportSession as T, type AppOptions as a, type AttachOptions as b, type ChannelMessage as c, ChannelMultiplexer as d, type ChannelName as e, type LayoutConfig as f, type LayoutContext as g, type SceneGraph as h, type Session as i, type SessionDocument as j, type SessionStartOptions as k, type SessionStream as l, type Store as m, type StoreOptions as n, type TransportSessionOptions as o, type TransportState as p };
@@ -1 +1 @@
1
- export { b as ChannelMessage, c as ChannelMultiplexer, d as ChannelName, j as SessionStartOptions, T as TransportSession, n as TransportSessionOptions } from './internal-Bur1pubT.mjs';
1
+ export { c as ChannelMessage, d as ChannelMultiplexer, e as ChannelName, k as SessionStartOptions, T as TransportSession, o as TransportSessionOptions } from './internal-ChJcuvFH.mjs';
@@ -1 +1 @@
1
- export { b as ChannelMessage, c as ChannelMultiplexer, d as ChannelName, j as SessionStartOptions, T as TransportSession, n as TransportSessionOptions } from './internal-Bur1pubT.js';
1
+ export { c as ChannelMessage, d as ChannelMultiplexer, e as ChannelName, k as SessionStartOptions, T as TransportSession, o as TransportSessionOptions } from './internal-ChJcuvFH.js';