@rpcbase/server 0.538.0 → 0.540.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/dist/email-DK8uUU4X.js +8045 -0
- package/dist/email-DK8uUU4X.js.map +1 -0
- package/dist/handler--FFBJMl6.js +153 -0
- package/dist/handler--FFBJMl6.js.map +1 -0
- package/dist/handler-0rPClEv4.js +663 -0
- package/dist/handler-0rPClEv4.js.map +1 -0
- package/dist/handler-COnCnprN.js +203 -0
- package/dist/handler-COnCnprN.js.map +1 -0
- package/dist/handler-ClQF4MOn.js +931 -0
- package/dist/handler-ClQF4MOn.js.map +1 -0
- package/dist/index.js +4988 -4830
- package/dist/index.js.map +1 -1
- package/dist/notifications.js +199 -134
- package/dist/notifications.js.map +1 -1
- package/dist/queryExecutor-Bol_iR8f.js +453 -0
- package/dist/queryExecutor-Bol_iR8f.js.map +1 -0
- package/dist/render_resend_false-MiC__Smr.js +6 -0
- package/dist/render_resend_false-MiC__Smr.js.map +1 -0
- package/dist/rts/index.d.ts +0 -1
- package/dist/rts/index.d.ts.map +1 -1
- package/dist/rts/index.js +1003 -842
- package/dist/rts/index.js.map +1 -1
- package/dist/schemas-Cjdjgehl.js +4225 -0
- package/dist/schemas-Cjdjgehl.js.map +1 -0
- package/dist/shared-nE84Or5W.js +111 -0
- package/dist/shared-nE84Or5W.js.map +1 -0
- package/dist/ssrMiddleware.d.ts +1 -1
- package/dist/uploads.js +99 -84
- package/dist/uploads.js.map +1 -1
- package/package.json +9 -9
- package/dist/email-H8nTAGxe.js +0 -12449
- package/dist/email-H8nTAGxe.js.map +0 -1
- package/dist/handler-BBzEodA0.js +0 -182
- package/dist/handler-BBzEodA0.js.map +0 -1
- package/dist/handler-BLwgdQv-.js +0 -544
- package/dist/handler-BLwgdQv-.js.map +0 -1
- package/dist/handler-CZD5p1Jv.js +0 -28
- package/dist/handler-CZD5p1Jv.js.map +0 -1
- package/dist/handler-Cq6MsoD4.js +0 -124
- package/dist/handler-Cq6MsoD4.js.map +0 -1
- package/dist/handler-DBtnVvP2.js +0 -756
- package/dist/handler-DBtnVvP2.js.map +0 -1
- package/dist/queryExecutor-JadZcQSQ.js +0 -318
- package/dist/queryExecutor-JadZcQSQ.js.map +0 -1
- package/dist/render_resend-DQANggpW.js +0 -7
- package/dist/render_resend-DQANggpW.js.map +0 -1
- package/dist/rts/api/cleanup/handler.d.ts +0 -9
- package/dist/rts/api/cleanup/handler.d.ts.map +0 -1
- package/dist/rts/api/cleanup/index.d.ts +0 -11
- package/dist/rts/api/cleanup/index.d.ts.map +0 -1
- package/dist/schemas-BR3K5Luo.js +0 -3824
- package/dist/schemas-BR3K5Luo.js.map +0 -1
- package/dist/shared-DhZ_rDdo.js +0 -87
- package/dist/shared-DhZ_rDdo.js.map +0 -1
package/dist/rts/index.js
CHANGED
|
@@ -1,845 +1,1006 @@
|
|
|
1
|
-
import { a as normalizeRtsQueryOptions, c as runRtsCount, l as runRtsQuery, n as RTS_USER_ID_HEADER, o as resolveRtsQueryDependencyModelNames, t as RTS_TENANT_ID_QUERY_PARAM } from "../queryExecutor-JadZcQSQ.js";
|
|
2
|
-
import { models } from "@rpcbase/db";
|
|
3
|
-
import { buildAbility, buildAbilityFromSession, getTenantRolesFromSessionUser } from "@rpcbase/db/acl";
|
|
4
1
|
import { randomUUID } from "node:crypto";
|
|
2
|
+
import { models } from "@rpcbase/db";
|
|
3
|
+
import { buildAbilityFromSession, getTenantRolesFromSessionUser, buildAbility } from "@rpcbase/db/acl";
|
|
5
4
|
import { WebSocketServer } from "ws";
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
return acc;
|
|
5
|
+
import { R as RTS_TENANT_ID_QUERY_PARAM, d as RTS_USER_ID_HEADER, n as normalizeRtsQueryOptions, e as resolveRtsQueryDependencyModelNames, a as runRtsQuery, c as runRtsCount } from "../queryExecutor-Bol_iR8f.js";
|
|
6
|
+
const routes = Object.entries({
|
|
7
|
+
.../* @__PURE__ */ Object.assign({ "./api/changes/handler.ts": () => import("../handler--FFBJMl6.js") })
|
|
8
|
+
}).reduce((acc, [path, mod]) => {
|
|
9
|
+
acc[path.replace("./api/", "@rpcbase/server/rts/api/")] = mod;
|
|
10
|
+
return acc;
|
|
13
11
|
}, {});
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
};
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
}
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
}
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
};
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
}
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
420
|
-
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
433
|
-
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
474
|
-
|
|
475
|
-
|
|
476
|
-
|
|
477
|
-
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
};
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
505
|
-
|
|
506
|
-
|
|
507
|
-
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
514
|
-
|
|
515
|
-
|
|
516
|
-
|
|
517
|
-
|
|
518
|
-
|
|
519
|
-
|
|
520
|
-
|
|
521
|
-
|
|
522
|
-
|
|
523
|
-
|
|
524
|
-
|
|
525
|
-
|
|
526
|
-
|
|
527
|
-
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
|
|
531
|
-
|
|
532
|
-
|
|
533
|
-
|
|
534
|
-
|
|
535
|
-
|
|
536
|
-
|
|
537
|
-
|
|
538
|
-
|
|
539
|
-
|
|
540
|
-
|
|
541
|
-
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
|
|
546
|
-
|
|
547
|
-
|
|
548
|
-
|
|
549
|
-
}
|
|
550
|
-
|
|
551
|
-
|
|
552
|
-
|
|
553
|
-
|
|
554
|
-
|
|
555
|
-
|
|
556
|
-
|
|
557
|
-
|
|
558
|
-
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
|
|
562
|
-
|
|
563
|
-
|
|
564
|
-
|
|
565
|
-
|
|
566
|
-
|
|
567
|
-
|
|
568
|
-
|
|
569
|
-
|
|
570
|
-
|
|
571
|
-
|
|
572
|
-
|
|
573
|
-
|
|
574
|
-
|
|
575
|
-
|
|
576
|
-
|
|
577
|
-
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
|
|
581
|
-
|
|
582
|
-
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
|
|
586
|
-
|
|
587
|
-
|
|
588
|
-
|
|
589
|
-
|
|
590
|
-
|
|
591
|
-
|
|
592
|
-
|
|
593
|
-
|
|
594
|
-
|
|
595
|
-
|
|
596
|
-
|
|
597
|
-
|
|
598
|
-
|
|
599
|
-
|
|
600
|
-
|
|
601
|
-
|
|
602
|
-
|
|
603
|
-
|
|
604
|
-
|
|
605
|
-
|
|
606
|
-
|
|
607
|
-
|
|
608
|
-
|
|
609
|
-
|
|
610
|
-
|
|
611
|
-
|
|
612
|
-
|
|
613
|
-
|
|
614
|
-
|
|
615
|
-
|
|
616
|
-
|
|
617
|
-
|
|
618
|
-
|
|
619
|
-
|
|
620
|
-
|
|
621
|
-
|
|
622
|
-
|
|
623
|
-
|
|
624
|
-
|
|
625
|
-
|
|
626
|
-
|
|
627
|
-
|
|
628
|
-
|
|
629
|
-
|
|
630
|
-
|
|
631
|
-
|
|
632
|
-
|
|
633
|
-
|
|
634
|
-
|
|
635
|
-
|
|
636
|
-
|
|
637
|
-
|
|
638
|
-
|
|
639
|
-
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
|
|
643
|
-
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
|
|
647
|
-
|
|
648
|
-
|
|
649
|
-
|
|
650
|
-
|
|
651
|
-
|
|
652
|
-
|
|
653
|
-
|
|
654
|
-
|
|
655
|
-
|
|
656
|
-
|
|
657
|
-
|
|
658
|
-
|
|
659
|
-
|
|
660
|
-
|
|
661
|
-
|
|
662
|
-
|
|
663
|
-
|
|
664
|
-
|
|
665
|
-
|
|
666
|
-
|
|
667
|
-
|
|
668
|
-
|
|
669
|
-
|
|
670
|
-
|
|
671
|
-
|
|
672
|
-
|
|
673
|
-
|
|
674
|
-
|
|
675
|
-
|
|
676
|
-
|
|
677
|
-
|
|
678
|
-
|
|
679
|
-
|
|
680
|
-
|
|
681
|
-
|
|
682
|
-
|
|
683
|
-
|
|
684
|
-
|
|
685
|
-
|
|
686
|
-
|
|
687
|
-
|
|
688
|
-
|
|
689
|
-
|
|
690
|
-
|
|
691
|
-
|
|
692
|
-
|
|
693
|
-
|
|
694
|
-
|
|
695
|
-
|
|
696
|
-
|
|
697
|
-
|
|
698
|
-
|
|
699
|
-
|
|
700
|
-
|
|
701
|
-
|
|
702
|
-
|
|
703
|
-
|
|
704
|
-
|
|
705
|
-
|
|
706
|
-
|
|
707
|
-
|
|
708
|
-
|
|
709
|
-
|
|
710
|
-
|
|
711
|
-
|
|
712
|
-
|
|
713
|
-
|
|
714
|
-
|
|
715
|
-
|
|
716
|
-
|
|
717
|
-
|
|
718
|
-
|
|
719
|
-
|
|
720
|
-
|
|
721
|
-
|
|
722
|
-
|
|
723
|
-
|
|
724
|
-
|
|
725
|
-
|
|
726
|
-
|
|
727
|
-
|
|
728
|
-
|
|
729
|
-
|
|
730
|
-
|
|
731
|
-
|
|
732
|
-
|
|
733
|
-
|
|
734
|
-
|
|
735
|
-
|
|
736
|
-
|
|
737
|
-
|
|
738
|
-
|
|
739
|
-
|
|
740
|
-
|
|
741
|
-
|
|
742
|
-
|
|
743
|
-
|
|
744
|
-
|
|
745
|
-
|
|
746
|
-
|
|
747
|
-
|
|
748
|
-
|
|
749
|
-
|
|
750
|
-
|
|
751
|
-
|
|
752
|
-
|
|
753
|
-
|
|
754
|
-
|
|
755
|
-
|
|
756
|
-
|
|
757
|
-
|
|
758
|
-
|
|
759
|
-
|
|
760
|
-
|
|
761
|
-
|
|
762
|
-
|
|
763
|
-
|
|
764
|
-
|
|
765
|
-
|
|
766
|
-
|
|
767
|
-
|
|
768
|
-
|
|
769
|
-
|
|
770
|
-
|
|
771
|
-
|
|
772
|
-
|
|
773
|
-
|
|
774
|
-
|
|
775
|
-
|
|
776
|
-
|
|
777
|
-
|
|
778
|
-
|
|
779
|
-
|
|
780
|
-
|
|
781
|
-
|
|
782
|
-
|
|
783
|
-
|
|
784
|
-
|
|
785
|
-
|
|
786
|
-
|
|
787
|
-
|
|
788
|
-
|
|
789
|
-
|
|
790
|
-
|
|
791
|
-
|
|
792
|
-
|
|
793
|
-
|
|
794
|
-
|
|
795
|
-
|
|
796
|
-
|
|
797
|
-
|
|
798
|
-
|
|
799
|
-
|
|
800
|
-
|
|
801
|
-
|
|
802
|
-
|
|
803
|
-
|
|
804
|
-
|
|
805
|
-
|
|
806
|
-
|
|
807
|
-
|
|
808
|
-
|
|
809
|
-
|
|
810
|
-
|
|
811
|
-
|
|
812
|
-
|
|
813
|
-
|
|
814
|
-
|
|
815
|
-
|
|
816
|
-
|
|
817
|
-
|
|
818
|
-
|
|
819
|
-
|
|
820
|
-
|
|
821
|
-
|
|
822
|
-
|
|
823
|
-
|
|
824
|
-
|
|
825
|
-
|
|
826
|
-
|
|
827
|
-
|
|
828
|
-
|
|
829
|
-
|
|
830
|
-
|
|
831
|
-
|
|
832
|
-
|
|
833
|
-
|
|
834
|
-
|
|
835
|
-
}
|
|
836
|
-
|
|
837
|
-
|
|
838
|
-
|
|
839
|
-
|
|
840
|
-
|
|
841
|
-
|
|
842
|
-
|
|
843
|
-
|
|
844
|
-
|
|
845
|
-
|
|
12
|
+
const QUERY_KEY_MAX_LEN = 4096;
|
|
13
|
+
const INTERNAL_MODEL_NAMES = /* @__PURE__ */ new Set(["RBRtsChange", "RBRtsCounter"]);
|
|
14
|
+
const DEFAULT_MAX_PAYLOAD_BYTES = 1024 * 1024;
|
|
15
|
+
const DEFAULT_MAX_SUBSCRIPTIONS_PER_SOCKET = 256;
|
|
16
|
+
const DEFAULT_DISPATCH_DEBOUNCE_MS = 25;
|
|
17
|
+
const initializedServers = /* @__PURE__ */ new WeakSet();
|
|
18
|
+
const customHandlers = [];
|
|
19
|
+
const sockets = /* @__PURE__ */ new Map();
|
|
20
|
+
const socketMeta = /* @__PURE__ */ new Map();
|
|
21
|
+
const socketWrappers = /* @__PURE__ */ new Map();
|
|
22
|
+
const socketCleanup = /* @__PURE__ */ new Map();
|
|
23
|
+
const socketSubscriptions = /* @__PURE__ */ new Map();
|
|
24
|
+
const socketCountSubscriptions = /* @__PURE__ */ new Map();
|
|
25
|
+
const subscriptions = /* @__PURE__ */ new Map();
|
|
26
|
+
const countSubscriptions = /* @__PURE__ */ new Map();
|
|
27
|
+
const changeStreams = /* @__PURE__ */ new Map();
|
|
28
|
+
const dispatchTimers = /* @__PURE__ */ new Map();
|
|
29
|
+
const upgradeMeta = /* @__PURE__ */ new WeakMap();
|
|
30
|
+
let maxPayloadBytes = DEFAULT_MAX_PAYLOAD_BYTES;
|
|
31
|
+
let maxSubscriptionsPerSocket = DEFAULT_MAX_SUBSCRIPTIONS_PER_SOCKET;
|
|
32
|
+
let dispatchDebounceMs = DEFAULT_DISPATCH_DEBOUNCE_MS;
|
|
33
|
+
let allowInternalModels = false;
|
|
34
|
+
class RtsSocket {
|
|
35
|
+
id;
|
|
36
|
+
tenantId;
|
|
37
|
+
userId;
|
|
38
|
+
ws;
|
|
39
|
+
handlers = /* @__PURE__ */ new Map();
|
|
40
|
+
constructor({
|
|
41
|
+
id,
|
|
42
|
+
ws,
|
|
43
|
+
meta
|
|
44
|
+
}) {
|
|
45
|
+
this.id = id;
|
|
46
|
+
this.ws = ws;
|
|
47
|
+
this.tenantId = meta.tenantId;
|
|
48
|
+
this.userId = meta.userId;
|
|
49
|
+
}
|
|
50
|
+
on(event, handler) {
|
|
51
|
+
const set = this.handlers.get(event) ?? /* @__PURE__ */ new Set();
|
|
52
|
+
set.add(handler);
|
|
53
|
+
this.handlers.set(event, set);
|
|
54
|
+
return () => this.off(event, handler);
|
|
55
|
+
}
|
|
56
|
+
off(event, handler) {
|
|
57
|
+
const set = this.handlers.get(event);
|
|
58
|
+
if (!set) return;
|
|
59
|
+
set.delete(handler);
|
|
60
|
+
if (!set.size) this.handlers.delete(event);
|
|
61
|
+
}
|
|
62
|
+
emit(event, payload) {
|
|
63
|
+
sendWs(this.ws, {
|
|
64
|
+
type: "event",
|
|
65
|
+
event,
|
|
66
|
+
payload
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
close() {
|
|
70
|
+
try {
|
|
71
|
+
this.ws.close();
|
|
72
|
+
} catch {
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
dispatch(event, payload) {
|
|
76
|
+
const set = this.handlers.get(event);
|
|
77
|
+
if (!set) return;
|
|
78
|
+
for (const handler of set) {
|
|
79
|
+
handler(payload);
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
const rawToText = (raw) => {
|
|
84
|
+
if (typeof raw === "string") return raw;
|
|
85
|
+
if (raw instanceof ArrayBuffer) return Buffer.from(raw).toString();
|
|
86
|
+
if (Array.isArray(raw)) return Buffer.concat(raw).toString();
|
|
87
|
+
return raw.toString();
|
|
88
|
+
};
|
|
89
|
+
const safeJsonParse = (raw) => JSON.parse(rawToText(raw));
|
|
90
|
+
const sendWs = (ws, message) => {
|
|
91
|
+
if (ws.readyState !== 1) return;
|
|
92
|
+
ws.send(JSON.stringify(message));
|
|
93
|
+
};
|
|
94
|
+
const ensureSocketErrorHandler = (socket, context) => {
|
|
95
|
+
if (socket.listenerCount("error") > 0) return;
|
|
96
|
+
socket.on("error", (err) => {
|
|
97
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
98
|
+
const name = err instanceof Error ? err.name : typeof err;
|
|
99
|
+
const stack = err instanceof Error ? err.stack : void 0;
|
|
100
|
+
const code = err && typeof err === "object" && "code" in err ? err.code : void 0;
|
|
101
|
+
if (code === "ECONNRESET" || code == null && message.includes("ECONNRESET")) return;
|
|
102
|
+
console.warn("[rb/rts] socket error", {
|
|
103
|
+
name,
|
|
104
|
+
code,
|
|
105
|
+
message,
|
|
106
|
+
stack,
|
|
107
|
+
remoteAddress: socket.remoteAddress,
|
|
108
|
+
remotePort: socket.remotePort,
|
|
109
|
+
localAddress: socket.localAddress,
|
|
110
|
+
localPort: socket.localPort,
|
|
111
|
+
...context?.()
|
|
112
|
+
});
|
|
113
|
+
});
|
|
114
|
+
};
|
|
115
|
+
const redactErrorMessage = (err) => {
|
|
116
|
+
const raw = err instanceof Error ? err.message : "Unknown error";
|
|
117
|
+
const trimmedModelList = raw.replace(/\.\s+Available models:[\s\S]*$/, "");
|
|
118
|
+
const maxLen = 256;
|
|
119
|
+
if (trimmedModelList.length <= maxLen) return trimmedModelList;
|
|
120
|
+
return trimmedModelList.slice(0, maxLen);
|
|
121
|
+
};
|
|
122
|
+
const unauthorized = (socket, message = "Unauthorized") => {
|
|
123
|
+
ensureSocketErrorHandler(socket);
|
|
124
|
+
try {
|
|
125
|
+
socket.write("HTTP/1.1 401 Unauthorized\r\n\r\n");
|
|
126
|
+
socket.write(`Error: ${message}\r
|
|
127
|
+
`);
|
|
128
|
+
socket.end();
|
|
129
|
+
} catch {
|
|
130
|
+
socket.destroy();
|
|
131
|
+
}
|
|
132
|
+
};
|
|
133
|
+
const badRequest = (socket, message = "Bad Request") => {
|
|
134
|
+
ensureSocketErrorHandler(socket);
|
|
135
|
+
try {
|
|
136
|
+
socket.write("HTTP/1.1 400 Bad Request\r\n\r\n");
|
|
137
|
+
socket.write(`Error: ${message}\r
|
|
138
|
+
`);
|
|
139
|
+
socket.end();
|
|
140
|
+
} catch {
|
|
141
|
+
socket.destroy();
|
|
142
|
+
}
|
|
143
|
+
};
|
|
144
|
+
const runSessionMiddleware = async (sessionMiddleware, req) => {
|
|
145
|
+
await new Promise((resolve, reject) => {
|
|
146
|
+
const next = (err) => {
|
|
147
|
+
if (err) reject(err);
|
|
148
|
+
else resolve();
|
|
149
|
+
};
|
|
150
|
+
sessionMiddleware(req, {}, next);
|
|
151
|
+
});
|
|
152
|
+
};
|
|
153
|
+
const parseUpgradeMeta = async ({
|
|
154
|
+
req,
|
|
155
|
+
url,
|
|
156
|
+
sessionMiddleware
|
|
157
|
+
}) => {
|
|
158
|
+
const tenantId = url.searchParams.get(RTS_TENANT_ID_QUERY_PARAM);
|
|
159
|
+
if (!tenantId) {
|
|
160
|
+
throw new Error("Missing rb-tenant-id query parameter");
|
|
161
|
+
}
|
|
162
|
+
if (sessionMiddleware) {
|
|
163
|
+
const upgradeReq = req;
|
|
164
|
+
try {
|
|
165
|
+
await runSessionMiddleware(sessionMiddleware, upgradeReq);
|
|
166
|
+
} catch {
|
|
167
|
+
throw new Error("Failed to load session for RTS");
|
|
168
|
+
}
|
|
169
|
+
const sessionUser = upgradeReq.session?.user;
|
|
170
|
+
const sessionUserId = sessionUser?.id;
|
|
171
|
+
if (!sessionUserId) {
|
|
172
|
+
throw new Error("Not signed in (missing session.user.id)");
|
|
173
|
+
}
|
|
174
|
+
const signedInTenants = sessionUser?.signedInTenants;
|
|
175
|
+
const currentTenantId = sessionUser?.currentTenantId;
|
|
176
|
+
if (Array.isArray(signedInTenants) && signedInTenants.length > 0) {
|
|
177
|
+
if (!signedInTenants.includes(tenantId)) {
|
|
178
|
+
throw new Error("Tenant not authorized for this session");
|
|
179
|
+
}
|
|
180
|
+
} else if (currentTenantId) {
|
|
181
|
+
if (currentTenantId !== tenantId) {
|
|
182
|
+
throw new Error("Tenant not authorized for this session");
|
|
183
|
+
}
|
|
184
|
+
} else {
|
|
185
|
+
throw new Error("Tenant not authorized for this session");
|
|
186
|
+
}
|
|
187
|
+
const ability2 = buildAbilityFromSession({
|
|
188
|
+
tenantId,
|
|
189
|
+
session: upgradeReq.session
|
|
190
|
+
});
|
|
191
|
+
return {
|
|
192
|
+
tenantId,
|
|
193
|
+
userId: sessionUserId,
|
|
194
|
+
ability: ability2
|
|
195
|
+
};
|
|
196
|
+
}
|
|
197
|
+
const raw = req.headers[RTS_USER_ID_HEADER];
|
|
198
|
+
const headerUserId = Array.isArray(raw) ? raw[0] : raw;
|
|
199
|
+
if (!headerUserId) {
|
|
200
|
+
throw new Error("Missing rb-user-id header (reverse-proxy) and no session middleware configured");
|
|
201
|
+
}
|
|
202
|
+
const rbCtx = {
|
|
203
|
+
req: {
|
|
204
|
+
session: null
|
|
205
|
+
}
|
|
206
|
+
};
|
|
207
|
+
const User = await models.getGlobal("RBUser", rbCtx);
|
|
208
|
+
const user = await User.findById(headerUserId, {
|
|
209
|
+
tenants: 1,
|
|
210
|
+
tenantRoles: 1
|
|
211
|
+
}).lean();
|
|
212
|
+
const tenantsRaw = user?.tenants;
|
|
213
|
+
const tenants = Array.isArray(tenantsRaw) ? tenantsRaw.map((t) => String(t)) : [];
|
|
214
|
+
if (!tenants.includes(tenantId)) {
|
|
215
|
+
throw new Error("Tenant not authorized for this session");
|
|
216
|
+
}
|
|
217
|
+
const roles = getTenantRolesFromSessionUser(user, tenantId);
|
|
218
|
+
const ability = buildAbility({
|
|
219
|
+
tenantId,
|
|
220
|
+
userId: headerUserId,
|
|
221
|
+
roles: roles.length ? roles : ["owner"]
|
|
222
|
+
});
|
|
223
|
+
return {
|
|
224
|
+
tenantId,
|
|
225
|
+
userId: headerUserId,
|
|
226
|
+
ability
|
|
227
|
+
};
|
|
228
|
+
};
|
|
229
|
+
const getTenantModel = async (tenantId, modelName) => {
|
|
230
|
+
const ctx = {
|
|
231
|
+
req: {
|
|
232
|
+
session: {
|
|
233
|
+
user: {
|
|
234
|
+
currentTenantId: tenantId
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
};
|
|
239
|
+
return models.getUnsafe(modelName, ctx);
|
|
240
|
+
};
|
|
241
|
+
const makeDispatchKey = (tenantId, modelName) => `${tenantId}:${modelName}`;
|
|
242
|
+
const clearDispatchTimer = (tenantId, modelName) => {
|
|
243
|
+
const key = makeDispatchKey(tenantId, modelName);
|
|
244
|
+
const timer = dispatchTimers.get(key);
|
|
245
|
+
if (!timer) return;
|
|
246
|
+
clearTimeout(timer);
|
|
247
|
+
dispatchTimers.delete(key);
|
|
248
|
+
};
|
|
249
|
+
const scheduleDispatchSubscriptionsForModel = (tenantId, modelName) => {
|
|
250
|
+
const key = makeDispatchKey(tenantId, modelName);
|
|
251
|
+
if (dispatchTimers.has(key)) return;
|
|
252
|
+
const delay = Math.max(0, Math.min(1e3, Math.floor(dispatchDebounceMs)));
|
|
253
|
+
dispatchTimers.set(key, setTimeout(() => {
|
|
254
|
+
dispatchTimers.delete(key);
|
|
255
|
+
void Promise.all([dispatchSubscriptionsForModel(tenantId, modelName), dispatchCountSubscriptionsForModel(tenantId, modelName)]).catch(() => {
|
|
256
|
+
});
|
|
257
|
+
}, delay));
|
|
258
|
+
};
|
|
259
|
+
const runAndSendQuery = async ({
|
|
260
|
+
tenantId,
|
|
261
|
+
targetSocketIds,
|
|
262
|
+
ability,
|
|
263
|
+
modelName,
|
|
264
|
+
queryKey,
|
|
265
|
+
query,
|
|
266
|
+
options
|
|
267
|
+
}) => {
|
|
268
|
+
const result = await runRtsQuery({
|
|
269
|
+
tenantId,
|
|
270
|
+
ability,
|
|
271
|
+
modelName,
|
|
272
|
+
query,
|
|
273
|
+
options,
|
|
274
|
+
allowInternalModels
|
|
275
|
+
});
|
|
276
|
+
const payload = {
|
|
277
|
+
type: "query-payload",
|
|
278
|
+
modelName,
|
|
279
|
+
queryKey,
|
|
280
|
+
data: result.data,
|
|
281
|
+
...result.pageInfo ? {
|
|
282
|
+
pageInfo: result.pageInfo
|
|
283
|
+
} : {},
|
|
284
|
+
...typeof result.totalCount === "number" ? {
|
|
285
|
+
totalCount: result.totalCount
|
|
286
|
+
} : {}
|
|
287
|
+
};
|
|
288
|
+
for (const socketId of targetSocketIds) {
|
|
289
|
+
const ws = sockets.get(socketId);
|
|
290
|
+
if (!ws) continue;
|
|
291
|
+
sendWs(ws, payload);
|
|
292
|
+
}
|
|
293
|
+
};
|
|
294
|
+
const runAndSendCount = async ({
|
|
295
|
+
tenantId,
|
|
296
|
+
targetSocketIds,
|
|
297
|
+
ability,
|
|
298
|
+
modelName,
|
|
299
|
+
queryKey,
|
|
300
|
+
query
|
|
301
|
+
}) => {
|
|
302
|
+
const count = await runRtsCount({
|
|
303
|
+
tenantId,
|
|
304
|
+
ability,
|
|
305
|
+
modelName,
|
|
306
|
+
query,
|
|
307
|
+
allowInternalModels
|
|
308
|
+
});
|
|
309
|
+
const payload = {
|
|
310
|
+
type: "count-payload",
|
|
311
|
+
modelName,
|
|
312
|
+
queryKey,
|
|
313
|
+
count
|
|
314
|
+
};
|
|
315
|
+
for (const socketId of targetSocketIds) {
|
|
316
|
+
const ws = sockets.get(socketId);
|
|
317
|
+
if (!ws) continue;
|
|
318
|
+
sendWs(ws, payload);
|
|
319
|
+
}
|
|
320
|
+
};
|
|
321
|
+
const subscriptionDependsOnModel = (changedModelName, ownerModelName, subscription) => {
|
|
322
|
+
if (ownerModelName === changedModelName) return true;
|
|
323
|
+
return subscription.dependencyModelNames.includes(changedModelName);
|
|
324
|
+
};
|
|
325
|
+
const hasAnySubscriptionsDependingOnModel = (tenantId, modelName) => {
|
|
326
|
+
const tenantSubs = subscriptions.get(tenantId);
|
|
327
|
+
if (!tenantSubs) return false;
|
|
328
|
+
for (const userSubs of tenantSubs.values()) {
|
|
329
|
+
for (const [ownerModelName, modelSubs] of userSubs.entries()) {
|
|
330
|
+
for (const subscription of modelSubs.values()) {
|
|
331
|
+
if (subscriptionDependsOnModel(modelName, ownerModelName, subscription)) {
|
|
332
|
+
return true;
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
}
|
|
337
|
+
return false;
|
|
338
|
+
};
|
|
339
|
+
const hasAnyCountSubscriptionsForModel = (tenantId, modelName) => {
|
|
340
|
+
const tenantSubs = countSubscriptions.get(tenantId);
|
|
341
|
+
if (!tenantSubs) return false;
|
|
342
|
+
for (const userSubs of tenantSubs.values()) {
|
|
343
|
+
const modelSubs = userSubs.get(modelName);
|
|
344
|
+
if (modelSubs?.size) {
|
|
345
|
+
return true;
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
return false;
|
|
349
|
+
};
|
|
350
|
+
const dispatchSubscriptionsForModel = async (tenantId, modelName) => {
|
|
351
|
+
const tenantSubs = subscriptions.get(tenantId);
|
|
352
|
+
if (!tenantSubs || !tenantSubs.size) return;
|
|
353
|
+
for (const userSubs of tenantSubs.values()) {
|
|
354
|
+
for (const [ownerModelName, modelSubs] of userSubs.entries()) {
|
|
355
|
+
for (const [queryKey, sub] of modelSubs.entries()) {
|
|
356
|
+
if (!subscriptionDependsOnModel(modelName, ownerModelName, sub)) continue;
|
|
357
|
+
const targetSocketIds = Array.from(sub.socketIds);
|
|
358
|
+
if (!targetSocketIds.length) continue;
|
|
359
|
+
const socketId = targetSocketIds[0];
|
|
360
|
+
const meta = socketMeta.get(socketId);
|
|
361
|
+
const ability = meta?.ability;
|
|
362
|
+
if (!ability) continue;
|
|
363
|
+
try {
|
|
364
|
+
await runAndSendQuery({
|
|
365
|
+
tenantId,
|
|
366
|
+
targetSocketIds,
|
|
367
|
+
ability,
|
|
368
|
+
modelName: ownerModelName,
|
|
369
|
+
queryKey,
|
|
370
|
+
query: sub.query,
|
|
371
|
+
options: sub.options
|
|
372
|
+
});
|
|
373
|
+
} catch (err) {
|
|
374
|
+
const error = redactErrorMessage(err);
|
|
375
|
+
const payload = {
|
|
376
|
+
type: "query-payload",
|
|
377
|
+
modelName: ownerModelName,
|
|
378
|
+
queryKey,
|
|
379
|
+
error
|
|
380
|
+
};
|
|
381
|
+
for (const socketId2 of targetSocketIds) {
|
|
382
|
+
const ws = sockets.get(socketId2);
|
|
383
|
+
if (!ws) continue;
|
|
384
|
+
sendWs(ws, payload);
|
|
385
|
+
}
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
}
|
|
390
|
+
};
|
|
391
|
+
const dispatchCountSubscriptionsForModel = async (tenantId, modelName) => {
|
|
392
|
+
const tenantSubs = countSubscriptions.get(tenantId);
|
|
393
|
+
if (!tenantSubs || !tenantSubs.size) return;
|
|
394
|
+
for (const userSubs of tenantSubs.values()) {
|
|
395
|
+
const modelSubs = userSubs.get(modelName);
|
|
396
|
+
if (!modelSubs) continue;
|
|
397
|
+
for (const [queryKey, sub] of modelSubs.entries()) {
|
|
398
|
+
const targetSocketIds = Array.from(sub.socketIds);
|
|
399
|
+
if (!targetSocketIds.length) continue;
|
|
400
|
+
const socketId = targetSocketIds[0];
|
|
401
|
+
const meta = socketMeta.get(socketId);
|
|
402
|
+
const ability = meta?.ability;
|
|
403
|
+
if (!ability) continue;
|
|
404
|
+
try {
|
|
405
|
+
await runAndSendCount({
|
|
406
|
+
tenantId,
|
|
407
|
+
targetSocketIds,
|
|
408
|
+
ability,
|
|
409
|
+
modelName,
|
|
410
|
+
queryKey,
|
|
411
|
+
query: sub.query
|
|
412
|
+
});
|
|
413
|
+
} catch (err) {
|
|
414
|
+
const error = redactErrorMessage(err);
|
|
415
|
+
const payload = {
|
|
416
|
+
type: "count-payload",
|
|
417
|
+
modelName,
|
|
418
|
+
queryKey,
|
|
419
|
+
error
|
|
420
|
+
};
|
|
421
|
+
for (const targetSocketId of targetSocketIds) {
|
|
422
|
+
const ws = sockets.get(targetSocketId);
|
|
423
|
+
if (!ws) continue;
|
|
424
|
+
sendWs(ws, payload);
|
|
425
|
+
}
|
|
426
|
+
}
|
|
427
|
+
}
|
|
428
|
+
}
|
|
429
|
+
};
|
|
430
|
+
const ensureChangeStream = async (tenantId, modelName) => {
|
|
431
|
+
const tenantStreams = changeStreams.get(tenantId) ?? /* @__PURE__ */ new Map();
|
|
432
|
+
changeStreams.set(tenantId, tenantStreams);
|
|
433
|
+
if (tenantStreams.has(modelName)) return;
|
|
434
|
+
const model = await getTenantModel(tenantId, modelName);
|
|
435
|
+
const stream = model.watch([], {
|
|
436
|
+
fullDocument: "updateLookup"
|
|
437
|
+
});
|
|
438
|
+
stream.on("change", () => {
|
|
439
|
+
scheduleDispatchSubscriptionsForModel(tenantId, modelName);
|
|
440
|
+
});
|
|
441
|
+
stream.on("close", () => {
|
|
442
|
+
clearDispatchTimer(tenantId, modelName);
|
|
443
|
+
const map = changeStreams.get(tenantId);
|
|
444
|
+
map?.delete(modelName);
|
|
445
|
+
if (map && map.size === 0) changeStreams.delete(tenantId);
|
|
446
|
+
});
|
|
447
|
+
stream.on("error", () => {
|
|
448
|
+
try {
|
|
449
|
+
clearDispatchTimer(tenantId, modelName);
|
|
450
|
+
stream.close();
|
|
451
|
+
} catch {
|
|
452
|
+
}
|
|
453
|
+
});
|
|
454
|
+
tenantStreams.set(modelName, stream);
|
|
455
|
+
};
|
|
456
|
+
const addSocketSubscription = ({
|
|
457
|
+
socketId,
|
|
458
|
+
tenantId,
|
|
459
|
+
userId,
|
|
460
|
+
modelName,
|
|
461
|
+
queryKey,
|
|
462
|
+
query,
|
|
463
|
+
options,
|
|
464
|
+
dependencyModelNames
|
|
465
|
+
}) => {
|
|
466
|
+
const tenantSubs = subscriptions.get(tenantId) ?? /* @__PURE__ */ new Map();
|
|
467
|
+
subscriptions.set(tenantId, tenantSubs);
|
|
468
|
+
const userSubs = tenantSubs.get(userId) ?? /* @__PURE__ */ new Map();
|
|
469
|
+
tenantSubs.set(userId, userSubs);
|
|
470
|
+
const modelSubs = userSubs.get(modelName) ?? /* @__PURE__ */ new Map();
|
|
471
|
+
userSubs.set(modelName, modelSubs);
|
|
472
|
+
const existing = modelSubs.get(queryKey);
|
|
473
|
+
if (existing) {
|
|
474
|
+
existing.socketIds.add(socketId);
|
|
475
|
+
existing.dependencyModelNames = Array.from(/* @__PURE__ */ new Set([...existing.dependencyModelNames, ...dependencyModelNames]));
|
|
476
|
+
} else {
|
|
477
|
+
modelSubs.set(queryKey, {
|
|
478
|
+
query,
|
|
479
|
+
options,
|
|
480
|
+
dependencyModelNames: Array.from(new Set(dependencyModelNames)),
|
|
481
|
+
socketIds: /* @__PURE__ */ new Set([socketId])
|
|
482
|
+
});
|
|
483
|
+
}
|
|
484
|
+
const byModel = socketSubscriptions.get(socketId) ?? /* @__PURE__ */ new Map();
|
|
485
|
+
socketSubscriptions.set(socketId, byModel);
|
|
486
|
+
const querySet = byModel.get(modelName) ?? /* @__PURE__ */ new Set();
|
|
487
|
+
byModel.set(modelName, querySet);
|
|
488
|
+
querySet.add(queryKey);
|
|
489
|
+
};
|
|
490
|
+
const addSocketCountSubscription = ({
|
|
491
|
+
socketId,
|
|
492
|
+
tenantId,
|
|
493
|
+
userId,
|
|
494
|
+
modelName,
|
|
495
|
+
queryKey,
|
|
496
|
+
query
|
|
497
|
+
}) => {
|
|
498
|
+
const tenantSubs = countSubscriptions.get(tenantId) ?? /* @__PURE__ */ new Map();
|
|
499
|
+
countSubscriptions.set(tenantId, tenantSubs);
|
|
500
|
+
const userSubs = tenantSubs.get(userId) ?? /* @__PURE__ */ new Map();
|
|
501
|
+
tenantSubs.set(userId, userSubs);
|
|
502
|
+
const modelSubs = userSubs.get(modelName) ?? /* @__PURE__ */ new Map();
|
|
503
|
+
userSubs.set(modelName, modelSubs);
|
|
504
|
+
const existing = modelSubs.get(queryKey);
|
|
505
|
+
if (existing) {
|
|
506
|
+
existing.socketIds.add(socketId);
|
|
507
|
+
} else {
|
|
508
|
+
modelSubs.set(queryKey, {
|
|
509
|
+
query,
|
|
510
|
+
socketIds: /* @__PURE__ */ new Set([socketId])
|
|
511
|
+
});
|
|
512
|
+
}
|
|
513
|
+
const byModel = socketCountSubscriptions.get(socketId) ?? /* @__PURE__ */ new Map();
|
|
514
|
+
socketCountSubscriptions.set(socketId, byModel);
|
|
515
|
+
const querySet = byModel.get(modelName) ?? /* @__PURE__ */ new Set();
|
|
516
|
+
byModel.set(modelName, querySet);
|
|
517
|
+
querySet.add(queryKey);
|
|
518
|
+
};
|
|
519
|
+
const removeSocketSubscription = ({
|
|
520
|
+
socketId,
|
|
521
|
+
tenantId,
|
|
522
|
+
userId,
|
|
523
|
+
modelName,
|
|
524
|
+
queryKey
|
|
525
|
+
}) => {
|
|
526
|
+
const tenantSubs = subscriptions.get(tenantId);
|
|
527
|
+
const userSubs = tenantSubs?.get(userId);
|
|
528
|
+
const modelSubs = userSubs?.get(modelName);
|
|
529
|
+
const sub = modelSubs?.get(queryKey);
|
|
530
|
+
const affectedModelNames = /* @__PURE__ */ new Set([modelName, ...sub?.dependencyModelNames ?? []]);
|
|
531
|
+
if (sub) {
|
|
532
|
+
sub.socketIds.delete(socketId);
|
|
533
|
+
if (!sub.socketIds.size) {
|
|
534
|
+
modelSubs?.delete(queryKey);
|
|
535
|
+
}
|
|
536
|
+
}
|
|
537
|
+
const byModel = socketSubscriptions.get(socketId);
|
|
538
|
+
const set = byModel?.get(modelName);
|
|
539
|
+
if (set) {
|
|
540
|
+
set.delete(queryKey);
|
|
541
|
+
if (!set.size) {
|
|
542
|
+
byModel?.delete(modelName);
|
|
543
|
+
}
|
|
544
|
+
}
|
|
545
|
+
if (modelSubs && modelSubs.size === 0) {
|
|
546
|
+
userSubs?.delete(modelName);
|
|
547
|
+
}
|
|
548
|
+
if (userSubs && userSubs.size === 0) {
|
|
549
|
+
tenantSubs?.delete(userId);
|
|
550
|
+
}
|
|
551
|
+
for (const affectedModelName of affectedModelNames) {
|
|
552
|
+
if (hasAnySubscriptionsDependingOnModel(tenantId, affectedModelName) || hasAnyCountSubscriptionsForModel(tenantId, affectedModelName)) {
|
|
553
|
+
continue;
|
|
554
|
+
}
|
|
555
|
+
const tenantStreams = changeStreams.get(tenantId);
|
|
556
|
+
const stream = tenantStreams?.get(affectedModelName);
|
|
557
|
+
if (stream) {
|
|
558
|
+
try {
|
|
559
|
+
stream.close();
|
|
560
|
+
} catch {
|
|
561
|
+
}
|
|
562
|
+
clearDispatchTimer(tenantId, affectedModelName);
|
|
563
|
+
tenantStreams?.delete(affectedModelName);
|
|
564
|
+
if (tenantStreams && tenantStreams.size === 0) changeStreams.delete(tenantId);
|
|
565
|
+
}
|
|
566
|
+
}
|
|
567
|
+
if (tenantSubs && tenantSubs.size === 0) subscriptions.delete(tenantId);
|
|
568
|
+
if (byModel && byModel.size === 0) socketSubscriptions.delete(socketId);
|
|
569
|
+
};
|
|
570
|
+
const removeSocketCountSubscription = ({
|
|
571
|
+
socketId,
|
|
572
|
+
tenantId,
|
|
573
|
+
userId,
|
|
574
|
+
modelName,
|
|
575
|
+
queryKey
|
|
576
|
+
}) => {
|
|
577
|
+
const tenantSubs = countSubscriptions.get(tenantId);
|
|
578
|
+
const userSubs = tenantSubs?.get(userId);
|
|
579
|
+
const modelSubs = userSubs?.get(modelName);
|
|
580
|
+
const sub = modelSubs?.get(queryKey);
|
|
581
|
+
if (sub) {
|
|
582
|
+
sub.socketIds.delete(socketId);
|
|
583
|
+
if (!sub.socketIds.size) {
|
|
584
|
+
modelSubs?.delete(queryKey);
|
|
585
|
+
}
|
|
586
|
+
}
|
|
587
|
+
const byModel = socketCountSubscriptions.get(socketId);
|
|
588
|
+
const set = byModel?.get(modelName);
|
|
589
|
+
if (set) {
|
|
590
|
+
set.delete(queryKey);
|
|
591
|
+
if (!set.size) {
|
|
592
|
+
byModel?.delete(modelName);
|
|
593
|
+
}
|
|
594
|
+
}
|
|
595
|
+
if (modelSubs && modelSubs.size === 0) {
|
|
596
|
+
userSubs?.delete(modelName);
|
|
597
|
+
}
|
|
598
|
+
if (userSubs && userSubs.size === 0) {
|
|
599
|
+
tenantSubs?.delete(userId);
|
|
600
|
+
}
|
|
601
|
+
if (!hasAnySubscriptionsDependingOnModel(tenantId, modelName) && !hasAnyCountSubscriptionsForModel(tenantId, modelName)) {
|
|
602
|
+
const tenantStreams = changeStreams.get(tenantId);
|
|
603
|
+
const stream = tenantStreams?.get(modelName);
|
|
604
|
+
if (stream) {
|
|
605
|
+
try {
|
|
606
|
+
stream.close();
|
|
607
|
+
} catch {
|
|
608
|
+
}
|
|
609
|
+
clearDispatchTimer(tenantId, modelName);
|
|
610
|
+
tenantStreams?.delete(modelName);
|
|
611
|
+
if (tenantStreams && tenantStreams.size === 0) changeStreams.delete(tenantId);
|
|
612
|
+
}
|
|
613
|
+
}
|
|
614
|
+
if (tenantSubs && tenantSubs.size === 0) countSubscriptions.delete(tenantId);
|
|
615
|
+
if (byModel && byModel.size === 0) socketCountSubscriptions.delete(socketId);
|
|
616
|
+
};
|
|
617
|
+
const getSocketSubscriptionCount = (socketId) => {
|
|
618
|
+
let count = 0;
|
|
619
|
+
const querySubscriptionsByModel = socketSubscriptions.get(socketId);
|
|
620
|
+
if (querySubscriptionsByModel) {
|
|
621
|
+
for (const set of querySubscriptionsByModel.values()) count += set.size;
|
|
622
|
+
}
|
|
623
|
+
const countSubscriptionsByModel = socketCountSubscriptions.get(socketId);
|
|
624
|
+
if (countSubscriptionsByModel) {
|
|
625
|
+
for (const set of countSubscriptionsByModel.values()) count += set.size;
|
|
626
|
+
}
|
|
627
|
+
return count;
|
|
628
|
+
};
|
|
629
|
+
const cleanupSocket = (socketId) => {
|
|
630
|
+
const meta = socketMeta.get(socketId);
|
|
631
|
+
if (meta) {
|
|
632
|
+
const byModel = socketSubscriptions.get(socketId);
|
|
633
|
+
if (byModel) {
|
|
634
|
+
for (const [modelName, keys] of byModel.entries()) {
|
|
635
|
+
for (const queryKey of keys.values()) {
|
|
636
|
+
removeSocketSubscription({
|
|
637
|
+
socketId,
|
|
638
|
+
tenantId: meta.tenantId,
|
|
639
|
+
userId: meta.userId,
|
|
640
|
+
modelName,
|
|
641
|
+
queryKey
|
|
642
|
+
});
|
|
643
|
+
}
|
|
644
|
+
}
|
|
645
|
+
}
|
|
646
|
+
const countByModel = socketCountSubscriptions.get(socketId);
|
|
647
|
+
if (countByModel) {
|
|
648
|
+
for (const [modelName, keys] of countByModel.entries()) {
|
|
649
|
+
for (const queryKey of keys.values()) {
|
|
650
|
+
removeSocketCountSubscription({
|
|
651
|
+
socketId,
|
|
652
|
+
tenantId: meta.tenantId,
|
|
653
|
+
userId: meta.userId,
|
|
654
|
+
modelName,
|
|
655
|
+
queryKey
|
|
656
|
+
});
|
|
657
|
+
}
|
|
658
|
+
}
|
|
659
|
+
}
|
|
660
|
+
}
|
|
661
|
+
socketSubscriptions.delete(socketId);
|
|
662
|
+
socketCountSubscriptions.delete(socketId);
|
|
663
|
+
const cleanupFns = socketCleanup.get(socketId) ?? [];
|
|
664
|
+
socketCleanup.delete(socketId);
|
|
665
|
+
for (const fn of cleanupFns) {
|
|
666
|
+
try {
|
|
667
|
+
fn();
|
|
668
|
+
} catch {
|
|
669
|
+
}
|
|
670
|
+
}
|
|
671
|
+
sockets.delete(socketId);
|
|
672
|
+
socketMeta.delete(socketId);
|
|
673
|
+
socketWrappers.delete(socketId);
|
|
674
|
+
};
|
|
675
|
+
const handleClientMessage = async ({
|
|
676
|
+
socketId,
|
|
677
|
+
meta,
|
|
678
|
+
message
|
|
679
|
+
}) => {
|
|
680
|
+
const ws = sockets.get(socketId);
|
|
681
|
+
if (!ws) return;
|
|
682
|
+
if (message.type === "event") {
|
|
683
|
+
const wrapper = socketWrappers.get(socketId);
|
|
684
|
+
wrapper?.dispatch(message.event, message.payload);
|
|
685
|
+
return;
|
|
686
|
+
}
|
|
687
|
+
const isCountMessage = message.type === "run-count" || message.type === "register-count" || message.type === "remove-count";
|
|
688
|
+
const sendAccessError = (error) => {
|
|
689
|
+
if (isCountMessage) {
|
|
690
|
+
sendWs(ws, {
|
|
691
|
+
type: "count-payload",
|
|
692
|
+
modelName: message.modelName ?? "",
|
|
693
|
+
queryKey: message.queryKey ?? "",
|
|
694
|
+
error
|
|
695
|
+
});
|
|
696
|
+
return;
|
|
697
|
+
}
|
|
698
|
+
sendWs(ws, {
|
|
699
|
+
type: "query-payload",
|
|
700
|
+
modelName: message.modelName ?? "",
|
|
701
|
+
queryKey: message.queryKey ?? "",
|
|
702
|
+
error
|
|
703
|
+
});
|
|
704
|
+
};
|
|
705
|
+
if (!message.modelName || typeof message.modelName !== "string") return;
|
|
706
|
+
if (!allowInternalModels && INTERNAL_MODEL_NAMES.has(message.modelName)) {
|
|
707
|
+
sendAccessError("Model not allowed");
|
|
708
|
+
return;
|
|
709
|
+
}
|
|
710
|
+
if (!message.queryKey || typeof message.queryKey !== "string") return;
|
|
711
|
+
if (message.queryKey.length > QUERY_KEY_MAX_LEN) return;
|
|
712
|
+
if (message.type === "remove-query") {
|
|
713
|
+
removeSocketSubscription({
|
|
714
|
+
socketId,
|
|
715
|
+
tenantId: meta.tenantId,
|
|
716
|
+
userId: meta.userId,
|
|
717
|
+
modelName: message.modelName,
|
|
718
|
+
queryKey: message.queryKey
|
|
719
|
+
});
|
|
720
|
+
return;
|
|
721
|
+
}
|
|
722
|
+
if (message.type === "remove-count") {
|
|
723
|
+
removeSocketCountSubscription({
|
|
724
|
+
socketId,
|
|
725
|
+
tenantId: meta.tenantId,
|
|
726
|
+
userId: meta.userId,
|
|
727
|
+
modelName: message.modelName,
|
|
728
|
+
queryKey: message.queryKey
|
|
729
|
+
});
|
|
730
|
+
return;
|
|
731
|
+
}
|
|
732
|
+
if (!message.query || typeof message.query !== "object") return;
|
|
733
|
+
const ability = meta.ability;
|
|
734
|
+
if (!ability.can("read", message.modelName)) {
|
|
735
|
+
sendAccessError("forbidden");
|
|
736
|
+
return;
|
|
737
|
+
}
|
|
738
|
+
if (message.type === "register-query" || message.type === "run-query") {
|
|
739
|
+
const options = normalizeRtsQueryOptions(message.options);
|
|
740
|
+
if (message.type === "register-query") {
|
|
741
|
+
const existing = socketSubscriptions.get(socketId)?.get(message.modelName)?.has(message.queryKey) ?? false;
|
|
742
|
+
if (!existing && getSocketSubscriptionCount(socketId) >= maxSubscriptionsPerSocket) {
|
|
743
|
+
sendWs(ws, {
|
|
744
|
+
type: "query-payload",
|
|
745
|
+
modelName: message.modelName,
|
|
746
|
+
queryKey: message.queryKey,
|
|
747
|
+
error: "Too many subscriptions"
|
|
748
|
+
});
|
|
749
|
+
return;
|
|
750
|
+
}
|
|
751
|
+
let dependencyModelNames = [];
|
|
752
|
+
if (options.populate !== void 0) {
|
|
753
|
+
try {
|
|
754
|
+
dependencyModelNames = await resolveRtsQueryDependencyModelNames({
|
|
755
|
+
tenantId: meta.tenantId,
|
|
756
|
+
ability,
|
|
757
|
+
modelName: message.modelName,
|
|
758
|
+
options,
|
|
759
|
+
allowInternalModels
|
|
760
|
+
});
|
|
761
|
+
} catch (err) {
|
|
762
|
+
const error = redactErrorMessage(err);
|
|
763
|
+
sendWs(ws, {
|
|
764
|
+
type: "query-payload",
|
|
765
|
+
modelName: message.modelName,
|
|
766
|
+
queryKey: message.queryKey,
|
|
767
|
+
error
|
|
768
|
+
});
|
|
769
|
+
return;
|
|
770
|
+
}
|
|
771
|
+
}
|
|
772
|
+
addSocketSubscription({
|
|
773
|
+
socketId,
|
|
774
|
+
tenantId: meta.tenantId,
|
|
775
|
+
userId: meta.userId,
|
|
776
|
+
modelName: message.modelName,
|
|
777
|
+
queryKey: message.queryKey,
|
|
778
|
+
query: message.query,
|
|
779
|
+
options,
|
|
780
|
+
dependencyModelNames
|
|
781
|
+
});
|
|
782
|
+
try {
|
|
783
|
+
const modelNamesToWatch = /* @__PURE__ */ new Set([message.modelName, ...dependencyModelNames]);
|
|
784
|
+
for (const modelName of modelNamesToWatch) {
|
|
785
|
+
await ensureChangeStream(meta.tenantId, modelName);
|
|
786
|
+
}
|
|
787
|
+
} catch (err) {
|
|
788
|
+
const error = redactErrorMessage(err);
|
|
789
|
+
sendWs(ws, {
|
|
790
|
+
type: "query-payload",
|
|
791
|
+
modelName: message.modelName,
|
|
792
|
+
queryKey: message.queryKey,
|
|
793
|
+
error
|
|
794
|
+
});
|
|
795
|
+
return;
|
|
796
|
+
}
|
|
797
|
+
if (message.runInitialQuery === false) {
|
|
798
|
+
return;
|
|
799
|
+
}
|
|
800
|
+
}
|
|
801
|
+
try {
|
|
802
|
+
await runAndSendQuery({
|
|
803
|
+
tenantId: meta.tenantId,
|
|
804
|
+
targetSocketIds: [socketId],
|
|
805
|
+
ability,
|
|
806
|
+
modelName: message.modelName,
|
|
807
|
+
queryKey: message.queryKey,
|
|
808
|
+
query: message.query,
|
|
809
|
+
options
|
|
810
|
+
});
|
|
811
|
+
} catch (err) {
|
|
812
|
+
const error = redactErrorMessage(err);
|
|
813
|
+
sendWs(ws, {
|
|
814
|
+
type: "query-payload",
|
|
815
|
+
modelName: message.modelName,
|
|
816
|
+
queryKey: message.queryKey,
|
|
817
|
+
error
|
|
818
|
+
});
|
|
819
|
+
}
|
|
820
|
+
return;
|
|
821
|
+
}
|
|
822
|
+
if (message.type === "register-count" || message.type === "run-count") {
|
|
823
|
+
if (message.type === "register-count") {
|
|
824
|
+
const existing = socketCountSubscriptions.get(socketId)?.get(message.modelName)?.has(message.queryKey) ?? false;
|
|
825
|
+
if (!existing && getSocketSubscriptionCount(socketId) >= maxSubscriptionsPerSocket) {
|
|
826
|
+
sendWs(ws, {
|
|
827
|
+
type: "count-payload",
|
|
828
|
+
modelName: message.modelName,
|
|
829
|
+
queryKey: message.queryKey,
|
|
830
|
+
error: "Too many subscriptions"
|
|
831
|
+
});
|
|
832
|
+
return;
|
|
833
|
+
}
|
|
834
|
+
addSocketCountSubscription({
|
|
835
|
+
socketId,
|
|
836
|
+
tenantId: meta.tenantId,
|
|
837
|
+
userId: meta.userId,
|
|
838
|
+
modelName: message.modelName,
|
|
839
|
+
queryKey: message.queryKey,
|
|
840
|
+
query: message.query
|
|
841
|
+
});
|
|
842
|
+
try {
|
|
843
|
+
await ensureChangeStream(meta.tenantId, message.modelName);
|
|
844
|
+
} catch (err) {
|
|
845
|
+
const error = redactErrorMessage(err);
|
|
846
|
+
sendWs(ws, {
|
|
847
|
+
type: "count-payload",
|
|
848
|
+
modelName: message.modelName,
|
|
849
|
+
queryKey: message.queryKey,
|
|
850
|
+
error
|
|
851
|
+
});
|
|
852
|
+
return;
|
|
853
|
+
}
|
|
854
|
+
if (message.runInitialQuery === false) {
|
|
855
|
+
return;
|
|
856
|
+
}
|
|
857
|
+
}
|
|
858
|
+
try {
|
|
859
|
+
await runAndSendCount({
|
|
860
|
+
tenantId: meta.tenantId,
|
|
861
|
+
targetSocketIds: [socketId],
|
|
862
|
+
ability,
|
|
863
|
+
modelName: message.modelName,
|
|
864
|
+
queryKey: message.queryKey,
|
|
865
|
+
query: message.query
|
|
866
|
+
});
|
|
867
|
+
} catch (err) {
|
|
868
|
+
const error = redactErrorMessage(err);
|
|
869
|
+
sendWs(ws, {
|
|
870
|
+
type: "count-payload",
|
|
871
|
+
modelName: message.modelName,
|
|
872
|
+
queryKey: message.queryKey,
|
|
873
|
+
error
|
|
874
|
+
});
|
|
875
|
+
}
|
|
876
|
+
}
|
|
877
|
+
};
|
|
878
|
+
const initRts = ({
|
|
879
|
+
server,
|
|
880
|
+
path = "/rts",
|
|
881
|
+
sessionMiddleware,
|
|
882
|
+
maxPayloadBytes: maxPayloadBytesArg,
|
|
883
|
+
maxSubscriptionsPerSocket: maxSubscriptionsPerSocketArg,
|
|
884
|
+
dispatchDebounceMs: dispatchDebounceMsArg,
|
|
885
|
+
allowInternalModels: allowInternalModelsArg
|
|
886
|
+
}) => {
|
|
887
|
+
if (initializedServers.has(server)) return;
|
|
888
|
+
initializedServers.add(server);
|
|
889
|
+
if (typeof maxPayloadBytesArg === "number" && Number.isFinite(maxPayloadBytesArg) && maxPayloadBytesArg > 0) {
|
|
890
|
+
maxPayloadBytes = Math.floor(maxPayloadBytesArg);
|
|
891
|
+
}
|
|
892
|
+
if (typeof maxSubscriptionsPerSocketArg === "number" && Number.isFinite(maxSubscriptionsPerSocketArg) && maxSubscriptionsPerSocketArg > 0) {
|
|
893
|
+
maxSubscriptionsPerSocket = Math.floor(maxSubscriptionsPerSocketArg);
|
|
894
|
+
}
|
|
895
|
+
if (typeof dispatchDebounceMsArg === "number" && Number.isFinite(dispatchDebounceMsArg) && dispatchDebounceMsArg >= 0) {
|
|
896
|
+
dispatchDebounceMs = Math.floor(dispatchDebounceMsArg);
|
|
897
|
+
}
|
|
898
|
+
allowInternalModels = Boolean(allowInternalModelsArg);
|
|
899
|
+
const wss = new WebSocketServer({
|
|
900
|
+
noServer: true,
|
|
901
|
+
maxPayload: maxPayloadBytes
|
|
902
|
+
});
|
|
903
|
+
server.on("upgrade", (req, socket, head) => {
|
|
904
|
+
ensureSocketErrorHandler(socket, () => ({
|
|
905
|
+
upgradeHost: req.headers.host ?? "",
|
|
906
|
+
upgradeUrl: req.url ?? "",
|
|
907
|
+
userAgent: typeof req.headers["user-agent"] === "string" ? req.headers["user-agent"] : ""
|
|
908
|
+
}));
|
|
909
|
+
upgradeMeta.delete(req);
|
|
910
|
+
let url;
|
|
911
|
+
try {
|
|
912
|
+
url = new URL(req.url ?? "", `http://${req.headers.host ?? "localhost"}`);
|
|
913
|
+
} catch {
|
|
914
|
+
badRequest(socket, "Invalid URL");
|
|
915
|
+
return;
|
|
916
|
+
}
|
|
917
|
+
if (url.pathname !== path) return;
|
|
918
|
+
void (async () => {
|
|
919
|
+
try {
|
|
920
|
+
const meta = await parseUpgradeMeta({
|
|
921
|
+
req,
|
|
922
|
+
url,
|
|
923
|
+
sessionMiddleware
|
|
924
|
+
});
|
|
925
|
+
upgradeMeta.set(req, meta);
|
|
926
|
+
wss.handleUpgrade(req, socket, head, (ws) => {
|
|
927
|
+
wss.emit("connection", ws, req);
|
|
928
|
+
});
|
|
929
|
+
} catch (err) {
|
|
930
|
+
const message = err instanceof Error ? err.message : "RTS upgrade failed";
|
|
931
|
+
if (message.startsWith("Missing rb-tenant-id")) {
|
|
932
|
+
badRequest(socket, message);
|
|
933
|
+
return;
|
|
934
|
+
}
|
|
935
|
+
unauthorized(socket, message);
|
|
936
|
+
return;
|
|
937
|
+
}
|
|
938
|
+
})().catch(() => {
|
|
939
|
+
badRequest(socket, "RTS upgrade failed");
|
|
940
|
+
});
|
|
941
|
+
});
|
|
942
|
+
wss.on("connection", (ws, req) => {
|
|
943
|
+
const meta = upgradeMeta.get(req);
|
|
944
|
+
upgradeMeta.delete(req);
|
|
945
|
+
if (!meta) {
|
|
946
|
+
try {
|
|
947
|
+
ws.close();
|
|
948
|
+
} catch {
|
|
949
|
+
}
|
|
950
|
+
return;
|
|
951
|
+
}
|
|
952
|
+
const socketId = randomUUID();
|
|
953
|
+
sockets.set(socketId, ws);
|
|
954
|
+
socketMeta.set(socketId, meta);
|
|
955
|
+
const wrapper = new RtsSocket({
|
|
956
|
+
id: socketId,
|
|
957
|
+
ws,
|
|
958
|
+
meta
|
|
959
|
+
});
|
|
960
|
+
socketWrappers.set(socketId, wrapper);
|
|
961
|
+
const cleanupFns = [];
|
|
962
|
+
for (const handler of customHandlers) {
|
|
963
|
+
try {
|
|
964
|
+
const cleanup = handler(wrapper);
|
|
965
|
+
if (typeof cleanup === "function") cleanupFns.push(cleanup);
|
|
966
|
+
} catch {
|
|
967
|
+
}
|
|
968
|
+
}
|
|
969
|
+
if (cleanupFns.length) socketCleanup.set(socketId, cleanupFns);
|
|
970
|
+
ws.on("message", (raw) => {
|
|
971
|
+
let parsed;
|
|
972
|
+
try {
|
|
973
|
+
parsed = safeJsonParse(raw);
|
|
974
|
+
} catch {
|
|
975
|
+
return;
|
|
976
|
+
}
|
|
977
|
+
if (!parsed || typeof parsed !== "object") return;
|
|
978
|
+
const message = parsed;
|
|
979
|
+
if (message.type !== "event" && message.type !== "run-query" && message.type !== "register-query" && message.type !== "remove-query" && message.type !== "run-count" && message.type !== "register-count" && message.type !== "remove-count") return;
|
|
980
|
+
void handleClientMessage({
|
|
981
|
+
socketId,
|
|
982
|
+
meta,
|
|
983
|
+
message
|
|
984
|
+
});
|
|
985
|
+
});
|
|
986
|
+
ws.on("close", () => {
|
|
987
|
+
cleanupSocket(socketId);
|
|
988
|
+
});
|
|
989
|
+
ws.on("error", () => {
|
|
990
|
+
cleanupSocket(socketId);
|
|
991
|
+
});
|
|
992
|
+
});
|
|
993
|
+
};
|
|
994
|
+
const registerRtsHandler = (handler) => {
|
|
995
|
+
customHandlers.push(handler);
|
|
996
|
+
};
|
|
997
|
+
const notifyRtsModelChanged = (tenantId, modelName) => {
|
|
998
|
+
scheduleDispatchSubscriptionsForModel(tenantId, modelName);
|
|
999
|
+
};
|
|
1000
|
+
export {
|
|
1001
|
+
initRts,
|
|
1002
|
+
notifyRtsModelChanged,
|
|
1003
|
+
registerRtsHandler,
|
|
1004
|
+
routes
|
|
1005
|
+
};
|
|
1006
|
+
//# sourceMappingURL=index.js.map
|