sonamu 0.9.4 → 0.9.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/ai/providers/rtzr/utils.js +2 -2
- package/dist/api/config.d.ts +13 -2
- package/dist/api/config.d.ts.map +1 -1
- package/dist/api/config.js +1 -1
- package/dist/api/context.d.ts +17 -7
- package/dist/api/context.d.ts.map +1 -1
- package/dist/api/context.js +1 -1
- package/dist/api/decorators.d.ts +18 -0
- package/dist/api/decorators.d.ts.map +1 -1
- package/dist/api/decorators.js +54 -3
- package/dist/api/index.js +8 -3
- package/dist/api/sonamu.d.ts +24 -9
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +365 -79
- package/dist/api/websocket-helpers.d.ts +24 -0
- package/dist/api/websocket-helpers.d.ts.map +1 -0
- package/dist/api/websocket-helpers.js +77 -0
- package/dist/bin/cli.js +12 -4
- package/dist/database/upsert-builder.js +4 -4
- package/dist/dict/sonamu-dictionary.js +6 -6
- package/dist/entity/entity-manager.js +1 -1
- package/dist/entity/entity.js +3 -3
- package/dist/index.d.ts +6 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +16 -4
- package/dist/migration/code-generation.d.ts.map +1 -1
- package/dist/migration/code-generation.js +8 -9
- package/dist/stream/index.d.ts +6 -0
- package/dist/stream/index.d.ts.map +1 -1
- package/dist/stream/index.js +13 -2
- package/dist/stream/ws-audience-resolver.d.ts +15 -0
- package/dist/stream/ws-audience-resolver.d.ts.map +1 -0
- package/dist/stream/ws-audience-resolver.js +31 -0
- package/dist/stream/ws-audience.d.ts +28 -0
- package/dist/stream/ws-audience.d.ts.map +1 -0
- package/dist/stream/ws-audience.js +46 -0
- package/dist/stream/ws-cluster-bus.d.ts +23 -0
- package/dist/stream/ws-cluster-bus.d.ts.map +1 -0
- package/dist/stream/ws-cluster-bus.js +18 -0
- package/dist/stream/ws-core.d.ts +15 -0
- package/dist/stream/ws-core.d.ts.map +1 -0
- package/dist/stream/ws-core.js +1 -0
- package/dist/stream/ws-delivery.d.ts +24 -0
- package/dist/stream/ws-delivery.d.ts.map +1 -0
- package/dist/stream/ws-delivery.js +103 -0
- package/dist/stream/ws-local-connection-store.d.ts +10 -0
- package/dist/stream/ws-local-connection-store.d.ts.map +1 -0
- package/dist/stream/ws-local-connection-store.js +44 -0
- package/dist/stream/ws-presence-store.d.ts +61 -0
- package/dist/stream/ws-presence-store.d.ts.map +1 -0
- package/dist/stream/ws-presence-store.js +236 -0
- package/dist/stream/ws-registry.d.ts +42 -0
- package/dist/stream/ws-registry.d.ts.map +1 -0
- package/dist/stream/ws-registry.js +108 -0
- package/dist/stream/ws.d.ts +52 -0
- package/dist/stream/ws.d.ts.map +1 -0
- package/dist/stream/ws.js +397 -0
- package/dist/syncer/api-parser.d.ts.map +1 -1
- package/dist/syncer/api-parser.js +72 -2
- package/dist/syncer/checksum.d.ts.map +1 -1
- package/dist/syncer/checksum.js +13 -12
- package/dist/syncer/code-generator.d.ts.map +1 -1
- package/dist/syncer/code-generator.js +7 -4
- package/dist/syncer/event-batcher.d.ts +27 -0
- package/dist/syncer/event-batcher.d.ts.map +1 -0
- package/dist/syncer/event-batcher.js +69 -0
- package/dist/syncer/file-patterns.d.ts +48 -26
- package/dist/syncer/file-patterns.d.ts.map +1 -1
- package/dist/syncer/file-patterns.js +71 -23
- package/dist/syncer/file-tracking.d.ts +13 -0
- package/dist/syncer/file-tracking.d.ts.map +1 -0
- package/dist/syncer/file-tracking.js +33 -0
- package/dist/syncer/index.js +2 -2
- package/dist/syncer/module-loader.d.ts +2 -11
- package/dist/syncer/module-loader.d.ts.map +1 -1
- package/dist/syncer/module-loader.js +3 -3
- package/dist/syncer/syncer-actions.d.ts +39 -6
- package/dist/syncer/syncer-actions.d.ts.map +1 -1
- package/dist/syncer/syncer-actions.js +125 -10
- package/dist/syncer/syncer.d.ts +33 -19
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +168 -168
- package/dist/syncer/watcher.d.ts +8 -0
- package/dist/syncer/watcher.d.ts.map +1 -0
- package/dist/syncer/watcher.js +105 -0
- package/dist/tasks/workflow-manager.d.ts.map +1 -1
- package/dist/tasks/workflow-manager.js +2 -1
- package/dist/template/implementations/services.template.d.ts.map +1 -1
- package/dist/template/implementations/services.template.js +36 -1
- package/dist/testing/bootstrap.d.ts.map +1 -1
- package/dist/testing/bootstrap.js +8 -1
- package/dist/testing/data-explorer.d.ts.map +1 -1
- package/dist/testing/data-explorer.js +5 -3
- package/dist/testing/fixture-manager.js +1 -1
- package/dist/types/types.d.ts +2 -1
- package/dist/types/types.d.ts.map +1 -1
- package/dist/types/types.js +2 -2
- package/dist/ui/api.d.ts.map +1 -1
- package/dist/ui/api.js +4 -3
- package/dist/ui/cdd-service.js +1 -1
- package/dist/ui-web/assets/{index-C5KUjXm0.js → index-BmThfg-s.js} +39 -39
- package/dist/ui-web/assets/index-D4rYm-Xz.css +1 -0
- package/dist/ui-web/index.html +2 -2
- package/dist/utils/async-utils.d.ts +27 -3
- package/dist/utils/async-utils.d.ts.map +1 -1
- package/dist/utils/async-utils.js +56 -6
- package/dist/utils/formatter.d.ts +7 -1
- package/dist/utils/formatter.d.ts.map +1 -1
- package/dist/utils/formatter.js +95 -60
- package/dist/utils/fs-utils.d.ts +2 -0
- package/dist/utils/fs-utils.d.ts.map +1 -1
- package/dist/utils/fs-utils.js +10 -2
- package/dist/utils/process-utils.d.ts +6 -0
- package/dist/utils/process-utils.d.ts.map +1 -1
- package/dist/utils/process-utils.js +16 -3
- package/dist/utils/utils.d.ts +1 -0
- package/dist/utils/utils.d.ts.map +1 -1
- package/dist/utils/utils.js +2 -2
- package/package.json +7 -5
- package/src/ai/providers/rtzr/utils.ts +1 -1
- package/src/api/__tests__/sonamu.websocket.test.ts +64 -0
- package/src/api/__tests__/websocket-context.types.test.ts +58 -0
- package/src/api/config.ts +28 -2
- package/src/api/context.ts +21 -7
- package/src/api/decorators.ts +101 -1
- package/src/api/sonamu.ts +529 -127
- package/src/api/websocket-helpers.ts +122 -0
- package/src/bin/cli.ts +10 -2
- package/src/database/upsert-builder.ts +3 -3
- package/src/dict/sonamu-dictionary.ts +3 -3
- package/src/entity/entity.ts +1 -1
- package/src/index.ts +6 -0
- package/src/migration/code-generation.ts +6 -11
- package/src/shared/app.shared.ts.txt +312 -4
- package/src/shared/web.shared.ts.txt +340 -4
- package/src/stream/__tests__/ws-contracts.test.ts +381 -0
- package/src/stream/__tests__/ws.test.ts +449 -0
- package/src/stream/index.ts +6 -0
- package/src/stream/ws-audience-resolver.ts +35 -0
- package/src/stream/ws-audience.ts +62 -0
- package/src/stream/ws-cluster-bus.ts +32 -0
- package/src/stream/ws-core.ts +16 -0
- package/src/stream/ws-delivery.ts +138 -0
- package/src/stream/ws-local-connection-store.ts +44 -0
- package/src/stream/ws-presence-store.ts +326 -0
- package/src/stream/ws-registry.ts +138 -0
- package/src/stream/ws.ts +591 -0
- package/src/syncer/__tests__/api-parser.websocket-type-ref.test.ts +78 -0
- package/src/syncer/api-parser.ts +112 -1
- package/src/syncer/checksum.ts +23 -29
- package/src/syncer/code-generator.ts +4 -1
- package/src/syncer/event-batcher.ts +72 -0
- package/src/syncer/file-patterns.ts +98 -30
- package/src/syncer/file-tracking.ts +27 -0
- package/src/syncer/module-loader.ts +5 -12
- package/src/syncer/syncer-actions.ts +179 -17
- package/src/syncer/syncer.ts +250 -287
- package/src/syncer/watcher.ts +128 -0
- package/src/tasks/workflow-manager.ts +1 -0
- package/src/template/__tests__/services.template.websocket.test.ts +79 -0
- package/src/template/implementations/services.template.ts +69 -0
- package/src/testing/bootstrap.ts +8 -1
- package/src/testing/data-explorer.ts +3 -2
- package/src/types/types.ts +20 -2
- package/src/ui/api.ts +10 -1
- package/src/utils/async-utils.ts +71 -4
- package/src/utils/formatter.ts +114 -75
- package/src/utils/fs-utils.ts +9 -0
- package/src/utils/process-utils.ts +17 -0
- package/src/utils/utils.ts +1 -1
- package/dist/ui-web/assets/index-Dr8pRJC_.css +0 -1
|
@@ -0,0 +1,236 @@
|
|
|
1
|
+
import { __esmMin } from "../_virtual/rolldown_runtime.js";
|
|
2
|
+
|
|
3
|
+
//#region src/stream/ws-presence-store.ts
|
|
4
|
+
function getBindingKey(namespace, id) {
|
|
5
|
+
return `${namespace}::${id}`;
|
|
6
|
+
}
|
|
7
|
+
function parseBindingKey(key) {
|
|
8
|
+
const [namespace, ...rest] = key.split("::");
|
|
9
|
+
return {
|
|
10
|
+
namespace,
|
|
11
|
+
id: rest.join("::")
|
|
12
|
+
};
|
|
13
|
+
}
|
|
14
|
+
var InMemoryWebSocketPresenceStore;
|
|
15
|
+
var init_ws_presence_store = __esmMin((() => {
|
|
16
|
+
InMemoryWebSocketPresenceStore = class {
|
|
17
|
+
connections = new Map();
|
|
18
|
+
rooms = new Map();
|
|
19
|
+
users = new Map();
|
|
20
|
+
register(input) {
|
|
21
|
+
const meta = {
|
|
22
|
+
sessionId: input.sessionId,
|
|
23
|
+
nodeId: input.nodeId,
|
|
24
|
+
namespace: input.namespace,
|
|
25
|
+
active: input.active ?? true,
|
|
26
|
+
rooms: new Set(),
|
|
27
|
+
connectedAt: new Date(),
|
|
28
|
+
lastSeenAt: new Date()
|
|
29
|
+
};
|
|
30
|
+
this.connections.set(input.sessionId, meta);
|
|
31
|
+
return meta;
|
|
32
|
+
}
|
|
33
|
+
activate(sessionId) {
|
|
34
|
+
const meta = this.connections.get(sessionId);
|
|
35
|
+
if (!meta) {
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
meta.active = true;
|
|
39
|
+
}
|
|
40
|
+
unregister(sessionId) {
|
|
41
|
+
const meta = this.connections.get(sessionId);
|
|
42
|
+
if (!meta) {
|
|
43
|
+
return undefined;
|
|
44
|
+
}
|
|
45
|
+
if (meta.userId !== undefined) {
|
|
46
|
+
this.removeUserBinding(meta.namespace, meta.userId, sessionId);
|
|
47
|
+
}
|
|
48
|
+
for (const roomId of meta.rooms) {
|
|
49
|
+
this.removeRoomBinding(meta.namespace, roomId, sessionId);
|
|
50
|
+
}
|
|
51
|
+
this.connections.delete(sessionId);
|
|
52
|
+
return meta;
|
|
53
|
+
}
|
|
54
|
+
touch(sessionId) {
|
|
55
|
+
const meta = this.connections.get(sessionId);
|
|
56
|
+
if (!meta) {
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
meta.lastSeenAt = new Date();
|
|
60
|
+
}
|
|
61
|
+
setUserId(sessionId, userId) {
|
|
62
|
+
const meta = this.connections.get(sessionId);
|
|
63
|
+
if (!meta) {
|
|
64
|
+
return;
|
|
65
|
+
}
|
|
66
|
+
if (meta.userId !== undefined) {
|
|
67
|
+
this.removeUserBinding(meta.namespace, meta.userId, sessionId);
|
|
68
|
+
}
|
|
69
|
+
meta.userId = userId;
|
|
70
|
+
const key = getBindingKey(meta.namespace, String(userId));
|
|
71
|
+
const bound = this.users.get(key) ?? new Set();
|
|
72
|
+
bound.add(sessionId);
|
|
73
|
+
this.users.set(key, bound);
|
|
74
|
+
}
|
|
75
|
+
clearUserId(sessionId) {
|
|
76
|
+
const meta = this.connections.get(sessionId);
|
|
77
|
+
if (!meta || meta.userId === undefined) {
|
|
78
|
+
return;
|
|
79
|
+
}
|
|
80
|
+
this.removeUserBinding(meta.namespace, meta.userId, sessionId);
|
|
81
|
+
delete meta.userId;
|
|
82
|
+
}
|
|
83
|
+
join(sessionId, roomId) {
|
|
84
|
+
const meta = this.connections.get(sessionId);
|
|
85
|
+
if (!meta) {
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
meta.rooms.add(roomId);
|
|
89
|
+
const key = getBindingKey(meta.namespace, roomId);
|
|
90
|
+
const members = this.rooms.get(key) ?? new Set();
|
|
91
|
+
members.add(sessionId);
|
|
92
|
+
this.rooms.set(key, members);
|
|
93
|
+
}
|
|
94
|
+
leave(sessionId, roomId) {
|
|
95
|
+
const meta = this.connections.get(sessionId);
|
|
96
|
+
if (!meta) {
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
meta.rooms.delete(roomId);
|
|
100
|
+
this.removeRoomBinding(meta.namespace, roomId, sessionId);
|
|
101
|
+
}
|
|
102
|
+
getConnection(sessionId) {
|
|
103
|
+
return this.connections.get(sessionId);
|
|
104
|
+
}
|
|
105
|
+
getConnectionCount(namespace) {
|
|
106
|
+
let count = 0;
|
|
107
|
+
for (const meta of this.connections.values()) {
|
|
108
|
+
if (!meta.active || namespace && meta.namespace !== namespace) {
|
|
109
|
+
continue;
|
|
110
|
+
}
|
|
111
|
+
count += 1;
|
|
112
|
+
}
|
|
113
|
+
return count;
|
|
114
|
+
}
|
|
115
|
+
getRoomMembers(roomId, namespace) {
|
|
116
|
+
return this.getBoundConnections(this.rooms, roomId, namespace);
|
|
117
|
+
}
|
|
118
|
+
getStats() {
|
|
119
|
+
const byNamespace = {};
|
|
120
|
+
for (const meta of this.connections.values()) {
|
|
121
|
+
if (!meta.active) {
|
|
122
|
+
continue;
|
|
123
|
+
}
|
|
124
|
+
byNamespace[meta.namespace] = (byNamespace[meta.namespace] ?? 0) + 1;
|
|
125
|
+
}
|
|
126
|
+
return {
|
|
127
|
+
totalConnections: Object.values(byNamespace).reduce((sum, count) => sum + count, 0),
|
|
128
|
+
totalRooms: this.rooms.size,
|
|
129
|
+
byNamespace
|
|
130
|
+
};
|
|
131
|
+
}
|
|
132
|
+
queryAudience(audience) {
|
|
133
|
+
switch (audience.type) {
|
|
134
|
+
case "all": return this.getActiveConnections(audience.namespace);
|
|
135
|
+
case "room": return this.getBoundConnections(this.rooms, audience.roomId, audience.namespace);
|
|
136
|
+
case "user": return this.getBoundConnections(this.users, String(audience.userId), audience.namespace);
|
|
137
|
+
case "connections": {
|
|
138
|
+
const metas = [];
|
|
139
|
+
for (const connectionId of audience.connectionIds) {
|
|
140
|
+
const meta = this.connections.get(connectionId);
|
|
141
|
+
if (!meta?.active) {
|
|
142
|
+
continue;
|
|
143
|
+
}
|
|
144
|
+
if (audience.namespace && meta.namespace !== audience.namespace) {
|
|
145
|
+
continue;
|
|
146
|
+
}
|
|
147
|
+
metas.push(meta);
|
|
148
|
+
}
|
|
149
|
+
return metas;
|
|
150
|
+
}
|
|
151
|
+
case "union": {
|
|
152
|
+
const seen = new Set();
|
|
153
|
+
const metas = [];
|
|
154
|
+
for (const item of audience.audiences) {
|
|
155
|
+
for (const meta of this.queryAudience(item)) {
|
|
156
|
+
if (seen.has(meta.sessionId)) {
|
|
157
|
+
continue;
|
|
158
|
+
}
|
|
159
|
+
seen.add(meta.sessionId);
|
|
160
|
+
metas.push(meta);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
return metas;
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
getActiveConnections(namespace) {
|
|
168
|
+
const metas = [];
|
|
169
|
+
for (const meta of this.connections.values()) {
|
|
170
|
+
if (!meta.active || namespace && meta.namespace !== namespace) {
|
|
171
|
+
continue;
|
|
172
|
+
}
|
|
173
|
+
metas.push(meta);
|
|
174
|
+
}
|
|
175
|
+
return metas;
|
|
176
|
+
}
|
|
177
|
+
getBoundConnections(bindings, id, namespace) {
|
|
178
|
+
const metas = [];
|
|
179
|
+
const seen = new Set();
|
|
180
|
+
for (const members of this.getScopedBindings(bindings, id, namespace)) {
|
|
181
|
+
for (const connectionId of members) {
|
|
182
|
+
if (seen.has(connectionId)) {
|
|
183
|
+
continue;
|
|
184
|
+
}
|
|
185
|
+
seen.add(connectionId);
|
|
186
|
+
const meta = this.connections.get(connectionId);
|
|
187
|
+
if (meta?.active) {
|
|
188
|
+
metas.push(meta);
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
return metas;
|
|
193
|
+
}
|
|
194
|
+
getScopedBindings(bindings, id, namespace) {
|
|
195
|
+
if (namespace) {
|
|
196
|
+
const exact = bindings.get(getBindingKey(namespace, id));
|
|
197
|
+
return exact ? [exact] : [];
|
|
198
|
+
}
|
|
199
|
+
const matches = [];
|
|
200
|
+
for (const [key, members] of bindings) {
|
|
201
|
+
const parsed = parseBindingKey(key);
|
|
202
|
+
if (parsed.id === id) {
|
|
203
|
+
matches.push(members);
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
return matches;
|
|
207
|
+
}
|
|
208
|
+
removeRoomBinding(namespace, roomId, sessionId) {
|
|
209
|
+
const key = getBindingKey(namespace, roomId);
|
|
210
|
+
const members = this.rooms.get(key);
|
|
211
|
+
if (!members) {
|
|
212
|
+
return;
|
|
213
|
+
}
|
|
214
|
+
members.delete(sessionId);
|
|
215
|
+
if (members.size === 0) {
|
|
216
|
+
this.rooms.delete(key);
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
removeUserBinding(namespace, userId, sessionId) {
|
|
220
|
+
const key = getBindingKey(namespace, String(userId));
|
|
221
|
+
const members = this.users.get(key);
|
|
222
|
+
if (!members) {
|
|
223
|
+
return;
|
|
224
|
+
}
|
|
225
|
+
members.delete(sessionId);
|
|
226
|
+
if (members.size === 0) {
|
|
227
|
+
this.users.delete(key);
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
};
|
|
231
|
+
}));
|
|
232
|
+
|
|
233
|
+
//#endregion
|
|
234
|
+
init_ws_presence_store();
|
|
235
|
+
export { InMemoryWebSocketPresenceStore, init_ws_presence_store };
|
|
236
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ws-presence-store.js","names":["meta: WebSocketSessionPresence","byNamespace: Record<string, number>","metas: WebSocketSessionPresence[]","matches: Set<string>[]"],"sources":["../../src/stream/ws-presence-store.ts"],"sourcesContent":["import { type WebSocketAudience } from \"./ws-audience\";\nimport { type WebSocketRegistryStats, type WebSocketRoomId, type WebSocketUserId } from \"./ws-core\";\n\nexport type WebSocketSessionPresence = {\n  sessionId: string;\n  nodeId: string;\n  namespace: string;\n  active: boolean;\n  rooms: Set<WebSocketRoomId>;\n  connectedAt: Date;\n  lastSeenAt: Date;\n  userId?: WebSocketUserId;\n};\n\nexport interface WebSocketPresenceStore {\n  register(input: {\n    sessionId: string;\n    nodeId: string;\n    namespace: string;\n    active?: boolean;\n  }): WebSocketSessionPresence;\n  activate(sessionId: string): void;\n  unregister(sessionId: string): WebSocketSessionPresence | undefined;\n  touch(sessionId: string): void;\n  setUserId(sessionId: string, userId: WebSocketUserId): void;\n  clearUserId(sessionId: string): void;\n  join(sessionId: string, roomId: WebSocketRoomId): void;\n  leave(sessionId: string, roomId: WebSocketRoomId): void;\n  getConnection(sessionId: string): WebSocketSessionPresence | undefined;\n  getConnectionCount(namespace?: string): number;\n  getRoomMembers(roomId: WebSocketRoomId, namespace?: string): WebSocketSessionPresence[];\n  getStats(): WebSocketRegistryStats;\n  queryAudience(audience: WebSocketAudience): WebSocketSessionPresence[];\n}\n\nexport class InMemoryWebSocketPresenceStore implements WebSocketPresenceStore {\n  private readonly connections = new Map<string, WebSocketSessionPresence>();\n  private readonly rooms = new Map<string, Set<string>>();\n  private readonly users = new Map<string, Set<string>>();\n\n  register(input: {\n    sessionId: string;\n    nodeId: string;\n    namespace: string;\n    active?: boolean;\n  }): WebSocketSessionPresence {\n    const meta: WebSocketSessionPresence = {\n      sessionId: input.sessionId,\n      nodeId: input.nodeId,\n      namespace: input.namespace,\n      active: input.active ?? true,\n      rooms: new Set(),\n      connectedAt: new Date(),\n      lastSeenAt: new Date(),\n    };\n    this.connections.set(input.sessionId, meta);\n    return meta;\n  }\n\n  activate(sessionId: string): void {\n    const meta = this.connections.get(sessionId);\n    if (!meta) {\n      return;\n    }\n\n    meta.active = true;\n  }\n\n  unregister(sessionId: string): WebSocketSessionPresence | undefined {\n    const meta = this.connections.get(sessionId);\n    if (!meta) {\n      return undefined;\n    }\n\n    if (meta.userId !== undefined) {\n      this.removeUserBinding(meta.namespace, meta.userId, sessionId);\n    }\n\n    for (const roomId of meta.rooms) {\n      this.removeRoomBinding(meta.namespace, roomId, sessionId);\n    }\n\n    this.connections.delete(sessionId);\n    return meta;\n  }\n\n  touch(sessionId: string): void {\n    const meta = this.connections.get(sessionId);\n    if (!meta) {\n      return;\n    }\n\n    meta.lastSeenAt = new Date();\n  }\n\n  setUserId(sessionId: string, userId: WebSocketUserId): void {\n    const meta = this.connections.get(sessionId);\n    if (!meta) {\n      return;\n    }\n\n    if (meta.userId !== undefined) {\n      this.removeUserBinding(meta.namespace, meta.userId, sessionId);\n    }\n\n    meta.userId = userId;\n    const key = getBindingKey(meta.namespace, String(userId));\n    const bound = this.users.get(key) ?? new Set<string>();\n    bound.add(sessionId);\n    this.users.set(key, bound);\n  }\n\n  clearUserId(sessionId: string): void {\n    const meta = this.connections.get(sessionId);\n    if (!meta || meta.userId === undefined) {\n      return;\n    }\n\n    this.removeUserBinding(meta.namespace, meta.userId, sessionId);\n    delete meta.userId;\n  }\n\n  join(sessionId: string, roomId: WebSocketRoomId): void {\n    const meta = this.connections.get(sessionId);\n    if (!meta) {\n      return;\n    }\n\n    meta.rooms.add(roomId);\n    const key = getBindingKey(meta.namespace, roomId);\n    const members = this.rooms.get(key) ?? new Set<string>();\n    members.add(sessionId);\n    this.rooms.set(key, members);\n  }\n\n  leave(sessionId: string, roomId: WebSocketRoomId): void {\n    const meta = this.connections.get(sessionId);\n    if (!meta) {\n      return;\n    }\n\n    meta.rooms.delete(roomId);\n    this.removeRoomBinding(meta.namespace, roomId, sessionId);\n  }\n\n  getConnection(sessionId: string): WebSocketSessionPresence | undefined {\n    return this.connections.get(sessionId);\n  }\n\n  getConnectionCount(namespace?: string): number {\n    let count = 0;\n\n    for (const meta of this.connections.values()) {\n      if (!meta.active || (namespace && meta.namespace !== namespace)) {\n        continue;\n      }\n\n      count += 1;\n    }\n\n    return count;\n  }\n\n  getRoomMembers(roomId: WebSocketRoomId, namespace?: string): WebSocketSessionPresence[] {\n    return this.getBoundConnections(this.rooms, roomId, namespace);\n  }\n\n  getStats(): WebSocketRegistryStats {\n    const byNamespace: Record<string, number> = {};\n\n    for (const meta of this.connections.values()) {\n      if (!meta.active) {\n        continue;\n      }\n      byNamespace[meta.namespace] = (byNamespace[meta.namespace] ?? 0) + 1;\n    }\n\n    return {\n      totalConnections: Object.values(byNamespace).reduce((sum, count) => sum + count, 0),\n      totalRooms: this.rooms.size,\n      byNamespace,\n    };\n  }\n\n  queryAudience(audience: WebSocketAudience): WebSocketSessionPresence[] {\n    switch (audience.type) {\n      case \"all\":\n        return this.getActiveConnections(audience.namespace);\n      case \"room\":\n        return this.getBoundConnections(this.rooms, audience.roomId, audience.namespace);\n      case \"user\":\n        return this.getBoundConnections(this.users, String(audience.userId), audience.namespace);\n      case \"connections\": {\n        const metas: WebSocketSessionPresence[] = [];\n\n        for (const connectionId of audience.connectionIds) {\n          const meta = this.connections.get(connectionId);\n          if (!meta?.active) {\n            continue;\n          }\n          if (audience.namespace && meta.namespace !== audience.namespace) {\n            continue;\n          }\n          metas.push(meta);\n        }\n\n        return metas;\n      }\n      case \"union\": {\n        const seen = new Set<string>();\n        const metas: WebSocketSessionPresence[] = [];\n\n        for (const item of audience.audiences) {\n          for (const meta of this.queryAudience(item)) {\n            if (seen.has(meta.sessionId)) {\n              continue;\n            }\n\n            seen.add(meta.sessionId);\n            metas.push(meta);\n          }\n        }\n\n        return metas;\n      }\n    }\n  }\n\n  private getActiveConnections(namespace?: string): WebSocketSessionPresence[] {\n    const metas: WebSocketSessionPresence[] = [];\n\n    for (const meta of this.connections.values()) {\n      if (!meta.active || (namespace && meta.namespace !== namespace)) {\n        continue;\n      }\n\n      metas.push(meta);\n    }\n\n    return metas;\n  }\n\n  private getBoundConnections(\n    bindings: Map<string, Set<string>>,\n    id: string,\n    namespace?: string,\n  ): WebSocketSessionPresence[] {\n    const metas: WebSocketSessionPresence[] = [];\n    const seen = new Set<string>();\n\n    for (const members of this.getScopedBindings(bindings, id, namespace)) {\n      for (const connectionId of members) {\n        if (seen.has(connectionId)) {\n          continue;\n        }\n        seen.add(connectionId);\n\n        const meta = this.connections.get(connectionId);\n        if (meta?.active) {\n          metas.push(meta);\n        }\n      }\n    }\n\n    return metas;\n  }\n\n  private getScopedBindings(\n    bindings: Map<string, Set<string>>,\n    id: string,\n    namespace?: string,\n  ): Set<string>[] {\n    if (namespace) {\n      const exact = bindings.get(getBindingKey(namespace, id));\n      return exact ? [exact] : [];\n    }\n\n    const matches: Set<string>[] = [];\n    for (const [key, members] of bindings) {\n      const parsed = parseBindingKey(key);\n      if (parsed.id === id) {\n        matches.push(members);\n      }\n    }\n\n    return matches;\n  }\n\n  private removeRoomBinding(namespace: string, roomId: WebSocketRoomId, sessionId: string): void {\n    const key = getBindingKey(namespace, roomId);\n    const members = this.rooms.get(key);\n    if (!members) {\n      return;\n    }\n\n    members.delete(sessionId);\n    if (members.size === 0) {\n      this.rooms.delete(key);\n    }\n  }\n\n  private removeUserBinding(namespace: string, userId: WebSocketUserId, sessionId: string): void {\n    const key = getBindingKey(namespace, String(userId));\n    const members = this.users.get(key);\n    if (!members) {\n      return;\n    }\n\n    members.delete(sessionId);\n    if (members.size === 0) {\n      this.users.delete(key);\n    }\n  }\n}\n\nfunction getBindingKey(namespace: string, id: string): string {\n  return `${namespace}::${id}`;\n}\n\nfunction parseBindingKey(key: string): { namespace: string; id: string } {\n  const [namespace, ...rest] = key.split(\"::\");\n  return {\n    namespace,\n    id: rest.join(\"::\"),\n  };\n}\n"],"mappings":"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"}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { type WebSocketAudience as WebSocketAudienceSpec } from "./ws-audience";
|
|
2
|
+
import { WebSocketAudienceResolver } from "./ws-audience-resolver";
|
|
3
|
+
import { type WebSocketClusterBus } from "./ws-cluster-bus";
|
|
4
|
+
import { type ManagedWebSocketConnection, type WebSocketRegistryStats, type WebSocketRoomId, type WebSocketUserId } from "./ws-core";
|
|
5
|
+
import { WebSocketDeliveryEngine } from "./ws-delivery";
|
|
6
|
+
import { WebSocketLocalConnectionStore } from "./ws-local-connection-store";
|
|
7
|
+
import { type WebSocketPresenceStore, type WebSocketSessionPresence } from "./ws-presence-store";
|
|
8
|
+
export type { ManagedWebSocketConnection, WebSocketRegistryStats, WebSocketRoomId, WebSocketUserId, };
|
|
9
|
+
export type WebSocketConnectionMeta = WebSocketSessionPresence;
|
|
10
|
+
export type WebSocketRegistryOptions = {
|
|
11
|
+
nodeId?: string;
|
|
12
|
+
presenceStore?: WebSocketPresenceStore;
|
|
13
|
+
clusterBus?: WebSocketClusterBus;
|
|
14
|
+
};
|
|
15
|
+
export declare class WebSocketRegistry {
|
|
16
|
+
readonly nodeId: string;
|
|
17
|
+
readonly localConnections: WebSocketLocalConnectionStore;
|
|
18
|
+
readonly presenceStore: WebSocketPresenceStore;
|
|
19
|
+
readonly clusterBus: WebSocketClusterBus;
|
|
20
|
+
readonly audienceResolver: WebSocketAudienceResolver;
|
|
21
|
+
readonly deliveryEngine: WebSocketDeliveryEngine;
|
|
22
|
+
constructor(options?: WebSocketRegistryOptions);
|
|
23
|
+
register(connection: ManagedWebSocketConnection, active?: boolean): WebSocketConnectionMeta;
|
|
24
|
+
activate(connectionId: string): void;
|
|
25
|
+
unregister(connectionId: string): void;
|
|
26
|
+
touch(connectionId: string): void;
|
|
27
|
+
setUserId(connectionId: string, userId: WebSocketUserId): void;
|
|
28
|
+
clearUserId(connectionId: string): void;
|
|
29
|
+
join(connectionId: string, roomId: WebSocketRoomId): void;
|
|
30
|
+
leave(connectionId: string, roomId: WebSocketRoomId): void;
|
|
31
|
+
broadcast(event: string, data: unknown, namespace?: string): void;
|
|
32
|
+
publishToRoom(roomId: WebSocketRoomId, event: string, data: unknown, namespace?: string): void;
|
|
33
|
+
publishToUser(userId: WebSocketUserId, event: string, data: unknown, namespace?: string): void;
|
|
34
|
+
publishToAudience(audience: WebSocketAudienceSpec, event: string, data: unknown): void;
|
|
35
|
+
getConnection(connectionId: string): WebSocketConnectionMeta | undefined;
|
|
36
|
+
getConnectionCount(namespace?: string): number;
|
|
37
|
+
getRoomMembers(roomId: WebSocketRoomId, namespace?: string): WebSocketConnectionMeta[];
|
|
38
|
+
getStats(): WebSocketRegistryStats;
|
|
39
|
+
closeAll(code?: number, reason?: string): void;
|
|
40
|
+
shutdown(code?: number, reason?: string): Promise<void>;
|
|
41
|
+
}
|
|
42
|
+
//# sourceMappingURL=ws-registry.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ws-registry.d.ts","sourceRoot":"","sources":["../../src/stream/ws-registry.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
import { __esmMin } from "../_virtual/rolldown_runtime.js";
|
|
2
|
+
import { WebSocketAudience, init_ws_audience } from "./ws-audience.js";
|
|
3
|
+
import { WebSocketAudienceResolver, init_ws_audience_resolver } from "./ws-audience-resolver.js";
|
|
4
|
+
import { NoopWebSocketClusterBus, init_ws_cluster_bus } from "./ws-cluster-bus.js";
|
|
5
|
+
import { WebSocketDeliveryEngine, init_ws_delivery } from "./ws-delivery.js";
|
|
6
|
+
import { WebSocketLocalConnectionStore, init_ws_local_connection_store } from "./ws-local-connection-store.js";
|
|
7
|
+
import { InMemoryWebSocketPresenceStore, init_ws_presence_store } from "./ws-presence-store.js";
|
|
8
|
+
|
|
9
|
+
//#region src/stream/ws-registry.ts
|
|
10
|
+
var WebSocketRegistry;
|
|
11
|
+
var init_ws_registry = __esmMin((() => {
|
|
12
|
+
init_ws_audience();
|
|
13
|
+
init_ws_audience_resolver();
|
|
14
|
+
init_ws_cluster_bus();
|
|
15
|
+
init_ws_delivery();
|
|
16
|
+
init_ws_local_connection_store();
|
|
17
|
+
init_ws_presence_store();
|
|
18
|
+
WebSocketRegistry = class {
|
|
19
|
+
nodeId;
|
|
20
|
+
localConnections = new WebSocketLocalConnectionStore();
|
|
21
|
+
presenceStore;
|
|
22
|
+
clusterBus;
|
|
23
|
+
audienceResolver;
|
|
24
|
+
deliveryEngine;
|
|
25
|
+
constructor(options = {}) {
|
|
26
|
+
this.nodeId = options.nodeId ?? "local";
|
|
27
|
+
this.presenceStore = options.presenceStore ?? new InMemoryWebSocketPresenceStore();
|
|
28
|
+
this.clusterBus = options.clusterBus ?? new NoopWebSocketClusterBus();
|
|
29
|
+
this.audienceResolver = new WebSocketAudienceResolver({
|
|
30
|
+
nodeId: this.nodeId,
|
|
31
|
+
presenceStore: this.presenceStore
|
|
32
|
+
});
|
|
33
|
+
this.deliveryEngine = new WebSocketDeliveryEngine({
|
|
34
|
+
nodeId: this.nodeId,
|
|
35
|
+
localConnections: this.localConnections,
|
|
36
|
+
audienceResolver: this.audienceResolver,
|
|
37
|
+
clusterBus: this.clusterBus
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
register(connection, active = true) {
|
|
41
|
+
this.localConnections.register(connection);
|
|
42
|
+
return this.presenceStore.register({
|
|
43
|
+
sessionId: connection.id,
|
|
44
|
+
nodeId: this.nodeId,
|
|
45
|
+
namespace: connection.namespace,
|
|
46
|
+
active
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
activate(connectionId) {
|
|
50
|
+
this.presenceStore.activate(connectionId);
|
|
51
|
+
}
|
|
52
|
+
unregister(connectionId) {
|
|
53
|
+
this.presenceStore.unregister(connectionId);
|
|
54
|
+
this.localConnections.unregister(connectionId);
|
|
55
|
+
}
|
|
56
|
+
touch(connectionId) {
|
|
57
|
+
this.presenceStore.touch(connectionId);
|
|
58
|
+
}
|
|
59
|
+
setUserId(connectionId, userId) {
|
|
60
|
+
this.presenceStore.setUserId(connectionId, userId);
|
|
61
|
+
}
|
|
62
|
+
clearUserId(connectionId) {
|
|
63
|
+
this.presenceStore.clearUserId(connectionId);
|
|
64
|
+
}
|
|
65
|
+
join(connectionId, roomId) {
|
|
66
|
+
this.presenceStore.join(connectionId, roomId);
|
|
67
|
+
}
|
|
68
|
+
leave(connectionId, roomId) {
|
|
69
|
+
this.presenceStore.leave(connectionId, roomId);
|
|
70
|
+
}
|
|
71
|
+
broadcast(event, data, namespace) {
|
|
72
|
+
this.publishToAudience(WebSocketAudience.all(namespace), event, data);
|
|
73
|
+
}
|
|
74
|
+
publishToRoom(roomId, event, data, namespace) {
|
|
75
|
+
this.publishToAudience(WebSocketAudience.room(roomId, namespace), event, data);
|
|
76
|
+
}
|
|
77
|
+
publishToUser(userId, event, data, namespace) {
|
|
78
|
+
this.publishToAudience(WebSocketAudience.user(userId, namespace), event, data);
|
|
79
|
+
}
|
|
80
|
+
publishToAudience(audience, event, data) {
|
|
81
|
+
this.deliveryEngine.publishToAudience(audience, event, data);
|
|
82
|
+
}
|
|
83
|
+
getConnection(connectionId) {
|
|
84
|
+
return this.presenceStore.getConnection(connectionId);
|
|
85
|
+
}
|
|
86
|
+
getConnectionCount(namespace) {
|
|
87
|
+
return this.presenceStore.getConnectionCount(namespace);
|
|
88
|
+
}
|
|
89
|
+
getRoomMembers(roomId, namespace) {
|
|
90
|
+
return this.presenceStore.getRoomMembers(roomId, namespace);
|
|
91
|
+
}
|
|
92
|
+
getStats() {
|
|
93
|
+
return this.presenceStore.getStats();
|
|
94
|
+
}
|
|
95
|
+
closeAll(code, reason) {
|
|
96
|
+
this.localConnections.closeAll(code, reason);
|
|
97
|
+
}
|
|
98
|
+
async shutdown(code, reason) {
|
|
99
|
+
this.closeAll(code, reason);
|
|
100
|
+
await this.deliveryEngine.shutdown();
|
|
101
|
+
}
|
|
102
|
+
};
|
|
103
|
+
}));
|
|
104
|
+
|
|
105
|
+
//#endregion
|
|
106
|
+
init_ws_registry();
|
|
107
|
+
export { WebSocketRegistry, init_ws_registry };
|
|
108
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ws-registry.js","names":[],"sources":["../../src/stream/ws-registry.ts"],"sourcesContent":["import { WebSocketAudience, type WebSocketAudience as WebSocketAudienceSpec } from \"./ws-audience\";\nimport { WebSocketAudienceResolver } from \"./ws-audience-resolver\";\nimport { NoopWebSocketClusterBus, type WebSocketClusterBus } from \"./ws-cluster-bus\";\nimport {\n  type ManagedWebSocketConnection,\n  type WebSocketRegistryStats,\n  type WebSocketRoomId,\n  type WebSocketUserId,\n} from \"./ws-core\";\nimport { WebSocketDeliveryEngine } from \"./ws-delivery\";\nimport { WebSocketLocalConnectionStore } from \"./ws-local-connection-store\";\nimport {\n  InMemoryWebSocketPresenceStore,\n  type WebSocketPresenceStore,\n  type WebSocketSessionPresence,\n} from \"./ws-presence-store\";\n\nexport type {\n  ManagedWebSocketConnection,\n  WebSocketRegistryStats,\n  WebSocketRoomId,\n  WebSocketUserId,\n};\nexport type WebSocketConnectionMeta = WebSocketSessionPresence;\n\nexport type WebSocketRegistryOptions = {\n  nodeId?: string;\n  presenceStore?: WebSocketPresenceStore;\n  clusterBus?: WebSocketClusterBus;\n};\n\nexport class WebSocketRegistry {\n  readonly nodeId: string;\n  readonly localConnections = new WebSocketLocalConnectionStore();\n  readonly presenceStore: WebSocketPresenceStore;\n  readonly clusterBus: WebSocketClusterBus;\n  readonly audienceResolver: WebSocketAudienceResolver;\n  readonly deliveryEngine: WebSocketDeliveryEngine;\n\n  constructor(options: WebSocketRegistryOptions = {}) {\n    this.nodeId = options.nodeId ?? \"local\";\n    this.presenceStore = options.presenceStore ?? new InMemoryWebSocketPresenceStore();\n    this.clusterBus = options.clusterBus ?? new NoopWebSocketClusterBus();\n    this.audienceResolver = new WebSocketAudienceResolver({\n      nodeId: this.nodeId,\n      presenceStore: this.presenceStore,\n    });\n    this.deliveryEngine = new WebSocketDeliveryEngine({\n      nodeId: this.nodeId,\n      localConnections: this.localConnections,\n      audienceResolver: this.audienceResolver,\n      clusterBus: this.clusterBus,\n    });\n  }\n\n  register(\n    connection: ManagedWebSocketConnection,\n    active: boolean = true,\n  ): WebSocketConnectionMeta {\n    this.localConnections.register(connection);\n    return this.presenceStore.register({\n      sessionId: connection.id,\n      nodeId: this.nodeId,\n      namespace: connection.namespace,\n      active,\n    });\n  }\n\n  activate(connectionId: string): void {\n    this.presenceStore.activate(connectionId);\n  }\n\n  unregister(connectionId: string): void {\n    this.presenceStore.unregister(connectionId);\n    this.localConnections.unregister(connectionId);\n  }\n\n  touch(connectionId: string): void {\n    this.presenceStore.touch(connectionId);\n  }\n\n  setUserId(connectionId: string, userId: WebSocketUserId): void {\n    this.presenceStore.setUserId(connectionId, userId);\n  }\n\n  clearUserId(connectionId: string): void {\n    this.presenceStore.clearUserId(connectionId);\n  }\n\n  join(connectionId: string, roomId: WebSocketRoomId): void {\n    this.presenceStore.join(connectionId, roomId);\n  }\n\n  leave(connectionId: string, roomId: WebSocketRoomId): void {\n    this.presenceStore.leave(connectionId, roomId);\n  }\n\n  broadcast(event: string, data: unknown, namespace?: string): void {\n    this.publishToAudience(WebSocketAudience.all(namespace), event, data);\n  }\n\n  publishToRoom(roomId: WebSocketRoomId, event: string, data: unknown, namespace?: string): void {\n    this.publishToAudience(WebSocketAudience.room(roomId, namespace), event, data);\n  }\n\n  publishToUser(userId: WebSocketUserId, event: string, data: unknown, namespace?: string): void {\n    this.publishToAudience(WebSocketAudience.user(userId, namespace), event, data);\n  }\n\n  publishToAudience(audience: WebSocketAudienceSpec, event: string, data: unknown): void {\n    this.deliveryEngine.publishToAudience(audience, event, data);\n  }\n\n  getConnection(connectionId: string): WebSocketConnectionMeta | undefined {\n    return this.presenceStore.getConnection(connectionId);\n  }\n\n  getConnectionCount(namespace?: string): number {\n    return this.presenceStore.getConnectionCount(namespace);\n  }\n\n  getRoomMembers(roomId: WebSocketRoomId, namespace?: string): WebSocketConnectionMeta[] {\n    return this.presenceStore.getRoomMembers(roomId, namespace);\n  }\n\n  getStats(): WebSocketRegistryStats {\n    return this.presenceStore.getStats();\n  }\n\n  closeAll(code?: number, reason?: string): void {\n    this.localConnections.closeAll(code, reason);\n  }\n\n  async shutdown(code?: number, reason?: string): Promise<void> {\n    this.closeAll(code, reason);\n    await this.deliveryEngine.shutdown();\n  }\n}\n"],"mappings":";;;;;;;;;;;mBAAmG;4BAChC;sBACkB;mBAO7B;iCACoB;yBAK/C;CAgBhB,oBAAb,MAA+B;EAC7B,AAAS;EACT,AAAS,mBAAmB,IAAI,+BAA+B;EAC/D,AAAS;EACT,AAAS;EACT,AAAS;EACT,AAAS;EAET,YAAY,UAAoC,EAAE,EAAE;AAClD,QAAK,SAAS,QAAQ,UAAU;AAChC,QAAK,gBAAgB,QAAQ,iBAAiB,IAAI,gCAAgC;AAClF,QAAK,aAAa,QAAQ,cAAc,IAAI,yBAAyB;AACrE,QAAK,mBAAmB,IAAI,0BAA0B;IACpD,QAAQ,KAAK;IACb,eAAe,KAAK;IACrB,CAAC;AACF,QAAK,iBAAiB,IAAI,wBAAwB;IAChD,QAAQ,KAAK;IACb,kBAAkB,KAAK;IACvB,kBAAkB,KAAK;IACvB,YAAY,KAAK;IAClB,CAAC;;EAGJ,SACE,YACA,SAAkB,MACO;AACzB,QAAK,iBAAiB,SAAS,WAAW;AAC1C,UAAO,KAAK,cAAc,SAAS;IACjC,WAAW,WAAW;IACtB,QAAQ,KAAK;IACb,WAAW,WAAW;IACtB;IACD,CAAC;;EAGJ,SAAS,cAA4B;AACnC,QAAK,cAAc,SAAS,aAAa;;EAG3C,WAAW,cAA4B;AACrC,QAAK,cAAc,WAAW,aAAa;AAC3C,QAAK,iBAAiB,WAAW,aAAa;;EAGhD,MAAM,cAA4B;AAChC,QAAK,cAAc,MAAM,aAAa;;EAGxC,UAAU,cAAsB,QAA+B;AAC7D,QAAK,cAAc,UAAU,cAAc,OAAO;;EAGpD,YAAY,cAA4B;AACtC,QAAK,cAAc,YAAY,aAAa;;EAG9C,KAAK,cAAsB,QAA+B;AACxD,QAAK,cAAc,KAAK,cAAc,OAAO;;EAG/C,MAAM,cAAsB,QAA+B;AACzD,QAAK,cAAc,MAAM,cAAc,OAAO;;EAGhD,UAAU,OAAe,MAAe,WAA0B;AAChE,QAAK,kBAAkB,kBAAkB,IAAI,UAAU,EAAE,OAAO,KAAK;;EAGvE,cAAc,QAAyB,OAAe,MAAe,WAA0B;AAC7F,QAAK,kBAAkB,kBAAkB,KAAK,QAAQ,UAAU,EAAE,OAAO,KAAK;;EAGhF,cAAc,QAAyB,OAAe,MAAe,WAA0B;AAC7F,QAAK,kBAAkB,kBAAkB,KAAK,QAAQ,UAAU,EAAE,OAAO,KAAK;;EAGhF,kBAAkB,UAAiC,OAAe,MAAqB;AACrF,QAAK,eAAe,kBAAkB,UAAU,OAAO,KAAK;;EAG9D,cAAc,cAA2D;AACvE,UAAO,KAAK,cAAc,cAAc,aAAa;;EAGvD,mBAAmB,WAA4B;AAC7C,UAAO,KAAK,cAAc,mBAAmB,UAAU;;EAGzD,eAAe,QAAyB,WAA+C;AACrF,UAAO,KAAK,cAAc,eAAe,QAAQ,UAAU;;EAG7D,WAAmC;AACjC,UAAO,KAAK,cAAc,UAAU;;EAGtC,SAAS,MAAe,QAAuB;AAC7C,QAAK,iBAAiB,SAAS,MAAM,OAAO;;EAG9C,MAAM,SAAS,MAAe,QAAgC;AAC5D,QAAK,SAAS,MAAM,OAAO;AAC3B,SAAM,KAAK,eAAe,UAAU"}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import { type WebSocket } from "ws";
|
|
2
|
+
import { z } from "zod";
|
|
3
|
+
import { type WebSocketAudience } from "./ws-audience";
|
|
4
|
+
import { type WebSocketClusterBus } from "./ws-cluster-bus";
|
|
5
|
+
import { type WebSocketPresenceStore } from "./ws-presence-store";
|
|
6
|
+
import { type ManagedWebSocketConnection, WebSocketRegistry, type WebSocketRoomId, type WebSocketUserId } from "./ws-registry";
|
|
7
|
+
type MessageHandler<T> = (data: T) => void | Promise<void>;
|
|
8
|
+
type CloseHandler = () => void | Promise<void>;
|
|
9
|
+
export type WebSocketEventMap = Record<string, unknown>;
|
|
10
|
+
type InferWebSocketEventMap<TSchema extends z.ZodRawShape> = z.infer<z.ZodObject<TSchema>>;
|
|
11
|
+
export type WebSocketOutEvents<TOut extends WebSocketEventMap = WebSocketEventMap> = TOut;
|
|
12
|
+
export type WebSocketInEvents<TIn extends WebSocketEventMap = WebSocketEventMap> = TIn;
|
|
13
|
+
export interface WebSocketConnection<TOut extends WebSocketEventMap = WebSocketEventMap, TIn extends WebSocketEventMap = WebSocketEventMap> extends ManagedWebSocketConnection {
|
|
14
|
+
transport: "ws";
|
|
15
|
+
onClose(callback: CloseHandler): void;
|
|
16
|
+
onMessage<K extends keyof WebSocketInEvents<TIn>>(event: K, handler: MessageHandler<WebSocketInEvents<TIn>[K]>): void;
|
|
17
|
+
publish<K extends keyof WebSocketOutEvents<TOut>>(event: K, data: WebSocketOutEvents<TOut>[K]): void;
|
|
18
|
+
waitForClose(): Promise<void>;
|
|
19
|
+
join(roomId: WebSocketRoomId): void;
|
|
20
|
+
leave(roomId: WebSocketRoomId): void;
|
|
21
|
+
setUserId(userId: WebSocketUserId): void;
|
|
22
|
+
clearUserId(): void;
|
|
23
|
+
}
|
|
24
|
+
export type AnyWebSocketConnection = WebSocketConnection<WebSocketEventMap, WebSocketEventMap>;
|
|
25
|
+
type WebSocketConnectionOptions<TOut extends z.ZodRawShape, TIn extends z.ZodRawShape> = {
|
|
26
|
+
namespace?: string;
|
|
27
|
+
heartbeat?: number;
|
|
28
|
+
maxPayload?: number;
|
|
29
|
+
active?: boolean;
|
|
30
|
+
outEvents: z.ZodObject<TOut>;
|
|
31
|
+
inEvents: z.ZodObject<TIn>;
|
|
32
|
+
registry: WebSocketRegistry;
|
|
33
|
+
};
|
|
34
|
+
export type WebSocketRuntimeOptions = {
|
|
35
|
+
nodeId?: string;
|
|
36
|
+
presenceStore?: WebSocketPresenceStore;
|
|
37
|
+
clusterBus?: WebSocketClusterBus;
|
|
38
|
+
};
|
|
39
|
+
export declare class WebSocketRuntime {
|
|
40
|
+
readonly registry: WebSocketRegistry;
|
|
41
|
+
constructor(options?: WebSocketRuntimeOptions);
|
|
42
|
+
registerConnection<TOutSchema extends z.ZodRawShape, TInSchema extends z.ZodRawShape>(socket: WebSocket, options: Omit<WebSocketConnectionOptions<TOutSchema, TInSchema>, "registry">): WebSocketConnection<InferWebSocketEventMap<TOutSchema>, InferWebSocketEventMap<TInSchema>>;
|
|
43
|
+
activateConnection(connectionId: string): void;
|
|
44
|
+
broadcast(event: string, data: unknown, namespace?: string): void;
|
|
45
|
+
publishToRoom(roomId: WebSocketRoomId, event: string, data: unknown, namespace?: string): void;
|
|
46
|
+
publishToUser(userId: WebSocketUserId, event: string, data: unknown, namespace?: string): void;
|
|
47
|
+
publishToAudience(audience: WebSocketAudience, event: string, data: unknown): void;
|
|
48
|
+
shutdown(code?: number, reason?: string): Promise<void>;
|
|
49
|
+
}
|
|
50
|
+
export declare function createWebSocketRuntime(options?: WebSocketRuntimeOptions): WebSocketRuntime;
|
|
51
|
+
export {};
|
|
52
|
+
//# sourceMappingURL=ws.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ws.d.ts","sourceRoot":"","sources":["../../src/stream/ws.ts"],"names":[],"mappings":"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"}
|