verani 0.4.4 → 0.4.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/verani.cjs +1 -1
- package/dist/verani.d.cts +47 -0
- package/dist/verani.d.mts +47 -0
- package/dist/verani.mjs +1 -1
- package/package.json +1 -1
package/dist/verani.cjs
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
const require_types=require(`./types-083oWz55.cjs`);let __cloudflare_actors=require(`@cloudflare/actors`);var RoomEventEmitterImpl=class{constructor(){this.handlers=new Map}on(
|
|
1
|
+
const require_types=require(`./types-083oWz55.cjs`);let __cloudflare_actors=require(`@cloudflare/actors`);var RoomEventEmitterImpl=class{constructor(){this.handlers=new Map}on(c,T){this.handlers.has(c)||this.handlers.set(c,new Set),this.handlers.get(c).add(T)}off(c,T){let E=this.handlers.get(c);E&&(T?(E.delete(T),E.size===0&&this.handlers.delete(c)):this.handlers.delete(c))}async emit(c,T,E){let D=this.handlers.get(c);if(D&&D.size>0){let c=[];for(let O of D)try{let D=O(T,E);D instanceof Promise&&c.push(D)}catch{}await Promise.all(c)}let O=this.handlers.get(`*`);if(O&&O.size>0){let c=[];for(let D of O)try{let O=D(T,E);O instanceof Promise&&c.push(O)}catch{}await Promise.all(c)}}hasHandlers(c){return this.handlers.has(c)&&this.handlers.get(c).size>0||this.handlers.has(`*`)&&this.handlers.get(`*`).size>0}getEventNames(){return Array.from(this.handlers.keys())}rebuildHandlers(c){this.handlers.clear();for(let[T,E]of c.entries())this.handlers.set(T,new Set(E))}};function createRoomEventEmitter(){return new RoomEventEmitterImpl}function defaultExtractMeta(c){let T=crypto.randomUUID(),E=crypto.randomUUID(),D=new URL(c.url).searchParams.get(`channels`);return{userId:T,clientId:E,channels:D?D.split(`,`).map(c=>c.trim()).filter(Boolean):[`default`]}}function defineRoom(c){let T=c.eventEmitter||createRoomEventEmitter(),E=c._staticHandlers||new Map;return{name:c.name,websocketPath:c.websocketPath,extractMeta:c.extractMeta||(c=>defaultExtractMeta(c)),onConnect:c.onConnect,onDisconnect:c.onDisconnect,onMessage:c.onMessage,onError:c.onError,onHibernationRestore:c.onHibernationRestore,eventEmitter:T,_staticHandlers:E,on(c,D){T.on(c,D),E.has(c)||E.set(c,new Set),E.get(c).add(D)},off(c,D){T.off(c,D);let O=E.get(c);O&&(D?(O.delete(D),O.size===0&&E.delete(c)):E.delete(c))}}}function cleanupStaleSessions(c){let T=0,E=[];for(let[T,D]of c.entries())T.readyState!==WebSocket.OPEN&&E.push(T);for(let D of E)c.delete(D),T++;return T}function decodeFrame$1(T){return require_types.o(T)??{type:`invalid`}}function encodeFrame$1(T){return require_types.r(T)}function broadcast(c,T,E,D){let O=0,k=encodeFrame$1({type:`event`,channel:T,data:E}),A=[];for(let{ws:E,meta:j}of c.values())if(j.channels.includes(T)&&!(D?.except&&E===D.except)&&!(D?.userIds&&!D.userIds.includes(j.userId))&&!(D?.clientIds&&!D.clientIds.includes(j.clientId))){if(E.readyState!==WebSocket.OPEN){A.push(E);continue}try{E.send(k),O++}catch{A.push(E)}}for(let T of A)c.delete(T);return A.length,O}function sendToUser(c,T,E,D){let O=0,k=encodeFrame$1({type:`event`,channel:E,data:D}),A=[];for(let{ws:D,meta:j}of c.values())if(j.userId===T&&j.channels.includes(E)){if(D.readyState!==WebSocket.OPEN){A.push(D);continue}try{D.send(k),O++}catch{A.push(D)}}for(let T of A)c.delete(T);return A.length,O}function getSessionCount(c){return c.size}function getConnectedUserIds(c){let T=new Set;for(let{meta:E}of c.values())T.add(E.userId);return Array.from(T)}function getUserSessions(c,T){let E=[];for(let{ws:D,meta:O}of c.values())O.userId===T&&E.push(D);return E}function getStorage(c){return c.storage}function sanitizeToClassName(c){return c.replace(/^\/+/,``).split(/[-_\/\s]+/).map(c=>c.replace(/[^a-zA-Z0-9]/g,``)).filter(c=>c.length>0).map(c=>c.charAt(0).toUpperCase()+c.slice(1).toLowerCase()).join(``)||`VeraniActor`}function createConfiguration(c){return function(T){return{locationHint:`me`,sockets:{upgradePath:c.websocketPath}}}}function isValidConnectionMeta(c){return!(!c||typeof c!=`object`||typeof c.userId!=`string`||!c.userId||typeof c.clientId!=`string`||!c.clientId||!Array.isArray(c.channels)||!c.channels.every(c=>typeof c==`string`))}function storeAttachment(c,T){c.serializeAttachment(T)}function restoreSessions(c){let T=0,E=0;for(let D of c.ctx.getWebSockets()){if(D.readyState!==WebSocket.OPEN){E++;continue}let O=D.deserializeAttachment();if(!O){E++;continue}if(!isValidConnectionMeta(O)){E++;continue}c.sessions.set(D,{ws:D,meta:O}),T++}}async function onInit(c,T){if(T.eventEmitter&&T._staticHandlers)try{T.eventEmitter.rebuildHandlers(T._staticHandlers)}catch{}try{restoreSessions(c)}catch{}if(T.onHibernationRestore&&c.sessions.size>0)try{await T.onHibernationRestore(c)}catch{}else T.onHibernationRestore&&c.sessions.size}function createUserEmitBuilder(c,T,E){return{emit(D,O){return sendToUser(T,c,E,{type:D,...O})}}}function createChannelEmitBuilder(c,T,E){return{emit(D,O){return broadcast(T,c,{type:D,...O},E)}}}function createSocketEmit(c){let T=c.meta.channels[0]||`default`;return{emit(E,D){if(c.ws.readyState===WebSocket.OPEN)try{let O={type:`event`,channel:T,data:{type:E,...D}};c.ws.send(encodeFrame$1(O))}catch{}},to(E){return c.meta.channels.includes(E)?createChannelEmitBuilder(E,c.actor.sessions,{except:c.ws}):createUserEmitBuilder(E,c.actor.sessions,T)}}}function createActorEmit(c){return{emit(T,E){let D={type:T,...E};return broadcast(c.sessions,`default`,D)},to(T){return createChannelEmitBuilder(T,c.sessions)}}}async function onWebSocketConnect(c,T,E,D){let O;try{O=T.extractMeta?await T.extractMeta(D):{userId:`anonymous`,clientId:crypto.randomUUID(),channels:[`default`]},storeAttachment(E,O);let k={actor:c,ws:E,meta:O,frame:{type:`connect`}};if(T.onConnect){let D={actor:c,ws:E,meta:O,emit:createSocketEmit(k)};await T.onConnect(D)}c.sessions.set(E,{ws:E,meta:O})}catch(D){if(T.onError&&O)try{let k={actor:c,ws:E,meta:O,frame:{type:`error`}};await T.onError(D,{actor:c,ws:E,meta:O,emit:createSocketEmit(k)})}catch{}E.close(1011,`Internal server error`)}}async function onWebSocketMessage(c,T,E,D){let O;try{let k=decodeFrame$1(D);if(k&&k.type===`ping`){if(E.readyState===WebSocket.OPEN)try{E.send(encodeFrame$1({type:`pong`}))}catch{}return}if(!k||k.type===`invalid`||(O=c.sessions.get(E),!O))return;let A={actor:c,ws:E,meta:O.meta,frame:k,emit:createSocketEmit({actor:c,ws:E,meta:O.meta,frame:k})},M=T.eventEmitter;M&&M.hasHandlers&&M.hasHandlers(k.type)?await M.emit(k.type,A,k.data||{}):T.onMessage&&await T.onMessage(A,k)}catch(D){if(T.onError&&O)try{await T.onError(D,{actor:c,ws:E,meta:O.meta,emit:createSocketEmit({actor:c,ws:E,meta:O.meta,frame:{type:`error`}})})}catch{}}}async function onWebSocketDisconnect(c,T,E){try{let D=c.sessions.get(E);if(c.sessions.delete(E),D&&T.onDisconnect){let O={actor:c,ws:E,meta:D.meta,frame:{type:`disconnect`}},k={actor:c,ws:E,meta:D.meta,emit:createSocketEmit(O)};await T.onDisconnect(k)}}catch{}}function createFetch(c,T){return async function(E){let D=new URL(E.url),O=E.headers.get(`Upgrade`);return D.pathname===c.websocketPath&&O===`websocket`&&await T.shouldUpgradeWebSocket(E)?T.onWebSocketUpgrade(E):T.onRequest(E)}}function createRpcChannelEmitBuilder(c,T){return{async emit(E,D){return c.emitToChannel(T,E,D)}}}function createRpcUserEmitBuilder(c,T,E=`default`){return{async emit(E,D){return c.emitToUser(T,E,D)}}}function createRpcEmitBuilder(c,T,E=`default`){let D=!1;for(let{meta:E}of c.sessions.values())if(E.channels.includes(T)){D=!0;break}return D?createRpcChannelEmitBuilder(c,T):createRpcUserEmitBuilder(c,T,E)}function createActorHandler(c){let E=sanitizeToClassName(c.name||c.websocketPath||`VeraniActor`);class D extends __cloudflare_actors.Actor{constructor(...T){super(...T),this.sessions=new Map,this.emit=createActorEmit(this),this.fetch=createFetch(c,this)}static{this.configuration=createConfiguration(c)}async shouldUpgradeWebSocket(c){return!0}async onInit(){await onInit(this,c)}async onWebSocketConnect(T,E){await onWebSocketConnect(this,c,T,E)}async onWebSocketMessage(T,E){await onWebSocketMessage(this,c,T,E)}async onWebSocketDisconnect(T){await onWebSocketDisconnect(this,c,T)}cleanupStaleSessions(){return cleanupStaleSessions(this.sessions)}broadcast(c,T,E){return broadcast(this.sessions,c,T,E)}getSessionCount(){return getSessionCount(this.sessions)}getConnectedUserIds(){return getConnectedUserIds(this.sessions)}getUserSessions(c){return getUserSessions(this.sessions,c)}sendToUser(c,T,E){return sendToUser(this.sessions,c,T,E)}emitToChannel(c,T,E){let D={type:T,...E};return broadcast(this.sessions,c,D)}emitToUser(c,T,E){let D={type:T,...E};return sendToUser(this.sessions,c,`default`,D)}toChannel(c){return createRpcChannelEmitBuilder(this,c)}toUser(c){return createRpcUserEmitBuilder(this,c)}to(c){return createRpcEmitBuilder(this,c)}getStorage(){return getStorage(this.ctx)}}return Object.defineProperty(D,`name`,{value:E,writable:!1,configurable:!0}),D}exports.PROTOCOL_VERSION=require_types.t,exports.createActorHandler=createActorHandler,exports.decodeClientMessage=require_types.a,exports.decodeFrame=require_types.o,exports.decodeServerMessage=require_types.s,exports.defineRoom=defineRoom,exports.encodeClientMessage=require_types.n,exports.encodeFrame=require_types.r,exports.encodeServerMessage=require_types.i,exports.restoreSessions=restoreSessions,exports.storeAttachment=storeAttachment;
|
package/dist/verani.d.cts
CHANGED
|
@@ -111,6 +111,13 @@ interface RoomEventEmitter<TMeta extends ConnectionMeta = ConnectionMeta, E = un
|
|
|
111
111
|
* @param data - Event data
|
|
112
112
|
*/
|
|
113
113
|
emit(event: string, ctx: MessageContext<TMeta, E>, data: any): Promise<void>;
|
|
114
|
+
/**
|
|
115
|
+
* Rebuild handlers from static storage.
|
|
116
|
+
* Called after hibernation to restore handlers from the room definition.
|
|
117
|
+
* This method MUST be implemented by all event emitter implementations.
|
|
118
|
+
* @param staticHandlers - Map of event names to handler sets from static storage
|
|
119
|
+
*/
|
|
120
|
+
rebuildHandlers(staticHandlers: Map<string, Set<EventHandler<TMeta, E>>>): void;
|
|
114
121
|
}
|
|
115
122
|
/**
|
|
116
123
|
* Builder interface for targeting specific scopes when emitting
|
|
@@ -161,6 +168,19 @@ interface ActorEmit<TMeta extends ConnectionMeta = ConnectionMeta, E = unknown>
|
|
|
161
168
|
*/
|
|
162
169
|
to(channel: string): EmitBuilder<TMeta, E>;
|
|
163
170
|
}
|
|
171
|
+
/**
|
|
172
|
+
* RPC-safe emit builder interface for use over RPC calls.
|
|
173
|
+
* Used by ActorStub to provide Socket.IO-like emit API over RPC.
|
|
174
|
+
*/
|
|
175
|
+
interface RpcEmitBuilder {
|
|
176
|
+
/**
|
|
177
|
+
* Emit an event to the targeted scope (channel or user)
|
|
178
|
+
* @param event - Event name
|
|
179
|
+
* @param data - Event data
|
|
180
|
+
* @returns Promise resolving to the number of connections that received the message
|
|
181
|
+
*/
|
|
182
|
+
emit(event: string, data?: any): Promise<number>;
|
|
183
|
+
}
|
|
164
184
|
/**
|
|
165
185
|
* Context provided to room lifecycle hooks
|
|
166
186
|
*/
|
|
@@ -237,6 +257,12 @@ interface RoomDefinition<TMeta extends ConnectionMeta = ConnectionMeta, E = unkn
|
|
|
237
257
|
* If not provided, a default event emitter will be created.
|
|
238
258
|
*/
|
|
239
259
|
eventEmitter?: RoomEventEmitter<TMeta, E>;
|
|
260
|
+
/**
|
|
261
|
+
* Static handler storage that persists across hibernation.
|
|
262
|
+
* Handlers registered via room.on() are stored here and rebuilt in onInit.
|
|
263
|
+
* @internal
|
|
264
|
+
*/
|
|
265
|
+
_staticHandlers?: Map<string, Set<EventHandler<TMeta, E>>>;
|
|
240
266
|
}
|
|
241
267
|
//#endregion
|
|
242
268
|
//#region src/actor/router.d.ts
|
|
@@ -279,6 +305,26 @@ interface ActorStub {
|
|
|
279
305
|
*/
|
|
280
306
|
fetch(request: Request): Promise<Response>;
|
|
281
307
|
/**
|
|
308
|
+
* Socket.IO-like emit API: Get a builder for emitting to a specific channel.
|
|
309
|
+
* Use `toChannel("default").emit(event, data)` to emit to the default channel.
|
|
310
|
+
* @param channel - Channel name
|
|
311
|
+
* @returns Promise resolving to an emit builder
|
|
312
|
+
*/
|
|
313
|
+
toChannel(channel: string): Promise<RpcEmitBuilder>;
|
|
314
|
+
/**
|
|
315
|
+
* Socket.IO-like emit API: Get a builder for emitting to a specific user.
|
|
316
|
+
* @param userId - User ID
|
|
317
|
+
* @returns Promise resolving to an emit builder
|
|
318
|
+
*/
|
|
319
|
+
toUser(userId: string): Promise<RpcEmitBuilder>;
|
|
320
|
+
/**
|
|
321
|
+
* Socket.IO-like emit API: Get a builder with smart routing (channel or user).
|
|
322
|
+
* @param target - Channel name or user ID
|
|
323
|
+
* @returns Promise resolving to an emit builder
|
|
324
|
+
*/
|
|
325
|
+
to(target: string): Promise<RpcEmitBuilder>;
|
|
326
|
+
/**
|
|
327
|
+
* @deprecated Use `emit()` or `toChannel().emit()` instead for Socket.IO-like API.
|
|
282
328
|
* Sends a message to a specific user (all their sessions) via RPC.
|
|
283
329
|
* @param userId - The user ID to send to
|
|
284
330
|
* @param channel - The channel to send to
|
|
@@ -287,6 +333,7 @@ interface ActorStub {
|
|
|
287
333
|
*/
|
|
288
334
|
sendToUser(userId: string, channel: string, data?: any): Promise<number>;
|
|
289
335
|
/**
|
|
336
|
+
* @deprecated Use `emit()` or `toChannel().emit()` instead for Socket.IO-like API.
|
|
290
337
|
* Broadcasts a message to all connections in a channel via RPC.
|
|
291
338
|
* Note: The `except` option from BroadcastOptions is not available over RPC
|
|
292
339
|
* since WebSocket cannot be serialized.
|
package/dist/verani.d.mts
CHANGED
|
@@ -111,6 +111,13 @@ interface RoomEventEmitter<TMeta extends ConnectionMeta = ConnectionMeta, E = un
|
|
|
111
111
|
* @param data - Event data
|
|
112
112
|
*/
|
|
113
113
|
emit(event: string, ctx: MessageContext<TMeta, E>, data: any): Promise<void>;
|
|
114
|
+
/**
|
|
115
|
+
* Rebuild handlers from static storage.
|
|
116
|
+
* Called after hibernation to restore handlers from the room definition.
|
|
117
|
+
* This method MUST be implemented by all event emitter implementations.
|
|
118
|
+
* @param staticHandlers - Map of event names to handler sets from static storage
|
|
119
|
+
*/
|
|
120
|
+
rebuildHandlers(staticHandlers: Map<string, Set<EventHandler<TMeta, E>>>): void;
|
|
114
121
|
}
|
|
115
122
|
/**
|
|
116
123
|
* Builder interface for targeting specific scopes when emitting
|
|
@@ -161,6 +168,19 @@ interface ActorEmit<TMeta extends ConnectionMeta = ConnectionMeta, E = unknown>
|
|
|
161
168
|
*/
|
|
162
169
|
to(channel: string): EmitBuilder<TMeta, E>;
|
|
163
170
|
}
|
|
171
|
+
/**
|
|
172
|
+
* RPC-safe emit builder interface for use over RPC calls.
|
|
173
|
+
* Used by ActorStub to provide Socket.IO-like emit API over RPC.
|
|
174
|
+
*/
|
|
175
|
+
interface RpcEmitBuilder {
|
|
176
|
+
/**
|
|
177
|
+
* Emit an event to the targeted scope (channel or user)
|
|
178
|
+
* @param event - Event name
|
|
179
|
+
* @param data - Event data
|
|
180
|
+
* @returns Promise resolving to the number of connections that received the message
|
|
181
|
+
*/
|
|
182
|
+
emit(event: string, data?: any): Promise<number>;
|
|
183
|
+
}
|
|
164
184
|
/**
|
|
165
185
|
* Context provided to room lifecycle hooks
|
|
166
186
|
*/
|
|
@@ -237,6 +257,12 @@ interface RoomDefinition<TMeta extends ConnectionMeta = ConnectionMeta, E = unkn
|
|
|
237
257
|
* If not provided, a default event emitter will be created.
|
|
238
258
|
*/
|
|
239
259
|
eventEmitter?: RoomEventEmitter<TMeta, E>;
|
|
260
|
+
/**
|
|
261
|
+
* Static handler storage that persists across hibernation.
|
|
262
|
+
* Handlers registered via room.on() are stored here and rebuilt in onInit.
|
|
263
|
+
* @internal
|
|
264
|
+
*/
|
|
265
|
+
_staticHandlers?: Map<string, Set<EventHandler<TMeta, E>>>;
|
|
240
266
|
}
|
|
241
267
|
//#endregion
|
|
242
268
|
//#region src/actor/router.d.ts
|
|
@@ -279,6 +305,26 @@ interface ActorStub {
|
|
|
279
305
|
*/
|
|
280
306
|
fetch(request: Request): Promise<Response>;
|
|
281
307
|
/**
|
|
308
|
+
* Socket.IO-like emit API: Get a builder for emitting to a specific channel.
|
|
309
|
+
* Use `toChannel("default").emit(event, data)` to emit to the default channel.
|
|
310
|
+
* @param channel - Channel name
|
|
311
|
+
* @returns Promise resolving to an emit builder
|
|
312
|
+
*/
|
|
313
|
+
toChannel(channel: string): Promise<RpcEmitBuilder>;
|
|
314
|
+
/**
|
|
315
|
+
* Socket.IO-like emit API: Get a builder for emitting to a specific user.
|
|
316
|
+
* @param userId - User ID
|
|
317
|
+
* @returns Promise resolving to an emit builder
|
|
318
|
+
*/
|
|
319
|
+
toUser(userId: string): Promise<RpcEmitBuilder>;
|
|
320
|
+
/**
|
|
321
|
+
* Socket.IO-like emit API: Get a builder with smart routing (channel or user).
|
|
322
|
+
* @param target - Channel name or user ID
|
|
323
|
+
* @returns Promise resolving to an emit builder
|
|
324
|
+
*/
|
|
325
|
+
to(target: string): Promise<RpcEmitBuilder>;
|
|
326
|
+
/**
|
|
327
|
+
* @deprecated Use `emit()` or `toChannel().emit()` instead for Socket.IO-like API.
|
|
282
328
|
* Sends a message to a specific user (all their sessions) via RPC.
|
|
283
329
|
* @param userId - The user ID to send to
|
|
284
330
|
* @param channel - The channel to send to
|
|
@@ -287,6 +333,7 @@ interface ActorStub {
|
|
|
287
333
|
*/
|
|
288
334
|
sendToUser(userId: string, channel: string, data?: any): Promise<number>;
|
|
289
335
|
/**
|
|
336
|
+
* @deprecated Use `emit()` or `toChannel().emit()` instead for Socket.IO-like API.
|
|
290
337
|
* Broadcasts a message to all connections in a channel via RPC.
|
|
291
338
|
* Note: The `except` option from BroadcastOptions is not available over RPC
|
|
292
339
|
* since WebSocket cannot be serialized.
|
package/dist/verani.mjs
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
import{a as decodeClientMessage,i as encodeServerMessage,n as encodeClientMessage,o as decodeFrame,r as encodeFrame,s as decodeServerMessage,t as PROTOCOL_VERSION}from"./types-CJLnZrA8.mjs";import{Actor}from"@cloudflare/actors";var RoomEventEmitterImpl=class{constructor(){this.handlers=new Map}on(u,
|
|
1
|
+
import{a as decodeClientMessage,i as encodeServerMessage,n as encodeClientMessage,o as decodeFrame,r as encodeFrame,s as decodeServerMessage,t as PROTOCOL_VERSION}from"./types-CJLnZrA8.mjs";import{Actor}from"@cloudflare/actors";var RoomEventEmitterImpl=class{constructor(){this.handlers=new Map}on(u,M){this.handlers.has(u)||this.handlers.set(u,new Set),this.handlers.get(u).add(M)}off(u,M){let N=this.handlers.get(u);N&&(M?(N.delete(M),N.size===0&&this.handlers.delete(u)):this.handlers.delete(u))}async emit(u,M,N){let P=this.handlers.get(u);if(P&&P.size>0){let u=[];for(let F of P)try{let P=F(M,N);P instanceof Promise&&u.push(P)}catch{}await Promise.all(u)}let F=this.handlers.get(`*`);if(F&&F.size>0){let u=[];for(let P of F)try{let F=P(M,N);F instanceof Promise&&u.push(F)}catch{}await Promise.all(u)}}hasHandlers(u){return this.handlers.has(u)&&this.handlers.get(u).size>0||this.handlers.has(`*`)&&this.handlers.get(`*`).size>0}getEventNames(){return Array.from(this.handlers.keys())}rebuildHandlers(u){this.handlers.clear();for(let[M,N]of u.entries())this.handlers.set(M,new Set(N))}};function createRoomEventEmitter(){return new RoomEventEmitterImpl}function defaultExtractMeta(u){let M=crypto.randomUUID(),N=crypto.randomUUID(),P=new URL(u.url).searchParams.get(`channels`);return{userId:M,clientId:N,channels:P?P.split(`,`).map(u=>u.trim()).filter(Boolean):[`default`]}}function defineRoom(u){let M=u.eventEmitter||createRoomEventEmitter(),N=u._staticHandlers||new Map;return{name:u.name,websocketPath:u.websocketPath,extractMeta:u.extractMeta||(u=>defaultExtractMeta(u)),onConnect:u.onConnect,onDisconnect:u.onDisconnect,onMessage:u.onMessage,onError:u.onError,onHibernationRestore:u.onHibernationRestore,eventEmitter:M,_staticHandlers:N,on(u,P){M.on(u,P),N.has(u)||N.set(u,new Set),N.get(u).add(P)},off(u,P){M.off(u,P);let F=N.get(u);F&&(P?(F.delete(P),F.size===0&&N.delete(u)):N.delete(u))}}}function cleanupStaleSessions(u){let M=0,N=[];for(let[M,P]of u.entries())M.readyState!==WebSocket.OPEN&&N.push(M);for(let P of N)u.delete(P),M++;return M}function decodeFrame$1(u){return decodeFrame(u)??{type:`invalid`}}function encodeFrame$1(u){return encodeFrame(u)}function broadcast(u,M,N,P){let F=0,I=encodeFrame$1({type:`event`,channel:M,data:N}),L=[];for(let{ws:N,meta:R}of u.values())if(R.channels.includes(M)&&!(P?.except&&N===P.except)&&!(P?.userIds&&!P.userIds.includes(R.userId))&&!(P?.clientIds&&!P.clientIds.includes(R.clientId))){if(N.readyState!==WebSocket.OPEN){L.push(N);continue}try{N.send(I),F++}catch{L.push(N)}}for(let M of L)u.delete(M);return L.length,F}function sendToUser(u,M,N,P){let F=0,I=encodeFrame$1({type:`event`,channel:N,data:P}),L=[];for(let{ws:P,meta:R}of u.values())if(R.userId===M&&R.channels.includes(N)){if(P.readyState!==WebSocket.OPEN){L.push(P);continue}try{P.send(I),F++}catch{L.push(P)}}for(let M of L)u.delete(M);return L.length,F}function getSessionCount(u){return u.size}function getConnectedUserIds(u){let M=new Set;for(let{meta:N}of u.values())M.add(N.userId);return Array.from(M)}function getUserSessions(u,M){let N=[];for(let{ws:P,meta:F}of u.values())F.userId===M&&N.push(P);return N}function getStorage(u){return u.storage}function sanitizeToClassName(u){return u.replace(/^\/+/,``).split(/[-_\/\s]+/).map(u=>u.replace(/[^a-zA-Z0-9]/g,``)).filter(u=>u.length>0).map(u=>u.charAt(0).toUpperCase()+u.slice(1).toLowerCase()).join(``)||`VeraniActor`}function createConfiguration(u){return function(M){return{locationHint:`me`,sockets:{upgradePath:u.websocketPath}}}}function isValidConnectionMeta(u){return!(!u||typeof u!=`object`||typeof u.userId!=`string`||!u.userId||typeof u.clientId!=`string`||!u.clientId||!Array.isArray(u.channels)||!u.channels.every(u=>typeof u==`string`))}function storeAttachment(u,M){u.serializeAttachment(M)}function restoreSessions(u){let M=0,N=0;for(let P of u.ctx.getWebSockets()){if(P.readyState!==WebSocket.OPEN){N++;continue}let F=P.deserializeAttachment();if(!F){N++;continue}if(!isValidConnectionMeta(F)){N++;continue}u.sessions.set(P,{ws:P,meta:F}),M++}}async function onInit(u,M){if(M.eventEmitter&&M._staticHandlers)try{M.eventEmitter.rebuildHandlers(M._staticHandlers)}catch{}try{restoreSessions(u)}catch{}if(M.onHibernationRestore&&u.sessions.size>0)try{await M.onHibernationRestore(u)}catch{}else M.onHibernationRestore&&u.sessions.size}function createUserEmitBuilder(u,M,N){return{emit(P,F){return sendToUser(M,u,N,{type:P,...F})}}}function createChannelEmitBuilder(u,M,N){return{emit(P,F){return broadcast(M,u,{type:P,...F},N)}}}function createSocketEmit(u){let M=u.meta.channels[0]||`default`;return{emit(N,P){if(u.ws.readyState===WebSocket.OPEN)try{let F={type:`event`,channel:M,data:{type:N,...P}};u.ws.send(encodeFrame$1(F))}catch{}},to(N){return u.meta.channels.includes(N)?createChannelEmitBuilder(N,u.actor.sessions,{except:u.ws}):createUserEmitBuilder(N,u.actor.sessions,M)}}}function createActorEmit(u){return{emit(M,N){let P={type:M,...N};return broadcast(u.sessions,`default`,P)},to(M){return createChannelEmitBuilder(M,u.sessions)}}}async function onWebSocketConnect(u,M,N,P){let F;try{F=M.extractMeta?await M.extractMeta(P):{userId:`anonymous`,clientId:crypto.randomUUID(),channels:[`default`]},storeAttachment(N,F);let I={actor:u,ws:N,meta:F,frame:{type:`connect`}};if(M.onConnect){let P={actor:u,ws:N,meta:F,emit:createSocketEmit(I)};await M.onConnect(P)}u.sessions.set(N,{ws:N,meta:F})}catch(P){if(M.onError&&F)try{let I={actor:u,ws:N,meta:F,frame:{type:`error`}};await M.onError(P,{actor:u,ws:N,meta:F,emit:createSocketEmit(I)})}catch{}N.close(1011,`Internal server error`)}}async function onWebSocketMessage(u,M,N,P){let F;try{let I=decodeFrame$1(P);if(I&&I.type===`ping`){if(N.readyState===WebSocket.OPEN)try{N.send(encodeFrame$1({type:`pong`}))}catch{}return}if(!I||I.type===`invalid`||(F=u.sessions.get(N),!F))return;let L={actor:u,ws:N,meta:F.meta,frame:I,emit:createSocketEmit({actor:u,ws:N,meta:F.meta,frame:I})},R=M.eventEmitter;R&&R.hasHandlers&&R.hasHandlers(I.type)?await R.emit(I.type,L,I.data||{}):M.onMessage&&await M.onMessage(L,I)}catch(P){if(M.onError&&F)try{await M.onError(P,{actor:u,ws:N,meta:F.meta,emit:createSocketEmit({actor:u,ws:N,meta:F.meta,frame:{type:`error`}})})}catch{}}}async function onWebSocketDisconnect(u,M,N){try{let P=u.sessions.get(N);if(u.sessions.delete(N),P&&M.onDisconnect){let F={actor:u,ws:N,meta:P.meta,frame:{type:`disconnect`}},I={actor:u,ws:N,meta:P.meta,emit:createSocketEmit(F)};await M.onDisconnect(I)}}catch{}}function createFetch(u,M){return async function(N){let P=new URL(N.url),F=N.headers.get(`Upgrade`);return P.pathname===u.websocketPath&&F===`websocket`&&await M.shouldUpgradeWebSocket(N)?M.onWebSocketUpgrade(N):M.onRequest(N)}}function createRpcChannelEmitBuilder(u,M){return{async emit(N,P){return u.emitToChannel(M,N,P)}}}function createRpcUserEmitBuilder(u,M,N=`default`){return{async emit(N,P){return u.emitToUser(M,N,P)}}}function createRpcEmitBuilder(u,M,N=`default`){let P=!1;for(let{meta:N}of u.sessions.values())if(N.channels.includes(M)){P=!0;break}return P?createRpcChannelEmitBuilder(u,M):createRpcUserEmitBuilder(u,M,N)}function createActorHandler(u){let M=sanitizeToClassName(u.name||u.websocketPath||`VeraniActor`);class N extends Actor{constructor(...M){super(...M),this.sessions=new Map,this.emit=createActorEmit(this),this.fetch=createFetch(u,this)}static{this.configuration=createConfiguration(u)}async shouldUpgradeWebSocket(u){return!0}async onInit(){await onInit(this,u)}async onWebSocketConnect(M,N){await onWebSocketConnect(this,u,M,N)}async onWebSocketMessage(M,N){await onWebSocketMessage(this,u,M,N)}async onWebSocketDisconnect(M){await onWebSocketDisconnect(this,u,M)}cleanupStaleSessions(){return cleanupStaleSessions(this.sessions)}broadcast(u,M,N){return broadcast(this.sessions,u,M,N)}getSessionCount(){return getSessionCount(this.sessions)}getConnectedUserIds(){return getConnectedUserIds(this.sessions)}getUserSessions(u){return getUserSessions(this.sessions,u)}sendToUser(u,M,N){return sendToUser(this.sessions,u,M,N)}emitToChannel(u,M,N){let P={type:M,...N};return broadcast(this.sessions,u,P)}emitToUser(u,M,N){let P={type:M,...N};return sendToUser(this.sessions,u,`default`,P)}toChannel(u){return createRpcChannelEmitBuilder(this,u)}toUser(u){return createRpcUserEmitBuilder(this,u)}to(u){return createRpcEmitBuilder(this,u)}getStorage(){return getStorage(this.ctx)}}return Object.defineProperty(N,`name`,{value:M,writable:!1,configurable:!0}),N}export{PROTOCOL_VERSION,createActorHandler,decodeClientMessage,decodeFrame,decodeServerMessage,defineRoom,encodeClientMessage,encodeFrame,encodeServerMessage,restoreSessions,storeAttachment};
|