@dobuki/hello-worker 1.0.67 → 1.0.69

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.
@@ -1,7 +1,8 @@
1
1
  import { EnterRoom } from "./signal-room";
2
2
  import { SigType, SigPayload } from "./webrtc-peer-collector";
3
3
  type UserListener = (user: string, action: "join" | "leave", users: string[]) => void;
4
- export declare function enterWorld<S extends string | ArrayBufferView = string | ArrayBufferView, R extends string | ArrayBufferLike = string | ArrayBufferLike>({ worldId, logLine, enterRoomFunction, peerlessUserExpiration, workerUrl, onRoomReady, onRoomClose, dataChannelOptions, }: {
4
+ export declare function enterWorld<S extends string | ArrayBufferView = string | ArrayBufferView, R extends string | ArrayBufferLike = string | ArrayBufferLike>({ userId: passedUserId, worldId, logLine, enterRoomFunction, peerlessUserExpiration, workerUrl, onRoomReady, onRoomClose, dataChannelOptions, }: {
5
+ userId?: string;
5
6
  worldId: string;
6
7
  logLine?: (direction: string, obj?: any) => void;
7
8
  enterRoomFunction?: EnterRoom<SigType, SigPayload>;
@@ -1 +1 @@
1
- {"version":3,"file":"enter-world.d.ts","sourceRoot":"","sources":["../src/browser/enter-world.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAa,MAAM,eAAe,CAAC;AACrD,OAAO,EACL,OAAO,EACP,UAAU,EAEX,MAAM,yBAAyB,CAAC;AAEjC,KAAK,YAAY,GAAG,CAClB,IAAI,EAAE,MAAM,EACZ,MAAM,EAAE,MAAM,GAAG,OAAO,EACxB,KAAK,EAAE,MAAM,EAAE,KACZ,IAAI,CAAC;AAEV,wBAAgB,UAAU,CACxB,CAAC,SAAS,MAAM,GAAG,eAAe,GAAG,MAAM,GAAG,eAAe,EAC7D,CAAC,SAAS,MAAM,GAAG,eAAe,GAAG,MAAM,GAAG,eAAe,EAC7D,EACA,OAAO,EACP,OAAuB,EACvB,iBAA6B,EAC7B,sBAAsB,EACtB,SAAS,EACT,WAAW,EACX,WAAW,EACX,kBAAkB,GACnB,EAAE;IACD,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,GAAG,KAAK,IAAI,CAAC;IACjD,iBAAiB,CAAC,EAAE,SAAS,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;IACnD,sBAAsB,CAAC,EAAE,MAAM,CAAC;IAChC,SAAS,CAAC,EAAE,GAAG,CAAC;IAChB,WAAW,CAAC,CAAC,IAAI,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI,CAAC;IACzD,WAAW,CAAC,CAAC,IAAI,EAAE;QACjB,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE,MAAM,CAAC;QACb,EAAE,EAAE,IAAI,CAAC,UAAU,EAAE,QAAQ,GAAG,MAAM,GAAG,UAAU,CAAC,CAAC;KACtD,GAAG,IAAI,CAAC;IACT,kBAAkB,CAAC,EAAE,kBAAkB,CAAC;CACzC;;iBA2FqB,CAAC,WAAW,MAAM;;;;;;;;;;;;mCAaA,CAAC,IAAI,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,KAAK,IAAI;sCAJ5B,CAAC,IAAI,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,KAAK,IAAI;gCAerC,YAAY;mCAJT,YAAY;;EAmCnD"}
1
+ {"version":3,"file":"enter-world.d.ts","sourceRoot":"","sources":["../src/browser/enter-world.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAa,MAAM,eAAe,CAAC;AACrD,OAAO,EACL,OAAO,EACP,UAAU,EAEX,MAAM,yBAAyB,CAAC;AAEjC,KAAK,YAAY,GAAG,CAClB,IAAI,EAAE,MAAM,EACZ,MAAM,EAAE,MAAM,GAAG,OAAO,EACxB,KAAK,EAAE,MAAM,EAAE,KACZ,IAAI,CAAC;AAEV,wBAAgB,UAAU,CACxB,CAAC,SAAS,MAAM,GAAG,eAAe,GAAG,MAAM,GAAG,eAAe,EAC7D,CAAC,SAAS,MAAM,GAAG,eAAe,GAAG,MAAM,GAAG,eAAe,EAC7D,EACA,MAAM,EAAE,YAAY,EACpB,OAAO,EACP,OAAO,EACP,iBAA6B,EAC7B,sBAAsB,EACtB,SAAS,EACT,WAAW,EACX,WAAW,EACX,kBAAkB,GACnB,EAAE;IACD,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,GAAG,KAAK,IAAI,CAAC;IACjD,iBAAiB,CAAC,EAAE,SAAS,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;IACnD,sBAAsB,CAAC,EAAE,MAAM,CAAC;IAChC,SAAS,CAAC,EAAE,GAAG,CAAC;IAChB,WAAW,CAAC,CAAC,IAAI,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI,CAAC;IACzD,WAAW,CAAC,CAAC,IAAI,EAAE;QACjB,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE,MAAM,CAAC;QACb,EAAE,EAAE,IAAI,CAAC,UAAU,EAAE,QAAQ,GAAG,MAAM,GAAG,UAAU,CAAC,CAAC;KACtD,GAAG,IAAI,CAAC;IACT,kBAAkB,CAAC,EAAE,kBAAkB,CAAC;CACzC;;iBA4FqB,CAAC,WAAW,MAAM;;;;;;;;;;;;mCAaA,CAAC,IAAI,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,KAAK,IAAI;sCAJ5B,CAAC,IAAI,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,KAAK,IAAI;gCAerC,YAAY;mCAJT,YAAY;;EAmCnD"}
@@ -1,4 +1,4 @@
1
- function c(F){let{userId:S,worldId:R,room:h,host:C,autoRejoin:O=!0,logLine:T}=F,E=!1,J=0,W,L,k=!0,K=new Map,A=`wss://${C}/room/${R}/${h}?userId=${encodeURIComponent(S)}`,q=[],b=0;function w(V,H,X){if(!W)return T?.("\uD83D\uDC64 ➡️ ❌","no ws available"),!1;if(E||W.readyState!==WebSocket.OPEN)return T?.("\uD83D\uDC64 ➡️ ❌","Not in opened state: "+W.readyState),!1;let G={type:V,to:H,payload:X};return q.push(G),T?.("\uD83D\uDC64 ➡️ \uD83D\uDDA5️",G),clearTimeout(b),b=setTimeout(()=>{W.send(JSON.stringify(q)),q.length=0}),!0}function P(){if(E)return;W=new WebSocket(A),W.onopen=()=>{if(k)F.onOpen?.(),k=!1;J=0},W.onmessage=(V)=>{try{let H=JSON.parse(V.data);(Array.isArray(H)?H:[H]).forEach((G)=>{if(T?.("\uD83D\uDDA5️ ➡️ \uD83D\uDC64",G),G.type==="peer-joined"||G.type==="peer-left")N(G.users);else if(G.type==="ice-server")F.onIceUrl?.(G.url,G.expiration);else if(G.userId)F.onMessage(G.type,G.payload,{userId:G.userId,receive:(z,D)=>w(z,G.userId,D)})})}catch{T?.("⚠️ ERROR",{error:"invalid-json"})}},W.onclose=(V)=>{let X=[1001,1006,1011,1012,1013].includes(V.code);if(O&&!E&&X){let G=Math.min(Math.pow(2,J)*1000,30000),z=Math.random()*1000,D=G+z;T?.("\uD83D\uDD04 RECONNECTING",{attempt:J+1,delayMs:Math.round(D)}),J++,L=setTimeout(P,D)}else F.onClose?.({code:V.code,reason:V.reason,wasClean:V.wasClean})},W.onerror=(V)=>{console.error("WS Error",V),F.onError?.()}}function N(V){let H=[],X=[],G=new Set;V.forEach(({userId:z})=>{if(z===S)return;if(!K.has(z)){let D={userId:z,receive:(y,Q)=>w(y,z,Q)};K.set(z,D),H.push(D)}G.add(z)});for(let z of K.keys())if(!G.has(z))K.delete(z),X.push({userId:z});if(H.length)F.onPeerJoined(H);if(X.length)F.onPeerLeft(X)}return P(),{sendToServer(V,H){w(V,"server",H)},exitRoom:()=>{E=!0,clearTimeout(L),W.close()}}}function I({userId:F,worldId:S,room:R,host:h,autoRejoin:C=!0,onOpen:O,onClose:T,onError:E,onPeerJoined:J,onPeerLeft:W,onIceUrl:L,onMessage:k,logLine:K,workerUrl:A}){if(!A)return console.warn("Warning: enterRoom called without workerUrl; this may cause issues in some environments. You should pass workerUrl explicitly. Use:","https://cdn.jsdelivr.net/npm/@dobuki/hello-worker/dist/signal-room.worker.min.js"),c({userId:F,worldId:S,room:R,host:h,autoRejoin:C,onOpen:O,onClose:T,onError:E,onPeerJoined:J,onPeerLeft:W,onIceUrl:L,onMessage:k});let q=new Worker(A,{type:"module"}),b=!1;function w({userId:N}){return{userId:N,receive:(V,H)=>{if(b)return!1;return q.postMessage({cmd:"send",toUserId:N,host:h,room:R,type:V,payload:H}),!0}}}let P=(N)=>{let V=N.data;if(V.kind==="open")O?.();else if(V.kind==="close")q.terminate(),T?.(V.ev);else if(V.kind==="error")E?.();else if(V.kind==="peer-joined")J(V.users.map((H)=>w({userId:H.userId})));else if(V.kind==="peer-left")W(V.users);else if(V.kind==="ice-server")L?.(V.url,V.expiration);else if(V.kind==="message")k(V.type,V.payload,w({userId:V.fromUserId}));else if(V.kind==="log")K?.(V.direction,V.obj)};return q.addEventListener("message",P),q.postMessage({cmd:"enter",userId:F,worldId:S,room:R,host:h,autoRejoin:C}),{exitRoom:()=>{b=!0,q.removeEventListener("message",P),q.postMessage({cmd:"exit"})},sendToServer:(N,V)=>{q.postMessage({cmd:"send",toUserId:"server",host:h,room:R,type:N,payload:V})}}}var l=I;function m({worldId:F,receivePeerConnection:S,peerlessUserExpiration:R=5000,fallbackRtcConfig:h={iceServers:[{urls:"stun:stun.l.google.com:19302"}]},enterRoomFunction:C=l,logLine:O=console.debug,onLeaveUser:T,workerUrl:E,onRoomReady:J,onRoomClose:W,onBroadcastMessage:L}){let k=`user-${crypto.randomUUID()}`,K=new Map,A=void 0,q={...h,timestamp:Date.now()},b=new Map;async function w(X){if(X)try{let G=await fetch(X);if(!G.ok)throw Error(`ICE endpoint failed: ${G.status}`);q=await G.json()}catch(G){console.warn("Using fallback rtcConfig:",G)}return q}function P(X){T?.(X);let G=K.get(X);if(!G)return;try{G.pc?.close()}catch{}K.delete(X)}async function N(X){if(!X.pc?.remoteDescription)return;let G=X.pendingRemoteIce;X.pendingRemoteIce=[];for(let z of G)try{await X.pc.addIceCandidate(z)}catch(D){O("⚠️ ERROR",{error:"add-ice-failed",userId:X.userId,detail:String(D)})}}function V({room:X,host:G}){let z=`${G}/room/${X}`,D=b.get(z);if(D)D.exitRoom(),b.delete(z)}function H({room:X,host:G}){return new Promise(async(z,D)=>{async function y(Z){let Y=!A||A.expiration-Date.now()<2000?await f():A;return Z.pc=new RTCPeerConnection(Date.now()-(q?.timestamp??0)<1e4?q:await w(Y.url)),Z.pc.onicecandidate=(B)=>{if(!B.candidate)return;Z.peer.receive("ice",B.candidate.toJSON())},Z.pc.onconnectionstatechange=()=>{O("\uD83D\uDCAC",{event:"pc-state",userId:Z.userId,state:Z.pc?.connectionState})},Z.pc}async function Q(Z){let Y=K.get(Z.userId),B=!1;if(!Y){let j={userId:Z.userId,pendingRemoteIce:[],peer:Z};await y(j),Y=j,K.set(Y.userId,Y),B=!0}else if(Y)clearTimeout(Y.expirationTimeout),Y.expirationTimeout=0;if(!Y.pc||Y.pc?.signalingState==="closed")await y(Y);return Y.peer=Z,[Y,B]}async function $(Z){let[Y]=await Q(Z),B=Y.pc,j=await B?.createOffer();await B?.setLocalDescription(j),Z.receive("offer",B?.localDescription?.toJSON())}let _;async function f(){let Z=await new Promise((Y)=>{_=Y,g("request-ice")});return _=void 0,Z}let{exitRoom:M,sendToServer:g}=C({userId:k,worldId:F,room:X,host:G,logLine:O,workerUrl:E,autoRejoin:!0,onOpen(){J?.({room:X,host:G}),z()},onError(){console.error("onError"),D()},onClose(Z){W?.({room:X,host:G,ev:Z})},onPeerJoined(Z){Z.forEach(async(Y)=>{let[B,j]=await Q(Y);if(!j){O("\uD83D\uDC64ℹ️","not a new peer: "+Y.userId);return}let x=B.pc;if(!x){O("\uD83D\uDC64ℹ️","no pc: "+Y.userId);return}async function U(){let v=K.get(Y.userId);if(v){v.pc=void 0;let u=await y(v);S({pc:u,userId:Y.userId,initiator:!0,restart:U}),await new Promise((p)=>setTimeout(p,3000)),await $(Y)}}S({pc:x,userId:Y.userId,initiator:!0,restart:U}),await $(Y)})},onPeerLeft(Z){Z.forEach(({userId:Y})=>{let B=K.get(Y);if(!B)return;B.expirationTimeout=setTimeout(()=>P(Y),R??0)})},onIceUrl(Z,Y){A={url:Z,expiration:Y},_?.(A)},async onMessage(Z,Y,B){let[j]=await Q(B),x=j.pc;if(!x)return;if(Z==="offer"){S({pc:x,userId:B.userId,initiator:!1,restart(){j.pc=void 0}}),await x.setRemoteDescription(Y);let U=await x.createAnswer();await x.setLocalDescription(U),B.receive("answer",x.localDescription?.toJSON()),await N(j);return}if(Z==="answer"){await x.setRemoteDescription(Y),await N(j);return}if(Z==="ice"){let U=Y;if(!x.remoteDescription){j.pendingRemoteIce.push(U);return}try{await x.addIceCandidate(U)}catch(v){O("⚠️ ERROR",{error:"add-ice-failed",userId:j.userId,detail:String(v)})}return}if(Z==="broadcast")L?.(Y,B.userId)}});b.set(`${G}/room/${X}`,{exitRoom:M,room:X,host:G,broadcast:(Z)=>{g("broadcast",Z)}})})}return{userId:k,enterRoom:H,exitRoom:V,leaveUser:P,broadcast(X){b.forEach((G)=>G.broadcast(X))},end(){b.forEach(({exitRoom:X})=>X()),b.clear(),K.forEach(({userId:X})=>P(X)),K.clear()}}}function VG({worldId:F,logLine:S=console.debug,enterRoomFunction:R=I,peerlessUserExpiration:h,workerUrl:C,onRoomReady:O,onRoomClose:T,dataChannelOptions:E}){let J=[],W=new Set;function L(Q,$,_,f){if(_){let M=Q.createDataChannel("data",E);K($,M,f),A.set($,M)}else{let M=function(g){let Z=g.channel;K($,Z,f),A.set($,Z)};return Q.addEventListener("datachannel",M),()=>{Q.removeEventListener("datachannel",M)}}}function k(Q,$){W.forEach((_)=>_(Q,$))}function K(Q,$,_){$.onopen=()=>{S("\uD83D\uDCAC",{event:"dc-open",userId:Q}),J.push(Q),q.forEach((M)=>M(Q,"join",J))};let f=({data:M})=>{k(M,Q)};$.addEventListener("message",f),$.addEventListener("close",()=>{S("\uD83D\uDCAC",{event:"dc-close",userId:Q}),J.splice(J.indexOf(Q),1),q.forEach((M)=>M(Q,"leave",J)),$.removeEventListener("message",f),_?.()}),$.onerror=()=>S("⚠️ ERROR",{error:"dc-error",userId:Q})}let A=new Map,q=new Set,{userId:b,enterRoom:w,exitRoom:P,leaveUser:N,broadcast:V,end:H}=m({worldId:F,enterRoomFunction:R,logLine:S,workerUrl:C,peerlessUserExpiration:h,onRoomReady:O,onRoomClose:T,onLeaveUser(Q){let $=A.get(Q);try{$?.close()}catch{}A.delete(Q)},receivePeerConnection({pc:Q,userId:$,initiator:_,restart:f}){L(Q,$,_,f)},onBroadcastMessage(Q,$){k(Q,$),S("\uD83D\uDCE2",{event:"broadcast",userId:b,data:Q})}});function X(Q,$){A.forEach((_,f)=>{if($&&f!==$)return;if(_.readyState==="open")_.send(Q)})}function G(Q){W.delete(Q)}function z(Q){return W.add(Q),()=>{G(Q)}}function D(Q){q.delete(Q)}function y(Q){return q.add(Q),()=>{D(Q)}}return{userId:b,send:X,broadcast:V,enterRoom:w,exitRoom:P,leaveUser:N,getUsers:()=>J,addMessageListener:z,removeMessageListener:G,addUserListener:y,removeUserListener:D,end(){A.forEach((Q)=>{try{Q.close()}catch{}}),A.clear(),H(),q.clear(),J.length=0}}}export{VG as enterWorld};
1
+ function c(S){let{userId:h,worldId:A,room:R,host:C,autoRejoin:y=!0,logLine:B}=S,x=!1,E=0,q,k,w=!0,T=new Map,J=`wss://${C}/room/${A}/${R}?userId=${encodeURIComponent(h)}`,$=[],K=0;function D(Q,W,M){if(!q)return B?.("\uD83D\uDC64 ➡️ ❌","no ws available"),!1;if(x||q.readyState!==WebSocket.OPEN)return B?.("\uD83D\uDC64 ➡️ ❌","Not in opened state: "+q.readyState),!1;let G={type:Q,to:W,payload:M};return $.push(G),B?.("\uD83D\uDC64 ➡️ \uD83D\uDDA5️",G),clearTimeout(K),K=setTimeout(()=>{q.send(JSON.stringify($)),$.length=0}),!0}function L(){if(x)return;q=new WebSocket(J),q.onopen=()=>{if(w)S.onOpen?.(),w=!1;E=0},q.onmessage=(Q)=>{try{let W=JSON.parse(Q.data);(Array.isArray(W)?W:[W]).forEach((G)=>{if(B?.("\uD83D\uDDA5️ ➡️ \uD83D\uDC64",G),G.type==="peer-joined"||G.type==="peer-left")b(G.users);else if(G.type==="ice-server")S.onIceUrl?.(G.url,G.expiration);else if(G.userId)S.onMessage(G.type,G.payload,{userId:G.userId,receive:(Y,F)=>D(Y,G.userId,F)})})}catch{B?.("⚠️ ERROR",{error:"invalid-json"})}},q.onclose=(Q)=>{let M=[1001,1006,1011,1012,1013].includes(Q.code);if(y&&!x&&M){let G=Math.min(Math.pow(2,E)*1000,30000),Y=Math.random()*1000,F=G+Y;B?.("\uD83D\uDD04 RECONNECTING",{attempt:E+1,delayMs:Math.round(F)}),E++,k=setTimeout(L,F)}else S.onClose?.({code:Q.code,reason:Q.reason,wasClean:Q.wasClean})},q.onerror=(Q)=>{console.error("WS Error",Q),S.onError?.()}}function b(Q){let W=[],M=[],G=new Set;Q.forEach(({userId:Y})=>{if(Y===h)return;if(!T.has(Y)){let F={userId:Y,receive:(P,v)=>D(P,Y,v)};T.set(Y,F),W.push(F)}G.add(Y)});for(let Y of T.keys())if(!G.has(Y))T.delete(Y),M.push({userId:Y});if(W.length)S.onPeerJoined(W);if(M.length)S.onPeerLeft(M)}return L(),{sendToServer(Q,W){D(Q,"server",W)},exitRoom:()=>{x=!0,clearTimeout(k),q.close()}}}function m({userId:S,worldId:h,room:A,host:R,autoRejoin:C=!0,onOpen:y,onClose:B,onError:x,onPeerJoined:E,onPeerLeft:q,onIceUrl:k,onMessage:w,logLine:T,workerUrl:J}){if(!J)return console.warn("Warning: enterRoom called without workerUrl; this may cause issues in some environments. You should pass workerUrl explicitly. Use:","https://cdn.jsdelivr.net/npm/@dobuki/hello-worker/dist/signal-room.worker.min.js"),c({userId:S,worldId:h,room:A,host:R,autoRejoin:C,onOpen:y,onClose:B,onError:x,onPeerJoined:E,onPeerLeft:q,onIceUrl:k,onMessage:w});let $=new Worker(J,{type:"module"}),K=!1;function D({userId:b}){return{userId:b,receive:(Q,W)=>{if(K)return!1;return $.postMessage({cmd:"send",toUserId:b,host:R,room:A,type:Q,payload:W}),!0}}}let L=(b)=>{let Q=b.data;if(Q.kind==="open")y?.();else if(Q.kind==="close")$.terminate(),B?.(Q.ev);else if(Q.kind==="error")x?.();else if(Q.kind==="peer-joined")E(Q.users.map((W)=>D({userId:W.userId})));else if(Q.kind==="peer-left")q(Q.users);else if(Q.kind==="ice-server")k?.(Q.url,Q.expiration);else if(Q.kind==="message")w(Q.type,Q.payload,D({userId:Q.fromUserId}));else if(Q.kind==="log")T?.(Q.direction,Q.obj)};return $.addEventListener("message",L),$.postMessage({cmd:"enter",userId:S,worldId:h,room:A,host:R,autoRejoin:C}),{exitRoom:()=>{K=!0,$.removeEventListener("message",L),$.postMessage({cmd:"exit"})},sendToServer:(b,Q)=>{$.postMessage({cmd:"send",toUserId:"server",host:R,room:A,type:b,payload:Q})}}}var i=m;function u({userId:S,worldId:h,receivePeerConnection:A,peerlessUserExpiration:R=5000,fallbackRtcConfig:C={iceServers:[{urls:"stun:stun.l.google.com:19302"}]},enterRoomFunction:y=i,logLine:B,onLeaveUser:x,workerUrl:E,onRoomReady:q,onRoomClose:k,onBroadcastMessage:w}){let T=S??`user-${crypto.randomUUID()}`,J=new Map,$=void 0,K={...C,timestamp:Date.now()},D=new Map;async function L(G){if(G)try{let Y=await fetch(G);if(!Y.ok)throw Error(`ICE endpoint failed: ${Y.status}`);K=await Y.json()}catch(Y){console.warn("Using fallback rtcConfig:",Y)}return K}function b(G){x?.(G);let Y=J.get(G);if(!Y)return;try{Y.pc?.close()}catch{}J.delete(G)}async function Q(G){if(!G.pc?.remoteDescription)return;let Y=G.pendingRemoteIce;G.pendingRemoteIce=[];for(let F of Y)try{await G.pc.addIceCandidate(F)}catch(P){B?.("⚠️ ERROR",{error:"add-ice-failed",userId:G.userId,detail:String(P)})}}function W({room:G,host:Y}){let F=`${Y}/room/${G}`,P=D.get(F);if(P)P.exitRoom(),D.delete(F)}function M({room:G,host:Y}){return new Promise(async(F,P)=>{async function v(Z){let X=Date.now();if(X-(K?.timestamp??0)>1e4){console.log("ICE expiration",$?.expiration,"-",X);let H=!$||$.expiration-X<2000?await f():$;K=await L(H.url)}return Z.pc=new RTCPeerConnection(K),Z.pc.onicecandidate=(H)=>{if(!H.candidate)return;Z.peer.receive("ice",H.candidate.toJSON())},Z.pc.onconnectionstatechange=()=>{B?.("\uD83D\uDCAC",{event:"pc-state",userId:Z.userId,state:Z.pc?.connectionState})},Z.pc}async function V(Z){let X=J.get(Z.userId),H=!1;if(!X){let O={userId:Z.userId,pendingRemoteIce:[],peer:Z};await v(O),X=O,J.set(X.userId,X),H=!0}else if(X)clearTimeout(X.expirationTimeout),X.expirationTimeout=0;if(!X.pc||X.pc?.signalingState==="closed")await v(X);return X.peer=Z,[X,H]}async function z(Z){let[X]=await V(Z),H=X.pc,O=await H?.createOffer();await H?.setLocalDescription(O),Z.receive("offer",H?.localDescription?.toJSON())}let N;async function f(){let Z=await new Promise((X)=>{N=X,I("request-ice")});return N=void 0,Z}let{exitRoom:_,sendToServer:I}=y({userId:T,worldId:h,room:G,host:Y,logLine:B,workerUrl:E,autoRejoin:!0,onOpen(){q?.({room:G,host:Y}),F()},onError(){console.error("onError"),P()},onClose(Z){k?.({room:G,host:Y,ev:Z})},onPeerJoined(Z){Z.forEach(async(X)=>{let[H,O]=await V(X);if(!O){B?.("\uD83D\uDC64ℹ️","not a new peer: "+X.userId);return}let j=H.pc;if(!j){B?.("\uD83D\uDC64ℹ️","no pc: "+X.userId);return}async function U(){let g=J.get(X.userId);if(g){g.pc=void 0;let p=await v(g);A({pc:p,userId:X.userId,initiator:!0,restart:U}),await new Promise((l)=>setTimeout(l,3000)),await z(X)}}A({pc:j,userId:X.userId,initiator:!0,restart:U}),await z(X)})},onPeerLeft(Z){Z.forEach(({userId:X})=>{let H=J.get(X);if(!H)return;H.expirationTimeout=setTimeout(()=>b(X),R??0)})},onIceUrl(Z,X){$={url:Z,expiration:X},N?.($)},async onMessage(Z,X,H){let[O]=await V(H),j=O.pc;if(!j)return;if(Z==="offer"){A({pc:j,userId:H.userId,initiator:!1,restart(){O.pc=void 0}}),await j.setRemoteDescription(X);let U=await j.createAnswer();await j.setLocalDescription(U),H.receive("answer",j.localDescription?.toJSON()),await Q(O);return}if(Z==="answer"){await j.setRemoteDescription(X),await Q(O);return}if(Z==="ice"){let U=X;if(!j.remoteDescription){O.pendingRemoteIce.push(U);return}try{await j.addIceCandidate(U)}catch(g){B?.("⚠️ ERROR",{error:"add-ice-failed",userId:O.userId,detail:String(g)})}return}if(Z==="broadcast")w?.(X,H.userId)}});D.set(`${Y}/room/${G}`,{exitRoom:_,room:G,host:Y,broadcast:(Z)=>{I("broadcast",Z)}})})}return{userId:T,enterRoom:M,exitRoom:W,leaveUser:b,broadcast(G){D.forEach((Y)=>Y.broadcast(G))},end(){D.forEach(({exitRoom:G})=>G()),D.clear(),J.forEach(({userId:G})=>b(G)),J.clear()}}}function XG({userId:S,worldId:h,logLine:A,enterRoomFunction:R=m,peerlessUserExpiration:C,workerUrl:y,onRoomReady:B,onRoomClose:x,dataChannelOptions:E}){let q=[],k=new Set;function w(V,z,N,f){if(N){let _=V.createDataChannel("data",E);J(z,_,f),$.set(z,_)}else{let _=function(I){let Z=I.channel;J(z,Z,f),$.set(z,Z)};return V.addEventListener("datachannel",_),()=>{V.removeEventListener("datachannel",_)}}}function T(V,z){k.forEach((N)=>N(V,z))}function J(V,z,N){z.onopen=()=>{A?.("\uD83D\uDCAC",{event:"dc-open",userId:V}),q.push(V),K.forEach((_)=>_(V,"join",q))};let f=({data:_})=>{T(_,V)};z.addEventListener("message",f),z.addEventListener("close",()=>{A?.("\uD83D\uDCAC",{event:"dc-close",userId:V}),q.splice(q.indexOf(V),1),K.forEach((_)=>_(V,"leave",q)),z.removeEventListener("message",f),N?.()}),z.onerror=()=>A?.("⚠️ ERROR",{error:"dc-error",userId:V})}let $=new Map,K=new Set,{userId:D,enterRoom:L,exitRoom:b,leaveUser:Q,broadcast:W,end:M}=u({userId:S,worldId:h,enterRoomFunction:R,logLine:A,workerUrl:y,peerlessUserExpiration:C,onRoomReady:B,onRoomClose:x,onLeaveUser(V){let z=$.get(V);try{z?.close()}catch{}$.delete(V)},receivePeerConnection({pc:V,userId:z,initiator:N,restart:f}){w(V,z,N,f)},onBroadcastMessage(V,z){T(V,z),A?.("\uD83D\uDCE2",{event:"broadcast",userId:D,data:V})}});function G(V,z){$.forEach((N,f)=>{if(z&&f!==z)return;if(N.readyState==="open")N.send(V)})}function Y(V){k.delete(V)}function F(V){return k.add(V),()=>{Y(V)}}function P(V){K.delete(V)}function v(V){return K.add(V),()=>{P(V)}}return{userId:D,send:G,broadcast:W,enterRoom:L,exitRoom:b,leaveUser:Q,getUsers:()=>q,addMessageListener:F,removeMessageListener:Y,addUserListener:v,removeUserListener:P,end(){$.forEach((V)=>{try{V.close()}catch{}}),$.clear(),M(),K.clear(),q.length=0}}}export{XG as enterWorld};
2
2
 
3
- //# debugId=C5894DD7F36420E164756E2164756E21
3
+ //# debugId=FC8366DF61052DAD64756E2164756E21
4
4
  //# sourceMappingURL=enter-world.js.map
@@ -4,10 +4,10 @@
4
4
  "sourcesContent": [
5
5
  "export interface IPeer<T extends string = string, P = any> {\n userId: string;\n receive(type: T, payload: P): boolean;\n}\n\ntype OutMessage = { type: string; to: string; payload: any };\n\n/**\n * enterRoom connects to the signaling room via WebSocket.\n */\nexport function enterRoom<T extends string, P = any>(params: {\n userId: string;\n worldId: string;\n room: string;\n host: string;\n onOpen?: () => void;\n onClose?: (ev: Pick<CloseEvent, \"code\" | \"reason\" | \"wasClean\">) => void;\n onError?: () => void;\n logLine?: (direction: string, obj?: any) => void;\n onPeerJoined(users: IPeer<T, P>[]): void;\n onPeerLeft(users: { userId: string }[]): void;\n onIceUrl?(url: string, expiration: number): void;\n onMessage(type: T, payload: P, from: IPeer<T, P>): void;\n autoRejoin?: boolean;\n}): {\n exitRoom: () => void;\n sendToServer: <P extends any>(type: T, payload?: P) => void;\n} {\n type Message = {\n type: \"peer-joined\" | \"peer-left\" | \"ice-server\" | T;\n userId: string;\n users: { userId: string }[];\n payload: P;\n url: string;\n expiration: number;\n };\n\n const { userId, worldId, room, host, autoRejoin = true, logLine } = params;\n\n let exited = false;\n let retryCount = 0;\n let ws: WebSocket;\n let timeoutId: ReturnType<typeof setTimeout>;\n let initialConnection = true;\n\n const peers = new Map<string, IPeer<T, P>>();\n const wsUrl = `wss://${host}/room/${worldId}/${room}?userId=${encodeURIComponent(\n userId,\n )}`;\n\n // Helper for sending (uses the current ws instance)\n const accumulatedMessages: OutMessage[] = [];\n let timeout: ReturnType<typeof setTimeout> = 0;\n function send(type: string, to: \"server\" | string, payload?: any) {\n if (!ws) {\n logLine?.(\"👤 ➡️ ❌\", \"no ws available\");\n return false;\n }\n if (exited || ws.readyState !== WebSocket.OPEN) {\n logLine?.(\"👤 ➡️ ❌\", \"Not in opened state: \" + ws.readyState);\n return false;\n }\n const obj: OutMessage = { type, to, payload };\n accumulatedMessages.push(obj);\n // ws.send(JSON.stringify(obj));\n logLine?.(\"👤 ➡️ 🖥️\", obj);\n clearTimeout(timeout);\n timeout = setTimeout(() => {\n ws.send(JSON.stringify(accumulatedMessages));\n accumulatedMessages.length = 0;\n });\n return true;\n }\n\n function connect() {\n if (exited) return;\n\n ws = new WebSocket(wsUrl);\n\n ws.onopen = () => {\n if (initialConnection) {\n params.onOpen?.();\n initialConnection = false;\n }\n retryCount = 0; // Reset backoff on successful connection\n };\n\n ws.onmessage = (e: MessageEvent) => {\n try {\n const result = JSON.parse(e.data);\n const msgs: Message[] = Array.isArray(result) ? result : [result];\n msgs.forEach((msg) => {\n logLine?.(\"🖥️ ➡️ 👤\", msg);\n if (msg.type === \"peer-joined\" || msg.type === \"peer-left\") {\n updatePeers(msg.users);\n } else if (msg.type === \"ice-server\") {\n params.onIceUrl?.(msg.url, msg.expiration);\n } else if (msg.userId) {\n params.onMessage(msg.type, msg.payload, {\n userId: msg.userId,\n receive: (type: T, payload: P) => send(type, msg.userId, payload),\n });\n }\n });\n } catch {\n logLine?.(\"⚠️ ERROR\", { error: \"invalid-json\" });\n }\n };\n\n ws.onclose = (ev: CloseEvent) => {\n // 1. Check if we should even try to reconnect\n const recoverableCodes = [1001, 1006, 1011, 1012, 1013];\n const isRecoverable = recoverableCodes.includes(ev.code);\n\n if (autoRejoin && !exited && isRecoverable) {\n // 2. Exponential Backoff: 1s, 2s, 4s, 8s... capped at 30s\n const backoff = Math.min(Math.pow(2, retryCount) * 1000, 30000);\n // 3. Add Jitter: +/- 1000ms randomness\n const jitter = Math.random() * 1000;\n const delay = backoff + jitter;\n\n logLine?.(\"🔄 RECONNECTING\", {\n attempt: retryCount + 1,\n delayMs: Math.round(delay),\n });\n\n retryCount++;\n timeoutId = setTimeout(connect, delay);\n } else {\n params.onClose?.({\n code: ev.code,\n reason: ev.reason,\n wasClean: ev.wasClean,\n });\n }\n };\n\n ws.onerror = (ev) => {\n console.error(\"WS Error\", ev);\n params.onError?.();\n };\n }\n\n // Helper for peer tracking (logic from your original code)\n function updatePeers(updatedUsers: { userId: string }[]) {\n const joined: IPeer<T, P>[] = [];\n const left: { userId: string }[] = [];\n const updatedPeerSet = new Set<string>();\n\n updatedUsers.forEach(({ userId: pUserId }) => {\n if (pUserId === userId) return;\n if (!peers.has(pUserId)) {\n const newPeer = {\n userId: pUserId,\n receive: (t: T, p: P) => send(t, pUserId, p),\n };\n peers.set(pUserId, newPeer);\n joined.push(newPeer);\n }\n updatedPeerSet.add(pUserId);\n });\n\n for (const pUserId of peers.keys()) {\n if (!updatedPeerSet.has(pUserId)) {\n peers.delete(pUserId);\n left.push({ userId: pUserId });\n }\n }\n // Notify peer joined first then peer left. (avoid disconnect in case the peer leaving / joining is on the same user).\n if (joined.length) params.onPeerJoined(joined);\n if (left.length) params.onPeerLeft(left);\n }\n\n // Start initial connection\n connect();\n\n return {\n sendToServer(type, payload) {\n send(type, \"server\", payload);\n },\n exitRoom: () => {\n exited = true;\n clearTimeout(timeoutId);\n ws.close();\n },\n };\n}\n",
6
6
  "import type { IPeer } from \"./impl/signal-room.js\";\nimport { enterRoom as baseEnterRoom } from \"./impl/signal-room.js\";\nimport { RoomEvent, WorkerCommand } from \"./signal-room.worker.js\";\n\nexport function enterRoom<T extends string, P = any>({\n userId,\n worldId,\n room,\n host,\n autoRejoin = true,\n onOpen,\n onClose,\n onError,\n onPeerJoined,\n onPeerLeft,\n onIceUrl,\n onMessage,\n logLine,\n workerUrl,\n}: {\n userId: string;\n worldId: string;\n room: string;\n host: string;\n autoRejoin?: boolean;\n onOpen?: () => void;\n onClose?: (ev: Pick<CloseEvent, \"code\" | \"reason\" | \"wasClean\">) => void;\n onError?: () => void;\n onPeerJoined: (users: IPeer<T, P>[]) => void;\n onPeerLeft: (users: { userId: string }[]) => void;\n onIceUrl?(url: string, expiration: number): void;\n onMessage: (type: T, payload: P, from: IPeer<T, P>) => void;\n logLine?: (direction: string, obj?: any) => void;\n\n // Pass the URL to your worker file (bundler will handle it)\n workerUrl?: URL;\n}): {\n exitRoom: () => void;\n sendToServer: <P extends any>(type: T, payload?: P) => void;\n} {\n if (!workerUrl) {\n const CDN_WORKER_URL = `https://cdn.jsdelivr.net/npm/@dobuki/hello-worker/dist/signal-room.worker.min.js`;\n\n console.warn(\n \"Warning: enterRoom called without workerUrl; this may cause issues in some environments. You should pass workerUrl explicitly. Use:\",\n CDN_WORKER_URL,\n );\n return baseEnterRoom<T, P>({\n userId,\n worldId,\n room,\n host,\n autoRejoin,\n onOpen,\n onClose,\n onError,\n onPeerJoined,\n onPeerLeft,\n onIceUrl,\n onMessage,\n });\n }\n const worker = new Worker(workerUrl, { type: \"module\" });\n let exited = false;\n\n function makeUser({ userId }: { userId: string }): IPeer<T, P> {\n return {\n userId,\n receive: (type: T, payload: P) => {\n if (exited) return false;\n worker.postMessage({\n cmd: \"send\",\n toUserId: userId,\n host,\n room,\n type,\n payload,\n } as WorkerCommand);\n return true;\n },\n };\n }\n\n const onWorkerMessage = (e: MessageEvent<RoomEvent<T, P>>) => {\n const ev = e.data;\n\n if (ev.kind === \"open\") onOpen?.();\n else if (ev.kind === \"close\") {\n worker.terminate();\n onClose?.(ev.ev);\n } else if (ev.kind === \"error\") onError?.();\n else if (ev.kind === \"peer-joined\")\n onPeerJoined(ev.users.map((ev) => makeUser({ userId: ev.userId })));\n else if (ev.kind === \"peer-left\") onPeerLeft(ev.users);\n else if (ev.kind === \"ice-server\") onIceUrl?.(ev.url, ev.expiration);\n else if (ev.kind === \"message\")\n onMessage(ev.type, ev.payload, makeUser({ userId: ev.fromUserId }));\n else if (ev.kind === \"log\") logLine?.(ev.direction, ev.obj);\n };\n\n worker.addEventListener(\"message\", onWorkerMessage);\n\n worker.postMessage({\n cmd: \"enter\",\n userId,\n worldId,\n room,\n host,\n autoRejoin,\n } as WorkerCommand);\n\n return {\n exitRoom: () => {\n exited = true;\n worker.removeEventListener(\"message\", onWorkerMessage);\n worker.postMessage({ cmd: \"exit\" } as WorkerCommand);\n },\n sendToServer: <P extends any>(type: T, payload?: P) => {\n worker.postMessage({\n cmd: \"send\",\n toUserId: \"server\",\n host,\n room,\n type,\n payload,\n } as WorkerCommand);\n },\n };\n}\n\nexport type EnterRoom<T extends string, P> = typeof enterRoom<T, P>;\n",
7
- "import { IPeer } from \"./impl/signal-room\";\nimport { EnterRoom, enterRoom } from \"./signal-room\";\n\nexport type SigType = \"offer\" | \"answer\" | \"ice\" | \"request-ice\" | \"broadcast\";\nexport type SigPayload = RTCSessionDescriptionInit | RTCIceCandidateInit;\n\ntype UserState = {\n userId: string;\n pc?: RTCPeerConnection;\n\n // ICE that arrived before we had remoteDescription\n pendingRemoteIce: RTCIceCandidateInit[];\n\n // the signaling \"user\" handle so we can send messages\n peer: IPeer<SigType, SigPayload>;\n\n expirationTimeout?: number;\n};\n\nconst DEFAULT_ENTER_ROOM = enterRoom;\n\nexport function collectPeerConnections({\n worldId,\n receivePeerConnection,\n peerlessUserExpiration = 5000,\n fallbackRtcConfig = {\n iceServers: [{ urls: \"stun:stun.l.google.com:19302\" }],\n },\n enterRoomFunction: enterRoom = DEFAULT_ENTER_ROOM,\n logLine = console.debug,\n onLeaveUser,\n workerUrl,\n onRoomReady,\n onRoomClose,\n onBroadcastMessage,\n}: {\n worldId: string;\n fallbackRtcConfig?: RTCConfiguration;\n enterRoomFunction?: EnterRoom<SigType, SigPayload>;\n onLeaveUser?: (userId: string) => void;\n logLine?: (direction: string, obj?: any) => void;\n workerUrl?: URL;\n peerlessUserExpiration?: number;\n receivePeerConnection(connection: {\n pc: RTCPeerConnection;\n userId: string;\n initiator: boolean;\n restart?: () => void;\n }): void;\n onRoomReady?(info: { host: string; room: string }): void;\n onRoomClose?(info: {\n host: string;\n room: string;\n ev: Pick<CloseEvent, \"reason\" | \"code\" | \"wasClean\">;\n }): void;\n onBroadcastMessage?<P extends any>(payload: P, from: string): void;\n}) {\n const userId = `user-${crypto.randomUUID()}`;\n const users: Map<string, UserState> = new Map();\n let iceUrl: { url: string; expiration: number } | undefined = undefined;\n let rtcConfig: RTCConfiguration & { timestamp: number } = {\n ...fallbackRtcConfig,\n timestamp: Date.now(),\n };\n\n const roomsEntered = new Map<\n string,\n {\n room: string;\n host: string;\n exitRoom: () => void;\n broadcast: <P extends any>(payload: P) => void;\n }\n >();\n\n async function getRtcConfig(iceUrl: string): Promise<RTCConfiguration> {\n if (iceUrl) {\n try {\n const r = await fetch(iceUrl);\n if (!r.ok) throw new Error(`ICE endpoint failed: ${r.status}`);\n rtcConfig = (await r.json()) as RTCConfiguration & {\n timestamp: number;\n };\n } catch (e) {\n console.warn(\"Using fallback rtcConfig:\", e);\n }\n }\n return rtcConfig;\n }\n\n function leaveUser(userId: string) {\n onLeaveUser?.(userId);\n const p = users.get(userId);\n if (!p) return;\n try {\n p.pc?.close();\n } catch {}\n users.delete(userId);\n }\n\n async function flushRemoteIce(state: UserState) {\n if (!state.pc?.remoteDescription) return;\n\n const queued = state.pendingRemoteIce;\n state.pendingRemoteIce = [];\n\n for (const ice of queued) {\n try {\n await state.pc.addIceCandidate(ice);\n } catch (e) {\n logLine(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n }\n }\n\n function exit({ room, host }: { room: string; host: string }) {\n const key = `${host}/room/${room}`;\n const session = roomsEntered.get(key);\n if (session) {\n session.exitRoom();\n roomsEntered.delete(key);\n }\n }\n\n function enter({ room, host }: { room: string; host: string }) {\n return new Promise<void>(async (resolve, reject) => {\n async function setupPC(state: UserState) {\n const ice =\n !iceUrl || iceUrl.expiration - Date.now() < 2000\n ? await requestIce()\n : iceUrl;\n state.pc = new RTCPeerConnection(\n Date.now() - (rtcConfig?.timestamp ?? 0) < 10000\n ? rtcConfig\n : await getRtcConfig(ice.url),\n );\n // Send local ICE candidates to this peer\n state.pc.onicecandidate = (ev) => {\n if (!ev.candidate) return;\n state.peer.receive(\"ice\", ev.candidate.toJSON());\n };\n\n state.pc.onconnectionstatechange = () => {\n logLine(\"💬\", {\n event: \"pc-state\",\n userId: state.userId,\n state: state.pc?.connectionState,\n });\n };\n return state.pc;\n }\n\n async function getPeer(\n peer: IPeer<SigType, SigPayload>,\n ): Promise<[UserState, boolean]> {\n let state = users.get(peer.userId);\n let isNewPeer = false;\n if (!state) {\n const newState: UserState = {\n userId: peer.userId,\n pendingRemoteIce: [],\n peer,\n };\n\n await setupPC(newState);\n state = newState;\n\n // New user\n users.set(state.userId, state);\n isNewPeer = true;\n } else if (state) {\n clearTimeout(state.expirationTimeout);\n state.expirationTimeout = 0;\n }\n if (!state.pc || state.pc?.signalingState === \"closed\") {\n await setupPC(state);\n }\n state.peer = peer;\n return [state, isNewPeer];\n }\n\n async function makeOffer(user: IPeer) {\n // Offer flow: createOffer -> setLocalDescription -> send localDescription\n const [state] = await getPeer(user);\n const pc = state.pc;\n const offer = await pc?.createOffer();\n await pc?.setLocalDescription(offer);\n user.receive(\"offer\", pc?.localDescription?.toJSON()!);\n }\n\n let icePromiseResolve:\n | undefined\n | ((url: { url: string; expiration: number }) => void);\n async function requestIce() {\n const iceUrl = await new Promise<{ url: string; expiration: number }>(\n (resolve) => {\n icePromiseResolve = resolve;\n sendToServer(\"request-ice\");\n },\n );\n icePromiseResolve = undefined;\n return iceUrl;\n }\n\n const { exitRoom, sendToServer } = enterRoom({\n userId,\n worldId,\n room,\n host,\n logLine,\n workerUrl,\n autoRejoin: true,\n\n onOpen() {\n onRoomReady?.({ room, host });\n resolve();\n },\n onError() {\n console.error(\"onError\");\n reject();\n },\n onClose(ev) {\n onRoomClose?.({ room, host, ev });\n },\n\n // Existing peers initiate to the newcomer\n onPeerJoined(joiningUsers: IPeer<SigType, SigPayload>[]) {\n joiningUsers.forEach(async (user) => {\n const [state, isNewPeer] = await getPeer(user);\n if (!isNewPeer) {\n logLine(\"👤ℹ️\", \"not a new peer: \" + user.userId);\n return;\n }\n const pc = state.pc;\n if (!pc) {\n logLine(\"👤ℹ️\", \"no pc: \" + user.userId);\n return;\n }\n\n async function restart() {\n const state = users.get(user.userId);\n if (state) {\n state.pc = undefined;\n const pc = await setupPC(state);\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await new Promise((resolve) => setTimeout(resolve, 3000));\n await makeOffer(user);\n }\n }\n\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await makeOffer(user);\n });\n },\n\n onPeerLeft(leavingUsers: { userId: string }[]) {\n leavingUsers.forEach(({ userId }) => {\n const state = users.get(userId);\n if (!state) return;\n state.expirationTimeout = setTimeout(\n () => leaveUser(userId),\n peerlessUserExpiration ?? 0,\n );\n });\n },\n\n onIceUrl(url, expiration) {\n iceUrl = { url, expiration };\n icePromiseResolve?.(iceUrl);\n },\n\n async onMessage(type: SigType, payload: any, from: IPeer) {\n const [state] = await getPeer(from);\n const pc = state.pc;\n if (!pc) return;\n\n if (type === \"offer\") {\n receivePeerConnection({\n pc,\n userId: from.userId,\n initiator: false,\n restart() {\n // reset PC\n state.pc = undefined;\n },\n });\n // Responder: set remote offer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n\n // Create and send answer\n const answer = await pc.createAnswer();\n await pc.setLocalDescription(answer);\n\n from.receive(\"answer\", pc.localDescription?.toJSON()!);\n\n // Now safe to apply any queued ICE from this peer\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"answer\") {\n // Initiator: set remote answer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"ice\") {\n const ice = payload as RTCIceCandidateInit;\n\n // If we don't have remoteDescription yet, queue it\n if (!pc.remoteDescription) {\n state.pendingRemoteIce.push(ice);\n return;\n }\n\n try {\n await pc.addIceCandidate(ice);\n } catch (e) {\n logLine(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n return;\n }\n\n if (type === \"broadcast\") {\n onBroadcastMessage?.(payload, from.userId);\n }\n },\n });\n roomsEntered.set(`${host}/room/${room}`, {\n exitRoom,\n room,\n host,\n broadcast: (payload) => {\n sendToServer(\"broadcast\", payload);\n },\n });\n });\n }\n\n return {\n userId,\n enterRoom: enter,\n exitRoom: exit,\n leaveUser,\n broadcast<P extends any>(payload: P) {\n roomsEntered.forEach((room) => room.broadcast(payload));\n },\n end() {\n roomsEntered.forEach(({ exitRoom }) => exitRoom());\n roomsEntered.clear();\n users.forEach(({ userId }) => leaveUser(userId));\n users.clear();\n },\n };\n}\n\n/*\nTurn Token ID\n<CF_TURN_TOKEN_ID>\n\nAPI Token\n<CF_RTC_API_TOKEN>\n\nCURL\ncurl \\\n\t-H \"Authorization: Bearer <CF_RTC_API_TOKEN>\" \\\n\t-H \"Content-Type: application/json\" -d '{\"ttl\": 86400}' \\\n\thttps://rtc.live.cloudflare.com/v1/turn/keys/<CF_TURN_TOKEN_ID>/credentials/generate-ice-servers\n\nJSON\n{\n\t\"iceServers\": [\n {\n \"urls\": [\n \"stun:stun.cloudflare.com:3478\",\n \"turn:turn.cloudflare.com:3478?transport=udp\",\n \"turn:turn.cloudflare.com:3478?transport=tcp\",\n \"turns:turn.cloudflare.com:5349?transport=tcp\"\n ],\n \"username\": \"xxxx\",\n \"credential\": \"yyyy\",\n }\n ]\n}\n\n*/\n",
8
- "import { EnterRoom, enterRoom } from \"./signal-room\";\nimport {\n SigType,\n SigPayload,\n collectPeerConnections,\n} from \"./webrtc-peer-collector\";\n\ntype UserListener = (\n user: string,\n action: \"join\" | \"leave\",\n users: string[],\n) => void;\n\nexport function enterWorld<\n S extends string | ArrayBufferView = string | ArrayBufferView,\n R extends string | ArrayBufferLike = string | ArrayBufferLike,\n>({\n worldId,\n logLine = console.debug,\n enterRoomFunction = enterRoom,\n peerlessUserExpiration,\n workerUrl,\n onRoomReady,\n onRoomClose,\n dataChannelOptions,\n}: {\n worldId: string;\n logLine?: (direction: string, obj?: any) => void;\n enterRoomFunction?: EnterRoom<SigType, SigPayload>;\n peerlessUserExpiration?: number;\n workerUrl?: URL;\n onRoomReady?(info: { host: string; room: string }): void;\n onRoomClose?(info: {\n host: string;\n room: string;\n ev: Pick<CloseEvent, \"reason\" | \"code\" | \"wasClean\">;\n }): void;\n dataChannelOptions?: RTCDataChannelInit;\n}) {\n const userIds: string[] = [];\n\n const messagesListeners = new Set<(data: R, from: string) => void>();\n\n function createDataChannel(\n pc: RTCPeerConnection,\n peerUserId: string,\n initiator: boolean,\n restart?: () => void,\n ) {\n if (initiator) {\n const dc = pc.createDataChannel(\"data\", dataChannelOptions);\n wireDataChannel(peerUserId, dc, restart);\n dataChannels.set(peerUserId, dc);\n } else {\n function listener(ev: RTCDataChannelEvent) {\n const dc = ev.channel;\n wireDataChannel(peerUserId, dc, restart);\n dataChannels.set(peerUserId, dc);\n }\n pc.addEventListener(\"datachannel\", listener);\n return () => {\n pc.removeEventListener(\"datachannel\", listener);\n };\n }\n }\n\n function conveyMessage(data: any, userId: string) {\n messagesListeners.forEach((listener) => listener(data, userId));\n }\n\n function wireDataChannel(\n userId: string,\n dc: RTCDataChannel,\n restart?: () => void,\n ) {\n dc.onopen = () => {\n logLine(\"💬\", { event: \"dc-open\", userId });\n userIds.push(userId);\n userListeners.forEach((listener) => listener(userId, \"join\", userIds));\n };\n const onmessage = ({ data }: MessageEvent) => {\n conveyMessage(data, userId);\n // logLine(\"💬\", { event: \"dc-message\", userId, data });\n };\n dc.addEventListener(\"message\", onmessage);\n dc.addEventListener(\"close\", () => {\n logLine(\"💬\", { event: \"dc-close\", userId });\n userIds.splice(userIds.indexOf(userId), 1);\n userListeners.forEach((listener) => listener(userId, \"leave\", userIds));\n dc.removeEventListener(\"message\", onmessage);\n restart?.();\n });\n dc.onerror = () => logLine(\"⚠️ ERROR\", { error: \"dc-error\", userId });\n }\n\n const dataChannels = new Map<string, RTCDataChannel>();\n const userListeners = new Set<UserListener>();\n\n const {\n userId,\n enterRoom,\n exitRoom,\n leaveUser,\n broadcast,\n end: endPeerCollection,\n } = collectPeerConnections({\n worldId,\n enterRoomFunction,\n logLine,\n workerUrl,\n peerlessUserExpiration,\n onRoomReady,\n onRoomClose,\n onLeaveUser(userId: string) {\n const dc = dataChannels.get(userId);\n try {\n dc?.close();\n } catch {}\n dataChannels.delete(userId);\n },\n receivePeerConnection({ pc, userId, initiator, restart }) {\n createDataChannel(pc, userId, initiator, restart);\n },\n onBroadcastMessage(payload, from) {\n conveyMessage(payload, from);\n logLine(\"📢\", { event: \"broadcast\", userId, data: payload });\n },\n });\n\n function send(data: S, userId?: string) {\n dataChannels.forEach((dataChannel, pUserId) => {\n if (userId && pUserId !== userId) return;\n if (dataChannel.readyState === \"open\") {\n dataChannel.send(data as any);\n }\n });\n }\n\n function removeMessageListener(listener: (data: R, from: string) => void) {\n messagesListeners.delete(listener);\n }\n\n function addMessageListener(listener: (data: R, from: string) => void) {\n messagesListeners.add(listener);\n return () => {\n removeMessageListener(listener);\n };\n }\n\n function removeUserListener(listener: UserListener) {\n userListeners.delete(listener);\n }\n\n function addUserListener(listener: UserListener) {\n userListeners.add(listener);\n return () => {\n removeUserListener(listener);\n };\n }\n\n return {\n userId,\n send,\n broadcast,\n enterRoom,\n exitRoom,\n leaveUser,\n getUsers: () => userIds,\n addMessageListener,\n removeMessageListener,\n addUserListener,\n removeUserListener,\n end() {\n dataChannels.forEach((dataChannel) => {\n try {\n dataChannel.close();\n } catch {}\n });\n dataChannels.clear();\n endPeerCollection();\n userListeners.clear();\n userIds.length = 0;\n },\n };\n}\n"
7
+ "import { IPeer } from \"./impl/signal-room\";\nimport { EnterRoom, enterRoom } from \"./signal-room\";\n\nexport type SigType = \"offer\" | \"answer\" | \"ice\" | \"request-ice\" | \"broadcast\";\nexport type SigPayload = RTCSessionDescriptionInit | RTCIceCandidateInit;\n\ntype UserState = {\n userId: string;\n pc?: RTCPeerConnection;\n\n // ICE that arrived before we had remoteDescription\n pendingRemoteIce: RTCIceCandidateInit[];\n\n // the signaling \"user\" handle so we can send messages\n peer: IPeer<SigType, SigPayload>;\n\n expirationTimeout?: number;\n};\n\nconst DEFAULT_ENTER_ROOM = enterRoom;\n\nexport function collectPeerConnections({\n userId: passedUserId,\n worldId,\n receivePeerConnection,\n peerlessUserExpiration = 5000,\n fallbackRtcConfig = {\n iceServers: [{ urls: \"stun:stun.l.google.com:19302\" }],\n },\n enterRoomFunction: enterRoom = DEFAULT_ENTER_ROOM,\n logLine,\n onLeaveUser,\n workerUrl,\n onRoomReady,\n onRoomClose,\n onBroadcastMessage,\n}: {\n userId?: string;\n worldId: string;\n fallbackRtcConfig?: RTCConfiguration;\n enterRoomFunction?: EnterRoom<SigType, SigPayload>;\n onLeaveUser?: (userId: string) => void;\n logLine?: (direction: string, obj?: any) => void;\n workerUrl?: URL;\n peerlessUserExpiration?: number;\n receivePeerConnection(connection: {\n pc: RTCPeerConnection;\n userId: string;\n initiator: boolean;\n restart?: () => void;\n }): void;\n onRoomReady?(info: { host: string; room: string }): void;\n onRoomClose?(info: {\n host: string;\n room: string;\n ev: Pick<CloseEvent, \"reason\" | \"code\" | \"wasClean\">;\n }): void;\n onBroadcastMessage?<P extends any>(payload: P, from: string): void;\n}) {\n const userId = passedUserId ?? `user-${crypto.randomUUID()}`;\n const users: Map<string, UserState> = new Map();\n let iceUrl: { url: string; expiration: number } | undefined = undefined;\n let rtcConfig: RTCConfiguration & { timestamp: number } = {\n ...fallbackRtcConfig,\n timestamp: Date.now(),\n };\n\n const roomsEntered = new Map<\n string,\n {\n room: string;\n host: string;\n exitRoom: () => void;\n broadcast: <P extends any>(payload: P) => void;\n }\n >();\n\n async function getRtcConfig(\n iceUrl: string,\n ): Promise<RTCConfiguration & { timestamp: number }> {\n if (iceUrl) {\n try {\n const r = await fetch(iceUrl);\n if (!r.ok) throw new Error(`ICE endpoint failed: ${r.status}`);\n rtcConfig = (await r.json()) as RTCConfiguration & {\n timestamp: number;\n };\n } catch (e) {\n console.warn(\"Using fallback rtcConfig:\", e);\n }\n }\n return rtcConfig;\n }\n\n function leaveUser(userId: string) {\n onLeaveUser?.(userId);\n const p = users.get(userId);\n if (!p) return;\n try {\n p.pc?.close();\n } catch {}\n users.delete(userId);\n }\n\n async function flushRemoteIce(state: UserState) {\n if (!state.pc?.remoteDescription) return;\n\n const queued = state.pendingRemoteIce;\n state.pendingRemoteIce = [];\n\n for (const ice of queued) {\n try {\n await state.pc.addIceCandidate(ice);\n } catch (e) {\n logLine?.(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n }\n }\n\n function exit({ room, host }: { room: string; host: string }) {\n const key = `${host}/room/${room}`;\n const session = roomsEntered.get(key);\n if (session) {\n session.exitRoom();\n roomsEntered.delete(key);\n }\n }\n\n function enter({ room, host }: { room: string; host: string }) {\n return new Promise<void>(async (resolve, reject) => {\n async function setupPC(state: UserState) {\n const now = Date.now();\n if (now - (rtcConfig?.timestamp ?? 0) > 10000) {\n console.log(\"ICE expiration\", iceUrl?.expiration, \"-\", now);\n const ice =\n !iceUrl || iceUrl.expiration - now < 2000\n ? await requestIce()\n : iceUrl;\n rtcConfig = await getRtcConfig(ice.url);\n }\n state.pc = new RTCPeerConnection(rtcConfig);\n // Send local ICE candidates to this peer\n state.pc.onicecandidate = (ev) => {\n if (!ev.candidate) return;\n state.peer.receive(\"ice\", ev.candidate.toJSON());\n };\n\n state.pc.onconnectionstatechange = () => {\n logLine?.(\"💬\", {\n event: \"pc-state\",\n userId: state.userId,\n state: state.pc?.connectionState,\n });\n };\n return state.pc;\n }\n\n async function getPeer(\n peer: IPeer<SigType, SigPayload>,\n ): Promise<[UserState, boolean]> {\n let state = users.get(peer.userId);\n let isNewPeer = false;\n if (!state) {\n const newState: UserState = {\n userId: peer.userId,\n pendingRemoteIce: [],\n peer,\n };\n\n await setupPC(newState);\n state = newState;\n\n // New user\n users.set(state.userId, state);\n isNewPeer = true;\n } else if (state) {\n clearTimeout(state.expirationTimeout);\n state.expirationTimeout = 0;\n }\n if (!state.pc || state.pc?.signalingState === \"closed\") {\n await setupPC(state);\n }\n state.peer = peer;\n return [state, isNewPeer];\n }\n\n async function makeOffer(user: IPeer) {\n // Offer flow: createOffer -> setLocalDescription -> send localDescription\n const [state] = await getPeer(user);\n const pc = state.pc;\n const offer = await pc?.createOffer();\n await pc?.setLocalDescription(offer);\n user.receive(\"offer\", pc?.localDescription?.toJSON()!);\n }\n\n let icePromiseResolve:\n | undefined\n | ((url: { url: string; expiration: number }) => void);\n async function requestIce() {\n const iceUrl = await new Promise<{ url: string; expiration: number }>(\n (resolve) => {\n icePromiseResolve = resolve;\n sendToServer(\"request-ice\");\n },\n );\n icePromiseResolve = undefined;\n return iceUrl;\n }\n\n const { exitRoom, sendToServer } = enterRoom({\n userId,\n worldId,\n room,\n host,\n logLine,\n workerUrl,\n autoRejoin: true,\n\n onOpen() {\n onRoomReady?.({ room, host });\n resolve();\n },\n onError() {\n console.error(\"onError\");\n reject();\n },\n onClose(ev) {\n onRoomClose?.({ room, host, ev });\n },\n\n // Existing peers initiate to the newcomer\n onPeerJoined(joiningUsers: IPeer<SigType, SigPayload>[]) {\n joiningUsers.forEach(async (user) => {\n const [state, isNewPeer] = await getPeer(user);\n if (!isNewPeer) {\n logLine?.(\"👤ℹ️\", \"not a new peer: \" + user.userId);\n return;\n }\n const pc = state.pc;\n if (!pc) {\n logLine?.(\"👤ℹ️\", \"no pc: \" + user.userId);\n return;\n }\n\n async function restart() {\n const state = users.get(user.userId);\n if (state) {\n state.pc = undefined;\n const pc = await setupPC(state);\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await new Promise((resolve) => setTimeout(resolve, 3000));\n await makeOffer(user);\n }\n }\n\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await makeOffer(user);\n });\n },\n\n onPeerLeft(leavingUsers: { userId: string }[]) {\n leavingUsers.forEach(({ userId }) => {\n const state = users.get(userId);\n if (!state) return;\n state.expirationTimeout = setTimeout(\n () => leaveUser(userId),\n peerlessUserExpiration ?? 0,\n );\n });\n },\n\n onIceUrl(url, expiration) {\n iceUrl = { url, expiration };\n icePromiseResolve?.(iceUrl);\n },\n\n async onMessage(type: SigType, payload: any, from: IPeer) {\n const [state] = await getPeer(from);\n const pc = state.pc;\n if (!pc) return;\n\n if (type === \"offer\") {\n receivePeerConnection({\n pc,\n userId: from.userId,\n initiator: false,\n restart() {\n // reset PC\n state.pc = undefined;\n },\n });\n // Responder: set remote offer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n\n // Create and send answer\n const answer = await pc.createAnswer();\n await pc.setLocalDescription(answer);\n\n from.receive(\"answer\", pc.localDescription?.toJSON()!);\n\n // Now safe to apply any queued ICE from this peer\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"answer\") {\n // Initiator: set remote answer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"ice\") {\n const ice = payload as RTCIceCandidateInit;\n\n // If we don't have remoteDescription yet, queue it\n if (!pc.remoteDescription) {\n state.pendingRemoteIce.push(ice);\n return;\n }\n\n try {\n await pc.addIceCandidate(ice);\n } catch (e) {\n logLine?.(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n return;\n }\n\n if (type === \"broadcast\") {\n onBroadcastMessage?.(payload, from.userId);\n }\n },\n });\n roomsEntered.set(`${host}/room/${room}`, {\n exitRoom,\n room,\n host,\n broadcast: (payload) => {\n sendToServer(\"broadcast\", payload);\n },\n });\n });\n }\n\n return {\n userId,\n enterRoom: enter,\n exitRoom: exit,\n leaveUser,\n broadcast<P extends any>(payload: P) {\n roomsEntered.forEach((room) => room.broadcast(payload));\n },\n end() {\n roomsEntered.forEach(({ exitRoom }) => exitRoom());\n roomsEntered.clear();\n users.forEach(({ userId }) => leaveUser(userId));\n users.clear();\n },\n };\n}\n\n/*\nTurn Token ID\n<CF_TURN_TOKEN_ID>\n\nAPI Token\n<CF_RTC_API_TOKEN>\n\nCURL\ncurl \\\n\t-H \"Authorization: Bearer <CF_RTC_API_TOKEN>\" \\\n\t-H \"Content-Type: application/json\" -d '{\"ttl\": 86400}' \\\n\thttps://rtc.live.cloudflare.com/v1/turn/keys/<CF_TURN_TOKEN_ID>/credentials/generate-ice-servers\n\nJSON\n{\n\t\"iceServers\": [\n {\n \"urls\": [\n \"stun:stun.cloudflare.com:3478\",\n \"turn:turn.cloudflare.com:3478?transport=udp\",\n \"turn:turn.cloudflare.com:3478?transport=tcp\",\n \"turns:turn.cloudflare.com:5349?transport=tcp\"\n ],\n \"username\": \"xxxx\",\n \"credential\": \"yyyy\",\n }\n ]\n}\n\n*/\n",
8
+ "import { EnterRoom, enterRoom } from \"./signal-room\";\nimport {\n SigType,\n SigPayload,\n collectPeerConnections,\n} from \"./webrtc-peer-collector\";\n\ntype UserListener = (\n user: string,\n action: \"join\" | \"leave\",\n users: string[],\n) => void;\n\nexport function enterWorld<\n S extends string | ArrayBufferView = string | ArrayBufferView,\n R extends string | ArrayBufferLike = string | ArrayBufferLike,\n>({\n userId: passedUserId,\n worldId,\n logLine,\n enterRoomFunction = enterRoom,\n peerlessUserExpiration,\n workerUrl,\n onRoomReady,\n onRoomClose,\n dataChannelOptions,\n}: {\n userId?: string;\n worldId: string;\n logLine?: (direction: string, obj?: any) => void;\n enterRoomFunction?: EnterRoom<SigType, SigPayload>;\n peerlessUserExpiration?: number;\n workerUrl?: URL;\n onRoomReady?(info: { host: string; room: string }): void;\n onRoomClose?(info: {\n host: string;\n room: string;\n ev: Pick<CloseEvent, \"reason\" | \"code\" | \"wasClean\">;\n }): void;\n dataChannelOptions?: RTCDataChannelInit;\n}) {\n const userIds: string[] = [];\n\n const messagesListeners = new Set<(data: R, from: string) => void>();\n\n function createDataChannel(\n pc: RTCPeerConnection,\n peerUserId: string,\n initiator: boolean,\n restart?: () => void,\n ) {\n if (initiator) {\n const dc = pc.createDataChannel(\"data\", dataChannelOptions);\n wireDataChannel(peerUserId, dc, restart);\n dataChannels.set(peerUserId, dc);\n } else {\n function listener(ev: RTCDataChannelEvent) {\n const dc = ev.channel;\n wireDataChannel(peerUserId, dc, restart);\n dataChannels.set(peerUserId, dc);\n }\n pc.addEventListener(\"datachannel\", listener);\n return () => {\n pc.removeEventListener(\"datachannel\", listener);\n };\n }\n }\n\n function conveyMessage(data: any, userId: string) {\n messagesListeners.forEach((listener) => listener(data, userId));\n }\n\n function wireDataChannel(\n userId: string,\n dc: RTCDataChannel,\n restart?: () => void,\n ) {\n dc.onopen = () => {\n logLine?.(\"💬\", { event: \"dc-open\", userId });\n userIds.push(userId);\n userListeners.forEach((listener) => listener(userId, \"join\", userIds));\n };\n const onmessage = ({ data }: MessageEvent) => {\n conveyMessage(data, userId);\n // logLine(\"💬\", { event: \"dc-message\", userId, data });\n };\n dc.addEventListener(\"message\", onmessage);\n dc.addEventListener(\"close\", () => {\n logLine?.(\"💬\", { event: \"dc-close\", userId });\n userIds.splice(userIds.indexOf(userId), 1);\n userListeners.forEach((listener) => listener(userId, \"leave\", userIds));\n dc.removeEventListener(\"message\", onmessage);\n restart?.();\n });\n dc.onerror = () => logLine?.(\"⚠️ ERROR\", { error: \"dc-error\", userId });\n }\n\n const dataChannels = new Map<string, RTCDataChannel>();\n const userListeners = new Set<UserListener>();\n\n const {\n userId,\n enterRoom,\n exitRoom,\n leaveUser,\n broadcast,\n end: endPeerCollection,\n } = collectPeerConnections({\n userId: passedUserId,\n worldId,\n enterRoomFunction,\n logLine,\n workerUrl,\n peerlessUserExpiration,\n onRoomReady,\n onRoomClose,\n onLeaveUser(userId: string) {\n const dc = dataChannels.get(userId);\n try {\n dc?.close();\n } catch {}\n dataChannels.delete(userId);\n },\n receivePeerConnection({ pc, userId, initiator, restart }) {\n createDataChannel(pc, userId, initiator, restart);\n },\n onBroadcastMessage(payload, from) {\n conveyMessage(payload, from);\n logLine?.(\"📢\", { event: \"broadcast\", userId, data: payload });\n },\n });\n\n function send(data: S, userId?: string) {\n dataChannels.forEach((dataChannel, pUserId) => {\n if (userId && pUserId !== userId) return;\n if (dataChannel.readyState === \"open\") {\n dataChannel.send(data as any);\n }\n });\n }\n\n function removeMessageListener(listener: (data: R, from: string) => void) {\n messagesListeners.delete(listener);\n }\n\n function addMessageListener(listener: (data: R, from: string) => void) {\n messagesListeners.add(listener);\n return () => {\n removeMessageListener(listener);\n };\n }\n\n function removeUserListener(listener: UserListener) {\n userListeners.delete(listener);\n }\n\n function addUserListener(listener: UserListener) {\n userListeners.add(listener);\n return () => {\n removeUserListener(listener);\n };\n }\n\n return {\n userId,\n send,\n broadcast,\n enterRoom,\n exitRoom,\n leaveUser,\n getUsers: () => userIds,\n addMessageListener,\n removeMessageListener,\n addUserListener,\n removeUserListener,\n end() {\n dataChannels.forEach((dataChannel) => {\n try {\n dataChannel.close();\n } catch {}\n });\n dataChannels.clear();\n endPeerCollection();\n userListeners.clear();\n userIds.length = 0;\n },\n };\n}\n"
9
9
  ],
10
- "mappings": "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",
11
- "debugId": "C5894DD7F36420E164756E2164756E21",
10
+ "mappings": "AAUO,SAAS,CAAoC,CAAC,EAiBnD,CAUA,IAAQ,SAAQ,UAAS,OAAM,OAAM,aAAa,GAAM,WAAY,EAEhE,EAAS,GACT,EAAa,EACb,EACA,EACA,EAAoB,GAElB,EAAQ,IAAI,IACZ,EAAQ,SAAS,UAAa,KAAW,YAAe,mBAC5D,CACF,IAGM,EAAoC,CAAC,EACvC,EAAyC,EAC7C,SAAS,CAAI,CAAC,EAAc,EAAuB,EAAe,CAChE,GAAI,CAAC,EAEH,OADA,IAAU,oBAAU,iBAAiB,EAC9B,GAET,GAAI,GAAU,EAAG,aAAe,UAAU,KAExC,OADA,IAAU,oBAAU,wBAA0B,EAAG,UAAU,EACpD,GAET,IAAM,EAAkB,CAAE,OAAM,KAAI,SAAQ,EAS5C,OARA,EAAoB,KAAK,CAAG,EAE5B,IAAU,gCAAY,CAAG,EACzB,aAAa,CAAO,EACpB,EAAU,WAAW,IAAM,CACzB,EAAG,KAAK,KAAK,UAAU,CAAmB,CAAC,EAC3C,EAAoB,OAAS,EAC9B,EACM,GAGT,SAAS,CAAO,EAAG,CACjB,GAAI,EAAQ,OAEZ,EAAK,IAAI,UAAU,CAAK,EAExB,EAAG,OAAS,IAAM,CAChB,GAAI,EACF,EAAO,SAAS,EAChB,EAAoB,GAEtB,EAAa,GAGf,EAAG,UAAY,CAAC,IAAoB,CAClC,GAAI,CACF,IAAM,EAAS,KAAK,MAAM,EAAE,IAAI,GACR,MAAM,QAAQ,CAAM,EAAI,EAAS,CAAC,CAAM,GAC3D,QAAQ,CAAC,IAAQ,CAEpB,GADA,IAAU,gCAAY,CAAG,EACrB,EAAI,OAAS,eAAiB,EAAI,OAAS,YAC7C,EAAY,EAAI,KAAK,EAChB,QAAI,EAAI,OAAS,aACtB,EAAO,WAAW,EAAI,IAAK,EAAI,UAAU,EACpC,QAAI,EAAI,OACb,EAAO,UAAU,EAAI,KAAM,EAAI,QAAS,CACtC,OAAQ,EAAI,OACZ,QAAS,CAAC,EAAS,IAAe,EAAK,EAAM,EAAI,OAAQ,CAAO,CAClE,CAAC,EAEJ,EACD,KAAM,CACN,IAAU,WAAW,CAAE,MAAO,cAAe,CAAC,IAIlD,EAAG,QAAU,CAAC,IAAmB,CAG/B,IAAM,EADmB,CAAC,KAAM,KAAM,KAAM,KAAM,IAAI,EACf,SAAS,EAAG,IAAI,EAEvD,GAAI,GAAc,CAAC,GAAU,EAAe,CAE1C,IAAM,EAAU,KAAK,IAAI,KAAK,IAAI,EAAG,CAAU,EAAI,KAAM,KAAK,EAExD,EAAS,KAAK,OAAO,EAAI,KACzB,EAAQ,EAAU,EAExB,IAAU,4BAAkB,CAC1B,QAAS,EAAa,EACtB,QAAS,KAAK,MAAM,CAAK,CAC3B,CAAC,EAED,IACA,EAAY,WAAW,EAAS,CAAK,EAErC,OAAO,UAAU,CACf,KAAM,EAAG,KACT,OAAQ,EAAG,OACX,SAAU,EAAG,QACf,CAAC,GAIL,EAAG,QAAU,CAAC,IAAO,CACnB,QAAQ,MAAM,WAAY,CAAE,EAC5B,EAAO,UAAU,GAKrB,SAAS,CAAW,CAAC,EAAoC,CACvD,IAAM,EAAwB,CAAC,EACzB,EAA6B,CAAC,EAC9B,EAAiB,IAAI,IAE3B,EAAa,QAAQ,EAAG,OAAQ,KAAc,CAC5C,GAAI,IAAY,EAAQ,OACxB,GAAI,CAAC,EAAM,IAAI,CAAO,EAAG,CACvB,IAAM,EAAU,CACd,OAAQ,EACR,QAAS,CAAC,EAAM,IAAS,EAAK,EAAG,EAAS,CAAC,CAC7C,EACA,EAAM,IAAI,EAAS,CAAO,EAC1B,EAAO,KAAK,CAAO,EAErB,EAAe,IAAI,CAAO,EAC3B,EAED,QAAW,KAAW,EAAM,KAAK,EAC/B,GAAI,CAAC,EAAe,IAAI,CAAO,EAC7B,EAAM,OAAO,CAAO,EACpB,EAAK,KAAK,CAAE,OAAQ,CAAQ,CAAC,EAIjC,GAAI,EAAO,OAAQ,EAAO,aAAa,CAAM,EAC7C,GAAI,EAAK,OAAQ,EAAO,WAAW,CAAI,EAMzC,OAFA,EAAQ,EAED,CACL,YAAY,CAAC,EAAM,EAAS,CAC1B,EAAK,EAAM,SAAU,CAAO,GAE9B,SAAU,IAAM,CACd,EAAS,GACT,aAAa,CAAS,EACtB,EAAG,MAAM,EAEb,ECrLK,SAAS,CAAoC,EAClD,SACA,UACA,OACA,OACA,aAAa,GACb,SACA,UACA,UACA,eACA,aACA,WACA,YACA,UACA,aAqBA,CACA,GAAI,CAAC,EAOH,OAJA,QAAQ,KACN,sIAHqB,kFAKvB,EACO,EAAoB,CACzB,SACA,UACA,OACA,OACA,aACA,SACA,UACA,UACA,eACA,aACA,WACA,WACF,CAAC,EAEH,IAAM,EAAS,IAAI,OAAO,EAAW,CAAE,KAAM,QAAS,CAAC,EACnD,EAAS,GAEb,SAAS,CAAQ,EAAG,UAA2C,CAC7D,MAAO,CACL,SACA,QAAS,CAAC,EAAS,IAAe,CAChC,GAAI,EAAQ,MAAO,GASnB,OARA,EAAO,YAAY,CACjB,IAAK,OACL,SAAU,EACV,OACA,OACA,OACA,SACF,CAAkB,EACX,GAEX,EAGF,IAAM,EAAkB,CAAC,IAAqC,CAC5D,IAAM,EAAK,EAAE,KAEb,GAAI,EAAG,OAAS,OAAQ,IAAS,EAC5B,QAAI,EAAG,OAAS,QACnB,EAAO,UAAU,EACjB,IAAU,EAAG,EAAE,EACV,QAAI,EAAG,OAAS,QAAS,IAAU,EACrC,QAAI,EAAG,OAAS,cACnB,EAAa,EAAG,MAAM,IAAI,CAAC,IAAO,EAAS,CAAE,OAAQ,EAAG,MAAO,CAAC,CAAC,CAAC,EAC/D,QAAI,EAAG,OAAS,YAAa,EAAW,EAAG,KAAK,EAChD,QAAI,EAAG,OAAS,aAAc,IAAW,EAAG,IAAK,EAAG,UAAU,EAC9D,QAAI,EAAG,OAAS,UACnB,EAAU,EAAG,KAAM,EAAG,QAAS,EAAS,CAAE,OAAQ,EAAG,UAAW,CAAC,CAAC,EAC/D,QAAI,EAAG,OAAS,MAAO,IAAU,EAAG,UAAW,EAAG,GAAG,GAc5D,OAXA,EAAO,iBAAiB,UAAW,CAAe,EAElD,EAAO,YAAY,CACjB,IAAK,QACL,SACA,UACA,OACA,OACA,YACF,CAAkB,EAEX,CACL,SAAU,IAAM,CACd,EAAS,GACT,EAAO,oBAAoB,UAAW,CAAe,EACrD,EAAO,YAAY,CAAE,IAAK,MAAO,CAAkB,GAErD,aAAc,CAAgB,EAAS,IAAgB,CACrD,EAAO,YAAY,CACjB,IAAK,OACL,SAAU,SACV,OACA,OACA,OACA,SACF,CAAkB,EAEtB,EC5GF,IAAM,EAAqB,EAEpB,SAAS,CAAsB,EACpC,OAAQ,EACR,UACA,wBACA,yBAAyB,KACzB,oBAAoB,CAClB,WAAY,CAAC,CAAE,KAAM,8BAA+B,CAAC,CACvD,EACA,kBAAmB,EAAY,EAC/B,UACA,cACA,YACA,cACA,cACA,sBAuBC,CACD,IAAM,EAAS,GAAgB,QAAQ,OAAO,WAAW,IACnD,EAAgC,IAAI,IACtC,EAA0D,OAC1D,EAAsD,IACrD,EACH,UAAW,KAAK,IAAI,CACtB,EAEM,EAAe,IAAI,IAUzB,eAAe,CAAY,CACzB,EACmD,CACnD,GAAI,EACF,GAAI,CACF,IAAM,EAAI,MAAM,MAAM,CAAM,EAC5B,GAAI,CAAC,EAAE,GAAI,MAAU,MAAM,wBAAwB,EAAE,QAAQ,EAC7D,EAAa,MAAM,EAAE,KAAK,EAG1B,MAAO,EAAG,CACV,QAAQ,KAAK,4BAA6B,CAAC,EAG/C,OAAO,EAGT,SAAS,CAAS,CAAC,EAAgB,CACjC,IAAc,CAAM,EACpB,IAAM,EAAI,EAAM,IAAI,CAAM,EAC1B,GAAI,CAAC,EAAG,OACR,GAAI,CACF,EAAE,IAAI,MAAM,EACZ,KAAM,EACR,EAAM,OAAO,CAAM,EAGrB,eAAe,CAAc,CAAC,EAAkB,CAC9C,GAAI,CAAC,EAAM,IAAI,kBAAmB,OAElC,IAAM,EAAS,EAAM,iBACrB,EAAM,iBAAmB,CAAC,EAE1B,QAAW,KAAO,EAChB,GAAI,CACF,MAAM,EAAM,GAAG,gBAAgB,CAAG,EAClC,MAAO,EAAG,CACV,IAAU,WAAW,CACnB,MAAO,iBACP,OAAQ,EAAM,OACd,OAAQ,OAAO,CAAC,CAClB,CAAC,GAKP,SAAS,CAAI,EAAG,OAAM,QAAwC,CAC5D,IAAM,EAAM,GAAG,UAAa,IACtB,EAAU,EAAa,IAAI,CAAG,EACpC,GAAI,EACF,EAAQ,SAAS,EACjB,EAAa,OAAO,CAAG,EAI3B,SAAS,CAAK,EAAG,OAAM,QAAwC,CAC7D,OAAO,IAAI,QAAc,MAAO,EAAS,IAAW,CAClD,eAAe,CAAO,CAAC,EAAkB,CACvC,IAAM,EAAM,KAAK,IAAI,EACrB,GAAI,GAAO,GAAW,WAAa,GAAK,IAAO,CAC7C,QAAQ,IAAI,iBAAkB,GAAQ,WAAY,IAAK,CAAG,EAC1D,IAAM,EACJ,CAAC,GAAU,EAAO,WAAa,EAAM,KACjC,MAAM,EAAW,EACjB,EACN,EAAY,MAAM,EAAa,EAAI,GAAG,EAgBxC,OAdA,EAAM,GAAK,IAAI,kBAAkB,CAAS,EAE1C,EAAM,GAAG,eAAiB,CAAC,IAAO,CAChC,GAAI,CAAC,EAAG,UAAW,OACnB,EAAM,KAAK,QAAQ,MAAO,EAAG,UAAU,OAAO,CAAC,GAGjD,EAAM,GAAG,wBAA0B,IAAM,CACvC,IAAU,eAAK,CACb,MAAO,WACP,OAAQ,EAAM,OACd,MAAO,EAAM,IAAI,eACnB,CAAC,GAEI,EAAM,GAGf,eAAe,CAAO,CACpB,EAC+B,CAC/B,IAAI,EAAQ,EAAM,IAAI,EAAK,MAAM,EAC7B,EAAY,GAChB,GAAI,CAAC,EAAO,CACV,IAAM,EAAsB,CAC1B,OAAQ,EAAK,OACb,iBAAkB,CAAC,EACnB,MACF,EAEA,MAAM,EAAQ,CAAQ,EACtB,EAAQ,EAGR,EAAM,IAAI,EAAM,OAAQ,CAAK,EAC7B,EAAY,GACP,QAAI,EACT,aAAa,EAAM,iBAAiB,EACpC,EAAM,kBAAoB,EAE5B,GAAI,CAAC,EAAM,IAAM,EAAM,IAAI,iBAAmB,SAC5C,MAAM,EAAQ,CAAK,EAGrB,OADA,EAAM,KAAO,EACN,CAAC,EAAO,CAAS,EAG1B,eAAe,CAAS,CAAC,EAAa,CAEpC,IAAO,GAAS,MAAM,EAAQ,CAAI,EAC5B,EAAK,EAAM,GACX,EAAQ,MAAM,GAAI,YAAY,EACpC,MAAM,GAAI,oBAAoB,CAAK,EACnC,EAAK,QAAQ,QAAS,GAAI,kBAAkB,OAAO,CAAE,EAGvD,IAAI,EAGJ,eAAe,CAAU,EAAG,CAC1B,IAAM,EAAS,MAAM,IAAI,QACvB,CAAC,IAAY,CACX,EAAoB,EACpB,EAAa,aAAa,EAE9B,EAEA,OADA,EAAoB,OACb,EAGT,IAAQ,WAAU,gBAAiB,EAAU,CAC3C,SACA,UACA,OACA,OACA,UACA,YACA,WAAY,GAEZ,MAAM,EAAG,CACP,IAAc,CAAE,OAAM,MAAK,CAAC,EAC5B,EAAQ,GAEV,OAAO,EAAG,CACR,QAAQ,MAAM,SAAS,EACvB,EAAO,GAET,OAAO,CAAC,EAAI,CACV,IAAc,CAAE,OAAM,OAAM,IAAG,CAAC,GAIlC,YAAY,CAAC,EAA4C,CACvD,EAAa,QAAQ,MAAO,IAAS,CACnC,IAAO,EAAO,GAAa,MAAM,EAAQ,CAAI,EAC7C,GAAI,CAAC,EAAW,CACd,IAAU,iBAAO,mBAAqB,EAAK,MAAM,EACjD,OAEF,IAAM,EAAK,EAAM,GACjB,GAAI,CAAC,EAAI,CACP,IAAU,iBAAO,UAAY,EAAK,MAAM,EACxC,OAGF,eAAe,CAAO,EAAG,CACvB,IAAM,EAAQ,EAAM,IAAI,EAAK,MAAM,EACnC,GAAI,EAAO,CACT,EAAM,GAAK,OACX,IAAM,EAAK,MAAM,EAAQ,CAAK,EAC9B,EAAsB,CACpB,KACA,OAAQ,EAAK,OACb,UAAW,GACX,SACF,CAAC,EACD,MAAM,IAAI,QAAQ,CAAC,IAAY,WAAW,EAAS,IAAI,CAAC,EACxD,MAAM,EAAU,CAAI,GAIxB,EAAsB,CACpB,KACA,OAAQ,EAAK,OACb,UAAW,GACX,SACF,CAAC,EACD,MAAM,EAAU,CAAI,EACrB,GAGH,UAAU,CAAC,EAAoC,CAC7C,EAAa,QAAQ,EAAG,YAAa,CACnC,IAAM,EAAQ,EAAM,IAAI,CAAM,EAC9B,GAAI,CAAC,EAAO,OACZ,EAAM,kBAAoB,WACxB,IAAM,EAAU,CAAM,EACtB,GAA0B,CAC5B,EACD,GAGH,QAAQ,CAAC,EAAK,EAAY,CACxB,EAAS,CAAE,MAAK,YAAW,EAC3B,IAAoB,CAAM,QAGtB,UAAS,CAAC,EAAe,EAAc,EAAa,CACxD,IAAO,GAAS,MAAM,EAAQ,CAAI,EAC5B,EAAK,EAAM,GACjB,GAAI,CAAC,EAAI,OAET,GAAI,IAAS,QAAS,CACpB,EAAsB,CACpB,KACA,OAAQ,EAAK,OACb,UAAW,GACX,OAAO,EAAG,CAER,EAAM,GAAK,OAEf,CAAC,EAED,MAAM,EAAG,qBAAqB,CAAoC,EAGlE,IAAM,EAAS,MAAM,EAAG,aAAa,EACrC,MAAM,EAAG,oBAAoB,CAAM,EAEnC,EAAK,QAAQ,SAAU,EAAG,kBAAkB,OAAO,CAAE,EAGrD,MAAM,EAAe,CAAK,EAC1B,OAGF,GAAI,IAAS,SAAU,CAErB,MAAM,EAAG,qBAAqB,CAAoC,EAClE,MAAM,EAAe,CAAK,EAC1B,OAGF,GAAI,IAAS,MAAO,CAClB,IAAM,EAAM,EAGZ,GAAI,CAAC,EAAG,kBAAmB,CACzB,EAAM,iBAAiB,KAAK,CAAG,EAC/B,OAGF,GAAI,CACF,MAAM,EAAG,gBAAgB,CAAG,EAC5B,MAAO,EAAG,CACV,IAAU,WAAW,CACnB,MAAO,iBACP,OAAQ,EAAM,OACd,OAAQ,OAAO,CAAC,CAClB,CAAC,EAEH,OAGF,GAAI,IAAS,YACX,IAAqB,EAAS,EAAK,MAAM,EAG/C,CAAC,EACD,EAAa,IAAI,GAAG,UAAa,IAAQ,CACvC,WACA,OACA,OACA,UAAW,CAAC,IAAY,CACtB,EAAa,YAAa,CAAO,EAErC,CAAC,EACF,EAGH,MAAO,CACL,SACA,UAAW,EACX,SAAU,EACV,YACA,SAAwB,CAAC,EAAY,CACnC,EAAa,QAAQ,CAAC,IAAS,EAAK,UAAU,CAAO,CAAC,GAExD,GAAG,EAAG,CACJ,EAAa,QAAQ,EAAG,cAAe,EAAS,CAAC,EACjD,EAAa,MAAM,EACnB,EAAM,QAAQ,EAAG,YAAa,EAAU,CAAM,CAAC,EAC/C,EAAM,MAAM,EAEhB,EC5WK,SAAS,EAGf,EACC,OAAQ,EACR,UACA,UACA,oBAAoB,EACpB,yBACA,YACA,cACA,cACA,sBAeC,CACD,IAAM,EAAoB,CAAC,EAErB,EAAoB,IAAI,IAE9B,SAAS,CAAiB,CACxB,EACA,EACA,EACA,EACA,CACA,GAAI,EAAW,CACb,IAAM,EAAK,EAAG,kBAAkB,OAAQ,CAAkB,EAC1D,EAAgB,EAAY,EAAI,CAAO,EACvC,EAAa,IAAI,EAAY,CAAE,EAC1B,KACL,IAAS,EAAT,QAAiB,CAAC,EAAyB,CACzC,IAAM,EAAK,EAAG,QACd,EAAgB,EAAY,EAAI,CAAO,EACvC,EAAa,IAAI,EAAY,CAAE,GAGjC,OADA,EAAG,iBAAiB,cAAe,CAAQ,EACpC,IAAM,CACX,EAAG,oBAAoB,cAAe,CAAQ,IAKpD,SAAS,CAAa,CAAC,EAAW,EAAgB,CAChD,EAAkB,QAAQ,CAAC,IAAa,EAAS,EAAM,CAAM,CAAC,EAGhE,SAAS,CAAe,CACtB,EACA,EACA,EACA,CACA,EAAG,OAAS,IAAM,CAChB,IAAU,eAAK,CAAE,MAAO,UAAW,QAAO,CAAC,EAC3C,EAAQ,KAAK,CAAM,EACnB,EAAc,QAAQ,CAAC,IAAa,EAAS,EAAQ,OAAQ,CAAO,CAAC,GAEvE,IAAM,EAAY,EAAG,UAAyB,CAC5C,EAAc,EAAM,CAAM,GAG5B,EAAG,iBAAiB,UAAW,CAAS,EACxC,EAAG,iBAAiB,QAAS,IAAM,CACjC,IAAU,eAAK,CAAE,MAAO,WAAY,QAAO,CAAC,EAC5C,EAAQ,OAAO,EAAQ,QAAQ,CAAM,EAAG,CAAC,EACzC,EAAc,QAAQ,CAAC,IAAa,EAAS,EAAQ,QAAS,CAAO,CAAC,EACtE,EAAG,oBAAoB,UAAW,CAAS,EAC3C,IAAU,EACX,EACD,EAAG,QAAU,IAAM,IAAU,WAAW,CAAE,MAAO,WAAY,QAAO,CAAC,EAGvE,IAAM,EAAe,IAAI,IACnB,EAAgB,IAAI,KAGxB,SACA,YACA,WACA,YACA,YACA,IAAK,GACH,EAAuB,CACzB,OAAQ,EACR,UACA,oBACA,UACA,YACA,yBACA,cACA,cACA,WAAW,CAAC,EAAgB,CAC1B,IAAM,EAAK,EAAa,IAAI,CAAM,EAClC,GAAI,CACF,GAAI,MAAM,EACV,KAAM,EACR,EAAa,OAAO,CAAM,GAE5B,qBAAqB,EAAG,KAAI,SAAQ,YAAW,WAAW,CACxD,EAAkB,EAAI,EAAQ,EAAW,CAAO,GAElD,kBAAkB,CAAC,EAAS,EAAM,CAChC,EAAc,EAAS,CAAI,EAC3B,IAAU,eAAK,CAAE,MAAO,YAAa,SAAQ,KAAM,CAAQ,CAAC,EAEhE,CAAC,EAED,SAAS,CAAI,CAAC,EAAS,EAAiB,CACtC,EAAa,QAAQ,CAAC,EAAa,IAAY,CAC7C,GAAI,GAAU,IAAY,EAAQ,OAClC,GAAI,EAAY,aAAe,OAC7B,EAAY,KAAK,CAAW,EAE/B,EAGH,SAAS,CAAqB,CAAC,EAA2C,CACxE,EAAkB,OAAO,CAAQ,EAGnC,SAAS,CAAkB,CAAC,EAA2C,CAErE,OADA,EAAkB,IAAI,CAAQ,EACvB,IAAM,CACX,EAAsB,CAAQ,GAIlC,SAAS,CAAkB,CAAC,EAAwB,CAClD,EAAc,OAAO,CAAQ,EAG/B,SAAS,CAAe,CAAC,EAAwB,CAE/C,OADA,EAAc,IAAI,CAAQ,EACnB,IAAM,CACX,EAAmB,CAAQ,GAI/B,MAAO,CACL,SACA,OACA,YACA,YACA,WACA,YACA,SAAU,IAAM,EAChB,qBACA,wBACA,kBACA,qBACA,GAAG,EAAG,CACJ,EAAa,QAAQ,CAAC,IAAgB,CACpC,GAAI,CACF,EAAY,MAAM,EAClB,KAAM,GACT,EACD,EAAa,MAAM,EACnB,EAAkB,EAClB,EAAc,MAAM,EACpB,EAAQ,OAAS,EAErB",
11
+ "debugId": "FC8366DF61052DAD64756E2164756E21",
12
12
  "names": []
13
13
  }
package/dist/index.js CHANGED
@@ -1,4 +1,4 @@
1
- function m(D){let{userId:F,worldId:R,room:h,host:C,autoRejoin:O=!0,logLine:T}=D,E=!1,J=0,H,L,k=!0,K=new Map,A=`wss://${C}/room/${R}/${h}?userId=${encodeURIComponent(F)}`,W=[],S=0;function w(V,q,X){if(!H)return T?.("\uD83D\uDC64 ➡️ ❌","no ws available"),!1;if(E||H.readyState!==WebSocket.OPEN)return T?.("\uD83D\uDC64 ➡️ ❌","Not in opened state: "+H.readyState),!1;let G={type:V,to:q,payload:X};return W.push(G),T?.("\uD83D\uDC64 ➡️ \uD83D\uDDA5️",G),clearTimeout(S),S=setTimeout(()=>{H.send(JSON.stringify(W)),W.length=0}),!0}function P(){if(E)return;H=new WebSocket(A),H.onopen=()=>{if(k)D.onOpen?.(),k=!1;J=0},H.onmessage=(V)=>{try{let q=JSON.parse(V.data);(Array.isArray(q)?q:[q]).forEach((G)=>{if(T?.("\uD83D\uDDA5️ ➡️ \uD83D\uDC64",G),G.type==="peer-joined"||G.type==="peer-left")N(G.users);else if(G.type==="ice-server")D.onIceUrl?.(G.url,G.expiration);else if(G.userId)D.onMessage(G.type,G.payload,{userId:G.userId,receive:(z,b)=>w(z,G.userId,b)})})}catch{T?.("⚠️ ERROR",{error:"invalid-json"})}},H.onclose=(V)=>{let X=[1001,1006,1011,1012,1013].includes(V.code);if(O&&!E&&X){let G=Math.min(Math.pow(2,J)*1000,30000),z=Math.random()*1000,b=G+z;T?.("\uD83D\uDD04 RECONNECTING",{attempt:J+1,delayMs:Math.round(b)}),J++,L=setTimeout(P,b)}else D.onClose?.({code:V.code,reason:V.reason,wasClean:V.wasClean})},H.onerror=(V)=>{console.error("WS Error",V),D.onError?.()}}function N(V){let q=[],X=[],G=new Set;V.forEach(({userId:z})=>{if(z===F)return;if(!K.has(z)){let b={userId:z,receive:(y,Q)=>w(y,z,Q)};K.set(z,b),q.push(b)}G.add(z)});for(let z of K.keys())if(!G.has(z))K.delete(z),X.push({userId:z});if(q.length)D.onPeerJoined(q);if(X.length)D.onPeerLeft(X)}return P(),{sendToServer(V,q){w(V,"server",q)},exitRoom:()=>{E=!0,clearTimeout(L),H.close()}}}function I({userId:D,worldId:F,room:R,host:h,autoRejoin:C=!0,onOpen:O,onClose:T,onError:E,onPeerJoined:J,onPeerLeft:H,onIceUrl:L,onMessage:k,logLine:K,workerUrl:A}){if(!A)return console.warn("Warning: enterRoom called without workerUrl; this may cause issues in some environments. You should pass workerUrl explicitly. Use:","https://cdn.jsdelivr.net/npm/@dobuki/hello-worker/dist/signal-room.worker.min.js"),m({userId:D,worldId:F,room:R,host:h,autoRejoin:C,onOpen:O,onClose:T,onError:E,onPeerJoined:J,onPeerLeft:H,onIceUrl:L,onMessage:k});let W=new Worker(A,{type:"module"}),S=!1;function w({userId:N}){return{userId:N,receive:(V,q)=>{if(S)return!1;return W.postMessage({cmd:"send",toUserId:N,host:h,room:R,type:V,payload:q}),!0}}}let P=(N)=>{let V=N.data;if(V.kind==="open")O?.();else if(V.kind==="close")W.terminate(),T?.(V.ev);else if(V.kind==="error")E?.();else if(V.kind==="peer-joined")J(V.users.map((q)=>w({userId:q.userId})));else if(V.kind==="peer-left")H(V.users);else if(V.kind==="ice-server")L?.(V.url,V.expiration);else if(V.kind==="message")k(V.type,V.payload,w({userId:V.fromUserId}));else if(V.kind==="log")K?.(V.direction,V.obj)};return W.addEventListener("message",P),W.postMessage({cmd:"enter",userId:D,worldId:F,room:R,host:h,autoRejoin:C}),{exitRoom:()=>{S=!0,W.removeEventListener("message",P),W.postMessage({cmd:"exit"})},sendToServer:(N,V)=>{W.postMessage({cmd:"send",toUserId:"server",host:h,room:R,type:N,payload:V})}}}var l=I;function c({worldId:D,receivePeerConnection:F,peerlessUserExpiration:R=5000,fallbackRtcConfig:h={iceServers:[{urls:"stun:stun.l.google.com:19302"}]},enterRoomFunction:C=l,logLine:O=console.debug,onLeaveUser:T,workerUrl:E,onRoomReady:J,onRoomClose:H,onBroadcastMessage:L}){let k=`user-${crypto.randomUUID()}`,K=new Map,A=void 0,W={...h,timestamp:Date.now()},S=new Map;async function w(X){if(X)try{let G=await fetch(X);if(!G.ok)throw Error(`ICE endpoint failed: ${G.status}`);W=await G.json()}catch(G){console.warn("Using fallback rtcConfig:",G)}return W}function P(X){T?.(X);let G=K.get(X);if(!G)return;try{G.pc?.close()}catch{}K.delete(X)}async function N(X){if(!X.pc?.remoteDescription)return;let G=X.pendingRemoteIce;X.pendingRemoteIce=[];for(let z of G)try{await X.pc.addIceCandidate(z)}catch(b){O("⚠️ ERROR",{error:"add-ice-failed",userId:X.userId,detail:String(b)})}}function V({room:X,host:G}){let z=`${G}/room/${X}`,b=S.get(z);if(b)b.exitRoom(),S.delete(z)}function q({room:X,host:G}){return new Promise(async(z,b)=>{async function y(Z){let Y=!A||A.expiration-Date.now()<2000?await f():A;return Z.pc=new RTCPeerConnection(Date.now()-(W?.timestamp??0)<1e4?W:await w(Y.url)),Z.pc.onicecandidate=(B)=>{if(!B.candidate)return;Z.peer.receive("ice",B.candidate.toJSON())},Z.pc.onconnectionstatechange=()=>{O("\uD83D\uDCAC",{event:"pc-state",userId:Z.userId,state:Z.pc?.connectionState})},Z.pc}async function Q(Z){let Y=K.get(Z.userId),B=!1;if(!Y){let x={userId:Z.userId,pendingRemoteIce:[],peer:Z};await y(x),Y=x,K.set(Y.userId,Y),B=!0}else if(Y)clearTimeout(Y.expirationTimeout),Y.expirationTimeout=0;if(!Y.pc||Y.pc?.signalingState==="closed")await y(Y);return Y.peer=Z,[Y,B]}async function $(Z){let[Y]=await Q(Z),B=Y.pc,x=await B?.createOffer();await B?.setLocalDescription(x),Z.receive("offer",B?.localDescription?.toJSON())}let _;async function f(){let Z=await new Promise((Y)=>{_=Y,g("request-ice")});return _=void 0,Z}let{exitRoom:M,sendToServer:g}=C({userId:k,worldId:D,room:X,host:G,logLine:O,workerUrl:E,autoRejoin:!0,onOpen(){J?.({room:X,host:G}),z()},onError(){console.error("onError"),b()},onClose(Z){H?.({room:X,host:G,ev:Z})},onPeerJoined(Z){Z.forEach(async(Y)=>{let[B,x]=await Q(Y);if(!x){O("\uD83D\uDC64ℹ️","not a new peer: "+Y.userId);return}let j=B.pc;if(!j){O("\uD83D\uDC64ℹ️","no pc: "+Y.userId);return}async function U(){let v=K.get(Y.userId);if(v){v.pc=void 0;let u=await y(v);F({pc:u,userId:Y.userId,initiator:!0,restart:U}),await new Promise((p)=>setTimeout(p,3000)),await $(Y)}}F({pc:j,userId:Y.userId,initiator:!0,restart:U}),await $(Y)})},onPeerLeft(Z){Z.forEach(({userId:Y})=>{let B=K.get(Y);if(!B)return;B.expirationTimeout=setTimeout(()=>P(Y),R??0)})},onIceUrl(Z,Y){A={url:Z,expiration:Y},_?.(A)},async onMessage(Z,Y,B){let[x]=await Q(B),j=x.pc;if(!j)return;if(Z==="offer"){F({pc:j,userId:B.userId,initiator:!1,restart(){x.pc=void 0}}),await j.setRemoteDescription(Y);let U=await j.createAnswer();await j.setLocalDescription(U),B.receive("answer",j.localDescription?.toJSON()),await N(x);return}if(Z==="answer"){await j.setRemoteDescription(Y),await N(x);return}if(Z==="ice"){let U=Y;if(!j.remoteDescription){x.pendingRemoteIce.push(U);return}try{await j.addIceCandidate(U)}catch(v){O("⚠️ ERROR",{error:"add-ice-failed",userId:x.userId,detail:String(v)})}return}if(Z==="broadcast")L?.(Y,B.userId)}});S.set(`${G}/room/${X}`,{exitRoom:M,room:X,host:G,broadcast:(Z)=>{g("broadcast",Z)}})})}return{userId:k,enterRoom:q,exitRoom:V,leaveUser:P,broadcast(X){S.forEach((G)=>G.broadcast(X))},end(){S.forEach(({exitRoom:X})=>X()),S.clear(),K.forEach(({userId:X})=>P(X)),K.clear()}}}function i({worldId:D,logLine:F=console.debug,enterRoomFunction:R=I,peerlessUserExpiration:h,workerUrl:C,onRoomReady:O,onRoomClose:T,dataChannelOptions:E}){let J=[],H=new Set;function L(Q,$,_,f){if(_){let M=Q.createDataChannel("data",E);K($,M,f),A.set($,M)}else{let M=function(g){let Z=g.channel;K($,Z,f),A.set($,Z)};return Q.addEventListener("datachannel",M),()=>{Q.removeEventListener("datachannel",M)}}}function k(Q,$){H.forEach((_)=>_(Q,$))}function K(Q,$,_){$.onopen=()=>{F("\uD83D\uDCAC",{event:"dc-open",userId:Q}),J.push(Q),W.forEach((M)=>M(Q,"join",J))};let f=({data:M})=>{k(M,Q)};$.addEventListener("message",f),$.addEventListener("close",()=>{F("\uD83D\uDCAC",{event:"dc-close",userId:Q}),J.splice(J.indexOf(Q),1),W.forEach((M)=>M(Q,"leave",J)),$.removeEventListener("message",f),_?.()}),$.onerror=()=>F("⚠️ ERROR",{error:"dc-error",userId:Q})}let A=new Map,W=new Set,{userId:S,enterRoom:w,exitRoom:P,leaveUser:N,broadcast:V,end:q}=c({worldId:D,enterRoomFunction:R,logLine:F,workerUrl:C,peerlessUserExpiration:h,onRoomReady:O,onRoomClose:T,onLeaveUser(Q){let $=A.get(Q);try{$?.close()}catch{}A.delete(Q)},receivePeerConnection({pc:Q,userId:$,initiator:_,restart:f}){L(Q,$,_,f)},onBroadcastMessage(Q,$){k(Q,$),F("\uD83D\uDCE2",{event:"broadcast",userId:S,data:Q})}});function X(Q,$){A.forEach((_,f)=>{if($&&f!==$)return;if(_.readyState==="open")_.send(Q)})}function G(Q){H.delete(Q)}function z(Q){return H.add(Q),()=>{G(Q)}}function b(Q){W.delete(Q)}function y(Q){return W.add(Q),()=>{b(Q)}}return{userId:S,send:X,broadcast:V,enterRoom:w,exitRoom:P,leaveUser:N,getUsers:()=>J,addMessageListener:z,removeMessageListener:G,addUserListener:y,removeUserListener:b,end(){A.forEach((Q)=>{try{Q.close()}catch{}}),A.clear(),q(),W.clear(),J.length=0}}}export{i as enterWorld,m as enterRoom,c as collectPeerConnections};
1
+ function c(F){let{userId:h,worldId:A,room:R,host:C,autoRejoin:y=!0,logLine:B}=F,j=!1,E=0,W,k,w=!0,T=new Map,J=`wss://${C}/room/${A}/${R}?userId=${encodeURIComponent(h)}`,$=[],K=0;function b(Q,H,M){if(!W)return B?.("\uD83D\uDC64 ➡️ ❌","no ws available"),!1;if(j||W.readyState!==WebSocket.OPEN)return B?.("\uD83D\uDC64 ➡️ ❌","Not in opened state: "+W.readyState),!1;let G={type:Q,to:H,payload:M};return $.push(G),B?.("\uD83D\uDC64 ➡️ \uD83D\uDDA5️",G),clearTimeout(K),K=setTimeout(()=>{W.send(JSON.stringify($)),$.length=0}),!0}function L(){if(j)return;W=new WebSocket(J),W.onopen=()=>{if(w)F.onOpen?.(),w=!1;E=0},W.onmessage=(Q)=>{try{let H=JSON.parse(Q.data);(Array.isArray(H)?H:[H]).forEach((G)=>{if(B?.("\uD83D\uDDA5️ ➡️ \uD83D\uDC64",G),G.type==="peer-joined"||G.type==="peer-left")S(G.users);else if(G.type==="ice-server")F.onIceUrl?.(G.url,G.expiration);else if(G.userId)F.onMessage(G.type,G.payload,{userId:G.userId,receive:(Y,D)=>b(Y,G.userId,D)})})}catch{B?.("⚠️ ERROR",{error:"invalid-json"})}},W.onclose=(Q)=>{let M=[1001,1006,1011,1012,1013].includes(Q.code);if(y&&!j&&M){let G=Math.min(Math.pow(2,E)*1000,30000),Y=Math.random()*1000,D=G+Y;B?.("\uD83D\uDD04 RECONNECTING",{attempt:E+1,delayMs:Math.round(D)}),E++,k=setTimeout(L,D)}else F.onClose?.({code:Q.code,reason:Q.reason,wasClean:Q.wasClean})},W.onerror=(Q)=>{console.error("WS Error",Q),F.onError?.()}}function S(Q){let H=[],M=[],G=new Set;Q.forEach(({userId:Y})=>{if(Y===h)return;if(!T.has(Y)){let D={userId:Y,receive:(P,v)=>b(P,Y,v)};T.set(Y,D),H.push(D)}G.add(Y)});for(let Y of T.keys())if(!G.has(Y))T.delete(Y),M.push({userId:Y});if(H.length)F.onPeerJoined(H);if(M.length)F.onPeerLeft(M)}return L(),{sendToServer(Q,H){b(Q,"server",H)},exitRoom:()=>{j=!0,clearTimeout(k),W.close()}}}function m({userId:F,worldId:h,room:A,host:R,autoRejoin:C=!0,onOpen:y,onClose:B,onError:j,onPeerJoined:E,onPeerLeft:W,onIceUrl:k,onMessage:w,logLine:T,workerUrl:J}){if(!J)return console.warn("Warning: enterRoom called without workerUrl; this may cause issues in some environments. You should pass workerUrl explicitly. Use:","https://cdn.jsdelivr.net/npm/@dobuki/hello-worker/dist/signal-room.worker.min.js"),c({userId:F,worldId:h,room:A,host:R,autoRejoin:C,onOpen:y,onClose:B,onError:j,onPeerJoined:E,onPeerLeft:W,onIceUrl:k,onMessage:w});let $=new Worker(J,{type:"module"}),K=!1;function b({userId:S}){return{userId:S,receive:(Q,H)=>{if(K)return!1;return $.postMessage({cmd:"send",toUserId:S,host:R,room:A,type:Q,payload:H}),!0}}}let L=(S)=>{let Q=S.data;if(Q.kind==="open")y?.();else if(Q.kind==="close")$.terminate(),B?.(Q.ev);else if(Q.kind==="error")j?.();else if(Q.kind==="peer-joined")E(Q.users.map((H)=>b({userId:H.userId})));else if(Q.kind==="peer-left")W(Q.users);else if(Q.kind==="ice-server")k?.(Q.url,Q.expiration);else if(Q.kind==="message")w(Q.type,Q.payload,b({userId:Q.fromUserId}));else if(Q.kind==="log")T?.(Q.direction,Q.obj)};return $.addEventListener("message",L),$.postMessage({cmd:"enter",userId:F,worldId:h,room:A,host:R,autoRejoin:C}),{exitRoom:()=>{K=!0,$.removeEventListener("message",L),$.postMessage({cmd:"exit"})},sendToServer:(S,Q)=>{$.postMessage({cmd:"send",toUserId:"server",host:R,room:A,type:S,payload:Q})}}}var i=m;function u({userId:F,worldId:h,receivePeerConnection:A,peerlessUserExpiration:R=5000,fallbackRtcConfig:C={iceServers:[{urls:"stun:stun.l.google.com:19302"}]},enterRoomFunction:y=i,logLine:B,onLeaveUser:j,workerUrl:E,onRoomReady:W,onRoomClose:k,onBroadcastMessage:w}){let T=F??`user-${crypto.randomUUID()}`,J=new Map,$=void 0,K={...C,timestamp:Date.now()},b=new Map;async function L(G){if(G)try{let Y=await fetch(G);if(!Y.ok)throw Error(`ICE endpoint failed: ${Y.status}`);K=await Y.json()}catch(Y){console.warn("Using fallback rtcConfig:",Y)}return K}function S(G){j?.(G);let Y=J.get(G);if(!Y)return;try{Y.pc?.close()}catch{}J.delete(G)}async function Q(G){if(!G.pc?.remoteDescription)return;let Y=G.pendingRemoteIce;G.pendingRemoteIce=[];for(let D of Y)try{await G.pc.addIceCandidate(D)}catch(P){B?.("⚠️ ERROR",{error:"add-ice-failed",userId:G.userId,detail:String(P)})}}function H({room:G,host:Y}){let D=`${Y}/room/${G}`,P=b.get(D);if(P)P.exitRoom(),b.delete(D)}function M({room:G,host:Y}){return new Promise(async(D,P)=>{async function v(Z){let X=Date.now();if(X-(K?.timestamp??0)>1e4){console.log("ICE expiration",$?.expiration,"-",X);let q=!$||$.expiration-X<2000?await f():$;K=await L(q.url)}return Z.pc=new RTCPeerConnection(K),Z.pc.onicecandidate=(q)=>{if(!q.candidate)return;Z.peer.receive("ice",q.candidate.toJSON())},Z.pc.onconnectionstatechange=()=>{B?.("\uD83D\uDCAC",{event:"pc-state",userId:Z.userId,state:Z.pc?.connectionState})},Z.pc}async function V(Z){let X=J.get(Z.userId),q=!1;if(!X){let O={userId:Z.userId,pendingRemoteIce:[],peer:Z};await v(O),X=O,J.set(X.userId,X),q=!0}else if(X)clearTimeout(X.expirationTimeout),X.expirationTimeout=0;if(!X.pc||X.pc?.signalingState==="closed")await v(X);return X.peer=Z,[X,q]}async function z(Z){let[X]=await V(Z),q=X.pc,O=await q?.createOffer();await q?.setLocalDescription(O),Z.receive("offer",q?.localDescription?.toJSON())}let N;async function f(){let Z=await new Promise((X)=>{N=X,I("request-ice")});return N=void 0,Z}let{exitRoom:_,sendToServer:I}=y({userId:T,worldId:h,room:G,host:Y,logLine:B,workerUrl:E,autoRejoin:!0,onOpen(){W?.({room:G,host:Y}),D()},onError(){console.error("onError"),P()},onClose(Z){k?.({room:G,host:Y,ev:Z})},onPeerJoined(Z){Z.forEach(async(X)=>{let[q,O]=await V(X);if(!O){B?.("\uD83D\uDC64ℹ️","not a new peer: "+X.userId);return}let x=q.pc;if(!x){B?.("\uD83D\uDC64ℹ️","no pc: "+X.userId);return}async function U(){let g=J.get(X.userId);if(g){g.pc=void 0;let p=await v(g);A({pc:p,userId:X.userId,initiator:!0,restart:U}),await new Promise((l)=>setTimeout(l,3000)),await z(X)}}A({pc:x,userId:X.userId,initiator:!0,restart:U}),await z(X)})},onPeerLeft(Z){Z.forEach(({userId:X})=>{let q=J.get(X);if(!q)return;q.expirationTimeout=setTimeout(()=>S(X),R??0)})},onIceUrl(Z,X){$={url:Z,expiration:X},N?.($)},async onMessage(Z,X,q){let[O]=await V(q),x=O.pc;if(!x)return;if(Z==="offer"){A({pc:x,userId:q.userId,initiator:!1,restart(){O.pc=void 0}}),await x.setRemoteDescription(X);let U=await x.createAnswer();await x.setLocalDescription(U),q.receive("answer",x.localDescription?.toJSON()),await Q(O);return}if(Z==="answer"){await x.setRemoteDescription(X),await Q(O);return}if(Z==="ice"){let U=X;if(!x.remoteDescription){O.pendingRemoteIce.push(U);return}try{await x.addIceCandidate(U)}catch(g){B?.("⚠️ ERROR",{error:"add-ice-failed",userId:O.userId,detail:String(g)})}return}if(Z==="broadcast")w?.(X,q.userId)}});b.set(`${Y}/room/${G}`,{exitRoom:_,room:G,host:Y,broadcast:(Z)=>{I("broadcast",Z)}})})}return{userId:T,enterRoom:M,exitRoom:H,leaveUser:S,broadcast(G){b.forEach((Y)=>Y.broadcast(G))},end(){b.forEach(({exitRoom:G})=>G()),b.clear(),J.forEach(({userId:G})=>S(G)),J.clear()}}}function d({userId:F,worldId:h,logLine:A,enterRoomFunction:R=m,peerlessUserExpiration:C,workerUrl:y,onRoomReady:B,onRoomClose:j,dataChannelOptions:E}){let W=[],k=new Set;function w(V,z,N,f){if(N){let _=V.createDataChannel("data",E);J(z,_,f),$.set(z,_)}else{let _=function(I){let Z=I.channel;J(z,Z,f),$.set(z,Z)};return V.addEventListener("datachannel",_),()=>{V.removeEventListener("datachannel",_)}}}function T(V,z){k.forEach((N)=>N(V,z))}function J(V,z,N){z.onopen=()=>{A?.("\uD83D\uDCAC",{event:"dc-open",userId:V}),W.push(V),K.forEach((_)=>_(V,"join",W))};let f=({data:_})=>{T(_,V)};z.addEventListener("message",f),z.addEventListener("close",()=>{A?.("\uD83D\uDCAC",{event:"dc-close",userId:V}),W.splice(W.indexOf(V),1),K.forEach((_)=>_(V,"leave",W)),z.removeEventListener("message",f),N?.()}),z.onerror=()=>A?.("⚠️ ERROR",{error:"dc-error",userId:V})}let $=new Map,K=new Set,{userId:b,enterRoom:L,exitRoom:S,leaveUser:Q,broadcast:H,end:M}=u({userId:F,worldId:h,enterRoomFunction:R,logLine:A,workerUrl:y,peerlessUserExpiration:C,onRoomReady:B,onRoomClose:j,onLeaveUser(V){let z=$.get(V);try{z?.close()}catch{}$.delete(V)},receivePeerConnection({pc:V,userId:z,initiator:N,restart:f}){w(V,z,N,f)},onBroadcastMessage(V,z){T(V,z),A?.("\uD83D\uDCE2",{event:"broadcast",userId:b,data:V})}});function G(V,z){$.forEach((N,f)=>{if(z&&f!==z)return;if(N.readyState==="open")N.send(V)})}function Y(V){k.delete(V)}function D(V){return k.add(V),()=>{Y(V)}}function P(V){K.delete(V)}function v(V){return K.add(V),()=>{P(V)}}return{userId:b,send:G,broadcast:H,enterRoom:L,exitRoom:S,leaveUser:Q,getUsers:()=>W,addMessageListener:D,removeMessageListener:Y,addUserListener:v,removeUserListener:P,end(){$.forEach((V)=>{try{V.close()}catch{}}),$.clear(),M(),K.clear(),W.length=0}}}export{d as enterWorld,c as enterRoom,u as collectPeerConnections};
2
2
 
3
- //# debugId=E580648F6184549964756E2164756E21
3
+ //# debugId=C7547F1FAAABA26F64756E2164756E21
4
4
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -4,10 +4,10 @@
4
4
  "sourcesContent": [
5
5
  "export interface IPeer<T extends string = string, P = any> {\n userId: string;\n receive(type: T, payload: P): boolean;\n}\n\ntype OutMessage = { type: string; to: string; payload: any };\n\n/**\n * enterRoom connects to the signaling room via WebSocket.\n */\nexport function enterRoom<T extends string, P = any>(params: {\n userId: string;\n worldId: string;\n room: string;\n host: string;\n onOpen?: () => void;\n onClose?: (ev: Pick<CloseEvent, \"code\" | \"reason\" | \"wasClean\">) => void;\n onError?: () => void;\n logLine?: (direction: string, obj?: any) => void;\n onPeerJoined(users: IPeer<T, P>[]): void;\n onPeerLeft(users: { userId: string }[]): void;\n onIceUrl?(url: string, expiration: number): void;\n onMessage(type: T, payload: P, from: IPeer<T, P>): void;\n autoRejoin?: boolean;\n}): {\n exitRoom: () => void;\n sendToServer: <P extends any>(type: T, payload?: P) => void;\n} {\n type Message = {\n type: \"peer-joined\" | \"peer-left\" | \"ice-server\" | T;\n userId: string;\n users: { userId: string }[];\n payload: P;\n url: string;\n expiration: number;\n };\n\n const { userId, worldId, room, host, autoRejoin = true, logLine } = params;\n\n let exited = false;\n let retryCount = 0;\n let ws: WebSocket;\n let timeoutId: ReturnType<typeof setTimeout>;\n let initialConnection = true;\n\n const peers = new Map<string, IPeer<T, P>>();\n const wsUrl = `wss://${host}/room/${worldId}/${room}?userId=${encodeURIComponent(\n userId,\n )}`;\n\n // Helper for sending (uses the current ws instance)\n const accumulatedMessages: OutMessage[] = [];\n let timeout: ReturnType<typeof setTimeout> = 0;\n function send(type: string, to: \"server\" | string, payload?: any) {\n if (!ws) {\n logLine?.(\"👤 ➡️ ❌\", \"no ws available\");\n return false;\n }\n if (exited || ws.readyState !== WebSocket.OPEN) {\n logLine?.(\"👤 ➡️ ❌\", \"Not in opened state: \" + ws.readyState);\n return false;\n }\n const obj: OutMessage = { type, to, payload };\n accumulatedMessages.push(obj);\n // ws.send(JSON.stringify(obj));\n logLine?.(\"👤 ➡️ 🖥️\", obj);\n clearTimeout(timeout);\n timeout = setTimeout(() => {\n ws.send(JSON.stringify(accumulatedMessages));\n accumulatedMessages.length = 0;\n });\n return true;\n }\n\n function connect() {\n if (exited) return;\n\n ws = new WebSocket(wsUrl);\n\n ws.onopen = () => {\n if (initialConnection) {\n params.onOpen?.();\n initialConnection = false;\n }\n retryCount = 0; // Reset backoff on successful connection\n };\n\n ws.onmessage = (e: MessageEvent) => {\n try {\n const result = JSON.parse(e.data);\n const msgs: Message[] = Array.isArray(result) ? result : [result];\n msgs.forEach((msg) => {\n logLine?.(\"🖥️ ➡️ 👤\", msg);\n if (msg.type === \"peer-joined\" || msg.type === \"peer-left\") {\n updatePeers(msg.users);\n } else if (msg.type === \"ice-server\") {\n params.onIceUrl?.(msg.url, msg.expiration);\n } else if (msg.userId) {\n params.onMessage(msg.type, msg.payload, {\n userId: msg.userId,\n receive: (type: T, payload: P) => send(type, msg.userId, payload),\n });\n }\n });\n } catch {\n logLine?.(\"⚠️ ERROR\", { error: \"invalid-json\" });\n }\n };\n\n ws.onclose = (ev: CloseEvent) => {\n // 1. Check if we should even try to reconnect\n const recoverableCodes = [1001, 1006, 1011, 1012, 1013];\n const isRecoverable = recoverableCodes.includes(ev.code);\n\n if (autoRejoin && !exited && isRecoverable) {\n // 2. Exponential Backoff: 1s, 2s, 4s, 8s... capped at 30s\n const backoff = Math.min(Math.pow(2, retryCount) * 1000, 30000);\n // 3. Add Jitter: +/- 1000ms randomness\n const jitter = Math.random() * 1000;\n const delay = backoff + jitter;\n\n logLine?.(\"🔄 RECONNECTING\", {\n attempt: retryCount + 1,\n delayMs: Math.round(delay),\n });\n\n retryCount++;\n timeoutId = setTimeout(connect, delay);\n } else {\n params.onClose?.({\n code: ev.code,\n reason: ev.reason,\n wasClean: ev.wasClean,\n });\n }\n };\n\n ws.onerror = (ev) => {\n console.error(\"WS Error\", ev);\n params.onError?.();\n };\n }\n\n // Helper for peer tracking (logic from your original code)\n function updatePeers(updatedUsers: { userId: string }[]) {\n const joined: IPeer<T, P>[] = [];\n const left: { userId: string }[] = [];\n const updatedPeerSet = new Set<string>();\n\n updatedUsers.forEach(({ userId: pUserId }) => {\n if (pUserId === userId) return;\n if (!peers.has(pUserId)) {\n const newPeer = {\n userId: pUserId,\n receive: (t: T, p: P) => send(t, pUserId, p),\n };\n peers.set(pUserId, newPeer);\n joined.push(newPeer);\n }\n updatedPeerSet.add(pUserId);\n });\n\n for (const pUserId of peers.keys()) {\n if (!updatedPeerSet.has(pUserId)) {\n peers.delete(pUserId);\n left.push({ userId: pUserId });\n }\n }\n // Notify peer joined first then peer left. (avoid disconnect in case the peer leaving / joining is on the same user).\n if (joined.length) params.onPeerJoined(joined);\n if (left.length) params.onPeerLeft(left);\n }\n\n // Start initial connection\n connect();\n\n return {\n sendToServer(type, payload) {\n send(type, \"server\", payload);\n },\n exitRoom: () => {\n exited = true;\n clearTimeout(timeoutId);\n ws.close();\n },\n };\n}\n",
6
6
  "import type { IPeer } from \"./impl/signal-room.js\";\nimport { enterRoom as baseEnterRoom } from \"./impl/signal-room.js\";\nimport { RoomEvent, WorkerCommand } from \"./signal-room.worker.js\";\n\nexport function enterRoom<T extends string, P = any>({\n userId,\n worldId,\n room,\n host,\n autoRejoin = true,\n onOpen,\n onClose,\n onError,\n onPeerJoined,\n onPeerLeft,\n onIceUrl,\n onMessage,\n logLine,\n workerUrl,\n}: {\n userId: string;\n worldId: string;\n room: string;\n host: string;\n autoRejoin?: boolean;\n onOpen?: () => void;\n onClose?: (ev: Pick<CloseEvent, \"code\" | \"reason\" | \"wasClean\">) => void;\n onError?: () => void;\n onPeerJoined: (users: IPeer<T, P>[]) => void;\n onPeerLeft: (users: { userId: string }[]) => void;\n onIceUrl?(url: string, expiration: number): void;\n onMessage: (type: T, payload: P, from: IPeer<T, P>) => void;\n logLine?: (direction: string, obj?: any) => void;\n\n // Pass the URL to your worker file (bundler will handle it)\n workerUrl?: URL;\n}): {\n exitRoom: () => void;\n sendToServer: <P extends any>(type: T, payload?: P) => void;\n} {\n if (!workerUrl) {\n const CDN_WORKER_URL = `https://cdn.jsdelivr.net/npm/@dobuki/hello-worker/dist/signal-room.worker.min.js`;\n\n console.warn(\n \"Warning: enterRoom called without workerUrl; this may cause issues in some environments. You should pass workerUrl explicitly. Use:\",\n CDN_WORKER_URL,\n );\n return baseEnterRoom<T, P>({\n userId,\n worldId,\n room,\n host,\n autoRejoin,\n onOpen,\n onClose,\n onError,\n onPeerJoined,\n onPeerLeft,\n onIceUrl,\n onMessage,\n });\n }\n const worker = new Worker(workerUrl, { type: \"module\" });\n let exited = false;\n\n function makeUser({ userId }: { userId: string }): IPeer<T, P> {\n return {\n userId,\n receive: (type: T, payload: P) => {\n if (exited) return false;\n worker.postMessage({\n cmd: \"send\",\n toUserId: userId,\n host,\n room,\n type,\n payload,\n } as WorkerCommand);\n return true;\n },\n };\n }\n\n const onWorkerMessage = (e: MessageEvent<RoomEvent<T, P>>) => {\n const ev = e.data;\n\n if (ev.kind === \"open\") onOpen?.();\n else if (ev.kind === \"close\") {\n worker.terminate();\n onClose?.(ev.ev);\n } else if (ev.kind === \"error\") onError?.();\n else if (ev.kind === \"peer-joined\")\n onPeerJoined(ev.users.map((ev) => makeUser({ userId: ev.userId })));\n else if (ev.kind === \"peer-left\") onPeerLeft(ev.users);\n else if (ev.kind === \"ice-server\") onIceUrl?.(ev.url, ev.expiration);\n else if (ev.kind === \"message\")\n onMessage(ev.type, ev.payload, makeUser({ userId: ev.fromUserId }));\n else if (ev.kind === \"log\") logLine?.(ev.direction, ev.obj);\n };\n\n worker.addEventListener(\"message\", onWorkerMessage);\n\n worker.postMessage({\n cmd: \"enter\",\n userId,\n worldId,\n room,\n host,\n autoRejoin,\n } as WorkerCommand);\n\n return {\n exitRoom: () => {\n exited = true;\n worker.removeEventListener(\"message\", onWorkerMessage);\n worker.postMessage({ cmd: \"exit\" } as WorkerCommand);\n },\n sendToServer: <P extends any>(type: T, payload?: P) => {\n worker.postMessage({\n cmd: \"send\",\n toUserId: \"server\",\n host,\n room,\n type,\n payload,\n } as WorkerCommand);\n },\n };\n}\n\nexport type EnterRoom<T extends string, P> = typeof enterRoom<T, P>;\n",
7
- "import { IPeer } from \"./impl/signal-room\";\nimport { EnterRoom, enterRoom } from \"./signal-room\";\n\nexport type SigType = \"offer\" | \"answer\" | \"ice\" | \"request-ice\" | \"broadcast\";\nexport type SigPayload = RTCSessionDescriptionInit | RTCIceCandidateInit;\n\ntype UserState = {\n userId: string;\n pc?: RTCPeerConnection;\n\n // ICE that arrived before we had remoteDescription\n pendingRemoteIce: RTCIceCandidateInit[];\n\n // the signaling \"user\" handle so we can send messages\n peer: IPeer<SigType, SigPayload>;\n\n expirationTimeout?: number;\n};\n\nconst DEFAULT_ENTER_ROOM = enterRoom;\n\nexport function collectPeerConnections({\n worldId,\n receivePeerConnection,\n peerlessUserExpiration = 5000,\n fallbackRtcConfig = {\n iceServers: [{ urls: \"stun:stun.l.google.com:19302\" }],\n },\n enterRoomFunction: enterRoom = DEFAULT_ENTER_ROOM,\n logLine = console.debug,\n onLeaveUser,\n workerUrl,\n onRoomReady,\n onRoomClose,\n onBroadcastMessage,\n}: {\n worldId: string;\n fallbackRtcConfig?: RTCConfiguration;\n enterRoomFunction?: EnterRoom<SigType, SigPayload>;\n onLeaveUser?: (userId: string) => void;\n logLine?: (direction: string, obj?: any) => void;\n workerUrl?: URL;\n peerlessUserExpiration?: number;\n receivePeerConnection(connection: {\n pc: RTCPeerConnection;\n userId: string;\n initiator: boolean;\n restart?: () => void;\n }): void;\n onRoomReady?(info: { host: string; room: string }): void;\n onRoomClose?(info: {\n host: string;\n room: string;\n ev: Pick<CloseEvent, \"reason\" | \"code\" | \"wasClean\">;\n }): void;\n onBroadcastMessage?<P extends any>(payload: P, from: string): void;\n}) {\n const userId = `user-${crypto.randomUUID()}`;\n const users: Map<string, UserState> = new Map();\n let iceUrl: { url: string; expiration: number } | undefined = undefined;\n let rtcConfig: RTCConfiguration & { timestamp: number } = {\n ...fallbackRtcConfig,\n timestamp: Date.now(),\n };\n\n const roomsEntered = new Map<\n string,\n {\n room: string;\n host: string;\n exitRoom: () => void;\n broadcast: <P extends any>(payload: P) => void;\n }\n >();\n\n async function getRtcConfig(iceUrl: string): Promise<RTCConfiguration> {\n if (iceUrl) {\n try {\n const r = await fetch(iceUrl);\n if (!r.ok) throw new Error(`ICE endpoint failed: ${r.status}`);\n rtcConfig = (await r.json()) as RTCConfiguration & {\n timestamp: number;\n };\n } catch (e) {\n console.warn(\"Using fallback rtcConfig:\", e);\n }\n }\n return rtcConfig;\n }\n\n function leaveUser(userId: string) {\n onLeaveUser?.(userId);\n const p = users.get(userId);\n if (!p) return;\n try {\n p.pc?.close();\n } catch {}\n users.delete(userId);\n }\n\n async function flushRemoteIce(state: UserState) {\n if (!state.pc?.remoteDescription) return;\n\n const queued = state.pendingRemoteIce;\n state.pendingRemoteIce = [];\n\n for (const ice of queued) {\n try {\n await state.pc.addIceCandidate(ice);\n } catch (e) {\n logLine(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n }\n }\n\n function exit({ room, host }: { room: string; host: string }) {\n const key = `${host}/room/${room}`;\n const session = roomsEntered.get(key);\n if (session) {\n session.exitRoom();\n roomsEntered.delete(key);\n }\n }\n\n function enter({ room, host }: { room: string; host: string }) {\n return new Promise<void>(async (resolve, reject) => {\n async function setupPC(state: UserState) {\n const ice =\n !iceUrl || iceUrl.expiration - Date.now() < 2000\n ? await requestIce()\n : iceUrl;\n state.pc = new RTCPeerConnection(\n Date.now() - (rtcConfig?.timestamp ?? 0) < 10000\n ? rtcConfig\n : await getRtcConfig(ice.url),\n );\n // Send local ICE candidates to this peer\n state.pc.onicecandidate = (ev) => {\n if (!ev.candidate) return;\n state.peer.receive(\"ice\", ev.candidate.toJSON());\n };\n\n state.pc.onconnectionstatechange = () => {\n logLine(\"💬\", {\n event: \"pc-state\",\n userId: state.userId,\n state: state.pc?.connectionState,\n });\n };\n return state.pc;\n }\n\n async function getPeer(\n peer: IPeer<SigType, SigPayload>,\n ): Promise<[UserState, boolean]> {\n let state = users.get(peer.userId);\n let isNewPeer = false;\n if (!state) {\n const newState: UserState = {\n userId: peer.userId,\n pendingRemoteIce: [],\n peer,\n };\n\n await setupPC(newState);\n state = newState;\n\n // New user\n users.set(state.userId, state);\n isNewPeer = true;\n } else if (state) {\n clearTimeout(state.expirationTimeout);\n state.expirationTimeout = 0;\n }\n if (!state.pc || state.pc?.signalingState === \"closed\") {\n await setupPC(state);\n }\n state.peer = peer;\n return [state, isNewPeer];\n }\n\n async function makeOffer(user: IPeer) {\n // Offer flow: createOffer -> setLocalDescription -> send localDescription\n const [state] = await getPeer(user);\n const pc = state.pc;\n const offer = await pc?.createOffer();\n await pc?.setLocalDescription(offer);\n user.receive(\"offer\", pc?.localDescription?.toJSON()!);\n }\n\n let icePromiseResolve:\n | undefined\n | ((url: { url: string; expiration: number }) => void);\n async function requestIce() {\n const iceUrl = await new Promise<{ url: string; expiration: number }>(\n (resolve) => {\n icePromiseResolve = resolve;\n sendToServer(\"request-ice\");\n },\n );\n icePromiseResolve = undefined;\n return iceUrl;\n }\n\n const { exitRoom, sendToServer } = enterRoom({\n userId,\n worldId,\n room,\n host,\n logLine,\n workerUrl,\n autoRejoin: true,\n\n onOpen() {\n onRoomReady?.({ room, host });\n resolve();\n },\n onError() {\n console.error(\"onError\");\n reject();\n },\n onClose(ev) {\n onRoomClose?.({ room, host, ev });\n },\n\n // Existing peers initiate to the newcomer\n onPeerJoined(joiningUsers: IPeer<SigType, SigPayload>[]) {\n joiningUsers.forEach(async (user) => {\n const [state, isNewPeer] = await getPeer(user);\n if (!isNewPeer) {\n logLine(\"👤ℹ️\", \"not a new peer: \" + user.userId);\n return;\n }\n const pc = state.pc;\n if (!pc) {\n logLine(\"👤ℹ️\", \"no pc: \" + user.userId);\n return;\n }\n\n async function restart() {\n const state = users.get(user.userId);\n if (state) {\n state.pc = undefined;\n const pc = await setupPC(state);\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await new Promise((resolve) => setTimeout(resolve, 3000));\n await makeOffer(user);\n }\n }\n\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await makeOffer(user);\n });\n },\n\n onPeerLeft(leavingUsers: { userId: string }[]) {\n leavingUsers.forEach(({ userId }) => {\n const state = users.get(userId);\n if (!state) return;\n state.expirationTimeout = setTimeout(\n () => leaveUser(userId),\n peerlessUserExpiration ?? 0,\n );\n });\n },\n\n onIceUrl(url, expiration) {\n iceUrl = { url, expiration };\n icePromiseResolve?.(iceUrl);\n },\n\n async onMessage(type: SigType, payload: any, from: IPeer) {\n const [state] = await getPeer(from);\n const pc = state.pc;\n if (!pc) return;\n\n if (type === \"offer\") {\n receivePeerConnection({\n pc,\n userId: from.userId,\n initiator: false,\n restart() {\n // reset PC\n state.pc = undefined;\n },\n });\n // Responder: set remote offer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n\n // Create and send answer\n const answer = await pc.createAnswer();\n await pc.setLocalDescription(answer);\n\n from.receive(\"answer\", pc.localDescription?.toJSON()!);\n\n // Now safe to apply any queued ICE from this peer\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"answer\") {\n // Initiator: set remote answer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"ice\") {\n const ice = payload as RTCIceCandidateInit;\n\n // If we don't have remoteDescription yet, queue it\n if (!pc.remoteDescription) {\n state.pendingRemoteIce.push(ice);\n return;\n }\n\n try {\n await pc.addIceCandidate(ice);\n } catch (e) {\n logLine(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n return;\n }\n\n if (type === \"broadcast\") {\n onBroadcastMessage?.(payload, from.userId);\n }\n },\n });\n roomsEntered.set(`${host}/room/${room}`, {\n exitRoom,\n room,\n host,\n broadcast: (payload) => {\n sendToServer(\"broadcast\", payload);\n },\n });\n });\n }\n\n return {\n userId,\n enterRoom: enter,\n exitRoom: exit,\n leaveUser,\n broadcast<P extends any>(payload: P) {\n roomsEntered.forEach((room) => room.broadcast(payload));\n },\n end() {\n roomsEntered.forEach(({ exitRoom }) => exitRoom());\n roomsEntered.clear();\n users.forEach(({ userId }) => leaveUser(userId));\n users.clear();\n },\n };\n}\n\n/*\nTurn Token ID\n<CF_TURN_TOKEN_ID>\n\nAPI Token\n<CF_RTC_API_TOKEN>\n\nCURL\ncurl \\\n\t-H \"Authorization: Bearer <CF_RTC_API_TOKEN>\" \\\n\t-H \"Content-Type: application/json\" -d '{\"ttl\": 86400}' \\\n\thttps://rtc.live.cloudflare.com/v1/turn/keys/<CF_TURN_TOKEN_ID>/credentials/generate-ice-servers\n\nJSON\n{\n\t\"iceServers\": [\n {\n \"urls\": [\n \"stun:stun.cloudflare.com:3478\",\n \"turn:turn.cloudflare.com:3478?transport=udp\",\n \"turn:turn.cloudflare.com:3478?transport=tcp\",\n \"turns:turn.cloudflare.com:5349?transport=tcp\"\n ],\n \"username\": \"xxxx\",\n \"credential\": \"yyyy\",\n }\n ]\n}\n\n*/\n",
8
- "import { EnterRoom, enterRoom } from \"./signal-room\";\nimport {\n SigType,\n SigPayload,\n collectPeerConnections,\n} from \"./webrtc-peer-collector\";\n\ntype UserListener = (\n user: string,\n action: \"join\" | \"leave\",\n users: string[],\n) => void;\n\nexport function enterWorld<\n S extends string | ArrayBufferView = string | ArrayBufferView,\n R extends string | ArrayBufferLike = string | ArrayBufferLike,\n>({\n worldId,\n logLine = console.debug,\n enterRoomFunction = enterRoom,\n peerlessUserExpiration,\n workerUrl,\n onRoomReady,\n onRoomClose,\n dataChannelOptions,\n}: {\n worldId: string;\n logLine?: (direction: string, obj?: any) => void;\n enterRoomFunction?: EnterRoom<SigType, SigPayload>;\n peerlessUserExpiration?: number;\n workerUrl?: URL;\n onRoomReady?(info: { host: string; room: string }): void;\n onRoomClose?(info: {\n host: string;\n room: string;\n ev: Pick<CloseEvent, \"reason\" | \"code\" | \"wasClean\">;\n }): void;\n dataChannelOptions?: RTCDataChannelInit;\n}) {\n const userIds: string[] = [];\n\n const messagesListeners = new Set<(data: R, from: string) => void>();\n\n function createDataChannel(\n pc: RTCPeerConnection,\n peerUserId: string,\n initiator: boolean,\n restart?: () => void,\n ) {\n if (initiator) {\n const dc = pc.createDataChannel(\"data\", dataChannelOptions);\n wireDataChannel(peerUserId, dc, restart);\n dataChannels.set(peerUserId, dc);\n } else {\n function listener(ev: RTCDataChannelEvent) {\n const dc = ev.channel;\n wireDataChannel(peerUserId, dc, restart);\n dataChannels.set(peerUserId, dc);\n }\n pc.addEventListener(\"datachannel\", listener);\n return () => {\n pc.removeEventListener(\"datachannel\", listener);\n };\n }\n }\n\n function conveyMessage(data: any, userId: string) {\n messagesListeners.forEach((listener) => listener(data, userId));\n }\n\n function wireDataChannel(\n userId: string,\n dc: RTCDataChannel,\n restart?: () => void,\n ) {\n dc.onopen = () => {\n logLine(\"💬\", { event: \"dc-open\", userId });\n userIds.push(userId);\n userListeners.forEach((listener) => listener(userId, \"join\", userIds));\n };\n const onmessage = ({ data }: MessageEvent) => {\n conveyMessage(data, userId);\n // logLine(\"💬\", { event: \"dc-message\", userId, data });\n };\n dc.addEventListener(\"message\", onmessage);\n dc.addEventListener(\"close\", () => {\n logLine(\"💬\", { event: \"dc-close\", userId });\n userIds.splice(userIds.indexOf(userId), 1);\n userListeners.forEach((listener) => listener(userId, \"leave\", userIds));\n dc.removeEventListener(\"message\", onmessage);\n restart?.();\n });\n dc.onerror = () => logLine(\"⚠️ ERROR\", { error: \"dc-error\", userId });\n }\n\n const dataChannels = new Map<string, RTCDataChannel>();\n const userListeners = new Set<UserListener>();\n\n const {\n userId,\n enterRoom,\n exitRoom,\n leaveUser,\n broadcast,\n end: endPeerCollection,\n } = collectPeerConnections({\n worldId,\n enterRoomFunction,\n logLine,\n workerUrl,\n peerlessUserExpiration,\n onRoomReady,\n onRoomClose,\n onLeaveUser(userId: string) {\n const dc = dataChannels.get(userId);\n try {\n dc?.close();\n } catch {}\n dataChannels.delete(userId);\n },\n receivePeerConnection({ pc, userId, initiator, restart }) {\n createDataChannel(pc, userId, initiator, restart);\n },\n onBroadcastMessage(payload, from) {\n conveyMessage(payload, from);\n logLine(\"📢\", { event: \"broadcast\", userId, data: payload });\n },\n });\n\n function send(data: S, userId?: string) {\n dataChannels.forEach((dataChannel, pUserId) => {\n if (userId && pUserId !== userId) return;\n if (dataChannel.readyState === \"open\") {\n dataChannel.send(data as any);\n }\n });\n }\n\n function removeMessageListener(listener: (data: R, from: string) => void) {\n messagesListeners.delete(listener);\n }\n\n function addMessageListener(listener: (data: R, from: string) => void) {\n messagesListeners.add(listener);\n return () => {\n removeMessageListener(listener);\n };\n }\n\n function removeUserListener(listener: UserListener) {\n userListeners.delete(listener);\n }\n\n function addUserListener(listener: UserListener) {\n userListeners.add(listener);\n return () => {\n removeUserListener(listener);\n };\n }\n\n return {\n userId,\n send,\n broadcast,\n enterRoom,\n exitRoom,\n leaveUser,\n getUsers: () => userIds,\n addMessageListener,\n removeMessageListener,\n addUserListener,\n removeUserListener,\n end() {\n dataChannels.forEach((dataChannel) => {\n try {\n dataChannel.close();\n } catch {}\n });\n dataChannels.clear();\n endPeerCollection();\n userListeners.clear();\n userIds.length = 0;\n },\n };\n}\n"
7
+ "import { IPeer } from \"./impl/signal-room\";\nimport { EnterRoom, enterRoom } from \"./signal-room\";\n\nexport type SigType = \"offer\" | \"answer\" | \"ice\" | \"request-ice\" | \"broadcast\";\nexport type SigPayload = RTCSessionDescriptionInit | RTCIceCandidateInit;\n\ntype UserState = {\n userId: string;\n pc?: RTCPeerConnection;\n\n // ICE that arrived before we had remoteDescription\n pendingRemoteIce: RTCIceCandidateInit[];\n\n // the signaling \"user\" handle so we can send messages\n peer: IPeer<SigType, SigPayload>;\n\n expirationTimeout?: number;\n};\n\nconst DEFAULT_ENTER_ROOM = enterRoom;\n\nexport function collectPeerConnections({\n userId: passedUserId,\n worldId,\n receivePeerConnection,\n peerlessUserExpiration = 5000,\n fallbackRtcConfig = {\n iceServers: [{ urls: \"stun:stun.l.google.com:19302\" }],\n },\n enterRoomFunction: enterRoom = DEFAULT_ENTER_ROOM,\n logLine,\n onLeaveUser,\n workerUrl,\n onRoomReady,\n onRoomClose,\n onBroadcastMessage,\n}: {\n userId?: string;\n worldId: string;\n fallbackRtcConfig?: RTCConfiguration;\n enterRoomFunction?: EnterRoom<SigType, SigPayload>;\n onLeaveUser?: (userId: string) => void;\n logLine?: (direction: string, obj?: any) => void;\n workerUrl?: URL;\n peerlessUserExpiration?: number;\n receivePeerConnection(connection: {\n pc: RTCPeerConnection;\n userId: string;\n initiator: boolean;\n restart?: () => void;\n }): void;\n onRoomReady?(info: { host: string; room: string }): void;\n onRoomClose?(info: {\n host: string;\n room: string;\n ev: Pick<CloseEvent, \"reason\" | \"code\" | \"wasClean\">;\n }): void;\n onBroadcastMessage?<P extends any>(payload: P, from: string): void;\n}) {\n const userId = passedUserId ?? `user-${crypto.randomUUID()}`;\n const users: Map<string, UserState> = new Map();\n let iceUrl: { url: string; expiration: number } | undefined = undefined;\n let rtcConfig: RTCConfiguration & { timestamp: number } = {\n ...fallbackRtcConfig,\n timestamp: Date.now(),\n };\n\n const roomsEntered = new Map<\n string,\n {\n room: string;\n host: string;\n exitRoom: () => void;\n broadcast: <P extends any>(payload: P) => void;\n }\n >();\n\n async function getRtcConfig(\n iceUrl: string,\n ): Promise<RTCConfiguration & { timestamp: number }> {\n if (iceUrl) {\n try {\n const r = await fetch(iceUrl);\n if (!r.ok) throw new Error(`ICE endpoint failed: ${r.status}`);\n rtcConfig = (await r.json()) as RTCConfiguration & {\n timestamp: number;\n };\n } catch (e) {\n console.warn(\"Using fallback rtcConfig:\", e);\n }\n }\n return rtcConfig;\n }\n\n function leaveUser(userId: string) {\n onLeaveUser?.(userId);\n const p = users.get(userId);\n if (!p) return;\n try {\n p.pc?.close();\n } catch {}\n users.delete(userId);\n }\n\n async function flushRemoteIce(state: UserState) {\n if (!state.pc?.remoteDescription) return;\n\n const queued = state.pendingRemoteIce;\n state.pendingRemoteIce = [];\n\n for (const ice of queued) {\n try {\n await state.pc.addIceCandidate(ice);\n } catch (e) {\n logLine?.(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n }\n }\n\n function exit({ room, host }: { room: string; host: string }) {\n const key = `${host}/room/${room}`;\n const session = roomsEntered.get(key);\n if (session) {\n session.exitRoom();\n roomsEntered.delete(key);\n }\n }\n\n function enter({ room, host }: { room: string; host: string }) {\n return new Promise<void>(async (resolve, reject) => {\n async function setupPC(state: UserState) {\n const now = Date.now();\n if (now - (rtcConfig?.timestamp ?? 0) > 10000) {\n console.log(\"ICE expiration\", iceUrl?.expiration, \"-\", now);\n const ice =\n !iceUrl || iceUrl.expiration - now < 2000\n ? await requestIce()\n : iceUrl;\n rtcConfig = await getRtcConfig(ice.url);\n }\n state.pc = new RTCPeerConnection(rtcConfig);\n // Send local ICE candidates to this peer\n state.pc.onicecandidate = (ev) => {\n if (!ev.candidate) return;\n state.peer.receive(\"ice\", ev.candidate.toJSON());\n };\n\n state.pc.onconnectionstatechange = () => {\n logLine?.(\"💬\", {\n event: \"pc-state\",\n userId: state.userId,\n state: state.pc?.connectionState,\n });\n };\n return state.pc;\n }\n\n async function getPeer(\n peer: IPeer<SigType, SigPayload>,\n ): Promise<[UserState, boolean]> {\n let state = users.get(peer.userId);\n let isNewPeer = false;\n if (!state) {\n const newState: UserState = {\n userId: peer.userId,\n pendingRemoteIce: [],\n peer,\n };\n\n await setupPC(newState);\n state = newState;\n\n // New user\n users.set(state.userId, state);\n isNewPeer = true;\n } else if (state) {\n clearTimeout(state.expirationTimeout);\n state.expirationTimeout = 0;\n }\n if (!state.pc || state.pc?.signalingState === \"closed\") {\n await setupPC(state);\n }\n state.peer = peer;\n return [state, isNewPeer];\n }\n\n async function makeOffer(user: IPeer) {\n // Offer flow: createOffer -> setLocalDescription -> send localDescription\n const [state] = await getPeer(user);\n const pc = state.pc;\n const offer = await pc?.createOffer();\n await pc?.setLocalDescription(offer);\n user.receive(\"offer\", pc?.localDescription?.toJSON()!);\n }\n\n let icePromiseResolve:\n | undefined\n | ((url: { url: string; expiration: number }) => void);\n async function requestIce() {\n const iceUrl = await new Promise<{ url: string; expiration: number }>(\n (resolve) => {\n icePromiseResolve = resolve;\n sendToServer(\"request-ice\");\n },\n );\n icePromiseResolve = undefined;\n return iceUrl;\n }\n\n const { exitRoom, sendToServer } = enterRoom({\n userId,\n worldId,\n room,\n host,\n logLine,\n workerUrl,\n autoRejoin: true,\n\n onOpen() {\n onRoomReady?.({ room, host });\n resolve();\n },\n onError() {\n console.error(\"onError\");\n reject();\n },\n onClose(ev) {\n onRoomClose?.({ room, host, ev });\n },\n\n // Existing peers initiate to the newcomer\n onPeerJoined(joiningUsers: IPeer<SigType, SigPayload>[]) {\n joiningUsers.forEach(async (user) => {\n const [state, isNewPeer] = await getPeer(user);\n if (!isNewPeer) {\n logLine?.(\"👤ℹ️\", \"not a new peer: \" + user.userId);\n return;\n }\n const pc = state.pc;\n if (!pc) {\n logLine?.(\"👤ℹ️\", \"no pc: \" + user.userId);\n return;\n }\n\n async function restart() {\n const state = users.get(user.userId);\n if (state) {\n state.pc = undefined;\n const pc = await setupPC(state);\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await new Promise((resolve) => setTimeout(resolve, 3000));\n await makeOffer(user);\n }\n }\n\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await makeOffer(user);\n });\n },\n\n onPeerLeft(leavingUsers: { userId: string }[]) {\n leavingUsers.forEach(({ userId }) => {\n const state = users.get(userId);\n if (!state) return;\n state.expirationTimeout = setTimeout(\n () => leaveUser(userId),\n peerlessUserExpiration ?? 0,\n );\n });\n },\n\n onIceUrl(url, expiration) {\n iceUrl = { url, expiration };\n icePromiseResolve?.(iceUrl);\n },\n\n async onMessage(type: SigType, payload: any, from: IPeer) {\n const [state] = await getPeer(from);\n const pc = state.pc;\n if (!pc) return;\n\n if (type === \"offer\") {\n receivePeerConnection({\n pc,\n userId: from.userId,\n initiator: false,\n restart() {\n // reset PC\n state.pc = undefined;\n },\n });\n // Responder: set remote offer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n\n // Create and send answer\n const answer = await pc.createAnswer();\n await pc.setLocalDescription(answer);\n\n from.receive(\"answer\", pc.localDescription?.toJSON()!);\n\n // Now safe to apply any queued ICE from this peer\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"answer\") {\n // Initiator: set remote answer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"ice\") {\n const ice = payload as RTCIceCandidateInit;\n\n // If we don't have remoteDescription yet, queue it\n if (!pc.remoteDescription) {\n state.pendingRemoteIce.push(ice);\n return;\n }\n\n try {\n await pc.addIceCandidate(ice);\n } catch (e) {\n logLine?.(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n return;\n }\n\n if (type === \"broadcast\") {\n onBroadcastMessage?.(payload, from.userId);\n }\n },\n });\n roomsEntered.set(`${host}/room/${room}`, {\n exitRoom,\n room,\n host,\n broadcast: (payload) => {\n sendToServer(\"broadcast\", payload);\n },\n });\n });\n }\n\n return {\n userId,\n enterRoom: enter,\n exitRoom: exit,\n leaveUser,\n broadcast<P extends any>(payload: P) {\n roomsEntered.forEach((room) => room.broadcast(payload));\n },\n end() {\n roomsEntered.forEach(({ exitRoom }) => exitRoom());\n roomsEntered.clear();\n users.forEach(({ userId }) => leaveUser(userId));\n users.clear();\n },\n };\n}\n\n/*\nTurn Token ID\n<CF_TURN_TOKEN_ID>\n\nAPI Token\n<CF_RTC_API_TOKEN>\n\nCURL\ncurl \\\n\t-H \"Authorization: Bearer <CF_RTC_API_TOKEN>\" \\\n\t-H \"Content-Type: application/json\" -d '{\"ttl\": 86400}' \\\n\thttps://rtc.live.cloudflare.com/v1/turn/keys/<CF_TURN_TOKEN_ID>/credentials/generate-ice-servers\n\nJSON\n{\n\t\"iceServers\": [\n {\n \"urls\": [\n \"stun:stun.cloudflare.com:3478\",\n \"turn:turn.cloudflare.com:3478?transport=udp\",\n \"turn:turn.cloudflare.com:3478?transport=tcp\",\n \"turns:turn.cloudflare.com:5349?transport=tcp\"\n ],\n \"username\": \"xxxx\",\n \"credential\": \"yyyy\",\n }\n ]\n}\n\n*/\n",
8
+ "import { EnterRoom, enterRoom } from \"./signal-room\";\nimport {\n SigType,\n SigPayload,\n collectPeerConnections,\n} from \"./webrtc-peer-collector\";\n\ntype UserListener = (\n user: string,\n action: \"join\" | \"leave\",\n users: string[],\n) => void;\n\nexport function enterWorld<\n S extends string | ArrayBufferView = string | ArrayBufferView,\n R extends string | ArrayBufferLike = string | ArrayBufferLike,\n>({\n userId: passedUserId,\n worldId,\n logLine,\n enterRoomFunction = enterRoom,\n peerlessUserExpiration,\n workerUrl,\n onRoomReady,\n onRoomClose,\n dataChannelOptions,\n}: {\n userId?: string;\n worldId: string;\n logLine?: (direction: string, obj?: any) => void;\n enterRoomFunction?: EnterRoom<SigType, SigPayload>;\n peerlessUserExpiration?: number;\n workerUrl?: URL;\n onRoomReady?(info: { host: string; room: string }): void;\n onRoomClose?(info: {\n host: string;\n room: string;\n ev: Pick<CloseEvent, \"reason\" | \"code\" | \"wasClean\">;\n }): void;\n dataChannelOptions?: RTCDataChannelInit;\n}) {\n const userIds: string[] = [];\n\n const messagesListeners = new Set<(data: R, from: string) => void>();\n\n function createDataChannel(\n pc: RTCPeerConnection,\n peerUserId: string,\n initiator: boolean,\n restart?: () => void,\n ) {\n if (initiator) {\n const dc = pc.createDataChannel(\"data\", dataChannelOptions);\n wireDataChannel(peerUserId, dc, restart);\n dataChannels.set(peerUserId, dc);\n } else {\n function listener(ev: RTCDataChannelEvent) {\n const dc = ev.channel;\n wireDataChannel(peerUserId, dc, restart);\n dataChannels.set(peerUserId, dc);\n }\n pc.addEventListener(\"datachannel\", listener);\n return () => {\n pc.removeEventListener(\"datachannel\", listener);\n };\n }\n }\n\n function conveyMessage(data: any, userId: string) {\n messagesListeners.forEach((listener) => listener(data, userId));\n }\n\n function wireDataChannel(\n userId: string,\n dc: RTCDataChannel,\n restart?: () => void,\n ) {\n dc.onopen = () => {\n logLine?.(\"💬\", { event: \"dc-open\", userId });\n userIds.push(userId);\n userListeners.forEach((listener) => listener(userId, \"join\", userIds));\n };\n const onmessage = ({ data }: MessageEvent) => {\n conveyMessage(data, userId);\n // logLine(\"💬\", { event: \"dc-message\", userId, data });\n };\n dc.addEventListener(\"message\", onmessage);\n dc.addEventListener(\"close\", () => {\n logLine?.(\"💬\", { event: \"dc-close\", userId });\n userIds.splice(userIds.indexOf(userId), 1);\n userListeners.forEach((listener) => listener(userId, \"leave\", userIds));\n dc.removeEventListener(\"message\", onmessage);\n restart?.();\n });\n dc.onerror = () => logLine?.(\"⚠️ ERROR\", { error: \"dc-error\", userId });\n }\n\n const dataChannels = new Map<string, RTCDataChannel>();\n const userListeners = new Set<UserListener>();\n\n const {\n userId,\n enterRoom,\n exitRoom,\n leaveUser,\n broadcast,\n end: endPeerCollection,\n } = collectPeerConnections({\n userId: passedUserId,\n worldId,\n enterRoomFunction,\n logLine,\n workerUrl,\n peerlessUserExpiration,\n onRoomReady,\n onRoomClose,\n onLeaveUser(userId: string) {\n const dc = dataChannels.get(userId);\n try {\n dc?.close();\n } catch {}\n dataChannels.delete(userId);\n },\n receivePeerConnection({ pc, userId, initiator, restart }) {\n createDataChannel(pc, userId, initiator, restart);\n },\n onBroadcastMessage(payload, from) {\n conveyMessage(payload, from);\n logLine?.(\"📢\", { event: \"broadcast\", userId, data: payload });\n },\n });\n\n function send(data: S, userId?: string) {\n dataChannels.forEach((dataChannel, pUserId) => {\n if (userId && pUserId !== userId) return;\n if (dataChannel.readyState === \"open\") {\n dataChannel.send(data as any);\n }\n });\n }\n\n function removeMessageListener(listener: (data: R, from: string) => void) {\n messagesListeners.delete(listener);\n }\n\n function addMessageListener(listener: (data: R, from: string) => void) {\n messagesListeners.add(listener);\n return () => {\n removeMessageListener(listener);\n };\n }\n\n function removeUserListener(listener: UserListener) {\n userListeners.delete(listener);\n }\n\n function addUserListener(listener: UserListener) {\n userListeners.add(listener);\n return () => {\n removeUserListener(listener);\n };\n }\n\n return {\n userId,\n send,\n broadcast,\n enterRoom,\n exitRoom,\n leaveUser,\n getUsers: () => userIds,\n addMessageListener,\n removeMessageListener,\n addUserListener,\n removeUserListener,\n end() {\n dataChannels.forEach((dataChannel) => {\n try {\n dataChannel.close();\n } catch {}\n });\n dataChannels.clear();\n endPeerCollection();\n userListeners.clear();\n userIds.length = 0;\n },\n };\n}\n"
9
9
  ],
10
- "mappings": "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",
11
- "debugId": "E580648F6184549964756E2164756E21",
10
+ "mappings": "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",
11
+ "debugId": "C7547F1FAAABA26F64756E2164756E21",
12
12
  "names": []
13
13
  }
@@ -1,7 +1,8 @@
1
1
  import { EnterRoom } from "./signal-room";
2
2
  export type SigType = "offer" | "answer" | "ice" | "request-ice" | "broadcast";
3
3
  export type SigPayload = RTCSessionDescriptionInit | RTCIceCandidateInit;
4
- export declare function collectPeerConnections({ worldId, receivePeerConnection, peerlessUserExpiration, fallbackRtcConfig, enterRoomFunction: enterRoom, logLine, onLeaveUser, workerUrl, onRoomReady, onRoomClose, onBroadcastMessage, }: {
4
+ export declare function collectPeerConnections({ userId: passedUserId, worldId, receivePeerConnection, peerlessUserExpiration, fallbackRtcConfig, enterRoomFunction: enterRoom, logLine, onLeaveUser, workerUrl, onRoomReady, onRoomClose, onBroadcastMessage, }: {
5
+ userId?: string;
5
6
  worldId: string;
6
7
  fallbackRtcConfig?: RTCConfiguration;
7
8
  enterRoomFunction?: EnterRoom<SigType, SigPayload>;
@@ -1 +1 @@
1
- {"version":3,"file":"webrtc-peer-collector.d.ts","sourceRoot":"","sources":["../src/browser/webrtc-peer-collector.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,SAAS,EAAa,MAAM,eAAe,CAAC;AAErD,MAAM,MAAM,OAAO,GAAG,OAAO,GAAG,QAAQ,GAAG,KAAK,GAAG,aAAa,GAAG,WAAW,CAAC;AAC/E,MAAM,MAAM,UAAU,GAAG,yBAAyB,GAAG,mBAAmB,CAAC;AAiBzE,wBAAgB,sBAAsB,CAAC,EACrC,OAAO,EACP,qBAAqB,EACrB,sBAA6B,EAC7B,iBAEC,EACD,iBAAiB,EAAE,SAA8B,EACjD,OAAuB,EACvB,WAAW,EACX,SAAS,EACT,WAAW,EACX,WAAW,EACX,kBAAkB,GACnB,EAAE;IACD,OAAO,EAAE,MAAM,CAAC;IAChB,iBAAiB,CAAC,EAAE,gBAAgB,CAAC;IACrC,iBAAiB,CAAC,EAAE,SAAS,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;IACnD,WAAW,CAAC,EAAE,CAAC,MAAM,EAAE,MAAM,KAAK,IAAI,CAAC;IACvC,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,GAAG,KAAK,IAAI,CAAC;IACjD,SAAS,CAAC,EAAE,GAAG,CAAC;IAChB,sBAAsB,CAAC,EAAE,MAAM,CAAC;IAChC,qBAAqB,CAAC,UAAU,EAAE;QAChC,EAAE,EAAE,iBAAiB,CAAC;QACtB,MAAM,EAAE,MAAM,CAAC;QACf,SAAS,EAAE,OAAO,CAAC;QACnB,OAAO,CAAC,EAAE,MAAM,IAAI,CAAC;KACtB,GAAG,IAAI,CAAC;IACT,WAAW,CAAC,CAAC,IAAI,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI,CAAC;IACzD,WAAW,CAAC,CAAC,IAAI,EAAE;QACjB,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE,MAAM,CAAC;QACb,EAAE,EAAE,IAAI,CAAC,UAAU,EAAE,QAAQ,GAAG,MAAM,GAAG,UAAU,CAAC,CAAC;KACtD,GAAG,IAAI,CAAC;IACT,kBAAkB,CAAC,CAAC,CAAC,SAAS,GAAG,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,GAAG,IAAI,CAAC;CACpE;;gCAwEgC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE;+BAT/B;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE;wBA7BjC,MAAM;cAiRrB,CAAC,2BAAuB,CAAC;;EAUtC"}
1
+ {"version":3,"file":"webrtc-peer-collector.d.ts","sourceRoot":"","sources":["../src/browser/webrtc-peer-collector.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,SAAS,EAAa,MAAM,eAAe,CAAC;AAErD,MAAM,MAAM,OAAO,GAAG,OAAO,GAAG,QAAQ,GAAG,KAAK,GAAG,aAAa,GAAG,WAAW,CAAC;AAC/E,MAAM,MAAM,UAAU,GAAG,yBAAyB,GAAG,mBAAmB,CAAC;AAiBzE,wBAAgB,sBAAsB,CAAC,EACrC,MAAM,EAAE,YAAY,EACpB,OAAO,EACP,qBAAqB,EACrB,sBAA6B,EAC7B,iBAEC,EACD,iBAAiB,EAAE,SAA8B,EACjD,OAAO,EACP,WAAW,EACX,SAAS,EACT,WAAW,EACX,WAAW,EACX,kBAAkB,GACnB,EAAE;IACD,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,iBAAiB,CAAC,EAAE,gBAAgB,CAAC;IACrC,iBAAiB,CAAC,EAAE,SAAS,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;IACnD,WAAW,CAAC,EAAE,CAAC,MAAM,EAAE,MAAM,KAAK,IAAI,CAAC;IACvC,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,GAAG,KAAK,IAAI,CAAC;IACjD,SAAS,CAAC,EAAE,GAAG,CAAC;IAChB,sBAAsB,CAAC,EAAE,MAAM,CAAC;IAChC,qBAAqB,CAAC,UAAU,EAAE;QAChC,EAAE,EAAE,iBAAiB,CAAC;QACtB,MAAM,EAAE,MAAM,CAAC;QACf,SAAS,EAAE,OAAO,CAAC;QACnB,OAAO,CAAC,EAAE,MAAM,IAAI,CAAC;KACtB,GAAG,IAAI,CAAC;IACT,WAAW,CAAC,CAAC,IAAI,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI,CAAC;IACzD,WAAW,CAAC,CAAC,IAAI,EAAE;QACjB,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE,MAAM,CAAC;QACb,EAAE,EAAE,IAAI,CAAC,UAAU,EAAE,QAAQ,GAAG,MAAM,GAAG,UAAU,CAAC,CAAC;KACtD,GAAG,IAAI,CAAC;IACT,kBAAkB,CAAC,CAAC,CAAC,SAAS,GAAG,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,GAAG,IAAI,CAAC;CACpE;;gCA0EgC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE;+BAT/B;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE;wBA7BjC,MAAM;cAkRrB,CAAC,2BAAuB,CAAC;;EAUtC"}
@@ -1,4 +1,4 @@
1
- function I(D){let{userId:N,worldId:x,room:j,host:P,autoRejoin:b=!0,logLine:S}=D,O=!1,_=0,H,L,E=!0,W=new Map,M=`wss://${P}/room/${x}/${j}?userId=${encodeURIComponent(N)}`,q=[],A=0;function k(Q,z,X){if(!H)return S?.("\uD83D\uDC64 ➡️ ❌","no ws available"),!1;if(O||H.readyState!==WebSocket.OPEN)return S?.("\uD83D\uDC64 ➡️ ❌","Not in opened state: "+H.readyState),!1;let G={type:Q,to:z,payload:X};return q.push(G),S?.("\uD83D\uDC64 ➡️ \uD83D\uDDA5️",G),clearTimeout(A),A=setTimeout(()=>{H.send(JSON.stringify(q)),q.length=0}),!0}function T(){if(O)return;H=new WebSocket(M),H.onopen=()=>{if(E)D.onOpen?.(),E=!1;_=0},H.onmessage=(Q)=>{try{let z=JSON.parse(Q.data);(Array.isArray(z)?z:[z]).forEach((G)=>{if(S?.("\uD83D\uDDA5️ ➡️ \uD83D\uDC64",G),G.type==="peer-joined"||G.type==="peer-left")F(G.users);else if(G.type==="ice-server")D.onIceUrl?.(G.url,G.expiration);else if(G.userId)D.onMessage(G.type,G.payload,{userId:G.userId,receive:(Z,B)=>k(Z,G.userId,B)})})}catch{S?.("⚠️ ERROR",{error:"invalid-json"})}},H.onclose=(Q)=>{let X=[1001,1006,1011,1012,1013].includes(Q.code);if(b&&!O&&X){let G=Math.min(Math.pow(2,_)*1000,30000),Z=Math.random()*1000,B=G+Z;S?.("\uD83D\uDD04 RECONNECTING",{attempt:_+1,delayMs:Math.round(B)}),_++,L=setTimeout(T,B)}else D.onClose?.({code:Q.code,reason:Q.reason,wasClean:Q.wasClean})},H.onerror=(Q)=>{console.error("WS Error",Q),D.onError?.()}}function F(Q){let z=[],X=[],G=new Set;Q.forEach(({userId:Z})=>{if(Z===N)return;if(!W.has(Z)){let B={userId:Z,receive:(R,C)=>k(R,Z,C)};W.set(Z,B),z.push(B)}G.add(Z)});for(let Z of W.keys())if(!G.has(Z))W.delete(Z),X.push({userId:Z});if(z.length)D.onPeerJoined(z);if(X.length)D.onPeerLeft(X)}return T(),{sendToServer(Q,z){k(Q,"server",z)},exitRoom:()=>{O=!0,clearTimeout(L),H.close()}}}function v({userId:D,worldId:N,room:x,host:j,autoRejoin:P=!0,onOpen:b,onClose:S,onError:O,onPeerJoined:_,onPeerLeft:H,onIceUrl:L,onMessage:E,logLine:W,workerUrl:M}){if(!M)return console.warn("Warning: enterRoom called without workerUrl; this may cause issues in some environments. You should pass workerUrl explicitly. Use:","https://cdn.jsdelivr.net/npm/@dobuki/hello-worker/dist/signal-room.worker.min.js"),I({userId:D,worldId:N,room:x,host:j,autoRejoin:P,onOpen:b,onClose:S,onError:O,onPeerJoined:_,onPeerLeft:H,onIceUrl:L,onMessage:E});let q=new Worker(M,{type:"module"}),A=!1;function k({userId:F}){return{userId:F,receive:(Q,z)=>{if(A)return!1;return q.postMessage({cmd:"send",toUserId:F,host:j,room:x,type:Q,payload:z}),!0}}}let T=(F)=>{let Q=F.data;if(Q.kind==="open")b?.();else if(Q.kind==="close")q.terminate(),S?.(Q.ev);else if(Q.kind==="error")O?.();else if(Q.kind==="peer-joined")_(Q.users.map((z)=>k({userId:z.userId})));else if(Q.kind==="peer-left")H(Q.users);else if(Q.kind==="ice-server")L?.(Q.url,Q.expiration);else if(Q.kind==="message")E(Q.type,Q.payload,k({userId:Q.fromUserId}));else if(Q.kind==="log")W?.(Q.direction,Q.obj)};return q.addEventListener("message",T),q.postMessage({cmd:"enter",userId:D,worldId:N,room:x,host:j,autoRejoin:P}),{exitRoom:()=>{A=!0,q.removeEventListener("message",T),q.postMessage({cmd:"exit"})},sendToServer:(F,Q)=>{q.postMessage({cmd:"send",toUserId:"server",host:j,room:x,type:F,payload:Q})}}}var d=v;function t({worldId:D,receivePeerConnection:N,peerlessUserExpiration:x=5000,fallbackRtcConfig:j={iceServers:[{urls:"stun:stun.l.google.com:19302"}]},enterRoomFunction:P=d,logLine:b=console.debug,onLeaveUser:S,workerUrl:O,onRoomReady:_,onRoomClose:H,onBroadcastMessage:L}){let E=`user-${crypto.randomUUID()}`,W=new Map,M=void 0,q={...j,timestamp:Date.now()},A=new Map;async function k(X){if(X)try{let G=await fetch(X);if(!G.ok)throw Error(`ICE endpoint failed: ${G.status}`);q=await G.json()}catch(G){console.warn("Using fallback rtcConfig:",G)}return q}function T(X){S?.(X);let G=W.get(X);if(!G)return;try{G.pc?.close()}catch{}W.delete(X)}async function F(X){if(!X.pc?.remoteDescription)return;let G=X.pendingRemoteIce;X.pendingRemoteIce=[];for(let Z of G)try{await X.pc.addIceCandidate(Z)}catch(B){b("⚠️ ERROR",{error:"add-ice-failed",userId:X.userId,detail:String(B)})}}function Q({room:X,host:G}){let Z=`${G}/room/${X}`,B=A.get(Z);if(B)B.exitRoom(),A.delete(Z)}function z({room:X,host:G}){return new Promise(async(Z,B)=>{async function R(Y){let V=!M||M.expiration-Date.now()<2000?await g():M;return Y.pc=new RTCPeerConnection(Date.now()-(q?.timestamp??0)<1e4?q:await k(V.url)),Y.pc.onicecandidate=($)=>{if(!$.candidate)return;Y.peer.receive("ice",$.candidate.toJSON())},Y.pc.onconnectionstatechange=()=>{b("\uD83D\uDCAC",{event:"pc-state",userId:Y.userId,state:Y.pc?.connectionState})},Y.pc}async function C(Y){let V=W.get(Y.userId),$=!1;if(!V){let K={userId:Y.userId,pendingRemoteIce:[],peer:Y};await R(K),V=K,W.set(V.userId,V),$=!0}else if(V)clearTimeout(V.expirationTimeout),V.expirationTimeout=0;if(!V.pc||V.pc?.signalingState==="closed")await R(V);return V.peer=Y,[V,$]}async function U(Y){let[V]=await C(Y),$=V.pc,K=await $?.createOffer();await $?.setLocalDescription(K),Y.receive("offer",$?.localDescription?.toJSON())}let f;async function g(){let Y=await new Promise((V)=>{f=V,y("request-ice")});return f=void 0,Y}let{exitRoom:c,sendToServer:y}=P({userId:E,worldId:D,room:X,host:G,logLine:b,workerUrl:O,autoRejoin:!0,onOpen(){_?.({room:X,host:G}),Z()},onError(){console.error("onError"),B()},onClose(Y){H?.({room:X,host:G,ev:Y})},onPeerJoined(Y){Y.forEach(async(V)=>{let[$,K]=await C(V);if(!K){b("\uD83D\uDC64ℹ️","not a new peer: "+V.userId);return}let J=$.pc;if(!J){b("\uD83D\uDC64ℹ️","no pc: "+V.userId);return}async function h(){let w=W.get(V.userId);if(w){w.pc=void 0;let u=await R(w);N({pc:u,userId:V.userId,initiator:!0,restart:h}),await new Promise((m)=>setTimeout(m,3000)),await U(V)}}N({pc:J,userId:V.userId,initiator:!0,restart:h}),await U(V)})},onPeerLeft(Y){Y.forEach(({userId:V})=>{let $=W.get(V);if(!$)return;$.expirationTimeout=setTimeout(()=>T(V),x??0)})},onIceUrl(Y,V){M={url:Y,expiration:V},f?.(M)},async onMessage(Y,V,$){let[K]=await C($),J=K.pc;if(!J)return;if(Y==="offer"){N({pc:J,userId:$.userId,initiator:!1,restart(){K.pc=void 0}}),await J.setRemoteDescription(V);let h=await J.createAnswer();await J.setLocalDescription(h),$.receive("answer",J.localDescription?.toJSON()),await F(K);return}if(Y==="answer"){await J.setRemoteDescription(V),await F(K);return}if(Y==="ice"){let h=V;if(!J.remoteDescription){K.pendingRemoteIce.push(h);return}try{await J.addIceCandidate(h)}catch(w){b("⚠️ ERROR",{error:"add-ice-failed",userId:K.userId,detail:String(w)})}return}if(Y==="broadcast")L?.(V,$.userId)}});A.set(`${G}/room/${X}`,{exitRoom:c,room:X,host:G,broadcast:(Y)=>{y("broadcast",Y)}})})}return{userId:E,enterRoom:z,exitRoom:Q,leaveUser:T,broadcast(X){A.forEach((G)=>G.broadcast(X))},end(){A.forEach(({exitRoom:X})=>X()),A.clear(),W.forEach(({userId:X})=>T(X)),W.clear()}}}export{t as collectPeerConnections};
1
+ function v(B){let{userId:j,worldId:S,room:T,host:P,autoRejoin:R=!0,logLine:q}=B,N=!1,O=0,H,E,k=!0,_=new Map,D=`wss://${P}/room/${S}/${T}?userId=${encodeURIComponent(j)}`,$=[],b=0;function K(Q,z,M){if(!H)return q?.("\uD83D\uDC64 ➡️ ❌","no ws available"),!1;if(N||H.readyState!==WebSocket.OPEN)return q?.("\uD83D\uDC64 ➡️ ❌","Not in opened state: "+H.readyState),!1;let G={type:Q,to:z,payload:M};return $.push(G),q?.("\uD83D\uDC64 ➡️ \uD83D\uDDA5️",G),clearTimeout(b),b=setTimeout(()=>{H.send(JSON.stringify($)),$.length=0}),!0}function L(){if(N)return;H=new WebSocket(D),H.onopen=()=>{if(k)B.onOpen?.(),k=!1;O=0},H.onmessage=(Q)=>{try{let z=JSON.parse(Q.data);(Array.isArray(z)?z:[z]).forEach((G)=>{if(q?.("\uD83D\uDDA5️ ➡️ \uD83D\uDC64",G),G.type==="peer-joined"||G.type==="peer-left")A(G.users);else if(G.type==="ice-server")B.onIceUrl?.(G.url,G.expiration);else if(G.userId)B.onMessage(G.type,G.payload,{userId:G.userId,receive:(X,W)=>K(X,G.userId,W)})})}catch{q?.("⚠️ ERROR",{error:"invalid-json"})}},H.onclose=(Q)=>{let M=[1001,1006,1011,1012,1013].includes(Q.code);if(R&&!N&&M){let G=Math.min(Math.pow(2,O)*1000,30000),X=Math.random()*1000,W=G+X;q?.("\uD83D\uDD04 RECONNECTING",{attempt:O+1,delayMs:Math.round(W)}),O++,E=setTimeout(L,W)}else B.onClose?.({code:Q.code,reason:Q.reason,wasClean:Q.wasClean})},H.onerror=(Q)=>{console.error("WS Error",Q),B.onError?.()}}function A(Q){let z=[],M=[],G=new Set;Q.forEach(({userId:X})=>{if(X===j)return;if(!_.has(X)){let W={userId:X,receive:(x,C)=>K(x,X,C)};_.set(X,W),z.push(W)}G.add(X)});for(let X of _.keys())if(!G.has(X))_.delete(X),M.push({userId:X});if(z.length)B.onPeerJoined(z);if(M.length)B.onPeerLeft(M)}return L(),{sendToServer(Q,z){K(Q,"server",z)},exitRoom:()=>{N=!0,clearTimeout(E),H.close()}}}function g({userId:B,worldId:j,room:S,host:T,autoRejoin:P=!0,onOpen:R,onClose:q,onError:N,onPeerJoined:O,onPeerLeft:H,onIceUrl:E,onMessage:k,logLine:_,workerUrl:D}){if(!D)return console.warn("Warning: enterRoom called without workerUrl; this may cause issues in some environments. You should pass workerUrl explicitly. Use:","https://cdn.jsdelivr.net/npm/@dobuki/hello-worker/dist/signal-room.worker.min.js"),v({userId:B,worldId:j,room:S,host:T,autoRejoin:P,onOpen:R,onClose:q,onError:N,onPeerJoined:O,onPeerLeft:H,onIceUrl:E,onMessage:k});let $=new Worker(D,{type:"module"}),b=!1;function K({userId:A}){return{userId:A,receive:(Q,z)=>{if(b)return!1;return $.postMessage({cmd:"send",toUserId:A,host:T,room:S,type:Q,payload:z}),!0}}}let L=(A)=>{let Q=A.data;if(Q.kind==="open")R?.();else if(Q.kind==="close")$.terminate(),q?.(Q.ev);else if(Q.kind==="error")N?.();else if(Q.kind==="peer-joined")O(Q.users.map((z)=>K({userId:z.userId})));else if(Q.kind==="peer-left")H(Q.users);else if(Q.kind==="ice-server")E?.(Q.url,Q.expiration);else if(Q.kind==="message")k(Q.type,Q.payload,K({userId:Q.fromUserId}));else if(Q.kind==="log")_?.(Q.direction,Q.obj)};return $.addEventListener("message",L),$.postMessage({cmd:"enter",userId:B,worldId:j,room:S,host:T,autoRejoin:P}),{exitRoom:()=>{b=!0,$.removeEventListener("message",L),$.postMessage({cmd:"exit"})},sendToServer:(A,Q)=>{$.postMessage({cmd:"send",toUserId:"server",host:T,room:S,type:A,payload:Q})}}}var d=g;function s({userId:B,worldId:j,receivePeerConnection:S,peerlessUserExpiration:T=5000,fallbackRtcConfig:P={iceServers:[{urls:"stun:stun.l.google.com:19302"}]},enterRoomFunction:R=d,logLine:q,onLeaveUser:N,workerUrl:O,onRoomReady:H,onRoomClose:E,onBroadcastMessage:k}){let _=B??`user-${crypto.randomUUID()}`,D=new Map,$=void 0,b={...P,timestamp:Date.now()},K=new Map;async function L(G){if(G)try{let X=await fetch(G);if(!X.ok)throw Error(`ICE endpoint failed: ${X.status}`);b=await X.json()}catch(X){console.warn("Using fallback rtcConfig:",X)}return b}function A(G){N?.(G);let X=D.get(G);if(!X)return;try{X.pc?.close()}catch{}D.delete(G)}async function Q(G){if(!G.pc?.remoteDescription)return;let X=G.pendingRemoteIce;G.pendingRemoteIce=[];for(let W of X)try{await G.pc.addIceCandidate(W)}catch(x){q?.("⚠️ ERROR",{error:"add-ice-failed",userId:G.userId,detail:String(x)})}}function z({room:G,host:X}){let W=`${X}/room/${G}`,x=K.get(W);if(x)x.exitRoom(),K.delete(W)}function M({room:G,host:X}){return new Promise(async(W,x)=>{async function C(Y){let V=Date.now();if(V-(b?.timestamp??0)>1e4){console.log("ICE expiration",$?.expiration,"-",V);let Z=!$||$.expiration-V<2000?await c():$;b=await L(Z.url)}return Y.pc=new RTCPeerConnection(b),Y.pc.onicecandidate=(Z)=>{if(!Z.candidate)return;Y.peer.receive("ice",Z.candidate.toJSON())},Y.pc.onconnectionstatechange=()=>{q?.("\uD83D\uDCAC",{event:"pc-state",userId:Y.userId,state:Y.pc?.connectionState})},Y.pc}async function w(Y){let V=D.get(Y.userId),Z=!1;if(!V){let F={userId:Y.userId,pendingRemoteIce:[],peer:Y};await C(F),V=F,D.set(V.userId,V),Z=!0}else if(V)clearTimeout(V.expirationTimeout),V.expirationTimeout=0;if(!V.pc||V.pc?.signalingState==="closed")await C(V);return V.peer=Y,[V,Z]}async function U(Y){let[V]=await w(Y),Z=V.pc,F=await Z?.createOffer();await Z?.setLocalDescription(F),Y.receive("offer",Z?.localDescription?.toJSON())}let y;async function c(){let Y=await new Promise((V)=>{y=V,I("request-ice")});return y=void 0,Y}let{exitRoom:u,sendToServer:I}=R({userId:_,worldId:j,room:G,host:X,logLine:q,workerUrl:O,autoRejoin:!0,onOpen(){H?.({room:G,host:X}),W()},onError(){console.error("onError"),x()},onClose(Y){E?.({room:G,host:X,ev:Y})},onPeerJoined(Y){Y.forEach(async(V)=>{let[Z,F]=await w(V);if(!F){q?.("\uD83D\uDC64ℹ️","not a new peer: "+V.userId);return}let J=Z.pc;if(!J){q?.("\uD83D\uDC64ℹ️","no pc: "+V.userId);return}async function h(){let f=D.get(V.userId);if(f){f.pc=void 0;let m=await C(f);S({pc:m,userId:V.userId,initiator:!0,restart:h}),await new Promise((p)=>setTimeout(p,3000)),await U(V)}}S({pc:J,userId:V.userId,initiator:!0,restart:h}),await U(V)})},onPeerLeft(Y){Y.forEach(({userId:V})=>{let Z=D.get(V);if(!Z)return;Z.expirationTimeout=setTimeout(()=>A(V),T??0)})},onIceUrl(Y,V){$={url:Y,expiration:V},y?.($)},async onMessage(Y,V,Z){let[F]=await w(Z),J=F.pc;if(!J)return;if(Y==="offer"){S({pc:J,userId:Z.userId,initiator:!1,restart(){F.pc=void 0}}),await J.setRemoteDescription(V);let h=await J.createAnswer();await J.setLocalDescription(h),Z.receive("answer",J.localDescription?.toJSON()),await Q(F);return}if(Y==="answer"){await J.setRemoteDescription(V),await Q(F);return}if(Y==="ice"){let h=V;if(!J.remoteDescription){F.pendingRemoteIce.push(h);return}try{await J.addIceCandidate(h)}catch(f){q?.("⚠️ ERROR",{error:"add-ice-failed",userId:F.userId,detail:String(f)})}return}if(Y==="broadcast")k?.(V,Z.userId)}});K.set(`${X}/room/${G}`,{exitRoom:u,room:G,host:X,broadcast:(Y)=>{I("broadcast",Y)}})})}return{userId:_,enterRoom:M,exitRoom:z,leaveUser:A,broadcast(G){K.forEach((X)=>X.broadcast(G))},end(){K.forEach(({exitRoom:G})=>G()),K.clear(),D.forEach(({userId:G})=>A(G)),D.clear()}}}export{s as collectPeerConnections};
2
2
 
3
- //# debugId=8DAA0A1DB167198B64756E2164756E21
3
+ //# debugId=14A51E1B5629C31A64756E2164756E21
4
4
  //# sourceMappingURL=webrtc-peer-collector.js.map
@@ -4,9 +4,9 @@
4
4
  "sourcesContent": [
5
5
  "export interface IPeer<T extends string = string, P = any> {\n userId: string;\n receive(type: T, payload: P): boolean;\n}\n\ntype OutMessage = { type: string; to: string; payload: any };\n\n/**\n * enterRoom connects to the signaling room via WebSocket.\n */\nexport function enterRoom<T extends string, P = any>(params: {\n userId: string;\n worldId: string;\n room: string;\n host: string;\n onOpen?: () => void;\n onClose?: (ev: Pick<CloseEvent, \"code\" | \"reason\" | \"wasClean\">) => void;\n onError?: () => void;\n logLine?: (direction: string, obj?: any) => void;\n onPeerJoined(users: IPeer<T, P>[]): void;\n onPeerLeft(users: { userId: string }[]): void;\n onIceUrl?(url: string, expiration: number): void;\n onMessage(type: T, payload: P, from: IPeer<T, P>): void;\n autoRejoin?: boolean;\n}): {\n exitRoom: () => void;\n sendToServer: <P extends any>(type: T, payload?: P) => void;\n} {\n type Message = {\n type: \"peer-joined\" | \"peer-left\" | \"ice-server\" | T;\n userId: string;\n users: { userId: string }[];\n payload: P;\n url: string;\n expiration: number;\n };\n\n const { userId, worldId, room, host, autoRejoin = true, logLine } = params;\n\n let exited = false;\n let retryCount = 0;\n let ws: WebSocket;\n let timeoutId: ReturnType<typeof setTimeout>;\n let initialConnection = true;\n\n const peers = new Map<string, IPeer<T, P>>();\n const wsUrl = `wss://${host}/room/${worldId}/${room}?userId=${encodeURIComponent(\n userId,\n )}`;\n\n // Helper for sending (uses the current ws instance)\n const accumulatedMessages: OutMessage[] = [];\n let timeout: ReturnType<typeof setTimeout> = 0;\n function send(type: string, to: \"server\" | string, payload?: any) {\n if (!ws) {\n logLine?.(\"👤 ➡️ ❌\", \"no ws available\");\n return false;\n }\n if (exited || ws.readyState !== WebSocket.OPEN) {\n logLine?.(\"👤 ➡️ ❌\", \"Not in opened state: \" + ws.readyState);\n return false;\n }\n const obj: OutMessage = { type, to, payload };\n accumulatedMessages.push(obj);\n // ws.send(JSON.stringify(obj));\n logLine?.(\"👤 ➡️ 🖥️\", obj);\n clearTimeout(timeout);\n timeout = setTimeout(() => {\n ws.send(JSON.stringify(accumulatedMessages));\n accumulatedMessages.length = 0;\n });\n return true;\n }\n\n function connect() {\n if (exited) return;\n\n ws = new WebSocket(wsUrl);\n\n ws.onopen = () => {\n if (initialConnection) {\n params.onOpen?.();\n initialConnection = false;\n }\n retryCount = 0; // Reset backoff on successful connection\n };\n\n ws.onmessage = (e: MessageEvent) => {\n try {\n const result = JSON.parse(e.data);\n const msgs: Message[] = Array.isArray(result) ? result : [result];\n msgs.forEach((msg) => {\n logLine?.(\"🖥️ ➡️ 👤\", msg);\n if (msg.type === \"peer-joined\" || msg.type === \"peer-left\") {\n updatePeers(msg.users);\n } else if (msg.type === \"ice-server\") {\n params.onIceUrl?.(msg.url, msg.expiration);\n } else if (msg.userId) {\n params.onMessage(msg.type, msg.payload, {\n userId: msg.userId,\n receive: (type: T, payload: P) => send(type, msg.userId, payload),\n });\n }\n });\n } catch {\n logLine?.(\"⚠️ ERROR\", { error: \"invalid-json\" });\n }\n };\n\n ws.onclose = (ev: CloseEvent) => {\n // 1. Check if we should even try to reconnect\n const recoverableCodes = [1001, 1006, 1011, 1012, 1013];\n const isRecoverable = recoverableCodes.includes(ev.code);\n\n if (autoRejoin && !exited && isRecoverable) {\n // 2. Exponential Backoff: 1s, 2s, 4s, 8s... capped at 30s\n const backoff = Math.min(Math.pow(2, retryCount) * 1000, 30000);\n // 3. Add Jitter: +/- 1000ms randomness\n const jitter = Math.random() * 1000;\n const delay = backoff + jitter;\n\n logLine?.(\"🔄 RECONNECTING\", {\n attempt: retryCount + 1,\n delayMs: Math.round(delay),\n });\n\n retryCount++;\n timeoutId = setTimeout(connect, delay);\n } else {\n params.onClose?.({\n code: ev.code,\n reason: ev.reason,\n wasClean: ev.wasClean,\n });\n }\n };\n\n ws.onerror = (ev) => {\n console.error(\"WS Error\", ev);\n params.onError?.();\n };\n }\n\n // Helper for peer tracking (logic from your original code)\n function updatePeers(updatedUsers: { userId: string }[]) {\n const joined: IPeer<T, P>[] = [];\n const left: { userId: string }[] = [];\n const updatedPeerSet = new Set<string>();\n\n updatedUsers.forEach(({ userId: pUserId }) => {\n if (pUserId === userId) return;\n if (!peers.has(pUserId)) {\n const newPeer = {\n userId: pUserId,\n receive: (t: T, p: P) => send(t, pUserId, p),\n };\n peers.set(pUserId, newPeer);\n joined.push(newPeer);\n }\n updatedPeerSet.add(pUserId);\n });\n\n for (const pUserId of peers.keys()) {\n if (!updatedPeerSet.has(pUserId)) {\n peers.delete(pUserId);\n left.push({ userId: pUserId });\n }\n }\n // Notify peer joined first then peer left. (avoid disconnect in case the peer leaving / joining is on the same user).\n if (joined.length) params.onPeerJoined(joined);\n if (left.length) params.onPeerLeft(left);\n }\n\n // Start initial connection\n connect();\n\n return {\n sendToServer(type, payload) {\n send(type, \"server\", payload);\n },\n exitRoom: () => {\n exited = true;\n clearTimeout(timeoutId);\n ws.close();\n },\n };\n}\n",
6
6
  "import type { IPeer } from \"./impl/signal-room.js\";\nimport { enterRoom as baseEnterRoom } from \"./impl/signal-room.js\";\nimport { RoomEvent, WorkerCommand } from \"./signal-room.worker.js\";\n\nexport function enterRoom<T extends string, P = any>({\n userId,\n worldId,\n room,\n host,\n autoRejoin = true,\n onOpen,\n onClose,\n onError,\n onPeerJoined,\n onPeerLeft,\n onIceUrl,\n onMessage,\n logLine,\n workerUrl,\n}: {\n userId: string;\n worldId: string;\n room: string;\n host: string;\n autoRejoin?: boolean;\n onOpen?: () => void;\n onClose?: (ev: Pick<CloseEvent, \"code\" | \"reason\" | \"wasClean\">) => void;\n onError?: () => void;\n onPeerJoined: (users: IPeer<T, P>[]) => void;\n onPeerLeft: (users: { userId: string }[]) => void;\n onIceUrl?(url: string, expiration: number): void;\n onMessage: (type: T, payload: P, from: IPeer<T, P>) => void;\n logLine?: (direction: string, obj?: any) => void;\n\n // Pass the URL to your worker file (bundler will handle it)\n workerUrl?: URL;\n}): {\n exitRoom: () => void;\n sendToServer: <P extends any>(type: T, payload?: P) => void;\n} {\n if (!workerUrl) {\n const CDN_WORKER_URL = `https://cdn.jsdelivr.net/npm/@dobuki/hello-worker/dist/signal-room.worker.min.js`;\n\n console.warn(\n \"Warning: enterRoom called without workerUrl; this may cause issues in some environments. You should pass workerUrl explicitly. Use:\",\n CDN_WORKER_URL,\n );\n return baseEnterRoom<T, P>({\n userId,\n worldId,\n room,\n host,\n autoRejoin,\n onOpen,\n onClose,\n onError,\n onPeerJoined,\n onPeerLeft,\n onIceUrl,\n onMessage,\n });\n }\n const worker = new Worker(workerUrl, { type: \"module\" });\n let exited = false;\n\n function makeUser({ userId }: { userId: string }): IPeer<T, P> {\n return {\n userId,\n receive: (type: T, payload: P) => {\n if (exited) return false;\n worker.postMessage({\n cmd: \"send\",\n toUserId: userId,\n host,\n room,\n type,\n payload,\n } as WorkerCommand);\n return true;\n },\n };\n }\n\n const onWorkerMessage = (e: MessageEvent<RoomEvent<T, P>>) => {\n const ev = e.data;\n\n if (ev.kind === \"open\") onOpen?.();\n else if (ev.kind === \"close\") {\n worker.terminate();\n onClose?.(ev.ev);\n } else if (ev.kind === \"error\") onError?.();\n else if (ev.kind === \"peer-joined\")\n onPeerJoined(ev.users.map((ev) => makeUser({ userId: ev.userId })));\n else if (ev.kind === \"peer-left\") onPeerLeft(ev.users);\n else if (ev.kind === \"ice-server\") onIceUrl?.(ev.url, ev.expiration);\n else if (ev.kind === \"message\")\n onMessage(ev.type, ev.payload, makeUser({ userId: ev.fromUserId }));\n else if (ev.kind === \"log\") logLine?.(ev.direction, ev.obj);\n };\n\n worker.addEventListener(\"message\", onWorkerMessage);\n\n worker.postMessage({\n cmd: \"enter\",\n userId,\n worldId,\n room,\n host,\n autoRejoin,\n } as WorkerCommand);\n\n return {\n exitRoom: () => {\n exited = true;\n worker.removeEventListener(\"message\", onWorkerMessage);\n worker.postMessage({ cmd: \"exit\" } as WorkerCommand);\n },\n sendToServer: <P extends any>(type: T, payload?: P) => {\n worker.postMessage({\n cmd: \"send\",\n toUserId: \"server\",\n host,\n room,\n type,\n payload,\n } as WorkerCommand);\n },\n };\n}\n\nexport type EnterRoom<T extends string, P> = typeof enterRoom<T, P>;\n",
7
- "import { IPeer } from \"./impl/signal-room\";\nimport { EnterRoom, enterRoom } from \"./signal-room\";\n\nexport type SigType = \"offer\" | \"answer\" | \"ice\" | \"request-ice\" | \"broadcast\";\nexport type SigPayload = RTCSessionDescriptionInit | RTCIceCandidateInit;\n\ntype UserState = {\n userId: string;\n pc?: RTCPeerConnection;\n\n // ICE that arrived before we had remoteDescription\n pendingRemoteIce: RTCIceCandidateInit[];\n\n // the signaling \"user\" handle so we can send messages\n peer: IPeer<SigType, SigPayload>;\n\n expirationTimeout?: number;\n};\n\nconst DEFAULT_ENTER_ROOM = enterRoom;\n\nexport function collectPeerConnections({\n worldId,\n receivePeerConnection,\n peerlessUserExpiration = 5000,\n fallbackRtcConfig = {\n iceServers: [{ urls: \"stun:stun.l.google.com:19302\" }],\n },\n enterRoomFunction: enterRoom = DEFAULT_ENTER_ROOM,\n logLine = console.debug,\n onLeaveUser,\n workerUrl,\n onRoomReady,\n onRoomClose,\n onBroadcastMessage,\n}: {\n worldId: string;\n fallbackRtcConfig?: RTCConfiguration;\n enterRoomFunction?: EnterRoom<SigType, SigPayload>;\n onLeaveUser?: (userId: string) => void;\n logLine?: (direction: string, obj?: any) => void;\n workerUrl?: URL;\n peerlessUserExpiration?: number;\n receivePeerConnection(connection: {\n pc: RTCPeerConnection;\n userId: string;\n initiator: boolean;\n restart?: () => void;\n }): void;\n onRoomReady?(info: { host: string; room: string }): void;\n onRoomClose?(info: {\n host: string;\n room: string;\n ev: Pick<CloseEvent, \"reason\" | \"code\" | \"wasClean\">;\n }): void;\n onBroadcastMessage?<P extends any>(payload: P, from: string): void;\n}) {\n const userId = `user-${crypto.randomUUID()}`;\n const users: Map<string, UserState> = new Map();\n let iceUrl: { url: string; expiration: number } | undefined = undefined;\n let rtcConfig: RTCConfiguration & { timestamp: number } = {\n ...fallbackRtcConfig,\n timestamp: Date.now(),\n };\n\n const roomsEntered = new Map<\n string,\n {\n room: string;\n host: string;\n exitRoom: () => void;\n broadcast: <P extends any>(payload: P) => void;\n }\n >();\n\n async function getRtcConfig(iceUrl: string): Promise<RTCConfiguration> {\n if (iceUrl) {\n try {\n const r = await fetch(iceUrl);\n if (!r.ok) throw new Error(`ICE endpoint failed: ${r.status}`);\n rtcConfig = (await r.json()) as RTCConfiguration & {\n timestamp: number;\n };\n } catch (e) {\n console.warn(\"Using fallback rtcConfig:\", e);\n }\n }\n return rtcConfig;\n }\n\n function leaveUser(userId: string) {\n onLeaveUser?.(userId);\n const p = users.get(userId);\n if (!p) return;\n try {\n p.pc?.close();\n } catch {}\n users.delete(userId);\n }\n\n async function flushRemoteIce(state: UserState) {\n if (!state.pc?.remoteDescription) return;\n\n const queued = state.pendingRemoteIce;\n state.pendingRemoteIce = [];\n\n for (const ice of queued) {\n try {\n await state.pc.addIceCandidate(ice);\n } catch (e) {\n logLine(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n }\n }\n\n function exit({ room, host }: { room: string; host: string }) {\n const key = `${host}/room/${room}`;\n const session = roomsEntered.get(key);\n if (session) {\n session.exitRoom();\n roomsEntered.delete(key);\n }\n }\n\n function enter({ room, host }: { room: string; host: string }) {\n return new Promise<void>(async (resolve, reject) => {\n async function setupPC(state: UserState) {\n const ice =\n !iceUrl || iceUrl.expiration - Date.now() < 2000\n ? await requestIce()\n : iceUrl;\n state.pc = new RTCPeerConnection(\n Date.now() - (rtcConfig?.timestamp ?? 0) < 10000\n ? rtcConfig\n : await getRtcConfig(ice.url),\n );\n // Send local ICE candidates to this peer\n state.pc.onicecandidate = (ev) => {\n if (!ev.candidate) return;\n state.peer.receive(\"ice\", ev.candidate.toJSON());\n };\n\n state.pc.onconnectionstatechange = () => {\n logLine(\"💬\", {\n event: \"pc-state\",\n userId: state.userId,\n state: state.pc?.connectionState,\n });\n };\n return state.pc;\n }\n\n async function getPeer(\n peer: IPeer<SigType, SigPayload>,\n ): Promise<[UserState, boolean]> {\n let state = users.get(peer.userId);\n let isNewPeer = false;\n if (!state) {\n const newState: UserState = {\n userId: peer.userId,\n pendingRemoteIce: [],\n peer,\n };\n\n await setupPC(newState);\n state = newState;\n\n // New user\n users.set(state.userId, state);\n isNewPeer = true;\n } else if (state) {\n clearTimeout(state.expirationTimeout);\n state.expirationTimeout = 0;\n }\n if (!state.pc || state.pc?.signalingState === \"closed\") {\n await setupPC(state);\n }\n state.peer = peer;\n return [state, isNewPeer];\n }\n\n async function makeOffer(user: IPeer) {\n // Offer flow: createOffer -> setLocalDescription -> send localDescription\n const [state] = await getPeer(user);\n const pc = state.pc;\n const offer = await pc?.createOffer();\n await pc?.setLocalDescription(offer);\n user.receive(\"offer\", pc?.localDescription?.toJSON()!);\n }\n\n let icePromiseResolve:\n | undefined\n | ((url: { url: string; expiration: number }) => void);\n async function requestIce() {\n const iceUrl = await new Promise<{ url: string; expiration: number }>(\n (resolve) => {\n icePromiseResolve = resolve;\n sendToServer(\"request-ice\");\n },\n );\n icePromiseResolve = undefined;\n return iceUrl;\n }\n\n const { exitRoom, sendToServer } = enterRoom({\n userId,\n worldId,\n room,\n host,\n logLine,\n workerUrl,\n autoRejoin: true,\n\n onOpen() {\n onRoomReady?.({ room, host });\n resolve();\n },\n onError() {\n console.error(\"onError\");\n reject();\n },\n onClose(ev) {\n onRoomClose?.({ room, host, ev });\n },\n\n // Existing peers initiate to the newcomer\n onPeerJoined(joiningUsers: IPeer<SigType, SigPayload>[]) {\n joiningUsers.forEach(async (user) => {\n const [state, isNewPeer] = await getPeer(user);\n if (!isNewPeer) {\n logLine(\"👤ℹ️\", \"not a new peer: \" + user.userId);\n return;\n }\n const pc = state.pc;\n if (!pc) {\n logLine(\"👤ℹ️\", \"no pc: \" + user.userId);\n return;\n }\n\n async function restart() {\n const state = users.get(user.userId);\n if (state) {\n state.pc = undefined;\n const pc = await setupPC(state);\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await new Promise((resolve) => setTimeout(resolve, 3000));\n await makeOffer(user);\n }\n }\n\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await makeOffer(user);\n });\n },\n\n onPeerLeft(leavingUsers: { userId: string }[]) {\n leavingUsers.forEach(({ userId }) => {\n const state = users.get(userId);\n if (!state) return;\n state.expirationTimeout = setTimeout(\n () => leaveUser(userId),\n peerlessUserExpiration ?? 0,\n );\n });\n },\n\n onIceUrl(url, expiration) {\n iceUrl = { url, expiration };\n icePromiseResolve?.(iceUrl);\n },\n\n async onMessage(type: SigType, payload: any, from: IPeer) {\n const [state] = await getPeer(from);\n const pc = state.pc;\n if (!pc) return;\n\n if (type === \"offer\") {\n receivePeerConnection({\n pc,\n userId: from.userId,\n initiator: false,\n restart() {\n // reset PC\n state.pc = undefined;\n },\n });\n // Responder: set remote offer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n\n // Create and send answer\n const answer = await pc.createAnswer();\n await pc.setLocalDescription(answer);\n\n from.receive(\"answer\", pc.localDescription?.toJSON()!);\n\n // Now safe to apply any queued ICE from this peer\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"answer\") {\n // Initiator: set remote answer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"ice\") {\n const ice = payload as RTCIceCandidateInit;\n\n // If we don't have remoteDescription yet, queue it\n if (!pc.remoteDescription) {\n state.pendingRemoteIce.push(ice);\n return;\n }\n\n try {\n await pc.addIceCandidate(ice);\n } catch (e) {\n logLine(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n return;\n }\n\n if (type === \"broadcast\") {\n onBroadcastMessage?.(payload, from.userId);\n }\n },\n });\n roomsEntered.set(`${host}/room/${room}`, {\n exitRoom,\n room,\n host,\n broadcast: (payload) => {\n sendToServer(\"broadcast\", payload);\n },\n });\n });\n }\n\n return {\n userId,\n enterRoom: enter,\n exitRoom: exit,\n leaveUser,\n broadcast<P extends any>(payload: P) {\n roomsEntered.forEach((room) => room.broadcast(payload));\n },\n end() {\n roomsEntered.forEach(({ exitRoom }) => exitRoom());\n roomsEntered.clear();\n users.forEach(({ userId }) => leaveUser(userId));\n users.clear();\n },\n };\n}\n\n/*\nTurn Token ID\n<CF_TURN_TOKEN_ID>\n\nAPI Token\n<CF_RTC_API_TOKEN>\n\nCURL\ncurl \\\n\t-H \"Authorization: Bearer <CF_RTC_API_TOKEN>\" \\\n\t-H \"Content-Type: application/json\" -d '{\"ttl\": 86400}' \\\n\thttps://rtc.live.cloudflare.com/v1/turn/keys/<CF_TURN_TOKEN_ID>/credentials/generate-ice-servers\n\nJSON\n{\n\t\"iceServers\": [\n {\n \"urls\": [\n \"stun:stun.cloudflare.com:3478\",\n \"turn:turn.cloudflare.com:3478?transport=udp\",\n \"turn:turn.cloudflare.com:3478?transport=tcp\",\n \"turns:turn.cloudflare.com:5349?transport=tcp\"\n ],\n \"username\": \"xxxx\",\n \"credential\": \"yyyy\",\n }\n ]\n}\n\n*/\n"
7
+ "import { IPeer } from \"./impl/signal-room\";\nimport { EnterRoom, enterRoom } from \"./signal-room\";\n\nexport type SigType = \"offer\" | \"answer\" | \"ice\" | \"request-ice\" | \"broadcast\";\nexport type SigPayload = RTCSessionDescriptionInit | RTCIceCandidateInit;\n\ntype UserState = {\n userId: string;\n pc?: RTCPeerConnection;\n\n // ICE that arrived before we had remoteDescription\n pendingRemoteIce: RTCIceCandidateInit[];\n\n // the signaling \"user\" handle so we can send messages\n peer: IPeer<SigType, SigPayload>;\n\n expirationTimeout?: number;\n};\n\nconst DEFAULT_ENTER_ROOM = enterRoom;\n\nexport function collectPeerConnections({\n userId: passedUserId,\n worldId,\n receivePeerConnection,\n peerlessUserExpiration = 5000,\n fallbackRtcConfig = {\n iceServers: [{ urls: \"stun:stun.l.google.com:19302\" }],\n },\n enterRoomFunction: enterRoom = DEFAULT_ENTER_ROOM,\n logLine,\n onLeaveUser,\n workerUrl,\n onRoomReady,\n onRoomClose,\n onBroadcastMessage,\n}: {\n userId?: string;\n worldId: string;\n fallbackRtcConfig?: RTCConfiguration;\n enterRoomFunction?: EnterRoom<SigType, SigPayload>;\n onLeaveUser?: (userId: string) => void;\n logLine?: (direction: string, obj?: any) => void;\n workerUrl?: URL;\n peerlessUserExpiration?: number;\n receivePeerConnection(connection: {\n pc: RTCPeerConnection;\n userId: string;\n initiator: boolean;\n restart?: () => void;\n }): void;\n onRoomReady?(info: { host: string; room: string }): void;\n onRoomClose?(info: {\n host: string;\n room: string;\n ev: Pick<CloseEvent, \"reason\" | \"code\" | \"wasClean\">;\n }): void;\n onBroadcastMessage?<P extends any>(payload: P, from: string): void;\n}) {\n const userId = passedUserId ?? `user-${crypto.randomUUID()}`;\n const users: Map<string, UserState> = new Map();\n let iceUrl: { url: string; expiration: number } | undefined = undefined;\n let rtcConfig: RTCConfiguration & { timestamp: number } = {\n ...fallbackRtcConfig,\n timestamp: Date.now(),\n };\n\n const roomsEntered = new Map<\n string,\n {\n room: string;\n host: string;\n exitRoom: () => void;\n broadcast: <P extends any>(payload: P) => void;\n }\n >();\n\n async function getRtcConfig(\n iceUrl: string,\n ): Promise<RTCConfiguration & { timestamp: number }> {\n if (iceUrl) {\n try {\n const r = await fetch(iceUrl);\n if (!r.ok) throw new Error(`ICE endpoint failed: ${r.status}`);\n rtcConfig = (await r.json()) as RTCConfiguration & {\n timestamp: number;\n };\n } catch (e) {\n console.warn(\"Using fallback rtcConfig:\", e);\n }\n }\n return rtcConfig;\n }\n\n function leaveUser(userId: string) {\n onLeaveUser?.(userId);\n const p = users.get(userId);\n if (!p) return;\n try {\n p.pc?.close();\n } catch {}\n users.delete(userId);\n }\n\n async function flushRemoteIce(state: UserState) {\n if (!state.pc?.remoteDescription) return;\n\n const queued = state.pendingRemoteIce;\n state.pendingRemoteIce = [];\n\n for (const ice of queued) {\n try {\n await state.pc.addIceCandidate(ice);\n } catch (e) {\n logLine?.(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n }\n }\n\n function exit({ room, host }: { room: string; host: string }) {\n const key = `${host}/room/${room}`;\n const session = roomsEntered.get(key);\n if (session) {\n session.exitRoom();\n roomsEntered.delete(key);\n }\n }\n\n function enter({ room, host }: { room: string; host: string }) {\n return new Promise<void>(async (resolve, reject) => {\n async function setupPC(state: UserState) {\n const now = Date.now();\n if (now - (rtcConfig?.timestamp ?? 0) > 10000) {\n console.log(\"ICE expiration\", iceUrl?.expiration, \"-\", now);\n const ice =\n !iceUrl || iceUrl.expiration - now < 2000\n ? await requestIce()\n : iceUrl;\n rtcConfig = await getRtcConfig(ice.url);\n }\n state.pc = new RTCPeerConnection(rtcConfig);\n // Send local ICE candidates to this peer\n state.pc.onicecandidate = (ev) => {\n if (!ev.candidate) return;\n state.peer.receive(\"ice\", ev.candidate.toJSON());\n };\n\n state.pc.onconnectionstatechange = () => {\n logLine?.(\"💬\", {\n event: \"pc-state\",\n userId: state.userId,\n state: state.pc?.connectionState,\n });\n };\n return state.pc;\n }\n\n async function getPeer(\n peer: IPeer<SigType, SigPayload>,\n ): Promise<[UserState, boolean]> {\n let state = users.get(peer.userId);\n let isNewPeer = false;\n if (!state) {\n const newState: UserState = {\n userId: peer.userId,\n pendingRemoteIce: [],\n peer,\n };\n\n await setupPC(newState);\n state = newState;\n\n // New user\n users.set(state.userId, state);\n isNewPeer = true;\n } else if (state) {\n clearTimeout(state.expirationTimeout);\n state.expirationTimeout = 0;\n }\n if (!state.pc || state.pc?.signalingState === \"closed\") {\n await setupPC(state);\n }\n state.peer = peer;\n return [state, isNewPeer];\n }\n\n async function makeOffer(user: IPeer) {\n // Offer flow: createOffer -> setLocalDescription -> send localDescription\n const [state] = await getPeer(user);\n const pc = state.pc;\n const offer = await pc?.createOffer();\n await pc?.setLocalDescription(offer);\n user.receive(\"offer\", pc?.localDescription?.toJSON()!);\n }\n\n let icePromiseResolve:\n | undefined\n | ((url: { url: string; expiration: number }) => void);\n async function requestIce() {\n const iceUrl = await new Promise<{ url: string; expiration: number }>(\n (resolve) => {\n icePromiseResolve = resolve;\n sendToServer(\"request-ice\");\n },\n );\n icePromiseResolve = undefined;\n return iceUrl;\n }\n\n const { exitRoom, sendToServer } = enterRoom({\n userId,\n worldId,\n room,\n host,\n logLine,\n workerUrl,\n autoRejoin: true,\n\n onOpen() {\n onRoomReady?.({ room, host });\n resolve();\n },\n onError() {\n console.error(\"onError\");\n reject();\n },\n onClose(ev) {\n onRoomClose?.({ room, host, ev });\n },\n\n // Existing peers initiate to the newcomer\n onPeerJoined(joiningUsers: IPeer<SigType, SigPayload>[]) {\n joiningUsers.forEach(async (user) => {\n const [state, isNewPeer] = await getPeer(user);\n if (!isNewPeer) {\n logLine?.(\"👤ℹ️\", \"not a new peer: \" + user.userId);\n return;\n }\n const pc = state.pc;\n if (!pc) {\n logLine?.(\"👤ℹ️\", \"no pc: \" + user.userId);\n return;\n }\n\n async function restart() {\n const state = users.get(user.userId);\n if (state) {\n state.pc = undefined;\n const pc = await setupPC(state);\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await new Promise((resolve) => setTimeout(resolve, 3000));\n await makeOffer(user);\n }\n }\n\n receivePeerConnection({\n pc,\n userId: user.userId,\n initiator: true,\n restart,\n });\n await makeOffer(user);\n });\n },\n\n onPeerLeft(leavingUsers: { userId: string }[]) {\n leavingUsers.forEach(({ userId }) => {\n const state = users.get(userId);\n if (!state) return;\n state.expirationTimeout = setTimeout(\n () => leaveUser(userId),\n peerlessUserExpiration ?? 0,\n );\n });\n },\n\n onIceUrl(url, expiration) {\n iceUrl = { url, expiration };\n icePromiseResolve?.(iceUrl);\n },\n\n async onMessage(type: SigType, payload: any, from: IPeer) {\n const [state] = await getPeer(from);\n const pc = state.pc;\n if (!pc) return;\n\n if (type === \"offer\") {\n receivePeerConnection({\n pc,\n userId: from.userId,\n initiator: false,\n restart() {\n // reset PC\n state.pc = undefined;\n },\n });\n // Responder: set remote offer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n\n // Create and send answer\n const answer = await pc.createAnswer();\n await pc.setLocalDescription(answer);\n\n from.receive(\"answer\", pc.localDescription?.toJSON()!);\n\n // Now safe to apply any queued ICE from this peer\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"answer\") {\n // Initiator: set remote answer\n await pc.setRemoteDescription(payload as RTCSessionDescriptionInit);\n await flushRemoteIce(state);\n return;\n }\n\n if (type === \"ice\") {\n const ice = payload as RTCIceCandidateInit;\n\n // If we don't have remoteDescription yet, queue it\n if (!pc.remoteDescription) {\n state.pendingRemoteIce.push(ice);\n return;\n }\n\n try {\n await pc.addIceCandidate(ice);\n } catch (e) {\n logLine?.(\"⚠️ ERROR\", {\n error: \"add-ice-failed\",\n userId: state.userId,\n detail: String(e),\n });\n }\n return;\n }\n\n if (type === \"broadcast\") {\n onBroadcastMessage?.(payload, from.userId);\n }\n },\n });\n roomsEntered.set(`${host}/room/${room}`, {\n exitRoom,\n room,\n host,\n broadcast: (payload) => {\n sendToServer(\"broadcast\", payload);\n },\n });\n });\n }\n\n return {\n userId,\n enterRoom: enter,\n exitRoom: exit,\n leaveUser,\n broadcast<P extends any>(payload: P) {\n roomsEntered.forEach((room) => room.broadcast(payload));\n },\n end() {\n roomsEntered.forEach(({ exitRoom }) => exitRoom());\n roomsEntered.clear();\n users.forEach(({ userId }) => leaveUser(userId));\n users.clear();\n },\n };\n}\n\n/*\nTurn Token ID\n<CF_TURN_TOKEN_ID>\n\nAPI Token\n<CF_RTC_API_TOKEN>\n\nCURL\ncurl \\\n\t-H \"Authorization: Bearer <CF_RTC_API_TOKEN>\" \\\n\t-H \"Content-Type: application/json\" -d '{\"ttl\": 86400}' \\\n\thttps://rtc.live.cloudflare.com/v1/turn/keys/<CF_TURN_TOKEN_ID>/credentials/generate-ice-servers\n\nJSON\n{\n\t\"iceServers\": [\n {\n \"urls\": [\n \"stun:stun.cloudflare.com:3478\",\n \"turn:turn.cloudflare.com:3478?transport=udp\",\n \"turn:turn.cloudflare.com:3478?transport=tcp\",\n \"turns:turn.cloudflare.com:5349?transport=tcp\"\n ],\n \"username\": \"xxxx\",\n \"credential\": \"yyyy\",\n }\n ]\n}\n\n*/\n"
8
8
  ],
9
- "mappings": "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",
10
- "debugId": "8DAA0A1DB167198B64756E2164756E21",
9
+ "mappings": "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",
10
+ "debugId": "14A51E1B5629C31A64756E2164756E21",
11
11
  "names": []
12
12
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@dobuki/hello-worker",
3
- "version": "1.0.67",
3
+ "version": "1.0.69",
4
4
  "type": "module",
5
5
  "main": "./dist/index.js",
6
6
  "types": "./dist/index.d.ts",