@forinda/kickjs-ws 3.2.0 → 4.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +21 -23
- package/dist/index.d.mts +37 -61
- package/dist/index.d.mts.map +1 -1
- package/dist/index.mjs +216 -217
- package/dist/index.mjs.map +1 -1
- package/package.json +7 -12
package/README.md
CHANGED
|
@@ -1,52 +1,50 @@
|
|
|
1
1
|
# @forinda/kickjs-ws
|
|
2
2
|
|
|
3
|
-
WebSocket
|
|
3
|
+
WebSocket adapter for KickJS — decorator-driven handlers (`@WsController`, `@OnConnect`, `@OnDisconnect`, `@OnMessage`, `@OnError`), namespaces, rooms, heartbeat, optional auth resolver.
|
|
4
4
|
|
|
5
5
|
## Install
|
|
6
6
|
|
|
7
7
|
```bash
|
|
8
|
-
# Using the KickJS CLI (recommended — auto-installs peer dependencies)
|
|
9
8
|
kick add ws
|
|
10
|
-
|
|
11
|
-
# Manual install
|
|
12
|
-
pnpm add @forinda/kickjs-ws ws
|
|
13
9
|
```
|
|
14
10
|
|
|
15
|
-
## Features
|
|
16
|
-
|
|
17
|
-
- `WsAdapter` — lifecycle adapter that attaches a WebSocket server to your app
|
|
18
|
-
- Decorator-driven handlers: `@WsController`, `@OnConnect`, `@OnDisconnect`, `@OnMessage`, `@OnError`
|
|
19
|
-
- `WsContext` — typed context for WebSocket handlers
|
|
20
|
-
- `RoomManager` — built-in room/namespace management
|
|
21
|
-
|
|
22
11
|
## Quick Example
|
|
23
12
|
|
|
24
|
-
```
|
|
25
|
-
|
|
13
|
+
```ts
|
|
14
|
+
// chat.ws-controller.ts
|
|
15
|
+
import { WsController, OnConnect, OnMessage, WsContext } from '@forinda/kickjs-ws'
|
|
26
16
|
|
|
27
17
|
@WsController('/chat')
|
|
28
|
-
class
|
|
18
|
+
export class ChatController {
|
|
29
19
|
@OnConnect()
|
|
30
20
|
onConnect(ctx: WsContext) {
|
|
31
|
-
|
|
21
|
+
ctx.send('welcome', { id: ctx.socketId })
|
|
32
22
|
}
|
|
33
23
|
|
|
34
|
-
@OnMessage('
|
|
35
|
-
|
|
36
|
-
ctx.broadcast(ctx.data)
|
|
24
|
+
@OnMessage('say')
|
|
25
|
+
onSay(ctx: WsContext) {
|
|
26
|
+
ctx.broadcast('say', ctx.data)
|
|
37
27
|
}
|
|
38
28
|
}
|
|
29
|
+
```
|
|
30
|
+
|
|
31
|
+
```ts
|
|
32
|
+
// src/index.ts
|
|
33
|
+
import { bootstrap } from '@forinda/kickjs'
|
|
34
|
+
import { WsAdapter } from '@forinda/kickjs-ws'
|
|
35
|
+
import { modules } from './modules'
|
|
39
36
|
|
|
40
|
-
|
|
41
|
-
bootstrap({
|
|
37
|
+
export const app = await bootstrap({
|
|
42
38
|
modules,
|
|
43
|
-
adapters: [
|
|
39
|
+
adapters: [WsAdapter({ path: '/ws' })],
|
|
44
40
|
})
|
|
45
41
|
```
|
|
46
42
|
|
|
43
|
+
Clients connect to `ws://localhost:3000/ws/chat`.
|
|
44
|
+
|
|
47
45
|
## Documentation
|
|
48
46
|
|
|
49
|
-
[
|
|
47
|
+
[forinda.github.io/kick-js/guide/websockets](https://forinda.github.io/kick-js/guide/websockets)
|
|
50
48
|
|
|
51
49
|
## License
|
|
52
50
|
|
package/dist/index.d.mts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
|
|
2
2
|
import { WebSocket, WebSocketServer } from "ws";
|
|
3
3
|
import * as _$_forinda_kickjs0 from "@forinda/kickjs";
|
|
4
|
-
import {
|
|
4
|
+
import { Ref } from "@forinda/kickjs";
|
|
5
5
|
import { IncomingMessage } from "node:http";
|
|
6
6
|
|
|
7
7
|
//#region src/room-manager.d.ts
|
|
@@ -101,48 +101,12 @@ declare const WS_USER_BROADCASTER: _$_forinda_kickjs0.InjectionToken<WsUserBroad
|
|
|
101
101
|
//#endregion
|
|
102
102
|
//#region src/ws-adapter.d.ts
|
|
103
103
|
/**
|
|
104
|
-
*
|
|
105
|
-
*
|
|
106
|
-
*
|
|
107
|
-
* @example
|
|
108
|
-
* ```ts
|
|
109
|
-
* import { WsAdapter } from '@forinda/kickjs-ws'
|
|
110
|
-
*
|
|
111
|
-
* bootstrap({
|
|
112
|
-
* modules: [ChatModule],
|
|
113
|
-
* adapters: [
|
|
114
|
-
* new WsAdapter({ path: '/ws' }),
|
|
115
|
-
* ],
|
|
116
|
-
* })
|
|
117
|
-
* ```
|
|
118
|
-
*
|
|
119
|
-
* Clients connect to: `ws://localhost:3000/ws/chat`
|
|
120
|
-
* Messages are JSON: `{ "event": "send", "data": { "text": "hello" } }`
|
|
104
|
+
* Public extension methods exposed by a WsAdapter instance — broadcast
|
|
105
|
+
* helpers, namespace stats, and reactive counters that DevTools and
|
|
106
|
+
* other adapters consume directly.
|
|
121
107
|
*/
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
private basePath;
|
|
125
|
-
private heartbeatInterval;
|
|
126
|
-
private maxPayload;
|
|
127
|
-
private auth?;
|
|
128
|
-
private userRoomPrefix;
|
|
129
|
-
private wss;
|
|
130
|
-
private container;
|
|
131
|
-
private namespaces;
|
|
132
|
-
private roomManager;
|
|
133
|
-
private heartbeatTimer;
|
|
134
|
-
/** Total WebSocket connections ever opened */
|
|
135
|
-
readonly totalConnections: Ref<number>;
|
|
136
|
-
/** Currently active connections */
|
|
137
|
-
readonly activeConnections: Ref<number>;
|
|
138
|
-
/** Total messages received */
|
|
139
|
-
readonly messagesReceived: Ref<number>;
|
|
140
|
-
/** Total messages sent */
|
|
141
|
-
readonly messagesSent: Ref<number>;
|
|
142
|
-
/** Total errors */
|
|
143
|
-
readonly wsErrors: Ref<number>;
|
|
144
|
-
constructor(options?: WsAdapterOptions);
|
|
145
|
-
/** Get a snapshot of WebSocket stats for DevTools */
|
|
108
|
+
interface WsAdapterExtensions {
|
|
109
|
+
/** Snapshot of WebSocket stats — consumed by DevTools / Swagger ws-server discovery. */
|
|
146
110
|
getStats(): {
|
|
147
111
|
totalConnections: number;
|
|
148
112
|
activeConnections: number;
|
|
@@ -153,31 +117,43 @@ declare class WsAdapter implements AppAdapter {
|
|
|
153
117
|
connections: number;
|
|
154
118
|
handlers: number;
|
|
155
119
|
}>;
|
|
156
|
-
rooms:
|
|
120
|
+
rooms: ReturnType<RoomManager['getAllRooms']>;
|
|
157
121
|
};
|
|
158
122
|
/** Room name used for per-user broadcasting. */
|
|
159
123
|
userRoom(userId: string): string;
|
|
160
124
|
/** Broadcast a single event to every socket in `user:<id>` (across namespaces). */
|
|
161
125
|
broadcastToUser(userId: string, event: string, data: unknown): void;
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
* Runs the configured auth hook against the upgrade request. Stashes the
|
|
173
|
-
* resolved user on the context (as `user`, plus mirrored keys) and, when
|
|
174
|
-
* `autoJoinUserRoom` is enabled, joins the socket to `user:<id>`.
|
|
175
|
-
* Returns `false` (and closes the socket with code 4401) on failure.
|
|
176
|
-
*/
|
|
177
|
-
private authenticate;
|
|
178
|
-
private invokeHandlers;
|
|
179
|
-
private safeInvoke;
|
|
126
|
+
/** Total WebSocket connections ever opened. */
|
|
127
|
+
readonly totalConnections: Ref<number>;
|
|
128
|
+
/** Currently active connections. */
|
|
129
|
+
readonly activeConnections: Ref<number>;
|
|
130
|
+
/** Total messages received. */
|
|
131
|
+
readonly messagesReceived: Ref<number>;
|
|
132
|
+
/** Total messages sent. */
|
|
133
|
+
readonly messagesSent: Ref<number>;
|
|
134
|
+
/** Total errors. */
|
|
135
|
+
readonly wsErrors: Ref<number>;
|
|
180
136
|
}
|
|
137
|
+
/**
|
|
138
|
+
* WebSocket adapter for KickJS. Attaches to the HTTP server and routes
|
|
139
|
+
* WebSocket connections to @WsController classes based on namespace paths.
|
|
140
|
+
*
|
|
141
|
+
* @example
|
|
142
|
+
* ```ts
|
|
143
|
+
* import { WsAdapter } from '@forinda/kickjs-ws'
|
|
144
|
+
*
|
|
145
|
+
* bootstrap({
|
|
146
|
+
* modules: [ChatModule],
|
|
147
|
+
* adapters: [
|
|
148
|
+
* WsAdapter({ path: '/ws' }),
|
|
149
|
+
* ],
|
|
150
|
+
* })
|
|
151
|
+
* ```
|
|
152
|
+
*
|
|
153
|
+
* Clients connect to: `ws://localhost:3000/ws/chat`
|
|
154
|
+
* Messages are JSON: `{ "event": "send", "data": { "text": "hello" } }`
|
|
155
|
+
*/
|
|
156
|
+
declare const WsAdapter: _$_forinda_kickjs0.AdapterFactory<WsAdapterOptions, WsAdapterExtensions>;
|
|
181
157
|
//#endregion
|
|
182
158
|
//#region src/ws-context.d.ts
|
|
183
159
|
/**
|
package/dist/index.d.mts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.mts","names":[],"sources":["../src/room-manager.ts","../src/interfaces.ts","../src/ws-adapter.ts","../src/ws-context.ts","../src/decorators.ts"],"mappings":";;;;;;;;;;;cAMa,WAAA;EAAA;EAAA,QAEH,WAAA;;UAEA,WAAA;EAER,IAAA,CAAK,QAAA,UAAkB,MAAA,EAAQ,SAAA,EAAW,IAAA;EAY1C,KAAA,CAAM,QAAA,UAAkB,IAAA;EAiCT;EAtBf,QAAA,CAAS,QAAA;EAaT,QAAA,CAAS,QAAA;EAIT,UAAA,CAAW,IAAA,WAAe,GAAA,SAAY,SAAA;EA1C9B;EA+CR,WAAA,CAAA,GAAe,MAAA;EA7CV;EAsDL,SAAA,CAAU,IAAA,UAAc,KAAA,UAAe,IAAA,OAAW,SAAA;AAAA;;;cC5DvC,WAAA;EAAA,SAGH,aAAA;EAAA,SAAA,WAAA;AAAA;AAAA,KAEE,aAAA;AAAA,UAEK,mBAAA;EACf,IAAA,EAAM,aAAA;EDsCgC;ECpCtC,KAAA;EDyCe;ECvCf,WAAA;AAAA;;;;;;UAQe,mBAAA;EACf,EAAA;EAAA,CACC,GAAA;AAAA;AAAA,UAGc,YAAA;EDIf;;;;;ECEA,WAAA,GACE,OAAA,EAAS,eAAA,KACN,OAAA,CAAQ,mBAAA,WAA8B,mBAAA;EDajB;;;;ECR1B,gBAAA;EDsBU;;;;ECjBV,cAAA;AAAA;AAAA,UAGe,gBAAA;;EAEf,IAAA;EAhDW;EAkDX,iBAAA;;EAEA,UAAA;;EAEA,IAAA,GAAO,YAAA;AAAA;;;;AA/CT;;;UAwDiB,iBAAA;EAvDf;EAyDA,MAAA,CAAO,MAAA;IAAmB,IAAA,CAAK,KAAA,UAAe,IAAA;EAAA;EArDnC;EAuDX,eAAA,CAAgB,MAAA,UAAgB,KAAA,UAAe,IAAA;EA/ChC;EAiDf,OAAA,CAAQ,MAAA;AAAA;;cAIG,UAAA,EAA8C,kBAAA,CAApC,cAAA;AAhDvB;AAAA,cAkDa,eAAA,EAAe,kBAAA,CAAA,cAAA,CAAA,WAAA;;cAEf,mBAAA,EAAmB,kBAAA,CAAA,cAAA,CAAA,iBAAA
|
|
1
|
+
{"version":3,"file":"index.d.mts","names":[],"sources":["../src/room-manager.ts","../src/interfaces.ts","../src/ws-adapter.ts","../src/ws-context.ts","../src/decorators.ts"],"mappings":";;;;;;;;;;;cAMa,WAAA;EAAA;EAAA,QAEH,WAAA;;UAEA,WAAA;EAER,IAAA,CAAK,QAAA,UAAkB,MAAA,EAAQ,SAAA,EAAW,IAAA;EAY1C,KAAA,CAAM,QAAA,UAAkB,IAAA;EAiCT;EAtBf,QAAA,CAAS,QAAA;EAaT,QAAA,CAAS,QAAA;EAIT,UAAA,CAAW,IAAA,WAAe,GAAA,SAAY,SAAA;EA1C9B;EA+CR,WAAA,CAAA,GAAe,MAAA;EA7CV;EAsDL,SAAA,CAAU,IAAA,UAAc,KAAA,UAAe,IAAA,OAAW,SAAA;AAAA;;;cC5DvC,WAAA;EAAA,SAGH,aAAA;EAAA,SAAA,WAAA;AAAA;AAAA,KAEE,aAAA;AAAA,UAEK,mBAAA;EACf,IAAA,EAAM,aAAA;EDsCgC;ECpCtC,KAAA;EDyCe;ECvCf,WAAA;AAAA;;;;;;UAQe,mBAAA;EACf,EAAA;EAAA,CACC,GAAA;AAAA;AAAA,UAGc,YAAA;EDIf;;;;;ECEA,WAAA,GACE,OAAA,EAAS,eAAA,KACN,OAAA,CAAQ,mBAAA,WAA8B,mBAAA;EDajB;;;;ECR1B,gBAAA;EDsBU;;;;ECjBV,cAAA;AAAA;AAAA,UAGe,gBAAA;;EAEf,IAAA;EAhDW;EAkDX,iBAAA;;EAEA,UAAA;;EAEA,IAAA,GAAO,YAAA;AAAA;;;;AA/CT;;;UAwDiB,iBAAA;EAvDf;EAyDA,MAAA,CAAO,MAAA;IAAmB,IAAA,CAAK,KAAA,UAAe,IAAA;EAAA;EArDnC;EAuDX,eAAA,CAAgB,MAAA,UAAgB,KAAA,UAAe,IAAA;EA/ChC;EAiDf,OAAA,CAAQ,MAAA;AAAA;;cAIG,UAAA,EAA8C,kBAAA,CAApC,cAAA;AAhDvB;AAAA,cAkDa,eAAA,EAAe,kBAAA,CAAA,cAAA,CAAA,WAAA;;cAEf,mBAAA,EAAmB,kBAAA,CAAA,cAAA,CAAA,iBAAA;;;;;;AD7EhC;;UEmCiB,mBAAA;EF7BgB;EE+B/B,QAAA;IACE,gBAAA;IACA,iBAAA;IACA,gBAAA;IACA,YAAA;IACA,MAAA;IACA,UAAA,EAAY,MAAA;MAAiB,WAAA;MAAqB,QAAA;IAAA;IAClD,KAAA,EAAO,UAAA,CAAW,WAAA;EAAA;EF1BpB;EE6BA,QAAA,CAAS,MAAA;EF7Be;EE+BxB,eAAA,CAAgB,MAAA,UAAgB,KAAA,UAAe,IAAA;EFpBtC;EAAA,SEsBA,gBAAA,EAAkB,GAAA;EFTlB;EAAA,SEWA,iBAAA,EAAmB,GAAA;EFPjB;EAAA,SESF,gBAAA,EAAkB,GAAA;EFTW;EAAA,SEW7B,YAAA,EAAc,GAAA;EFNR;EAAA,SEQN,QAAA,EAAU,GAAA;AAAA;;;;;;;;;AD3DrB;;;;;AAKA;;;;;AAEA;cC0Ea,SAAA,EAAS,kBAAA,CAAA,cAAA,CAAA,gBAAA,EAAA,mBAAA;;;;;;AFjFtB;;;;;;;;;;;;;cGca,SAAA;EAAA,SAgBA,MAAA,EAAQ,SAAA;EAAA,SACR,MAAA,EAAQ,eAAA;EAAA,iBACA,WAAA;EAAA,iBACA,gBAAA;EHfK;EAAA,SGFf,EAAA;EHaA;EGXT,IAAA;EHwBS;EGtBT,KAAA;EH0BW;EAAA,SGxBF,SAAA;EHwB6B;;EAAA,SGrB7B,OAAA,EAAS,eAAA;EAAA,QAEV,QAAA;cAGG,MAAA,EAAQ,SAAA,EACR,MAAA,EAAQ,eAAA,EACA,WAAA,EAAa,WAAA,EACb,gBAAA,EAAkB,GAAA,SAAY,SAAA,GAC/C,EAAA,UACA,SAAA,UACA,OAAA,EAAS,eAAA;EHwBa;EAAA,IGdpB,OAAA,CAAA,GAAW,MAAA;EHcmC;EGClD,GAAA,SAAA,CAAa,GAAA,WAAc,CAAA;EHDyC;EGMpE,GAAA,CAAI,GAAA,UAAa,KAAA;;EAKjB,IAAA,CAAK,KAAA,UAAe,IAAA;EFvET;EE8EX,SAAA,CAAU,KAAA,UAAe,IAAA;;EAUzB,YAAA,CAAa,KAAA,UAAe,IAAA;;EAU5B,IAAA,CAAK,IAAA;EF7FkB;EEkGvB,KAAA,CAAM,IAAA;EFlGiB;EEuGvB,KAAA,CAAA;EFrGe;EE0Gf,EAAA,CAAG,IAAA;IAAiB,IAAA,CAAK,KAAA,UAAe,IAAA;EAAA;AAAA;;;;;;;;;AHjH1C;;;;;;;;;;iBIagB,YAAA,CAAa,SAAA,YAAqB,cAAA;;;;;;;;;;;;cA+BrC,SAAA,QAtBA,eAAA;;;;;;;;;;;;cAmCA,YAAA,QAnCA,eAAA;;;;;AHtBb;;;;;AAKA;;cGiEa,OAAA,QAhDA,eAAA;;;AHfb;;;;;;;;;;AAaA;;;;;AAKA;;iBGkEgB,SAAA,CAAU,KAAA,WAAgB,eAAA"}
|
package/dist/index.mjs
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* @forinda/kickjs-ws
|
|
2
|
+
* @forinda/kickjs-ws v4.0.0
|
|
3
3
|
*
|
|
4
4
|
* Copyright (c) Felix Orinda
|
|
5
5
|
*
|
|
@@ -10,7 +10,7 @@
|
|
|
10
10
|
*/
|
|
11
11
|
import { randomUUID } from "node:crypto";
|
|
12
12
|
import { WebSocketServer } from "ws";
|
|
13
|
-
import { Service, createLogger, createToken, getClassMeta, getClassMetaOrUndefined, pushClassMeta, ref, setClassMeta } from "@forinda/kickjs";
|
|
13
|
+
import { Service, createLogger, createToken, defineAdapter, getClassMeta, getClassMetaOrUndefined, pushClassMeta, ref, setClassMeta } from "@forinda/kickjs";
|
|
14
14
|
//#region src/interfaces.ts
|
|
15
15
|
const WS_METADATA = {
|
|
16
16
|
WS_CONTROLLER: Symbol("kick:ws:controller"),
|
|
@@ -198,7 +198,7 @@ const log = createLogger("WsAdapter");
|
|
|
198
198
|
* bootstrap({
|
|
199
199
|
* modules: [ChatModule],
|
|
200
200
|
* adapters: [
|
|
201
|
-
*
|
|
201
|
+
* WsAdapter({ path: '/ws' }),
|
|
202
202
|
* ],
|
|
203
203
|
* })
|
|
204
204
|
* ```
|
|
@@ -206,230 +206,229 @@ const log = createLogger("WsAdapter");
|
|
|
206
206
|
* Clients connect to: `ws://localhost:3000/ws/chat`
|
|
207
207
|
* Messages are JSON: `{ "event": "send", "data": { "text": "hello" } }`
|
|
208
208
|
*/
|
|
209
|
-
|
|
210
|
-
name
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
this.maxPayload = options.maxPayload;
|
|
235
|
-
this.auth = options.auth;
|
|
236
|
-
this.userRoomPrefix = options.auth?.userRoomPrefix ?? "user:";
|
|
237
|
-
this.totalConnections = ref(0);
|
|
238
|
-
this.activeConnections = ref(0);
|
|
239
|
-
this.messagesReceived = ref(0);
|
|
240
|
-
this.messagesSent = ref(0);
|
|
241
|
-
this.wsErrors = ref(0);
|
|
242
|
-
}
|
|
243
|
-
/** Get a snapshot of WebSocket stats for DevTools */
|
|
244
|
-
getStats() {
|
|
245
|
-
const namespaceStats = {};
|
|
246
|
-
for (const [path, entry] of this.namespaces) namespaceStats[path] = {
|
|
247
|
-
connections: entry.sockets.size,
|
|
248
|
-
handlers: entry.handlers.length
|
|
249
|
-
};
|
|
250
|
-
return {
|
|
251
|
-
totalConnections: this.totalConnections.value,
|
|
252
|
-
activeConnections: this.activeConnections.value,
|
|
253
|
-
messagesReceived: this.messagesReceived.value,
|
|
254
|
-
messagesSent: this.messagesSent.value,
|
|
255
|
-
errors: this.wsErrors.value,
|
|
256
|
-
namespaces: namespaceStats,
|
|
257
|
-
rooms: this.roomManager.getAllRooms()
|
|
258
|
-
};
|
|
259
|
-
}
|
|
260
|
-
/** Room name used for per-user broadcasting. */
|
|
261
|
-
userRoom(userId) {
|
|
262
|
-
return this.userRoomPrefix + userId;
|
|
263
|
-
}
|
|
264
|
-
/** Broadcast a single event to every socket in `user:<id>` (across namespaces). */
|
|
265
|
-
broadcastToUser(userId, event, data) {
|
|
266
|
-
this.roomManager.broadcast(this.userRoom(userId), event, data);
|
|
267
|
-
}
|
|
268
|
-
buildUserBroadcaster() {
|
|
269
|
-
const self = this;
|
|
270
|
-
return {
|
|
271
|
-
roomFor: (id) => self.userRoom(id),
|
|
272
|
-
broadcastToUser: (id, event, data) => self.broadcastToUser(id, event, data),
|
|
273
|
-
toUser: (id) => ({ send: (event, data) => self.broadcastToUser(id, event, data) })
|
|
209
|
+
const WsAdapter = defineAdapter({
|
|
210
|
+
name: "WsAdapter",
|
|
211
|
+
defaults: {
|
|
212
|
+
path: "/ws",
|
|
213
|
+
heartbeatInterval: 3e4
|
|
214
|
+
},
|
|
215
|
+
build: (options) => {
|
|
216
|
+
const basePath = options.path;
|
|
217
|
+
const heartbeatInterval = options.heartbeatInterval;
|
|
218
|
+
const maxPayload = options.maxPayload;
|
|
219
|
+
const auth = options.auth;
|
|
220
|
+
const userRoomPrefix = options.auth?.userRoomPrefix ?? "user:";
|
|
221
|
+
let wss = null;
|
|
222
|
+
let container = null;
|
|
223
|
+
const namespaces = /* @__PURE__ */ new Map();
|
|
224
|
+
const roomManager = new RoomManager();
|
|
225
|
+
let heartbeatTimer = null;
|
|
226
|
+
const totalConnections = ref(0);
|
|
227
|
+
const activeConnections = ref(0);
|
|
228
|
+
const messagesReceived = ref(0);
|
|
229
|
+
const messagesSent = ref(0);
|
|
230
|
+
const wsErrors = ref(0);
|
|
231
|
+
const userRoom = (userId) => userRoomPrefix + userId;
|
|
232
|
+
const broadcastToUser = (userId, event, data) => {
|
|
233
|
+
roomManager.broadcast(userRoom(userId), event, data);
|
|
274
234
|
};
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
container.registerInstance(WS_ROOM_MANAGER, this.roomManager);
|
|
280
|
-
container.registerInstance(WS_USER_BROADCASTER, this.buildUserBroadcaster());
|
|
281
|
-
for (const controllerClass of wsControllerRegistry) {
|
|
282
|
-
const namespace = getClassMetaOrUndefined(WS_METADATA.WS_CONTROLLER, controllerClass);
|
|
283
|
-
if (namespace === void 0) continue;
|
|
284
|
-
const handlers = getClassMeta(WS_METADATA.WS_HANDLERS, controllerClass, []);
|
|
285
|
-
const fullPath = this.basePath + (namespace === "/" ? "" : namespace);
|
|
286
|
-
this.namespaces.set(fullPath, {
|
|
287
|
-
namespace,
|
|
288
|
-
controllerClass,
|
|
289
|
-
handlers,
|
|
290
|
-
sockets: /* @__PURE__ */ new Map(),
|
|
291
|
-
contexts: /* @__PURE__ */ new Map()
|
|
292
|
-
});
|
|
293
|
-
log.info(`Registered WS namespace: ${fullPath} (${controllerClass.name})`);
|
|
294
|
-
}
|
|
295
|
-
}
|
|
296
|
-
afterStart({ server }) {
|
|
297
|
-
if (!server) return;
|
|
298
|
-
this.wss = new WebSocketServer({
|
|
299
|
-
noServer: true,
|
|
300
|
-
maxPayload: this.maxPayload
|
|
301
|
-
});
|
|
302
|
-
server.on("upgrade", (request, socket, head) => {
|
|
303
|
-
const pathname = (request.url || "/").split("?")[0];
|
|
304
|
-
const entry = this.namespaces.get(pathname);
|
|
305
|
-
if (!entry) {
|
|
306
|
-
socket.write("HTTP/1.1 404 Not Found\r\n\r\n");
|
|
307
|
-
socket.destroy();
|
|
308
|
-
return;
|
|
309
|
-
}
|
|
310
|
-
this.wss.handleUpgrade(request, socket, head, (ws) => {
|
|
311
|
-
this.handleConnection(ws, entry, request);
|
|
312
|
-
});
|
|
235
|
+
const buildUserBroadcaster = () => ({
|
|
236
|
+
roomFor: (id) => userRoom(id),
|
|
237
|
+
broadcastToUser: (id, event, data) => broadcastToUser(id, event, data),
|
|
238
|
+
toUser: (id) => ({ send: (event, data) => broadcastToUser(id, event, data) })
|
|
313
239
|
});
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
240
|
+
const getStats = () => {
|
|
241
|
+
const namespaceStats = {};
|
|
242
|
+
for (const [path, entry] of namespaces) namespaceStats[path] = {
|
|
243
|
+
connections: entry.sockets.size,
|
|
244
|
+
handlers: entry.handlers.length
|
|
245
|
+
};
|
|
246
|
+
return {
|
|
247
|
+
totalConnections: totalConnections.value,
|
|
248
|
+
activeConnections: activeConnections.value,
|
|
249
|
+
messagesReceived: messagesReceived.value,
|
|
250
|
+
messagesSent: messagesSent.value,
|
|
251
|
+
errors: wsErrors.value,
|
|
252
|
+
namespaces: namespaceStats,
|
|
253
|
+
rooms: roomManager.getAllRooms()
|
|
254
|
+
};
|
|
255
|
+
};
|
|
256
|
+
const safeInvoke = (controller, method, ctx) => {
|
|
257
|
+
try {
|
|
258
|
+
const result = controller[method](ctx);
|
|
259
|
+
if (result instanceof Promise) result.catch((err) => {
|
|
260
|
+
log.error({ err }, `WS handler error in ${method}`);
|
|
261
|
+
});
|
|
262
|
+
} catch (err) {
|
|
263
|
+
log.error({ err }, `WS handler error in ${method}`);
|
|
322
264
|
}
|
|
323
|
-
}
|
|
324
|
-
const
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
}
|
|
336
|
-
handleConnection(ws, entry, request) {
|
|
337
|
-
const socketId = randomUUID();
|
|
338
|
-
ws.__alive = true;
|
|
339
|
-
entry.sockets.set(socketId, ws);
|
|
340
|
-
this.totalConnections.value++;
|
|
341
|
-
this.activeConnections.value++;
|
|
342
|
-
const ctx = new WsContext(ws, this.wss, this.roomManager, entry.sockets, socketId, entry.namespace, request);
|
|
343
|
-
entry.contexts.set(socketId, ctx);
|
|
344
|
-
const controller = this.container.resolve(entry.controllerClass);
|
|
345
|
-
ws.on("pong", () => {
|
|
346
|
-
ws.__alive = true;
|
|
347
|
-
});
|
|
348
|
-
let authed = this.auth === void 0;
|
|
349
|
-
(this.auth ? this.authenticate(ctx) : Promise.resolve(true)).then((ok) => {
|
|
350
|
-
if (!ok) return;
|
|
351
|
-
authed = true;
|
|
352
|
-
this.invokeHandlers(controller, entry.handlers, "connect", ctx);
|
|
353
|
-
});
|
|
354
|
-
ws.on("message", (raw) => {
|
|
355
|
-
if (!authed) return;
|
|
356
|
-
this.messagesReceived.value++;
|
|
265
|
+
};
|
|
266
|
+
const invokeHandlers = (controller, handlers, type, ctx) => {
|
|
267
|
+
for (const handler of handlers) if (handler.type === type) safeInvoke(controller, handler.handlerName, ctx);
|
|
268
|
+
};
|
|
269
|
+
/**
|
|
270
|
+
* Runs the configured auth hook against the upgrade request. Stashes the
|
|
271
|
+
* resolved user on the context (as `user`, plus mirrored keys) and, when
|
|
272
|
+
* `autoJoinUserRoom` is enabled, joins the socket to `user:<id>`.
|
|
273
|
+
* Returns `false` (and closes the socket with code 4401) on failure.
|
|
274
|
+
*/
|
|
275
|
+
const authenticate = async (ctx) => {
|
|
276
|
+
if (!auth) return true;
|
|
357
277
|
try {
|
|
358
|
-
const
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
ctx.send("error", { message: "Invalid message format: missing \"event\" field" });
|
|
363
|
-
return;
|
|
364
|
-
}
|
|
365
|
-
ctx.event = event;
|
|
366
|
-
ctx.data = data;
|
|
367
|
-
const handler = entry.handlers.find((h) => h.type === "message" && h.event === event);
|
|
368
|
-
if (handler) this.safeInvoke(controller, handler.handlerName, ctx);
|
|
369
|
-
else {
|
|
370
|
-
const catchAll = entry.handlers.find((h) => h.type === "message" && h.event === "*");
|
|
371
|
-
if (catchAll) this.safeInvoke(controller, catchAll.handlerName, ctx);
|
|
278
|
+
const user = await auth.resolveUser(ctx.request);
|
|
279
|
+
if (!user || !user.id) {
|
|
280
|
+
ctx.socket.close(4401, "Unauthorized");
|
|
281
|
+
return false;
|
|
372
282
|
}
|
|
373
|
-
|
|
374
|
-
ctx.
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
this.activeConnections.value--;
|
|
380
|
-
this.invokeHandlers(controller, entry.handlers, "disconnect", ctx);
|
|
381
|
-
this.roomManager.leaveAll(socketId);
|
|
382
|
-
entry.sockets.delete(socketId);
|
|
383
|
-
entry.contexts.delete(socketId);
|
|
384
|
-
});
|
|
385
|
-
ws.on("error", (err) => {
|
|
386
|
-
this.wsErrors.value++;
|
|
387
|
-
ctx.data = {
|
|
388
|
-
message: err.message,
|
|
389
|
-
name: err.name
|
|
390
|
-
};
|
|
391
|
-
this.invokeHandlers(controller, entry.handlers, "error", ctx);
|
|
392
|
-
});
|
|
393
|
-
}
|
|
394
|
-
/**
|
|
395
|
-
* Runs the configured auth hook against the upgrade request. Stashes the
|
|
396
|
-
* resolved user on the context (as `user`, plus mirrored keys) and, when
|
|
397
|
-
* `autoJoinUserRoom` is enabled, joins the socket to `user:<id>`.
|
|
398
|
-
* Returns `false` (and closes the socket with code 4401) on failure.
|
|
399
|
-
*/
|
|
400
|
-
async authenticate(ctx) {
|
|
401
|
-
const cfg = this.auth;
|
|
402
|
-
if (!cfg) return true;
|
|
403
|
-
try {
|
|
404
|
-
const user = await cfg.resolveUser(ctx.request);
|
|
405
|
-
if (!user || !user.id) {
|
|
283
|
+
ctx.set("user", user);
|
|
284
|
+
ctx.set("userId", user.id);
|
|
285
|
+
if (auth.autoJoinUserRoom !== false) ctx.join(userRoom(user.id));
|
|
286
|
+
return true;
|
|
287
|
+
} catch (err) {
|
|
288
|
+
log.warn("WS auth failed", err);
|
|
406
289
|
ctx.socket.close(4401, "Unauthorized");
|
|
407
290
|
return false;
|
|
408
291
|
}
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
420
|
-
|
|
421
|
-
}
|
|
422
|
-
safeInvoke(controller, method, ctx) {
|
|
423
|
-
try {
|
|
424
|
-
const result = controller[method](ctx);
|
|
425
|
-
if (result instanceof Promise) result.catch((err) => {
|
|
426
|
-
log.error({ err }, `WS handler error in ${method}`);
|
|
292
|
+
};
|
|
293
|
+
const handleConnection = (ws, entry, request) => {
|
|
294
|
+
const socketId = randomUUID();
|
|
295
|
+
ws.__alive = true;
|
|
296
|
+
entry.sockets.set(socketId, ws);
|
|
297
|
+
totalConnections.value++;
|
|
298
|
+
activeConnections.value++;
|
|
299
|
+
const ctx = new WsContext(ws, wss, roomManager, entry.sockets, socketId, entry.namespace, request);
|
|
300
|
+
entry.contexts.set(socketId, ctx);
|
|
301
|
+
const controller = container.resolve(entry.controllerClass);
|
|
302
|
+
ws.on("pong", () => {
|
|
303
|
+
ws.__alive = true;
|
|
427
304
|
});
|
|
428
|
-
|
|
429
|
-
|
|
430
|
-
|
|
305
|
+
let authed = auth === void 0;
|
|
306
|
+
(auth ? authenticate(ctx) : Promise.resolve(true)).then((ok) => {
|
|
307
|
+
if (!ok) return;
|
|
308
|
+
authed = true;
|
|
309
|
+
invokeHandlers(controller, entry.handlers, "connect", ctx);
|
|
310
|
+
});
|
|
311
|
+
ws.on("message", (raw) => {
|
|
312
|
+
if (!authed) return;
|
|
313
|
+
messagesReceived.value++;
|
|
314
|
+
try {
|
|
315
|
+
const parsed = JSON.parse(raw.toString());
|
|
316
|
+
const event = parsed.event;
|
|
317
|
+
const data = parsed.data;
|
|
318
|
+
if (!event || typeof event !== "string") {
|
|
319
|
+
ctx.send("error", { message: "Invalid message format: missing \"event\" field" });
|
|
320
|
+
return;
|
|
321
|
+
}
|
|
322
|
+
ctx.event = event;
|
|
323
|
+
ctx.data = data;
|
|
324
|
+
const handler = entry.handlers.find((h) => h.type === "message" && h.event === event);
|
|
325
|
+
if (handler) safeInvoke(controller, handler.handlerName, ctx);
|
|
326
|
+
else {
|
|
327
|
+
const catchAll = entry.handlers.find((h) => h.type === "message" && h.event === "*");
|
|
328
|
+
if (catchAll) safeInvoke(controller, catchAll.handlerName, ctx);
|
|
329
|
+
}
|
|
330
|
+
} catch {
|
|
331
|
+
ctx.data = { message: "Invalid JSON" };
|
|
332
|
+
invokeHandlers(controller, entry.handlers, "error", ctx);
|
|
333
|
+
}
|
|
334
|
+
});
|
|
335
|
+
ws.on("close", () => {
|
|
336
|
+
activeConnections.value--;
|
|
337
|
+
invokeHandlers(controller, entry.handlers, "disconnect", ctx);
|
|
338
|
+
roomManager.leaveAll(socketId);
|
|
339
|
+
entry.sockets.delete(socketId);
|
|
340
|
+
entry.contexts.delete(socketId);
|
|
341
|
+
});
|
|
342
|
+
ws.on("error", (err) => {
|
|
343
|
+
wsErrors.value++;
|
|
344
|
+
ctx.data = {
|
|
345
|
+
message: err.message,
|
|
346
|
+
name: err.name
|
|
347
|
+
};
|
|
348
|
+
invokeHandlers(controller, entry.handlers, "error", ctx);
|
|
349
|
+
});
|
|
350
|
+
};
|
|
351
|
+
return {
|
|
352
|
+
getStats,
|
|
353
|
+
userRoom,
|
|
354
|
+
broadcastToUser,
|
|
355
|
+
totalConnections,
|
|
356
|
+
activeConnections,
|
|
357
|
+
messagesReceived,
|
|
358
|
+
messagesSent,
|
|
359
|
+
wsErrors,
|
|
360
|
+
beforeStart({ container: containerArg }) {
|
|
361
|
+
container = containerArg;
|
|
362
|
+
container.registerInstance(WS_ADAPTER, {
|
|
363
|
+
getStats,
|
|
364
|
+
userRoom,
|
|
365
|
+
broadcastToUser,
|
|
366
|
+
totalConnections,
|
|
367
|
+
activeConnections,
|
|
368
|
+
messagesReceived,
|
|
369
|
+
messagesSent,
|
|
370
|
+
wsErrors
|
|
371
|
+
});
|
|
372
|
+
container.registerInstance(WS_ROOM_MANAGER, roomManager);
|
|
373
|
+
container.registerInstance(WS_USER_BROADCASTER, buildUserBroadcaster());
|
|
374
|
+
for (const controllerClass of wsControllerRegistry) {
|
|
375
|
+
const namespace = getClassMetaOrUndefined(WS_METADATA.WS_CONTROLLER, controllerClass);
|
|
376
|
+
if (namespace === void 0) continue;
|
|
377
|
+
const handlers = getClassMeta(WS_METADATA.WS_HANDLERS, controllerClass, []);
|
|
378
|
+
const fullPath = basePath + (namespace === "/" ? "" : namespace);
|
|
379
|
+
namespaces.set(fullPath, {
|
|
380
|
+
namespace,
|
|
381
|
+
controllerClass,
|
|
382
|
+
handlers,
|
|
383
|
+
sockets: /* @__PURE__ */ new Map(),
|
|
384
|
+
contexts: /* @__PURE__ */ new Map()
|
|
385
|
+
});
|
|
386
|
+
log.info(`Registered WS namespace: ${fullPath} (${controllerClass.name})`);
|
|
387
|
+
}
|
|
388
|
+
},
|
|
389
|
+
afterStart({ server }) {
|
|
390
|
+
if (!server) return;
|
|
391
|
+
wss = new WebSocketServer({
|
|
392
|
+
noServer: true,
|
|
393
|
+
maxPayload
|
|
394
|
+
});
|
|
395
|
+
server.on("upgrade", (request, socket, head) => {
|
|
396
|
+
const pathname = (request.url || "/").split("?")[0];
|
|
397
|
+
const entry = namespaces.get(pathname);
|
|
398
|
+
if (!entry) {
|
|
399
|
+
socket.write("HTTP/1.1 404 Not Found\r\n\r\n");
|
|
400
|
+
socket.destroy();
|
|
401
|
+
return;
|
|
402
|
+
}
|
|
403
|
+
wss.handleUpgrade(request, socket, head, (ws) => {
|
|
404
|
+
handleConnection(ws, entry, request);
|
|
405
|
+
});
|
|
406
|
+
});
|
|
407
|
+
if (heartbeatInterval > 0) heartbeatTimer = setInterval(() => {
|
|
408
|
+
for (const [, entry] of namespaces) for (const [, socket] of entry.sockets) {
|
|
409
|
+
if (socket.__alive === false) {
|
|
410
|
+
socket.terminate();
|
|
411
|
+
continue;
|
|
412
|
+
}
|
|
413
|
+
socket.__alive = false;
|
|
414
|
+
socket.ping();
|
|
415
|
+
}
|
|
416
|
+
}, heartbeatInterval);
|
|
417
|
+
const totalHandlers = Array.from(namespaces.values()).reduce((sum, e) => sum + e.handlers.length, 0);
|
|
418
|
+
log.info(`WebSocket ready — ${namespaces.size} namespace(s), ${totalHandlers} handler(s)`);
|
|
419
|
+
},
|
|
420
|
+
shutdown() {
|
|
421
|
+
if (heartbeatTimer) clearInterval(heartbeatTimer);
|
|
422
|
+
for (const [, entry] of namespaces) {
|
|
423
|
+
for (const [, socket] of entry.sockets) socket.close(1001, "Server shutting down");
|
|
424
|
+
entry.sockets.clear();
|
|
425
|
+
entry.contexts.clear();
|
|
426
|
+
}
|
|
427
|
+
wss?.close();
|
|
428
|
+
}
|
|
429
|
+
};
|
|
431
430
|
}
|
|
432
|
-
};
|
|
431
|
+
});
|
|
433
432
|
//#endregion
|
|
434
433
|
//#region src/decorators.ts
|
|
435
434
|
/**
|
package/dist/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.mjs","names":[],"sources":["../src/interfaces.ts","../src/ws-context.ts","../src/room-manager.ts","../src/ws-adapter.ts","../src/decorators.ts"],"sourcesContent":["import type { IncomingMessage } from 'node:http'\nimport { createToken } from '@forinda/kickjs'\nimport type { RoomManager } from './room-manager'\n\ntype Constructor = new (...args: any[]) => any\n\nexport const WS_METADATA = {\n WS_CONTROLLER: Symbol('kick:ws:controller'),\n WS_HANDLERS: Symbol('kick:ws:handlers'),\n} as const\n\nexport type WsHandlerType = 'connect' | 'disconnect' | 'message' | 'error'\n\nexport interface WsHandlerDefinition {\n type: WsHandlerType\n /** Event name — only for 'message' type */\n event?: string\n /** Method name on the controller class */\n handlerName: string\n}\n\n/**\n * Resolved principal returned from {@link WsAuthConfig.resolveUser}. Only `id`\n * is required — everything else is free-form metadata stashed on the\n * `WsContext` under `user:<field>` keys for handler access.\n */\nexport interface WsAuthenticatedUser {\n id: string\n [key: string]: unknown\n}\n\nexport interface WsAuthConfig {\n /**\n * Resolve a user from the upgrade request. Called once per socket during\n * handshake, before any `@OnConnect` handler fires. Return `null` or throw\n * to reject the socket with HTTP 401.\n */\n resolveUser: (\n request: IncomingMessage,\n ) => Promise<WsAuthenticatedUser | null> | WsAuthenticatedUser | null\n /**\n * When true, each authenticated socket auto-joins `user:<id>` immediately\n * after `resolveUser` resolves. Pairs with {@link WsUserBroadcaster}.\n */\n autoJoinUserRoom?: boolean\n /**\n * Room name prefix for per-user broadcasting (default: `'user:'`).\n * Must match what `@forinda/kickjs-ws`'s `WsUserBroadcaster` targets.\n */\n userRoomPrefix?: string\n}\n\nexport interface WsAdapterOptions {\n /** Base path for WebSocket upgrade (default: '/ws') */\n path?: string\n /** Heartbeat ping interval in ms (default: 30000). Set to 0 to disable. */\n heartbeatInterval?: number\n /** Maximum message payload size in bytes */\n maxPayload?: number\n /** Optional authenticated-handshake configuration. */\n auth?: WsAuthConfig\n}\n\n/**\n * Per-user broadcasting across all WS namespaces. Registered on the DI\n * container when {@link WsAdapterOptions.auth} is configured, but the\n * underlying room (`user:<id>`) can also be joined manually by any\n * controller — the helper works either way.\n */\nexport interface WsUserBroadcaster {\n /** Send a single event to every socket bound to this user. */\n toUser(userId: string): { send(event: string, data: unknown): void }\n /** Convenience — `toUser(id).send(event, data)` in one call. */\n broadcastToUser(userId: string, event: string, data: unknown): void\n /** Room name for a given user (respects `userRoomPrefix`). */\n roomFor(userId: string): string\n}\n\n/** DI token for the live {@link WsAdapter} instance. */\nexport const WS_ADAPTER = createToken<unknown>('WsAdapter')\n/** DI token for the shared {@link RoomManager}. */\nexport const WS_ROOM_MANAGER = createToken<RoomManager>('WsRoomManager')\n/** DI token for the per-user broadcaster helper. */\nexport const WS_USER_BROADCASTER = createToken<WsUserBroadcaster>('WsUserBroadcaster')\n\n/** Registry of all @WsController classes — populated at decorator time */\nexport const wsControllerRegistry = new Set<Constructor>()\n","import type { IncomingMessage } from 'node:http'\nimport type { WebSocket, WebSocketServer } from 'ws'\nimport type { RoomManager } from './room-manager'\n\n/**\n * Context object passed to WebSocket handler methods.\n * Analogous to RequestContext for HTTP controllers.\n *\n * @example\n * ```ts\n * @OnMessage('chat:send')\n * handleSend(ctx: WsContext) {\n * console.log(ctx.data) // parsed message payload\n * ctx.send('chat:ack', { ok: true })\n * ctx.broadcast('chat:receive', ctx.data)\n * ctx.join('room-1')\n * ctx.to('room-1').send('chat:receive', ctx.data)\n * }\n * ```\n */\nexport class WsContext {\n /** Unique connection ID */\n readonly id: string\n /** Parsed message payload (set for @OnMessage handlers) */\n data: any\n /** Event name from the message envelope (set for @OnMessage handlers) */\n event: string\n /** The namespace this connection belongs to */\n readonly namespace: string\n /** The HTTP upgrade request — available from @OnConnect onward. Use to read\n * cookies, headers, query string, and client IP for authenticated handshakes. */\n readonly request: IncomingMessage\n\n private metadata = new Map<string, any>()\n\n constructor(\n readonly socket: WebSocket,\n readonly server: WebSocketServer,\n private readonly roomManager: RoomManager,\n private readonly namespaceSockets: Map<string, WebSocket>,\n id: string,\n namespace: string,\n request: IncomingMessage,\n ) {\n this.id = id\n this.namespace = namespace\n this.request = request\n this.data = null\n this.event = ''\n }\n\n /** Parsed cookies from the upgrade request (raw `Cookie` header parse). */\n get cookies(): Record<string, string> {\n const header = this.request.headers.cookie\n if (!header) return {}\n const out: Record<string, string> = {}\n for (const part of header.split(';')) {\n const idx = part.indexOf('=')\n if (idx === -1) continue\n const k = part.slice(0, idx).trim()\n const v = part.slice(idx + 1).trim()\n if (k) out[k] = decodeURIComponent(v)\n }\n return out\n }\n\n /** Get a metadata value */\n get<T = any>(key: string): T | undefined {\n return this.metadata.get(key)\n }\n\n /** Set a metadata value (persists for the lifetime of the connection) */\n set(key: string, value: any): void {\n this.metadata.set(key, value)\n }\n\n /** Send a message to this socket */\n send(event: string, data: any): void {\n if (this.socket.readyState === this.socket.OPEN) {\n this.socket.send(JSON.stringify({ event, data }))\n }\n }\n\n /** Send to all sockets in the same namespace except this one */\n broadcast(event: string, data: any): void {\n const message = JSON.stringify({ event, data })\n for (const [id, socket] of this.namespaceSockets) {\n if (id !== this.id && socket.readyState === socket.OPEN) {\n socket.send(message)\n }\n }\n }\n\n /** Send to all sockets in the same namespace including this one */\n broadcastAll(event: string, data: any): void {\n const message = JSON.stringify({ event, data })\n for (const [, socket] of this.namespaceSockets) {\n if (socket.readyState === socket.OPEN) {\n socket.send(message)\n }\n }\n }\n\n /** Join a room */\n join(room: string): void {\n this.roomManager.join(this.id, this.socket, room)\n }\n\n /** Leave a room */\n leave(room: string): void {\n this.roomManager.leave(this.id, room)\n }\n\n /** Get all rooms this socket is in */\n rooms(): string[] {\n return this.roomManager.getRooms(this.id)\n }\n\n /** Send to all sockets in a room */\n to(room: string): { send(event: string, data: any): void } {\n return {\n send: (event: string, data: any) => {\n this.roomManager.broadcast(room, event, data)\n },\n }\n }\n}\n","import type { WebSocket } from 'ws'\n\n/**\n * Manages WebSocket room membership and broadcasting.\n * Standalone from ws/socket.io — can be swapped for socket.io's built-in rooms.\n */\nexport class RoomManager {\n /** socketId → set of room names */\n private socketRooms = new Map<string, Set<string>>()\n /** room name → set of { socketId, socket } */\n private roomSockets = new Map<string, Map<string, WebSocket>>()\n\n join(socketId: string, socket: WebSocket, room: string): void {\n if (!this.socketRooms.has(socketId)) {\n this.socketRooms.set(socketId, new Set())\n }\n this.socketRooms.get(socketId)!.add(room)\n\n if (!this.roomSockets.has(room)) {\n this.roomSockets.set(room, new Map())\n }\n this.roomSockets.get(room)!.set(socketId, socket)\n }\n\n leave(socketId: string, room: string): void {\n this.socketRooms.get(socketId)?.delete(room)\n this.roomSockets.get(room)?.delete(socketId)\n\n // Clean up empty rooms\n if (this.roomSockets.get(room)?.size === 0) {\n this.roomSockets.delete(room)\n }\n }\n\n /** Remove socket from all rooms (called on disconnect) */\n leaveAll(socketId: string): void {\n const rooms = this.socketRooms.get(socketId)\n if (rooms) {\n for (const room of rooms) {\n this.roomSockets.get(room)?.delete(socketId)\n if (this.roomSockets.get(room)?.size === 0) {\n this.roomSockets.delete(room)\n }\n }\n }\n this.socketRooms.delete(socketId)\n }\n\n getRooms(socketId: string): string[] {\n return Array.from(this.socketRooms.get(socketId) ?? [])\n }\n\n getSockets(room: string): Map<string, WebSocket> {\n return this.roomSockets.get(room) ?? new Map()\n }\n\n /** Get all rooms with their member counts */\n getAllRooms(): Record<string, number> {\n const result: Record<string, number> = {}\n for (const [room, sockets] of this.roomSockets) {\n result[room] = sockets.size\n }\n return result\n }\n\n /** Broadcast to all sockets in a room, optionally excluding one */\n broadcast(room: string, event: string, data: any, excludeId?: string): void {\n const sockets = this.roomSockets.get(room)\n if (!sockets) return\n\n const message = JSON.stringify({ event, data })\n for (const [id, socket] of sockets) {\n if (id !== excludeId && socket.readyState === socket.OPEN) {\n socket.send(message)\n }\n }\n }\n}\n","import { randomUUID } from 'node:crypto'\nimport { WebSocketServer, type WebSocket } from 'ws'\nimport type { IncomingMessage } from 'node:http'\nimport type { Duplex } from 'node:stream'\nimport {\n type AppAdapter,\n type AdapterContext,\n type Container,\n createLogger,\n ref,\n type Ref,\n getClassMetaOrUndefined,\n getClassMeta,\n} from '@forinda/kickjs'\nimport {\n WS_ADAPTER,\n WS_METADATA,\n WS_ROOM_MANAGER,\n WS_USER_BROADCASTER,\n wsControllerRegistry,\n type WsAdapterOptions,\n type WsAuthConfig,\n type WsHandlerDefinition,\n type WsUserBroadcaster,\n} from './interfaces'\nimport { WsContext } from './ws-context'\nimport { RoomManager } from './room-manager'\n\nconst log = createLogger('WsAdapter')\n\ninterface NamespaceEntry {\n namespace: string\n controllerClass: any\n handlers: WsHandlerDefinition[]\n sockets: Map<string, WebSocket>\n contexts: Map<string, WsContext>\n}\n\n/**\n * WebSocket adapter for KickJS. Attaches to the HTTP server and routes\n * WebSocket connections to @WsController classes based on namespace paths.\n *\n * @example\n * ```ts\n * import { WsAdapter } from '@forinda/kickjs-ws'\n *\n * bootstrap({\n * modules: [ChatModule],\n * adapters: [\n * new WsAdapter({ path: '/ws' }),\n * ],\n * })\n * ```\n *\n * Clients connect to: `ws://localhost:3000/ws/chat`\n * Messages are JSON: `{ \"event\": \"send\", \"data\": { \"text\": \"hello\" } }`\n */\nexport class WsAdapter implements AppAdapter {\n readonly name = 'WsAdapter'\n\n private basePath: string\n private heartbeatInterval: number\n private maxPayload: number | undefined\n private auth?: WsAuthConfig\n private userRoomPrefix: string\n private wss: WebSocketServer | null = null\n private container: Container | null = null\n private namespaces = new Map<string, NamespaceEntry>()\n private roomManager = new RoomManager()\n private heartbeatTimer: ReturnType<typeof setInterval> | null = null\n\n // ── Reactive Stats (exposed for DevToolsAdapter) ─────────────────────\n /** Total WebSocket connections ever opened */\n readonly totalConnections: Ref<number>\n /** Currently active connections */\n readonly activeConnections: Ref<number>\n /** Total messages received */\n readonly messagesReceived: Ref<number>\n /** Total messages sent */\n readonly messagesSent: Ref<number>\n /** Total errors */\n readonly wsErrors: Ref<number>\n\n constructor(options: WsAdapterOptions = {}) {\n this.basePath = options.path ?? '/ws'\n this.heartbeatInterval = options.heartbeatInterval ?? 30000\n this.maxPayload = options.maxPayload\n this.auth = options.auth\n this.userRoomPrefix = options.auth?.userRoomPrefix ?? 'user:'\n\n this.totalConnections = ref(0)\n this.activeConnections = ref(0)\n this.messagesReceived = ref(0)\n this.messagesSent = ref(0)\n this.wsErrors = ref(0)\n }\n\n /** Get a snapshot of WebSocket stats for DevTools */\n getStats() {\n const namespaceStats: Record<string, { connections: number; handlers: number }> = {}\n for (const [path, entry] of this.namespaces) {\n namespaceStats[path] = {\n connections: entry.sockets.size,\n handlers: entry.handlers.length,\n }\n }\n return {\n totalConnections: this.totalConnections.value,\n activeConnections: this.activeConnections.value,\n messagesReceived: this.messagesReceived.value,\n messagesSent: this.messagesSent.value,\n errors: this.wsErrors.value,\n namespaces: namespaceStats,\n rooms: this.roomManager.getAllRooms(),\n }\n }\n\n /** Room name used for per-user broadcasting. */\n userRoom(userId: string): string {\n return this.userRoomPrefix + userId\n }\n\n /** Broadcast a single event to every socket in `user:<id>` (across namespaces). */\n broadcastToUser(userId: string, event: string, data: unknown): void {\n this.roomManager.broadcast(this.userRoom(userId), event, data)\n }\n\n private buildUserBroadcaster(): WsUserBroadcaster {\n const self = this\n return {\n roomFor: (id) => self.userRoom(id),\n broadcastToUser: (id, event, data) => self.broadcastToUser(id, event, data),\n toUser: (id) => ({\n send: (event, data) => self.broadcastToUser(id, event, data),\n }),\n }\n }\n\n beforeStart({ container }: AdapterContext): void {\n this.container = container\n\n container.registerInstance(WS_ADAPTER, this)\n container.registerInstance(WS_ROOM_MANAGER, this.roomManager)\n container.registerInstance(WS_USER_BROADCASTER, this.buildUserBroadcaster())\n\n // Discover all @WsController classes and build routing table\n for (const controllerClass of wsControllerRegistry) {\n const namespace = getClassMetaOrUndefined<string>(WS_METADATA.WS_CONTROLLER, controllerClass)\n if (namespace === undefined) continue\n\n const handlers = getClassMeta<WsHandlerDefinition[]>(\n WS_METADATA.WS_HANDLERS,\n controllerClass,\n [],\n )\n\n const fullPath = this.basePath + (namespace === '/' ? '' : namespace)\n\n this.namespaces.set(fullPath, {\n namespace,\n controllerClass,\n handlers,\n sockets: new Map(),\n contexts: new Map(),\n })\n\n log.info(`Registered WS namespace: ${fullPath} (${controllerClass.name})`)\n }\n }\n\n afterStart({ server }: AdapterContext): void {\n if (!server) return\n\n this.wss = new WebSocketServer({\n noServer: true,\n maxPayload: this.maxPayload,\n })\n\n // Handle upgrade requests — route to correct namespace\n server.on('upgrade', (request: IncomingMessage, socket: Duplex, head: Buffer) => {\n const url = request.url || '/'\n // Parse pathname without relying on host header\n const pathname = url.split('?')[0]\n\n const entry = this.namespaces.get(pathname)\n if (!entry) {\n socket.write('HTTP/1.1 404 Not Found\\r\\n\\r\\n')\n socket.destroy()\n return\n }\n\n this.wss!.handleUpgrade(request, socket, head, (ws) => {\n this.handleConnection(ws, entry, request)\n })\n })\n\n // Heartbeat ping/pong\n if (this.heartbeatInterval > 0) {\n this.heartbeatTimer = setInterval(() => {\n for (const [, entry] of this.namespaces) {\n for (const [, socket] of entry.sockets) {\n if ((socket as any).__alive === false) {\n socket.terminate()\n continue\n }\n ;(socket as any).__alive = false\n socket.ping()\n }\n }\n }, this.heartbeatInterval)\n }\n\n const totalHandlers = Array.from(this.namespaces.values()).reduce(\n (sum, e) => sum + e.handlers.length,\n 0,\n )\n log.info(`WebSocket ready — ${this.namespaces.size} namespace(s), ${totalHandlers} handler(s)`)\n }\n\n shutdown(): void {\n if (this.heartbeatTimer) {\n clearInterval(this.heartbeatTimer)\n }\n\n // Close all connections\n for (const [, entry] of this.namespaces) {\n for (const [, socket] of entry.sockets) {\n socket.close(1001, 'Server shutting down')\n }\n entry.sockets.clear()\n entry.contexts.clear()\n }\n\n this.wss?.close()\n }\n\n // ── Internal ───────────────────────────────────────────────────────────\n\n private handleConnection(ws: WebSocket, entry: NamespaceEntry, request: IncomingMessage): void {\n const socketId = randomUUID()\n ;(ws as any).__alive = true\n\n entry.sockets.set(socketId, ws)\n this.totalConnections.value++\n this.activeConnections.value++\n\n const ctx = new WsContext(\n ws,\n this.wss!,\n this.roomManager,\n entry.sockets,\n socketId,\n entry.namespace,\n request,\n )\n entry.contexts.set(socketId, ctx)\n\n // Resolve controller from DI\n const controller = this.container!.resolve(entry.controllerClass)\n\n // Pong handler for heartbeat\n ws.on('pong', () => {\n ;(ws as any).__alive = true\n })\n\n // Authenticated handshake (optional). Messages received before auth\n // resolves are buffered on the socket; we gate dispatch on `authed`.\n let authed = this.auth === undefined\n const authPromise = this.auth ? this.authenticate(ctx) : Promise.resolve(true)\n\n authPromise.then((ok) => {\n if (!ok) return\n authed = true\n // @OnConnect handlers\n this.invokeHandlers(controller, entry.handlers, 'connect', ctx)\n })\n\n // Message handler\n ws.on('message', (raw: Buffer | string) => {\n if (!authed) return\n this.messagesReceived.value++\n try {\n const parsed = JSON.parse(raw.toString())\n const event = parsed.event as string\n const data = parsed.data\n\n if (!event || typeof event !== 'string') {\n ctx.send('error', { message: 'Invalid message format: missing \"event\" field' })\n return\n }\n\n ctx.event = event\n ctx.data = data\n\n // Find matching @OnMessage handler\n const handler = entry.handlers.find((h) => h.type === 'message' && h.event === event)\n\n if (handler) {\n this.safeInvoke(controller, handler.handlerName, ctx)\n } else {\n // Try catch-all @OnMessage('*')\n const catchAll = entry.handlers.find((h) => h.type === 'message' && h.event === '*')\n if (catchAll) {\n this.safeInvoke(controller, catchAll.handlerName, ctx)\n }\n }\n } catch {\n ctx.data = { message: 'Invalid JSON' }\n this.invokeHandlers(controller, entry.handlers, 'error', ctx)\n }\n })\n\n // Close handler\n ws.on('close', () => {\n this.activeConnections.value--\n this.invokeHandlers(controller, entry.handlers, 'disconnect', ctx)\n this.roomManager.leaveAll(socketId)\n entry.sockets.delete(socketId)\n entry.contexts.delete(socketId)\n })\n\n // Error handler\n ws.on('error', (err: Error) => {\n this.wsErrors.value++\n ctx.data = { message: err.message, name: err.name }\n this.invokeHandlers(controller, entry.handlers, 'error', ctx)\n })\n }\n\n /**\n * Runs the configured auth hook against the upgrade request. Stashes the\n * resolved user on the context (as `user`, plus mirrored keys) and, when\n * `autoJoinUserRoom` is enabled, joins the socket to `user:<id>`.\n * Returns `false` (and closes the socket with code 4401) on failure.\n */\n private async authenticate(ctx: WsContext): Promise<boolean> {\n const cfg = this.auth\n if (!cfg) return true\n try {\n const user = await cfg.resolveUser(ctx.request)\n if (!user || !user.id) {\n ctx.socket.close(4401, 'Unauthorized')\n return false\n }\n ctx.set('user', user)\n ctx.set('userId', user.id)\n if (cfg.autoJoinUserRoom !== false) {\n ctx.join(this.userRoom(user.id))\n }\n return true\n } catch (err) {\n log.warn('WS auth failed', err)\n ctx.socket.close(4401, 'Unauthorized')\n return false\n }\n }\n\n private invokeHandlers(\n controller: any,\n handlers: WsHandlerDefinition[],\n type: WsHandlerDefinition['type'],\n ctx: WsContext,\n ): void {\n for (const handler of handlers) {\n if (handler.type === type) {\n this.safeInvoke(controller, handler.handlerName, ctx)\n }\n }\n }\n\n private safeInvoke(controller: any, method: string, ctx: WsContext): void {\n try {\n const result = controller[method](ctx)\n if (result instanceof Promise) {\n result.catch((err: Error) => {\n log.error({ err }, `WS handler error in ${method}`)\n })\n }\n } catch (err) {\n log.error({ err }, `WS handler error in ${method}`)\n }\n }\n}\n","import { Service, setClassMeta, pushClassMeta } from '@forinda/kickjs'\nimport { WS_METADATA, wsControllerRegistry, type WsHandlerDefinition } from './interfaces'\n\n/**\n * Mark a class as a WebSocket controller with a namespace path.\n * Registers the class in the DI container and the WS controller registry.\n *\n * @example\n * ```ts\n * @WsController('/chat')\n * export class ChatController {\n * @OnConnect()\n * handleConnect(ctx: WsContext) { }\n *\n * @OnMessage('send')\n * handleSend(ctx: WsContext) { }\n * }\n * ```\n */\nexport function WsController(namespace?: string): ClassDecorator {\n return (target: any) => {\n Service()(target)\n setClassMeta(WS_METADATA.WS_CONTROLLER, namespace || '/', target)\n wsControllerRegistry.add(target)\n }\n}\n\nfunction createWsHandlerDecorator(type: WsHandlerDefinition['type'], event?: string) {\n return (): MethodDecorator => {\n return (target, propertyKey) => {\n pushClassMeta<WsHandlerDefinition>(WS_METADATA.WS_HANDLERS, target.constructor, {\n type,\n event,\n handlerName: propertyKey as string,\n })\n }\n }\n}\n\n/**\n * Handle new WebSocket connections.\n *\n * @example\n * ```ts\n * @OnConnect()\n * handleConnect(ctx: WsContext) {\n * console.log(`Client ${ctx.id} connected`)\n * }\n * ```\n */\nexport const OnConnect = createWsHandlerDecorator('connect')\n\n/**\n * Handle WebSocket disconnections.\n *\n * @example\n * ```ts\n * @OnDisconnect()\n * handleDisconnect(ctx: WsContext) {\n * console.log(`Client ${ctx.id} disconnected`)\n * }\n * ```\n */\nexport const OnDisconnect = createWsHandlerDecorator('disconnect')\n\n/**\n * Handle WebSocket errors.\n *\n * @example\n * ```ts\n * @OnError()\n * handleError(ctx: WsContext) {\n * console.error('WS error:', ctx.data)\n * }\n * ```\n */\nexport const OnError = createWsHandlerDecorator('error')\n\n/**\n * Handle a specific WebSocket message event.\n * Use '*' as a catch-all for unmatched events.\n *\n * Messages must be JSON: `{ \"event\": \"chat:send\", \"data\": { ... } }`\n *\n * @example\n * ```ts\n * @OnMessage('chat:send')\n * handleChatSend(ctx: WsContext) {\n * ctx.broadcast('chat:receive', ctx.data)\n * }\n *\n * @OnMessage('*')\n * handleUnknown(ctx: WsContext) {\n * ctx.send('error', { message: `Unknown event: ${ctx.event}` })\n * }\n * ```\n */\nexport function OnMessage(event: string): MethodDecorator {\n return (target, propertyKey) => {\n pushClassMeta<WsHandlerDefinition>(WS_METADATA.WS_HANDLERS, target.constructor, {\n type: 'message',\n event,\n handlerName: propertyKey as string,\n })\n }\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.mjs","names":[],"sources":["../src/interfaces.ts","../src/ws-context.ts","../src/room-manager.ts","../src/ws-adapter.ts","../src/decorators.ts"],"sourcesContent":["import type { IncomingMessage } from 'node:http'\nimport { createToken } from '@forinda/kickjs'\nimport type { RoomManager } from './room-manager'\n\ntype Constructor = new (...args: any[]) => any\n\nexport const WS_METADATA = {\n WS_CONTROLLER: Symbol('kick:ws:controller'),\n WS_HANDLERS: Symbol('kick:ws:handlers'),\n} as const\n\nexport type WsHandlerType = 'connect' | 'disconnect' | 'message' | 'error'\n\nexport interface WsHandlerDefinition {\n type: WsHandlerType\n /** Event name — only for 'message' type */\n event?: string\n /** Method name on the controller class */\n handlerName: string\n}\n\n/**\n * Resolved principal returned from {@link WsAuthConfig.resolveUser}. Only `id`\n * is required — everything else is free-form metadata stashed on the\n * `WsContext` under `user:<field>` keys for handler access.\n */\nexport interface WsAuthenticatedUser {\n id: string\n [key: string]: unknown\n}\n\nexport interface WsAuthConfig {\n /**\n * Resolve a user from the upgrade request. Called once per socket during\n * handshake, before any `@OnConnect` handler fires. Return `null` or throw\n * to reject the socket with HTTP 401.\n */\n resolveUser: (\n request: IncomingMessage,\n ) => Promise<WsAuthenticatedUser | null> | WsAuthenticatedUser | null\n /**\n * When true, each authenticated socket auto-joins `user:<id>` immediately\n * after `resolveUser` resolves. Pairs with {@link WsUserBroadcaster}.\n */\n autoJoinUserRoom?: boolean\n /**\n * Room name prefix for per-user broadcasting (default: `'user:'`).\n * Must match what `@forinda/kickjs-ws`'s `WsUserBroadcaster` targets.\n */\n userRoomPrefix?: string\n}\n\nexport interface WsAdapterOptions {\n /** Base path for WebSocket upgrade (default: '/ws') */\n path?: string\n /** Heartbeat ping interval in ms (default: 30000). Set to 0 to disable. */\n heartbeatInterval?: number\n /** Maximum message payload size in bytes */\n maxPayload?: number\n /** Optional authenticated-handshake configuration. */\n auth?: WsAuthConfig\n}\n\n/**\n * Per-user broadcasting across all WS namespaces. Registered on the DI\n * container when {@link WsAdapterOptions.auth} is configured, but the\n * underlying room (`user:<id>`) can also be joined manually by any\n * controller — the helper works either way.\n */\nexport interface WsUserBroadcaster {\n /** Send a single event to every socket bound to this user. */\n toUser(userId: string): { send(event: string, data: unknown): void }\n /** Convenience — `toUser(id).send(event, data)` in one call. */\n broadcastToUser(userId: string, event: string, data: unknown): void\n /** Room name for a given user (respects `userRoomPrefix`). */\n roomFor(userId: string): string\n}\n\n/** DI token for the live {@link WsAdapter} instance. */\nexport const WS_ADAPTER = createToken<unknown>('WsAdapter')\n/** DI token for the shared {@link RoomManager}. */\nexport const WS_ROOM_MANAGER = createToken<RoomManager>('WsRoomManager')\n/** DI token for the per-user broadcaster helper. */\nexport const WS_USER_BROADCASTER = createToken<WsUserBroadcaster>('WsUserBroadcaster')\n\n/** Registry of all @WsController classes — populated at decorator time */\nexport const wsControllerRegistry = new Set<Constructor>()\n","import type { IncomingMessage } from 'node:http'\nimport type { WebSocket, WebSocketServer } from 'ws'\nimport type { RoomManager } from './room-manager'\n\n/**\n * Context object passed to WebSocket handler methods.\n * Analogous to RequestContext for HTTP controllers.\n *\n * @example\n * ```ts\n * @OnMessage('chat:send')\n * handleSend(ctx: WsContext) {\n * console.log(ctx.data) // parsed message payload\n * ctx.send('chat:ack', { ok: true })\n * ctx.broadcast('chat:receive', ctx.data)\n * ctx.join('room-1')\n * ctx.to('room-1').send('chat:receive', ctx.data)\n * }\n * ```\n */\nexport class WsContext {\n /** Unique connection ID */\n readonly id: string\n /** Parsed message payload (set for @OnMessage handlers) */\n data: any\n /** Event name from the message envelope (set for @OnMessage handlers) */\n event: string\n /** The namespace this connection belongs to */\n readonly namespace: string\n /** The HTTP upgrade request — available from @OnConnect onward. Use to read\n * cookies, headers, query string, and client IP for authenticated handshakes. */\n readonly request: IncomingMessage\n\n private metadata = new Map<string, any>()\n\n constructor(\n readonly socket: WebSocket,\n readonly server: WebSocketServer,\n private readonly roomManager: RoomManager,\n private readonly namespaceSockets: Map<string, WebSocket>,\n id: string,\n namespace: string,\n request: IncomingMessage,\n ) {\n this.id = id\n this.namespace = namespace\n this.request = request\n this.data = null\n this.event = ''\n }\n\n /** Parsed cookies from the upgrade request (raw `Cookie` header parse). */\n get cookies(): Record<string, string> {\n const header = this.request.headers.cookie\n if (!header) return {}\n const out: Record<string, string> = {}\n for (const part of header.split(';')) {\n const idx = part.indexOf('=')\n if (idx === -1) continue\n const k = part.slice(0, idx).trim()\n const v = part.slice(idx + 1).trim()\n if (k) out[k] = decodeURIComponent(v)\n }\n return out\n }\n\n /** Get a metadata value */\n get<T = any>(key: string): T | undefined {\n return this.metadata.get(key)\n }\n\n /** Set a metadata value (persists for the lifetime of the connection) */\n set(key: string, value: any): void {\n this.metadata.set(key, value)\n }\n\n /** Send a message to this socket */\n send(event: string, data: any): void {\n if (this.socket.readyState === this.socket.OPEN) {\n this.socket.send(JSON.stringify({ event, data }))\n }\n }\n\n /** Send to all sockets in the same namespace except this one */\n broadcast(event: string, data: any): void {\n const message = JSON.stringify({ event, data })\n for (const [id, socket] of this.namespaceSockets) {\n if (id !== this.id && socket.readyState === socket.OPEN) {\n socket.send(message)\n }\n }\n }\n\n /** Send to all sockets in the same namespace including this one */\n broadcastAll(event: string, data: any): void {\n const message = JSON.stringify({ event, data })\n for (const [, socket] of this.namespaceSockets) {\n if (socket.readyState === socket.OPEN) {\n socket.send(message)\n }\n }\n }\n\n /** Join a room */\n join(room: string): void {\n this.roomManager.join(this.id, this.socket, room)\n }\n\n /** Leave a room */\n leave(room: string): void {\n this.roomManager.leave(this.id, room)\n }\n\n /** Get all rooms this socket is in */\n rooms(): string[] {\n return this.roomManager.getRooms(this.id)\n }\n\n /** Send to all sockets in a room */\n to(room: string): { send(event: string, data: any): void } {\n return {\n send: (event: string, data: any) => {\n this.roomManager.broadcast(room, event, data)\n },\n }\n }\n}\n","import type { WebSocket } from 'ws'\n\n/**\n * Manages WebSocket room membership and broadcasting.\n * Standalone from ws/socket.io — can be swapped for socket.io's built-in rooms.\n */\nexport class RoomManager {\n /** socketId → set of room names */\n private socketRooms = new Map<string, Set<string>>()\n /** room name → set of { socketId, socket } */\n private roomSockets = new Map<string, Map<string, WebSocket>>()\n\n join(socketId: string, socket: WebSocket, room: string): void {\n if (!this.socketRooms.has(socketId)) {\n this.socketRooms.set(socketId, new Set())\n }\n this.socketRooms.get(socketId)!.add(room)\n\n if (!this.roomSockets.has(room)) {\n this.roomSockets.set(room, new Map())\n }\n this.roomSockets.get(room)!.set(socketId, socket)\n }\n\n leave(socketId: string, room: string): void {\n this.socketRooms.get(socketId)?.delete(room)\n this.roomSockets.get(room)?.delete(socketId)\n\n // Clean up empty rooms\n if (this.roomSockets.get(room)?.size === 0) {\n this.roomSockets.delete(room)\n }\n }\n\n /** Remove socket from all rooms (called on disconnect) */\n leaveAll(socketId: string): void {\n const rooms = this.socketRooms.get(socketId)\n if (rooms) {\n for (const room of rooms) {\n this.roomSockets.get(room)?.delete(socketId)\n if (this.roomSockets.get(room)?.size === 0) {\n this.roomSockets.delete(room)\n }\n }\n }\n this.socketRooms.delete(socketId)\n }\n\n getRooms(socketId: string): string[] {\n return Array.from(this.socketRooms.get(socketId) ?? [])\n }\n\n getSockets(room: string): Map<string, WebSocket> {\n return this.roomSockets.get(room) ?? new Map()\n }\n\n /** Get all rooms with their member counts */\n getAllRooms(): Record<string, number> {\n const result: Record<string, number> = {}\n for (const [room, sockets] of this.roomSockets) {\n result[room] = sockets.size\n }\n return result\n }\n\n /** Broadcast to all sockets in a room, optionally excluding one */\n broadcast(room: string, event: string, data: any, excludeId?: string): void {\n const sockets = this.roomSockets.get(room)\n if (!sockets) return\n\n const message = JSON.stringify({ event, data })\n for (const [id, socket] of sockets) {\n if (id !== excludeId && socket.readyState === socket.OPEN) {\n socket.send(message)\n }\n }\n }\n}\n","import { randomUUID } from 'node:crypto'\nimport { WebSocketServer, type WebSocket } from 'ws'\nimport type { IncomingMessage } from 'node:http'\nimport type { Duplex } from 'node:stream'\nimport {\n defineAdapter,\n type Container,\n createLogger,\n ref,\n type Ref,\n getClassMetaOrUndefined,\n getClassMeta,\n} from '@forinda/kickjs'\nimport {\n WS_ADAPTER,\n WS_METADATA,\n WS_ROOM_MANAGER,\n WS_USER_BROADCASTER,\n wsControllerRegistry,\n type WsAdapterOptions,\n type WsHandlerDefinition,\n type WsUserBroadcaster,\n} from './interfaces'\nimport { WsContext } from './ws-context'\nimport { RoomManager } from './room-manager'\n\nconst log = createLogger('WsAdapter')\n\ninterface NamespaceEntry {\n namespace: string\n controllerClass: any\n handlers: WsHandlerDefinition[]\n sockets: Map<string, WebSocket>\n contexts: Map<string, WsContext>\n}\n\n/**\n * Public extension methods exposed by a WsAdapter instance — broadcast\n * helpers, namespace stats, and reactive counters that DevTools and\n * other adapters consume directly.\n */\nexport interface WsAdapterExtensions {\n /** Snapshot of WebSocket stats — consumed by DevTools / Swagger ws-server discovery. */\n getStats(): {\n totalConnections: number\n activeConnections: number\n messagesReceived: number\n messagesSent: number\n errors: number\n namespaces: Record<string, { connections: number; handlers: number }>\n rooms: ReturnType<RoomManager['getAllRooms']>\n }\n /** Room name used for per-user broadcasting. */\n userRoom(userId: string): string\n /** Broadcast a single event to every socket in `user:<id>` (across namespaces). */\n broadcastToUser(userId: string, event: string, data: unknown): void\n /** Total WebSocket connections ever opened. */\n readonly totalConnections: Ref<number>\n /** Currently active connections. */\n readonly activeConnections: Ref<number>\n /** Total messages received. */\n readonly messagesReceived: Ref<number>\n /** Total messages sent. */\n readonly messagesSent: Ref<number>\n /** Total errors. */\n readonly wsErrors: Ref<number>\n}\n\n/**\n * WebSocket adapter for KickJS. Attaches to the HTTP server and routes\n * WebSocket connections to @WsController classes based on namespace paths.\n *\n * @example\n * ```ts\n * import { WsAdapter } from '@forinda/kickjs-ws'\n *\n * bootstrap({\n * modules: [ChatModule],\n * adapters: [\n * WsAdapter({ path: '/ws' }),\n * ],\n * })\n * ```\n *\n * Clients connect to: `ws://localhost:3000/ws/chat`\n * Messages are JSON: `{ \"event\": \"send\", \"data\": { \"text\": \"hello\" } }`\n */\nexport const WsAdapter = defineAdapter<WsAdapterOptions, WsAdapterExtensions>({\n name: 'WsAdapter',\n defaults: {\n path: '/ws',\n heartbeatInterval: 30000,\n },\n build: (options) => {\n const basePath = options.path!\n const heartbeatInterval = options.heartbeatInterval!\n const maxPayload = options.maxPayload\n const auth = options.auth\n const userRoomPrefix = options.auth?.userRoomPrefix ?? 'user:'\n\n let wss: WebSocketServer | null = null\n let container: Container | null = null\n const namespaces = new Map<string, NamespaceEntry>()\n const roomManager = new RoomManager()\n let heartbeatTimer: ReturnType<typeof setInterval> | null = null\n\n const totalConnections = ref(0)\n const activeConnections = ref(0)\n const messagesReceived = ref(0)\n const messagesSent = ref(0)\n const wsErrors = ref(0)\n\n const userRoom = (userId: string): string => userRoomPrefix + userId\n\n const broadcastToUser = (userId: string, event: string, data: unknown): void => {\n roomManager.broadcast(userRoom(userId), event, data)\n }\n\n const buildUserBroadcaster = (): WsUserBroadcaster => ({\n roomFor: (id) => userRoom(id),\n broadcastToUser: (id, event, data) => broadcastToUser(id, event, data),\n toUser: (id) => ({\n send: (event, data) => broadcastToUser(id, event, data),\n }),\n })\n\n const getStats = () => {\n const namespaceStats: Record<string, { connections: number; handlers: number }> = {}\n for (const [path, entry] of namespaces) {\n namespaceStats[path] = {\n connections: entry.sockets.size,\n handlers: entry.handlers.length,\n }\n }\n return {\n totalConnections: totalConnections.value,\n activeConnections: activeConnections.value,\n messagesReceived: messagesReceived.value,\n messagesSent: messagesSent.value,\n errors: wsErrors.value,\n namespaces: namespaceStats,\n rooms: roomManager.getAllRooms(),\n }\n }\n\n const safeInvoke = (controller: any, method: string, ctx: WsContext): void => {\n try {\n const result = controller[method](ctx)\n if (result instanceof Promise) {\n result.catch((err: Error) => {\n log.error({ err }, `WS handler error in ${method}`)\n })\n }\n } catch (err) {\n log.error({ err }, `WS handler error in ${method}`)\n }\n }\n\n const invokeHandlers = (\n controller: any,\n handlers: WsHandlerDefinition[],\n type: WsHandlerDefinition['type'],\n ctx: WsContext,\n ): void => {\n for (const handler of handlers) {\n if (handler.type === type) {\n safeInvoke(controller, handler.handlerName, ctx)\n }\n }\n }\n\n /**\n * Runs the configured auth hook against the upgrade request. Stashes the\n * resolved user on the context (as `user`, plus mirrored keys) and, when\n * `autoJoinUserRoom` is enabled, joins the socket to `user:<id>`.\n * Returns `false` (and closes the socket with code 4401) on failure.\n */\n const authenticate = async (ctx: WsContext): Promise<boolean> => {\n if (!auth) return true\n try {\n const user = await auth.resolveUser(ctx.request)\n if (!user || !user.id) {\n ctx.socket.close(4401, 'Unauthorized')\n return false\n }\n ctx.set('user', user)\n ctx.set('userId', user.id)\n if (auth.autoJoinUserRoom !== false) {\n ctx.join(userRoom(user.id))\n }\n return true\n } catch (err) {\n log.warn('WS auth failed', err)\n ctx.socket.close(4401, 'Unauthorized')\n return false\n }\n }\n\n const handleConnection = (\n ws: WebSocket,\n entry: NamespaceEntry,\n request: IncomingMessage,\n ): void => {\n const socketId = randomUUID()\n ;(ws as any).__alive = true\n\n entry.sockets.set(socketId, ws)\n totalConnections.value++\n activeConnections.value++\n\n const ctx = new WsContext(\n ws,\n wss!,\n roomManager,\n entry.sockets,\n socketId,\n entry.namespace,\n request,\n )\n entry.contexts.set(socketId, ctx)\n\n const controller = container!.resolve(entry.controllerClass)\n\n ws.on('pong', () => {\n ;(ws as any).__alive = true\n })\n\n // Authenticated handshake (optional). Messages received before auth\n // resolves are buffered on the socket; we gate dispatch on `authed`.\n let authed = auth === undefined\n const authPromise = auth ? authenticate(ctx) : Promise.resolve(true)\n\n authPromise.then((ok) => {\n if (!ok) return\n authed = true\n invokeHandlers(controller, entry.handlers, 'connect', ctx)\n })\n\n ws.on('message', (raw: Buffer | string) => {\n if (!authed) return\n messagesReceived.value++\n try {\n const parsed = JSON.parse(raw.toString())\n const event = parsed.event as string\n const data = parsed.data\n\n if (!event || typeof event !== 'string') {\n ctx.send('error', { message: 'Invalid message format: missing \"event\" field' })\n return\n }\n\n ctx.event = event\n ctx.data = data\n\n const handler = entry.handlers.find((h) => h.type === 'message' && h.event === event)\n\n if (handler) {\n safeInvoke(controller, handler.handlerName, ctx)\n } else {\n const catchAll = entry.handlers.find((h) => h.type === 'message' && h.event === '*')\n if (catchAll) {\n safeInvoke(controller, catchAll.handlerName, ctx)\n }\n }\n } catch {\n ctx.data = { message: 'Invalid JSON' }\n invokeHandlers(controller, entry.handlers, 'error', ctx)\n }\n })\n\n ws.on('close', () => {\n activeConnections.value--\n invokeHandlers(controller, entry.handlers, 'disconnect', ctx)\n roomManager.leaveAll(socketId)\n entry.sockets.delete(socketId)\n entry.contexts.delete(socketId)\n })\n\n ws.on('error', (err: Error) => {\n wsErrors.value++\n ctx.data = { message: err.message, name: err.name }\n invokeHandlers(controller, entry.handlers, 'error', ctx)\n })\n }\n\n return {\n getStats,\n userRoom,\n broadcastToUser,\n totalConnections,\n activeConnections,\n messagesReceived,\n messagesSent,\n wsErrors,\n\n beforeStart({ container: containerArg }) {\n container = containerArg\n\n // The factory's mutate-name pattern means `this` inside lifecycle\n // hooks does not reach the returned adapter object; pass the\n // adapter itself via WS_ADAPTER for code that wants the full\n // surface (broadcast helpers, stats, refs).\n // We don't have a `this` reference for the WS_ADAPTER token, so\n // we rebuild the externally-visible surface inline here.\n container.registerInstance(WS_ADAPTER, {\n getStats,\n userRoom,\n broadcastToUser,\n totalConnections,\n activeConnections,\n messagesReceived,\n messagesSent,\n wsErrors,\n })\n container.registerInstance(WS_ROOM_MANAGER, roomManager)\n container.registerInstance(WS_USER_BROADCASTER, buildUserBroadcaster())\n\n // Discover all @WsController classes and build routing table\n for (const controllerClass of wsControllerRegistry) {\n const namespace = getClassMetaOrUndefined<string>(\n WS_METADATA.WS_CONTROLLER,\n controllerClass,\n )\n if (namespace === undefined) continue\n\n const handlers = getClassMeta<WsHandlerDefinition[]>(\n WS_METADATA.WS_HANDLERS,\n controllerClass,\n [],\n )\n\n const fullPath = basePath + (namespace === '/' ? '' : namespace)\n\n namespaces.set(fullPath, {\n namespace,\n controllerClass,\n handlers,\n sockets: new Map(),\n contexts: new Map(),\n })\n\n log.info(`Registered WS namespace: ${fullPath} (${controllerClass.name})`)\n }\n },\n\n afterStart({ server }) {\n if (!server) return\n\n wss = new WebSocketServer({\n noServer: true,\n maxPayload,\n })\n\n // Handle upgrade requests — route to correct namespace\n server.on('upgrade', (request: IncomingMessage, socket: Duplex, head: Buffer) => {\n const url = request.url || '/'\n // Parse pathname without relying on host header\n const pathname = url.split('?')[0]\n\n const entry = namespaces.get(pathname)\n if (!entry) {\n socket.write('HTTP/1.1 404 Not Found\\r\\n\\r\\n')\n socket.destroy()\n return\n }\n\n wss!.handleUpgrade(request, socket, head, (ws) => {\n handleConnection(ws, entry, request)\n })\n })\n\n // Heartbeat ping/pong\n if (heartbeatInterval > 0) {\n heartbeatTimer = setInterval(() => {\n for (const [, entry] of namespaces) {\n for (const [, socket] of entry.sockets) {\n if ((socket as any).__alive === false) {\n socket.terminate()\n continue\n }\n ;(socket as any).__alive = false\n socket.ping()\n }\n }\n }, heartbeatInterval)\n }\n\n const totalHandlers = Array.from(namespaces.values()).reduce(\n (sum, e) => sum + e.handlers.length,\n 0,\n )\n log.info(`WebSocket ready — ${namespaces.size} namespace(s), ${totalHandlers} handler(s)`)\n },\n\n shutdown() {\n if (heartbeatTimer) {\n clearInterval(heartbeatTimer)\n }\n\n // Close all connections\n for (const [, entry] of namespaces) {\n for (const [, socket] of entry.sockets) {\n socket.close(1001, 'Server shutting down')\n }\n entry.sockets.clear()\n entry.contexts.clear()\n }\n\n wss?.close()\n },\n }\n },\n})\n","import { Service, setClassMeta, pushClassMeta } from '@forinda/kickjs'\nimport { WS_METADATA, wsControllerRegistry, type WsHandlerDefinition } from './interfaces'\n\n/**\n * Mark a class as a WebSocket controller with a namespace path.\n * Registers the class in the DI container and the WS controller registry.\n *\n * @example\n * ```ts\n * @WsController('/chat')\n * export class ChatController {\n * @OnConnect()\n * handleConnect(ctx: WsContext) { }\n *\n * @OnMessage('send')\n * handleSend(ctx: WsContext) { }\n * }\n * ```\n */\nexport function WsController(namespace?: string): ClassDecorator {\n return (target: any) => {\n Service()(target)\n setClassMeta(WS_METADATA.WS_CONTROLLER, namespace || '/', target)\n wsControllerRegistry.add(target)\n }\n}\n\nfunction createWsHandlerDecorator(type: WsHandlerDefinition['type'], event?: string) {\n return (): MethodDecorator => {\n return (target, propertyKey) => {\n pushClassMeta<WsHandlerDefinition>(WS_METADATA.WS_HANDLERS, target.constructor, {\n type,\n event,\n handlerName: propertyKey as string,\n })\n }\n }\n}\n\n/**\n * Handle new WebSocket connections.\n *\n * @example\n * ```ts\n * @OnConnect()\n * handleConnect(ctx: WsContext) {\n * console.log(`Client ${ctx.id} connected`)\n * }\n * ```\n */\nexport const OnConnect = createWsHandlerDecorator('connect')\n\n/**\n * Handle WebSocket disconnections.\n *\n * @example\n * ```ts\n * @OnDisconnect()\n * handleDisconnect(ctx: WsContext) {\n * console.log(`Client ${ctx.id} disconnected`)\n * }\n * ```\n */\nexport const OnDisconnect = createWsHandlerDecorator('disconnect')\n\n/**\n * Handle WebSocket errors.\n *\n * @example\n * ```ts\n * @OnError()\n * handleError(ctx: WsContext) {\n * console.error('WS error:', ctx.data)\n * }\n * ```\n */\nexport const OnError = createWsHandlerDecorator('error')\n\n/**\n * Handle a specific WebSocket message event.\n * Use '*' as a catch-all for unmatched events.\n *\n * Messages must be JSON: `{ \"event\": \"chat:send\", \"data\": { ... } }`\n *\n * @example\n * ```ts\n * @OnMessage('chat:send')\n * handleChatSend(ctx: WsContext) {\n * ctx.broadcast('chat:receive', ctx.data)\n * }\n *\n * @OnMessage('*')\n * handleUnknown(ctx: WsContext) {\n * ctx.send('error', { message: `Unknown event: ${ctx.event}` })\n * }\n * ```\n */\nexport function OnMessage(event: string): MethodDecorator {\n return (target, propertyKey) => {\n pushClassMeta<WsHandlerDefinition>(WS_METADATA.WS_HANDLERS, target.constructor, {\n type: 'message',\n event,\n handlerName: propertyKey as string,\n })\n }\n}\n"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@forinda/kickjs-ws",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "4.0.0",
|
|
4
4
|
"description": "WebSocket support with decorators, namespaces, rooms, and DI integration for KickJS",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"kickjs",
|
|
@@ -22,13 +22,10 @@
|
|
|
22
22
|
"@forinda/kickjs",
|
|
23
23
|
"@forinda/kickjs-auth",
|
|
24
24
|
"@forinda/kickjs-cli",
|
|
25
|
-
"@forinda/kickjs-config",
|
|
26
|
-
"@forinda/kickjs-core",
|
|
27
25
|
"@forinda/kickjs-cron",
|
|
28
26
|
"@forinda/kickjs-devtools",
|
|
29
27
|
"@forinda/kickjs-drizzle",
|
|
30
28
|
"@forinda/kickjs-graphql",
|
|
31
|
-
"@forinda/kickjs-http",
|
|
32
29
|
"@forinda/kickjs-mailer",
|
|
33
30
|
"@forinda/kickjs-multi-tenant",
|
|
34
31
|
"@forinda/kickjs-notifications",
|
|
@@ -64,9 +61,7 @@
|
|
|
64
61
|
"output": [
|
|
65
62
|
"dist/**"
|
|
66
63
|
],
|
|
67
|
-
"dependencies": [
|
|
68
|
-
"../core:build"
|
|
69
|
-
]
|
|
64
|
+
"dependencies": []
|
|
70
65
|
}
|
|
71
66
|
},
|
|
72
67
|
"dependencies": {
|
|
@@ -74,12 +69,12 @@
|
|
|
74
69
|
"ws": "^8.20.0"
|
|
75
70
|
},
|
|
76
71
|
"devDependencies": {
|
|
77
|
-
"@swc/core": "^1.15.
|
|
78
|
-
"@types/node": "^25.
|
|
72
|
+
"@swc/core": "^1.15.30",
|
|
73
|
+
"@types/node": "^25.6.0",
|
|
79
74
|
"@types/ws": "^8.18.0",
|
|
80
|
-
"typescript": "^
|
|
81
|
-
"vitest": "^4.1.
|
|
82
|
-
"@forinda/kickjs": "
|
|
75
|
+
"typescript": "^6.0.3",
|
|
76
|
+
"vitest": "^4.1.5",
|
|
77
|
+
"@forinda/kickjs": "4.0.0"
|
|
83
78
|
},
|
|
84
79
|
"publishConfig": {
|
|
85
80
|
"access": "public"
|