@cloudbase/realtime 1.2.2-alpha.0 → 1.8.1
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/.eslintrc.js +15 -0
- package/dist/cjs/common.js +1 -1
- package/dist/cjs/error.js +17 -16
- package/dist/cjs/index.js +5 -5
- package/dist/cjs/listener.js +2 -4
- package/dist/cjs/message.js +2 -2
- package/dist/cjs/snapshot.js +7 -7
- package/dist/cjs/utils.js +3 -2
- package/dist/cjs/virtual-websocket-client.js +86 -83
- package/dist/cjs/websocket-client.js +162 -176
- package/dist/cjs/ws-event.js +27 -26
- package/dist/esm/common.js +1 -1
- package/dist/esm/error.js +13 -15
- package/dist/esm/index.js +3 -3
- package/dist/esm/listener.js +2 -4
- package/dist/esm/message.js +2 -2
- package/dist/esm/snapshot.js +7 -7
- package/dist/esm/utils.js +1 -1
- package/dist/esm/virtual-websocket-client.js +75 -72
- package/dist/esm/websocket-client.js +156 -170
- package/dist/esm/ws-event.js +25 -25
- package/package.json +6 -22
- package/src/common.ts +5 -5
- package/src/error.ts +34 -40
- package/src/index.ts +15 -15
- package/src/listener.ts +11 -13
- package/src/message.ts +5 -7
- package/src/snapshot.ts +20 -20
- package/src/utils.ts +1 -1
- package/src/virtual-websocket-client.ts +446 -514
- package/src/websocket-client.ts +493 -518
- package/src/ws-event.ts +28 -28
- package/tsconfig.esm.json +2 -20
- package/tsconfig.json +2 -20
- package/.eslintrc +0 -30
|
@@ -25,7 +25,7 @@ var __generator = (this && this.__generator) || function (thisArg, body) {
|
|
|
25
25
|
function verb(n) { return function (v) { return step([n, v]); }; }
|
|
26
26
|
function step(op) {
|
|
27
27
|
if (f) throw new TypeError("Generator is already executing.");
|
|
28
|
-
while (_) try {
|
|
28
|
+
while (g && (g = 0, op[0] && (_ = 0)), _) try {
|
|
29
29
|
if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
|
|
30
30
|
if (y = 0, t) op = [op[0] & 2, t.value];
|
|
31
31
|
switch (op[0]) {
|
|
@@ -58,7 +58,7 @@ var WS_READY_STATE = {
|
|
|
58
58
|
CONNECTING: 0,
|
|
59
59
|
OPEN: 1,
|
|
60
60
|
CLOSING: 2,
|
|
61
|
-
CLOSED: 3
|
|
61
|
+
CLOSED: 3,
|
|
62
62
|
};
|
|
63
63
|
var MAX_RTT_OBSERVED = 3;
|
|
64
64
|
var DEFAULT_EXPECTED_EVENT_WAIT_TIME = 5000;
|
|
@@ -107,7 +107,7 @@ var RealtimeWebSocketClient = (function () {
|
|
|
107
107
|
switch (_a.label) {
|
|
108
108
|
case 0:
|
|
109
109
|
if (!(!_hasResolved || !_hasRejected)) return [3, 2];
|
|
110
|
-
return [4, utils_1.sleep(0)];
|
|
110
|
+
return [4, (0, utils_1.sleep)(0)];
|
|
111
111
|
case 1:
|
|
112
112
|
_a.sent();
|
|
113
113
|
if (!_hasResolved || !_hasRejected) {
|
|
@@ -133,14 +133,14 @@ var RealtimeWebSocketClient = (function () {
|
|
|
133
133
|
return [2];
|
|
134
134
|
}
|
|
135
135
|
if (this._ws.readyState !== WS_READY_STATE.OPEN) {
|
|
136
|
-
reject(new Error("ws readyState invalid: "
|
|
136
|
+
reject(new Error("ws readyState invalid: ".concat(this._ws.readyState, ", can not send message")));
|
|
137
137
|
return [2];
|
|
138
138
|
}
|
|
139
139
|
if (opts.waitResponse) {
|
|
140
140
|
this._wsResponseWait.set(opts.msg.requestId, {
|
|
141
141
|
resolve: resolve,
|
|
142
142
|
reject: reject,
|
|
143
|
-
skipOnMessage: opts.skipOnMessage
|
|
143
|
+
skipOnMessage: opts.skipOnMessage,
|
|
144
144
|
});
|
|
145
145
|
}
|
|
146
146
|
_a.label = 4;
|
|
@@ -150,7 +150,7 @@ var RealtimeWebSocketClient = (function () {
|
|
|
150
150
|
case 5:
|
|
151
151
|
_a.sent();
|
|
152
152
|
if (!opts.waitResponse) {
|
|
153
|
-
resolve();
|
|
153
|
+
resolve(undefined);
|
|
154
154
|
}
|
|
155
155
|
return [3, 7];
|
|
156
156
|
case 6:
|
|
@@ -179,13 +179,11 @@ var RealtimeWebSocketClient = (function () {
|
|
|
179
179
|
});
|
|
180
180
|
};
|
|
181
181
|
this.pauseClients = function (clients) {
|
|
182
|
-
;
|
|
183
182
|
(clients || _this._virtualWSClient).forEach(function (client) {
|
|
184
183
|
client.pause();
|
|
185
184
|
});
|
|
186
185
|
};
|
|
187
186
|
this.resumeClients = function (clients) {
|
|
188
|
-
;
|
|
189
187
|
(clients || _this._virtualWSClient).forEach(function (client) {
|
|
190
188
|
client.resume();
|
|
191
189
|
});
|
|
@@ -223,9 +221,9 @@ var RealtimeWebSocketClient = (function () {
|
|
|
223
221
|
wsSign_1 = _a.sent();
|
|
224
222
|
return [4, new Promise(function (success) {
|
|
225
223
|
var url = wsSign_1.wsUrl || 'wss://tcb-ws.tencentcloudapi.com';
|
|
226
|
-
var wsClass = common_1.getWsClass();
|
|
224
|
+
var wsClass = (0, common_1.getWsClass)();
|
|
227
225
|
_this._ws = wsClass ? new wsClass(url) : new WebSocket(url);
|
|
228
|
-
success();
|
|
226
|
+
success(undefined);
|
|
229
227
|
})];
|
|
230
228
|
case 2:
|
|
231
229
|
_a.sent();
|
|
@@ -250,7 +248,7 @@ var RealtimeWebSocketClient = (function () {
|
|
|
250
248
|
isConnected = true;
|
|
251
249
|
this._wsInitPromise = undefined;
|
|
252
250
|
if (!isConnected) return [3, 8];
|
|
253
|
-
return [4, utils_1.sleep(this._reconnectInterval)];
|
|
251
|
+
return [4, (0, utils_1.sleep)(this._reconnectInterval)];
|
|
254
252
|
case 7:
|
|
255
253
|
_a.sent();
|
|
256
254
|
if (reconnect) {
|
|
@@ -265,7 +263,7 @@ var RealtimeWebSocketClient = (function () {
|
|
|
265
263
|
if (reconnect) {
|
|
266
264
|
this.closeAllClients(new error_1.CloudSDKError({
|
|
267
265
|
errCode: error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_RECONNECT_WATCH_FAIL,
|
|
268
|
-
errMsg: e_3
|
|
266
|
+
errMsg: e_3,
|
|
269
267
|
}));
|
|
270
268
|
}
|
|
271
269
|
_a.label = 10;
|
|
@@ -301,158 +299,152 @@ var RealtimeWebSocketClient = (function () {
|
|
|
301
299
|
});
|
|
302
300
|
});
|
|
303
301
|
};
|
|
304
|
-
this.initWebSocketEvent = function () {
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
302
|
+
this.initWebSocketEvent = function () { return new Promise(function (resolve, reject) {
|
|
303
|
+
if (!_this._ws) {
|
|
304
|
+
throw new Error('can not initWebSocketEvent, ws not exists');
|
|
305
|
+
}
|
|
306
|
+
var wsOpened = false;
|
|
307
|
+
_this._ws.onopen = function (event) {
|
|
308
|
+
console.warn('[realtime] ws event: open', event);
|
|
309
|
+
wsOpened = true;
|
|
310
|
+
resolve();
|
|
311
|
+
};
|
|
312
|
+
_this._ws.onerror = function (event) {
|
|
313
|
+
_this._logins = new Map();
|
|
314
|
+
if (!wsOpened) {
|
|
315
|
+
console.error('[realtime] ws open failed with ws event: error', event);
|
|
316
|
+
reject(event);
|
|
317
|
+
}
|
|
318
|
+
else {
|
|
319
|
+
console.error('[realtime] ws event: error', event);
|
|
320
|
+
_this.clearHeartbeat();
|
|
321
|
+
_this._virtualWSClient.forEach(function (client) { return client.closeWithError(new error_1.CloudSDKError({
|
|
322
|
+
errCode: error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_WEBSOCKET_CONNECTION_ERROR,
|
|
323
|
+
errMsg: event,
|
|
324
|
+
})); });
|
|
308
325
|
}
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
if (!wsOpened) {
|
|
318
|
-
console.error('[realtime] ws open failed with ws event: error', event);
|
|
319
|
-
reject(event);
|
|
326
|
+
};
|
|
327
|
+
_this._ws.onclose = function (closeEvent) {
|
|
328
|
+
console.warn('[realtime] ws event: close', closeEvent);
|
|
329
|
+
_this._logins = new Map();
|
|
330
|
+
_this.clearHeartbeat();
|
|
331
|
+
switch (closeEvent.code) {
|
|
332
|
+
case ws_event_1.CLOSE_EVENT_CODE.ReconnectWebSocket: {
|
|
333
|
+
break;
|
|
320
334
|
}
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
_this.clearHeartbeat();
|
|
324
|
-
_this._virtualWSClient.forEach(function (client) {
|
|
325
|
-
return client.closeWithError(new error_1.CloudSDKError({
|
|
326
|
-
errCode: error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_WEBSOCKET_CONNECTION_ERROR,
|
|
327
|
-
errMsg: event
|
|
328
|
-
}));
|
|
329
|
-
});
|
|
335
|
+
case ws_event_1.CLOSE_EVENT_CODE.NoRealtimeListeners: {
|
|
336
|
+
break;
|
|
330
337
|
}
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
case ws_event_1.CLOSE_EVENT_CODE.ReconnectWebSocket: {
|
|
338
|
-
break;
|
|
338
|
+
case ws_event_1.CLOSE_EVENT_CODE.HeartbeatPingError:
|
|
339
|
+
case ws_event_1.CLOSE_EVENT_CODE.HeartbeatPongTimeoutError:
|
|
340
|
+
case ws_event_1.CLOSE_EVENT_CODE.NormalClosure:
|
|
341
|
+
case ws_event_1.CLOSE_EVENT_CODE.AbnormalClosure: {
|
|
342
|
+
if (_this._maxReconnect > 0) {
|
|
343
|
+
_this.initWebSocketConnection(true, _this._maxReconnect);
|
|
339
344
|
}
|
|
340
|
-
|
|
341
|
-
|
|
345
|
+
else {
|
|
346
|
+
_this.closeAllClients((0, ws_event_1.getWSCloseError)(closeEvent.code));
|
|
342
347
|
}
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
}
|
|
353
|
-
break;
|
|
348
|
+
break;
|
|
349
|
+
}
|
|
350
|
+
case ws_event_1.CLOSE_EVENT_CODE.NoAuthentication: {
|
|
351
|
+
_this.closeAllClients((0, ws_event_1.getWSCloseError)(closeEvent.code, closeEvent.reason));
|
|
352
|
+
break;
|
|
353
|
+
}
|
|
354
|
+
default: {
|
|
355
|
+
if (_this._maxReconnect > 0) {
|
|
356
|
+
_this.initWebSocketConnection(true, _this._maxReconnect);
|
|
354
357
|
}
|
|
355
|
-
|
|
356
|
-
_this.closeAllClients(ws_event_1.getWSCloseError(closeEvent.code
|
|
357
|
-
break;
|
|
358
|
+
else {
|
|
359
|
+
_this.closeAllClients((0, ws_event_1.getWSCloseError)(closeEvent.code));
|
|
358
360
|
}
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
361
|
+
}
|
|
362
|
+
}
|
|
363
|
+
};
|
|
364
|
+
_this._ws.onmessage = function (res) {
|
|
365
|
+
var rawMsg = res.data;
|
|
366
|
+
_this.heartbeat();
|
|
367
|
+
var msg;
|
|
368
|
+
try {
|
|
369
|
+
msg = JSON.parse(rawMsg);
|
|
370
|
+
}
|
|
371
|
+
catch (e) {
|
|
372
|
+
throw new Error("[realtime] onMessage parse res.data error: ".concat(e));
|
|
373
|
+
}
|
|
374
|
+
if (msg.msgType === 'ERROR') {
|
|
375
|
+
var virtualWatch_1 = null;
|
|
376
|
+
_this._virtualWSClient.forEach(function (item) {
|
|
377
|
+
if (item.watchId === msg.watchId) {
|
|
378
|
+
virtualWatch_1 = item;
|
|
366
379
|
}
|
|
380
|
+
});
|
|
381
|
+
if (virtualWatch_1) {
|
|
382
|
+
virtualWatch_1.listener.onError(msg);
|
|
367
383
|
}
|
|
368
|
-
}
|
|
369
|
-
_this.
|
|
370
|
-
|
|
371
|
-
_this.heartbeat();
|
|
372
|
-
var msg;
|
|
384
|
+
}
|
|
385
|
+
var responseWaitSpec = _this._wsResponseWait.get(msg.requestId);
|
|
386
|
+
if (responseWaitSpec) {
|
|
373
387
|
try {
|
|
374
|
-
msg
|
|
388
|
+
if (msg.msgType === 'ERROR') {
|
|
389
|
+
responseWaitSpec.reject(new error_1.RealtimeErrorMessageError(msg));
|
|
390
|
+
}
|
|
391
|
+
else {
|
|
392
|
+
responseWaitSpec.resolve(msg);
|
|
393
|
+
}
|
|
375
394
|
}
|
|
376
395
|
catch (e) {
|
|
377
|
-
|
|
396
|
+
console.error('ws onMessage responseWaitSpec.resolve(msg) errored:', e);
|
|
378
397
|
}
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
_this._virtualWSClient.forEach(function (item) {
|
|
382
|
-
if (item.watchId === msg.watchId) {
|
|
383
|
-
virtualWatch_1 = item;
|
|
384
|
-
}
|
|
385
|
-
});
|
|
386
|
-
if (virtualWatch_1) {
|
|
387
|
-
virtualWatch_1.listener.onError(msg);
|
|
388
|
-
}
|
|
398
|
+
finally {
|
|
399
|
+
_this._wsResponseWait.delete(msg.requestId);
|
|
389
400
|
}
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
}
|
|
400
|
-
catch (e) {
|
|
401
|
-
console.error('ws onMessage responseWaitSpec.resolve(msg) errored:', e);
|
|
402
|
-
}
|
|
403
|
-
finally {
|
|
404
|
-
_this._wsResponseWait.delete(msg.requestId);
|
|
405
|
-
}
|
|
406
|
-
if (responseWaitSpec.skipOnMessage) {
|
|
401
|
+
if (responseWaitSpec.skipOnMessage) {
|
|
402
|
+
return;
|
|
403
|
+
}
|
|
404
|
+
}
|
|
405
|
+
if (msg.msgType === 'PONG') {
|
|
406
|
+
if (_this._lastPingSendTS) {
|
|
407
|
+
var rtt = Date.now() - _this._lastPingSendTS;
|
|
408
|
+
if (rtt > DEFAULT_UNTRUSTED_RTT_THRESHOLD) {
|
|
409
|
+
console.warn("[realtime] untrusted rtt observed: ".concat(rtt));
|
|
407
410
|
return;
|
|
408
411
|
}
|
|
412
|
+
if (_this._rttObserved.length >= MAX_RTT_OBSERVED) {
|
|
413
|
+
_this._rttObserved.splice(0, _this._rttObserved.length - MAX_RTT_OBSERVED + 1);
|
|
414
|
+
}
|
|
415
|
+
_this._rttObserved.push(rtt);
|
|
409
416
|
}
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
417
|
+
return;
|
|
418
|
+
}
|
|
419
|
+
var client = msg.watchId && _this._watchIdClientMap.get(msg.watchId);
|
|
420
|
+
if (client) {
|
|
421
|
+
client.onMessage(msg);
|
|
422
|
+
}
|
|
423
|
+
else {
|
|
424
|
+
console.error("[realtime] no realtime listener found responsible for watchId ".concat(msg.watchId, ": "), msg);
|
|
425
|
+
switch (msg.msgType) {
|
|
426
|
+
case 'INIT_EVENT':
|
|
427
|
+
case 'NEXT_EVENT':
|
|
428
|
+
case 'CHECK_EVENT': {
|
|
429
|
+
client = _this._queryIdClientMap.get(msg.msgData.queryID);
|
|
430
|
+
if (client) {
|
|
431
|
+
client.onMessage(msg);
|
|
419
432
|
}
|
|
420
|
-
|
|
433
|
+
break;
|
|
421
434
|
}
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
|
|
425
|
-
|
|
426
|
-
client.onMessage(msg);
|
|
427
|
-
}
|
|
428
|
-
else {
|
|
429
|
-
console.error("[realtime] no realtime listener found responsible for watchId " + msg.watchId + ": ", msg);
|
|
430
|
-
switch (msg.msgType) {
|
|
431
|
-
case 'INIT_EVENT':
|
|
432
|
-
case 'NEXT_EVENT':
|
|
433
|
-
case 'CHECK_EVENT': {
|
|
434
|
-
client = _this._queryIdClientMap.get(msg.msgData.queryID);
|
|
435
|
-
if (client) {
|
|
436
|
-
client.onMessage(msg);
|
|
437
|
-
}
|
|
435
|
+
default: {
|
|
436
|
+
for (var _i = 0, _a = Array.from(_this._watchIdClientMap.entries()); _i < _a.length; _i++) {
|
|
437
|
+
var _b = _a[_i], client_1 = _b[1];
|
|
438
|
+
client_1.onMessage(msg);
|
|
438
439
|
break;
|
|
439
440
|
}
|
|
440
|
-
default: {
|
|
441
|
-
for (var _i = 0, _a = Array.from(_this._watchIdClientMap.entries()); _i < _a.length; _i++) {
|
|
442
|
-
var _b = _a[_i], client_1 = _b[1];
|
|
443
|
-
client_1.onMessage(msg);
|
|
444
|
-
break;
|
|
445
|
-
}
|
|
446
|
-
}
|
|
447
441
|
}
|
|
448
442
|
}
|
|
449
|
-
}
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
};
|
|
453
|
-
this.isWSConnected = function () {
|
|
454
|
-
return Boolean(_this._ws && _this._ws.readyState === WS_READY_STATE.OPEN);
|
|
455
|
-
};
|
|
443
|
+
}
|
|
444
|
+
};
|
|
445
|
+
_this.heartbeat();
|
|
446
|
+
}); };
|
|
447
|
+
this.isWSConnected = function () { return Boolean(_this._ws && _this._ws.readyState === WS_READY_STATE.OPEN); };
|
|
456
448
|
this.onceWSConnected = function () { return __awaiter(_this, void 0, void 0, function () {
|
|
457
449
|
var _this = this;
|
|
458
450
|
return __generator(this, function (_a) {
|
|
@@ -465,7 +457,7 @@ var RealtimeWebSocketClient = (function () {
|
|
|
465
457
|
return [2, new Promise(function (resolve, reject) {
|
|
466
458
|
_this._wsReadySubsribers.push({
|
|
467
459
|
resolve: resolve,
|
|
468
|
-
reject: reject
|
|
460
|
+
reject: reject,
|
|
469
461
|
});
|
|
470
462
|
})];
|
|
471
463
|
});
|
|
@@ -483,7 +475,7 @@ var RealtimeWebSocketClient = (function () {
|
|
|
483
475
|
if (loginInfo_1.loggedIn && loginInfo_1.loginResult) {
|
|
484
476
|
return [2, loginInfo_1.loginResult];
|
|
485
477
|
}
|
|
486
|
-
|
|
478
|
+
if (loginInfo_1.loggingInPromise) {
|
|
487
479
|
return [2, loginInfo_1.loggingInPromise];
|
|
488
480
|
}
|
|
489
481
|
}
|
|
@@ -509,34 +501,34 @@ var RealtimeWebSocketClient = (function () {
|
|
|
509
501
|
accessToken: '',
|
|
510
502
|
referrer: 'web',
|
|
511
503
|
sdkVersion: '',
|
|
512
|
-
dataVersion: ''
|
|
504
|
+
dataVersion: '',
|
|
513
505
|
};
|
|
514
506
|
loginMsg = {
|
|
515
507
|
watchId: undefined,
|
|
516
|
-
requestId: message_1.genRequestId(),
|
|
508
|
+
requestId: (0, message_1.genRequestId)(),
|
|
517
509
|
msgType: 'LOGIN',
|
|
518
510
|
msgData: msgData,
|
|
519
511
|
exMsgData: {
|
|
520
|
-
runtime: common_1.getRuntime(),
|
|
512
|
+
runtime: (0, common_1.getRuntime)(),
|
|
521
513
|
signStr: wsSign.signStr,
|
|
522
|
-
secretVersion: wsSign.secretVersion
|
|
523
|
-
}
|
|
514
|
+
secretVersion: wsSign.secretVersion,
|
|
515
|
+
},
|
|
524
516
|
};
|
|
525
517
|
return [4, this.send({
|
|
526
518
|
msg: loginMsg,
|
|
527
519
|
waitResponse: true,
|
|
528
520
|
skipOnMessage: true,
|
|
529
|
-
timeout: DEFAULT_LOGIN_TIMEOUT
|
|
521
|
+
timeout: DEFAULT_LOGIN_TIMEOUT,
|
|
530
522
|
})];
|
|
531
523
|
case 2:
|
|
532
524
|
loginResMsg = _a.sent();
|
|
533
525
|
if (!loginResMsg.msgData.code) {
|
|
534
526
|
resolve({
|
|
535
|
-
envId: wsSign.envId
|
|
527
|
+
envId: wsSign.envId,
|
|
536
528
|
});
|
|
537
529
|
}
|
|
538
530
|
else {
|
|
539
|
-
reject(new Error(loginResMsg.msgData.code
|
|
531
|
+
reject(new Error("".concat(loginResMsg.msgData.code, " ").concat(loginResMsg.msgData.message)));
|
|
540
532
|
}
|
|
541
533
|
return [3, 4];
|
|
542
534
|
case 3:
|
|
@@ -558,7 +550,7 @@ var RealtimeWebSocketClient = (function () {
|
|
|
558
550
|
loginInfo = {
|
|
559
551
|
loggedIn: false,
|
|
560
552
|
loggingInPromise: promise,
|
|
561
|
-
loginStartTS: loginStartTS
|
|
553
|
+
loginStartTS: loginStartTS,
|
|
562
554
|
};
|
|
563
555
|
this._logins.set(envId || '', loginInfo);
|
|
564
556
|
}
|
|
@@ -569,25 +561,23 @@ var RealtimeWebSocketClient = (function () {
|
|
|
569
561
|
case 2:
|
|
570
562
|
loginResult = _a.sent();
|
|
571
563
|
curLoginInfo = envId && this._logins.get(envId);
|
|
572
|
-
if (curLoginInfo
|
|
573
|
-
curLoginInfo === loginInfo
|
|
574
|
-
curLoginInfo.loginStartTS === loginStartTS) {
|
|
564
|
+
if (curLoginInfo
|
|
565
|
+
&& curLoginInfo === loginInfo
|
|
566
|
+
&& curLoginInfo.loginStartTS === loginStartTS) {
|
|
575
567
|
loginInfo.loggedIn = true;
|
|
576
568
|
loginInfo.loggingInPromise = undefined;
|
|
577
569
|
loginInfo.loginStartTS = undefined;
|
|
578
570
|
loginInfo.loginResult = loginResult;
|
|
579
571
|
return [2, loginResult];
|
|
580
572
|
}
|
|
581
|
-
|
|
573
|
+
if (curLoginInfo) {
|
|
582
574
|
if (curLoginInfo.loggedIn && curLoginInfo.loginResult) {
|
|
583
575
|
return [2, curLoginInfo.loginResult];
|
|
584
576
|
}
|
|
585
|
-
|
|
577
|
+
if (curLoginInfo.loggingInPromise) {
|
|
586
578
|
return [2, curLoginInfo.loggingInPromise];
|
|
587
579
|
}
|
|
588
|
-
|
|
589
|
-
throw new Error('ws unexpected login info');
|
|
590
|
-
}
|
|
580
|
+
throw new Error('ws unexpected login info');
|
|
591
581
|
}
|
|
592
582
|
else {
|
|
593
583
|
throw new Error('ws login info reset');
|
|
@@ -613,11 +603,11 @@ var RealtimeWebSocketClient = (function () {
|
|
|
613
603
|
return [2, this._wsSign];
|
|
614
604
|
}
|
|
615
605
|
expiredTs = Date.now() + 60000;
|
|
616
|
-
return [4, this._context.appConfig.request.send('auth.wsWebSign', { runtime: common_1.getRuntime() })];
|
|
606
|
+
return [4, this._context.appConfig.request.send('auth.wsWebSign', { runtime: (0, common_1.getRuntime)() })];
|
|
617
607
|
case 1:
|
|
618
608
|
res = _b.sent();
|
|
619
609
|
if (res.code) {
|
|
620
|
-
throw new Error("[tcb-js-sdk] \u83B7\u53D6\u5B9E\u65F6\u6570\u636E\u63A8\u9001\u767B\u5F55\u7968\u636E\u5931\u8D25: "
|
|
610
|
+
throw new Error("[tcb-js-sdk] \u83B7\u53D6\u5B9E\u65F6\u6570\u636E\u63A8\u9001\u767B\u5F55\u7968\u636E\u5931\u8D25: ".concat(res.code));
|
|
621
611
|
}
|
|
622
612
|
if (res.data) {
|
|
623
613
|
_a = res.data, signStr = _a.signStr, wsUrl = _a.wsUrl, secretVersion = _a.secretVersion, envId = _a.envId;
|
|
@@ -626,13 +616,10 @@ var RealtimeWebSocketClient = (function () {
|
|
|
626
616
|
wsUrl: wsUrl,
|
|
627
617
|
secretVersion: secretVersion,
|
|
628
618
|
envId: envId,
|
|
629
|
-
expiredTs: expiredTs
|
|
619
|
+
expiredTs: expiredTs,
|
|
630
620
|
}];
|
|
631
621
|
}
|
|
632
|
-
|
|
633
|
-
throw new Error('[tcb-js-sdk] 获取实时数据推送登录票据失败');
|
|
634
|
-
}
|
|
635
|
-
return [2];
|
|
622
|
+
throw new Error('[tcb-js-sdk] 获取实时数据推送登录票据失败');
|
|
636
623
|
}
|
|
637
624
|
});
|
|
638
625
|
}); };
|
|
@@ -640,9 +627,9 @@ var RealtimeWebSocketClient = (function () {
|
|
|
640
627
|
if (!_this._rttObserved.length) {
|
|
641
628
|
return DEFAULT_EXPECTED_EVENT_WAIT_TIME;
|
|
642
629
|
}
|
|
643
|
-
return ((_this._rttObserved.reduce(function (acc, cur) { return acc + cur; })
|
|
644
|
-
_this._rttObserved.length)
|
|
645
|
-
1.5);
|
|
630
|
+
return ((_this._rttObserved.reduce(function (acc, cur) { return acc + cur; })
|
|
631
|
+
/ _this._rttObserved.length)
|
|
632
|
+
* 1.5);
|
|
646
633
|
};
|
|
647
634
|
this.ping = function () { return __awaiter(_this, void 0, void 0, function () {
|
|
648
635
|
var msg;
|
|
@@ -651,12 +638,12 @@ var RealtimeWebSocketClient = (function () {
|
|
|
651
638
|
case 0:
|
|
652
639
|
msg = {
|
|
653
640
|
watchId: undefined,
|
|
654
|
-
requestId: message_1.genRequestId(),
|
|
641
|
+
requestId: (0, message_1.genRequestId)(),
|
|
655
642
|
msgType: 'PING',
|
|
656
|
-
msgData: null
|
|
643
|
+
msgData: null,
|
|
657
644
|
};
|
|
658
645
|
return [4, this.send({
|
|
659
|
-
msg: msg
|
|
646
|
+
msg: msg,
|
|
660
647
|
})];
|
|
661
648
|
case 1:
|
|
662
649
|
_a.sent();
|
|
@@ -678,8 +665,7 @@ var RealtimeWebSocketClient = (function () {
|
|
|
678
665
|
}
|
|
679
666
|
};
|
|
680
667
|
this._maxReconnect = options.maxReconnect || DEFAULT_MAX_RECONNECT;
|
|
681
|
-
this._reconnectInterval =
|
|
682
|
-
options.reconnectInterval || DEFAULT_WS_RECONNECT_INTERVAL;
|
|
668
|
+
this._reconnectInterval = options.reconnectInterval || DEFAULT_WS_RECONNECT_INTERVAL;
|
|
683
669
|
this._context = options.context;
|
|
684
670
|
}
|
|
685
671
|
RealtimeWebSocketClient.prototype.clearHeartbeat = function () {
|
|
@@ -749,4 +735,4 @@ var RealtimeWebSocketClient = (function () {
|
|
|
749
735
|
return RealtimeWebSocketClient;
|
|
750
736
|
}());
|
|
751
737
|
exports.RealtimeWebSocketClient = RealtimeWebSocketClient;
|
|
752
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-client.js","sourceRoot":"","sources":["../../src/websocket-client.ts"],"names":[],"mappings":"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","sourcesContent":["import { VirtualWebSocketClient } from './virtual-websocket-client'\nimport { genRequestId } from './message'\nimport {\n  IDatabaseServiceContext,\n} from '@cloudbase/types/database'\nimport {\n  IWatchOptions,\n  DBRealtimeListener,\n  IRequestMessage,\n  IResponseMessage,\n  IRequestMessagePingMsg,\n  IRequestMessageLoginMsg,\n  IResponseMessageLoginResMsg,\n  IRequestMessageLoginData\n} from '@cloudbase/types/realtime'\nimport {\n  CLOSE_EVENT_CODE,\n  CLOSE_EVENT_CODE_INFO,\n  getWSCloseError\n} from './ws-event'\n\nimport { ERR_CODE, TimeoutError, RealtimeErrorMessageError,CloudSDKError } from './error'\nimport { getWsClass, getRuntime } from './common'\nimport { sleep } from './utils'\n\nexport interface IRealtimeWebSocketClientConstructorOptions {\n  maxReconnect?: number\n  reconnectInterval?: number\n  context: IDatabaseServiceContext\n}\n\nexport interface ISignature {\n  envId: string\n  secretVersion: number\n  signStr: string\n  wsUrl: string\n  expireTS: number\n}\n\nexport interface ILoginInfo {\n  loggedIn: boolean\n  loggingInPromise?: Promise<ILoginResult>\n  loginStartTS?: number\n  loginResult?: ILoginResult\n}\n\nexport interface ILoginResult {\n  envId: string\n}\n\nexport interface IWSSendOptions {\n  msg: IRequestMessage\n  waitResponse?: boolean\n  // when waitResponse is set to true, if skipOnMessage is true, general onMessage handler will be skipped\n  skipOnMessage?: boolean\n  timeout?: number\n}\n\nexport interface IWSWatchOptions extends IWatchOptions {\n  envId?: string\n  collectionName: string\n  query: string\n  limit?: number\n  orderBy?: Record<string, string>\n}\n\ninterface IResolveReject {\n  resolve: (value?: any | PromiseLike<any> | undefined) => void\n  reject: (reason?: any) => void\n}\n\ninterface IResponseWaitSpec extends IResolveReject {\n  skipOnMessage?: boolean\n}\n\ninterface IWsSign {\n  signStr: string,\n  wsUrl: string,\n  secretVersion: string\n  envId: string\n  expiredTs: number\n}\n\nconst WS_READY_STATE = {\n  CONNECTING: 0,\n  OPEN: 1,\n  CLOSING: 2,\n  CLOSED: 3\n}\n\nconst MAX_RTT_OBSERVED = 3\nconst DEFAULT_EXPECTED_EVENT_WAIT_TIME = 5000\nconst DEFAULT_UNTRUSTED_RTT_THRESHOLD = 10000\nconst DEFAULT_MAX_RECONNECT = 5\nconst DEFAULT_WS_RECONNECT_INTERVAL = 10000\n// const DEFAULT_WS_RECONNECT_MAX_VALID_INTERVAL = 3 * 60 * 1000\nconst DEFAULT_PING_FAIL_TOLERANCE = 2\nconst DEFAULT_PONG_MISS_TOLERANCE = 2\nconst DEFAULT_LOGIN_TIMEOUT = 5000\n\nexport class RealtimeWebSocketClient {\n  private _virtualWSClient: Set<VirtualWebSocketClient> = new Set()\n  // after listener initWatch, the listener has the queryID and can store it here\n  private _queryIdClientMap: Map<string, VirtualWebSocketClient> = new Map()\n  private _watchIdClientMap: Map<string, VirtualWebSocketClient> = new Map()\n  private _maxReconnect: number\n  // private _availableRetries: number\n  private _reconnectInterval: number\n  private _context: IDatabaseServiceContext\n  // private _ws?: WXNS.Socket.ISocketTask\n  private _ws?: any\n  private _lastPingSendTS?: number\n  private _pingFailed = 0\n  private _pongMissed = 0\n  private _pingTimeoutId?: number\n  private _pongTimeoutId?: number\n  private _logins: Map<string /* envId */, ILoginInfo> = new Map()\n  // private _loginInfo: ILoginInfo\n  // private _signatures: Map<string /* envId */, ISignature> = new Map()\n  private _wsInitPromise?: Promise<void>\n  private _wsReadySubsribers: IResolveReject[] = []\n  private _wsResponseWait: Map<\n    string /* requestId */,\n    IResponseWaitSpec\n  > = new Map()\n  private _rttObserved: number[] = []\n  private _reconnectState: boolean\n  // obtained from the first getSignature with no envId provided\n  // private _defaultEnvId?: string\n  private _wsSign: IWsSign\n\n  constructor(options: IRealtimeWebSocketClientConstructorOptions) {\n    this._maxReconnect = options.maxReconnect || DEFAULT_MAX_RECONNECT\n    // this._availableRetries = this._maxReconnect\n    this._reconnectInterval =\n      options.reconnectInterval || DEFAULT_WS_RECONNECT_INTERVAL\n    this._context = options.context\n  }\n\n  clearHeartbeat() {\n    this._pingTimeoutId && clearTimeout(this._pingTimeoutId)\n    this._pongTimeoutId && clearTimeout(this._pongTimeoutId)\n  }\n\n  send = async <T = any>(opts: IWSSendOptions): Promise<T> =>\n    new Promise<T>(async (_resolve, _reject) => {\n      let timeoutId: number\n      let _hasResolved = false\n      let _hasRejected = false\n\n      const resolve: typeof _resolve = (\n        value?: T | PromiseLike<T> | undefined\n      ) => {\n        _hasResolved = true\n        timeoutId && clearTimeout(timeoutId)\n        _resolve(value)\n      }\n\n      const reject: typeof _reject = (error: any) => {\n        _hasRejected = true\n        timeoutId && clearTimeout(timeoutId)\n        _reject(error)\n      }\n\n      if (opts.timeout) {\n        // @ts-ignore\n        timeoutId = setTimeout(async () => {\n          if (!_hasResolved || !_hasRejected) {\n            // wait another immediate timeout to allow the success/fail callback to be invoked if ws has already got the result,\n            // this is because the timer is registered before ws.send\n            await sleep(0)\n            if (!_hasResolved || !_hasRejected) {\n              reject(new TimeoutError('wsclient.send timedout'))\n            }\n          }\n        }, opts.timeout)\n      }\n\n      try {\n        // if (this._context.debug) {\n        // console.log(`[realtime] ws send (${new Date()}): `, opts)\n        // console.log(\n        //   `[realtime] ws send ${\n        //     opts.msg.msgType\n        //   } (${new Date().toLocaleString()}): `,\n        //   opts\n        // )\n        // }\n\n        if (this._wsInitPromise) {\n          await this._wsInitPromise\n        }\n\n        if (!this._ws) {\n          reject(\n            new Error(\n              'invalid state: ws connection not exists, can not send message'\n            )\n          )\n          return\n        }\n\n        if (this._ws.readyState !== WS_READY_STATE.OPEN) {\n          reject(\n            new Error(\n              `ws readyState invalid: ${this._ws.readyState}, can not send message`\n            )\n          )\n          return\n        }\n\n        if (opts.waitResponse) {\n          this._wsResponseWait.set(opts.msg.requestId, {\n            resolve,\n            reject,\n            skipOnMessage: opts.skipOnMessage\n          } as IResponseWaitSpec)\n        }\n\n        // console.log('send msg:', opts.msg)\n        try {\n          await this._ws.send(JSON.stringify(opts.msg))\n          if (!opts.waitResponse) {\n            resolve()\n          }\n        } catch (err) {\n          if (err) {\n            reject(err)\n            if (opts.waitResponse) {\n              this._wsResponseWait.delete(opts.msg.requestId)\n            }\n          }\n        }\n        // this._ws.send(JSON.stringify(opts.msg), err => {\n        //   if (err) {\n        //     reject(err)\n        //     if (opts.waitResponse) {\n        //       this._wsResponseWait.delete(opts.msg.requestId)\n        //     }\n        //     return\n        //   }\n\n        //   if (!opts.waitResponse) {\n        //     resolve()\n        //   }\n        // })\n\n        // this._ws.send({\n        //   data: JSON.stringify(opts.msg),\n        //   success: res => {\n        //     if (!opts.waitResponse) {\n        //       resolve(res)\n        //     }\n        //   },\n        //   fail: e => {\n        //     reject(e)\n        //     if (opts.waitResponse) {\n        //       this._wsResponseWait.delete(opts.msg.requestId)\n        //     }\n        //   }\n        // })\n      } catch (e) {\n        reject(e)\n      }\n    })\n\n  close(code: CLOSE_EVENT_CODE) {\n    this.clearHeartbeat()\n\n    if (this._ws) {\n      this._ws.close(code, CLOSE_EVENT_CODE_INFO[code].name)\n      this._ws = undefined\n    }\n  }\n\n  closeAllClients = (error: any) => {\n    this._virtualWSClient.forEach(client => {\n      client.closeWithError(error)\n    })\n  }\n\n  pauseClients = (clients?: Set<VirtualWebSocketClient>) => {\n    ;(clients || this._virtualWSClient).forEach(client => {\n      client.pause()\n    })\n  }\n\n  resumeClients = (clients?: Set<VirtualWebSocketClient>) => {\n    ;(clients || this._virtualWSClient).forEach(client => {\n      client.resume()\n    })\n  }\n\n  watch(options: IWSWatchOptions): DBRealtimeListener {\n    if (!this._ws && !this._wsInitPromise) {\n      this.initWebSocketConnection(false)\n    }\n\n    const virtualClient = new VirtualWebSocketClient({\n      ...options,\n      send: this.send,\n      login: this.webLogin,\n      isWSConnected: this.isWSConnected,\n      onceWSConnected: this.onceWSConnected,\n      getWaitExpectedTimeoutLength: this.getWaitExpectedTimeoutLength,\n      onWatchStart: this.onWatchStart,\n      onWatchClose: this.onWatchClose,\n      debug: true\n    })\n    this._virtualWSClient.add(virtualClient)\n    this._watchIdClientMap.set(virtualClient.watchId, virtualClient)\n    return virtualClient.listener\n  }\n\n  private initWebSocketConnection = async (\n    reconnect: boolean,\n    availableRetries: number = this._maxReconnect\n  ): Promise<void> => {\n    // 当前处于正在重连中的状态\n    if (reconnect && this._reconnectState) {\n      return // 忽略\n    }\n\n    if (reconnect) {\n      this._reconnectState = true // 重连状态开始\n    }\n\n    if (this._wsInitPromise) {\n      // there already exists a websocket initiation, just wait for it\n      return this._wsInitPromise\n    }\n\n    // if (process.env.DEBUG) {\n    // console.log(\n    //   `[realtime] initWebSocketConnection reconnect ${reconnect} availableRetries ${availableRetries}`\n    // )\n    // }\n\n    if (reconnect) {\n      this.pauseClients()\n    }\n\n    this.close(CLOSE_EVENT_CODE.ReconnectWebSocket)\n\n    this._wsInitPromise = new Promise<void>(async (resolve, reject) => {\n      try {\n        // if (process.env.DEBUG) {\n        // console.log(\n        //   '[realtime] initWebSocketConnection start throwErrorIfNetworkOffline'\n        // )\n        // }\n\n        // 暂不检查网络态\n        // await throwErrorIfNetworkOffline()\n\n        // if (process.env.DEBUG) {\n        // console.log('[realtime] initWebSocketConnection start getSignature')\n        // }\n\n        // const signature = await this.getSignature()\n        const wsSign = await this.getWsSign()\n\n        // if (process.env.DEBUG) {\n        // console.log('[realtime] initWebSocketConnection getSignature success')\n        // console.log('[realtime] initWebSocketConnection start connectSocket')\n        // }\n\n        await new Promise(success => {\n          // this._ws = getSDK(this._context.identifiers)\n          //   ._socketSkipCheckDomainFactory()\n          //   .connectSocket({\n          //     url: signature.wsUrl,\n          //     header: {\n          //       \"content-type\": \"application/json\"\n          //     },\n          //     success: () => success(),\n          //     fail\n          //   })\n\n          const url = wsSign.wsUrl || 'wss://tcb-ws.tencentcloudapi.com';\n          const wsClass = getWsClass();\n          this._ws = wsClass ? new wsClass(url) : new WebSocket(url)\n          success()\n        })\n\n        if(this._ws.connect){\n          await this._ws.connect()\n        }\n\n        // if (process.env.DEBUG) {\n        // console.log(\n        //   '[realtime] initWebSocketConnection connectSocket successfully fired'\n        // )\n        // }\n\n        await this.initWebSocketEvent()\n        resolve()\n\n        if (reconnect) {\n          this.resumeClients()\n          this._reconnectState = false // 重连状态结束\n        }\n      } catch (e) {\n        // if (process.env.DEBUG) {\n        console.error('[realtime] initWebSocketConnection connect fail', e)\n        // }\n\n        if (availableRetries > 0) {\n          // this is an optimization, in case of network offline, we don't need to stubbornly sleep for sometime,\n          // we only need to wait for the network to be back online, this ensures minimum downtime\n          // const { isConnected } = await getNetworkStatus()\n          const isConnected = true\n\n          // if (process.env.DEBUG) {\n          // console.log(\n          //   '[realtime] initWebSocketConnection waiting for network online'\n          // )\n          // }\n\n          // auto wait until network online, cause' it would be meaningless to reconnect while network is offline\n\n          // await onceNetworkOnline()\n\n          // COMPATIBILITY: wait for ide state update\n          // if (isDevTools()) {\n          //   await sleep(0)\n          // }\n\n          // if (process.env.DEBUG) {\n          // console.log('[realtime] initWebSocketConnection network online')\n          // }\n\n          this._wsInitPromise = undefined\n\n          if (isConnected) {\n            // if (process.env.DEBUG) {\n            // console.log(\n            //   `[realtime] initWebSocketConnection sleep ${this._reconnectInterval}ms`\n            // )\n            // }\n            await sleep(this._reconnectInterval)\n            if (reconnect) {\n              this._reconnectState = false // 重连异常也算重连状态结束\n            }\n          }\n\n          resolve(this.initWebSocketConnection(reconnect, availableRetries - 1))\n        } else {\n          reject(e)\n\n          if (reconnect) {\n            this.closeAllClients(\n              new CloudSDKError({\n                errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_RECONNECT_WATCH_FAIL as string,\n                errMsg: e\n              })\n            )\n          }\n        }\n      }\n    })\n\n    // let success = false\n\n    try {\n      await this._wsInitPromise\n      // success = true\n      this._wsReadySubsribers.forEach(({ resolve }) => resolve())\n    } catch (e) {\n      this._wsReadySubsribers.forEach(({ reject }) => reject())\n    } finally {\n      this._wsInitPromise = undefined\n      this._wsReadySubsribers = []\n    }\n\n    // if (process.env.DEBUG) {\n    // console.log(\n    //   `[realtime] initWebSocketConnection ${success ? 'success' : 'fail'}`\n    // )\n    // }\n  }\n\n  private initWebSocketEvent = () =>\n    new Promise<void>((resolve, reject) => {\n      if (!this._ws) {\n        throw new Error('can not initWebSocketEvent, ws not exists')\n      }\n\n      let wsOpened = false\n\n      this._ws.onopen = event => {\n        // this._ws.onOpen(() => {\n        // this._ws.on(\"open\", () => {\n        // this._context.debug &&\n        console.warn('[realtime] ws event: open', event)\n        wsOpened = true\n        resolve()\n      }\n\n      this._ws.onerror = event => {\n        // this._ws.on(\"error\", error => {\n        // this._ws.onError(error => {\n        // all logins are invalid after disconnection\n        this._logins = new Map()\n\n        // error写进file\n\n        if (!wsOpened) {\n          // this._context.debug &&\n          console.error('[realtime] ws open failed with ws event: error', event)\n          // writeToFile(\n          //   \"wserror.txt\",\n          //   `${\n          //     this.specialNumber\n          //   } [realtime] ws open failed with ws event: error ${error} \\n`\n          // )\n\n          reject(event)\n        } else {\n          // this._context.debug &&\n          console.error('[realtime] ws event: error', event)\n\n          // writeToFile(\n          //   \"wserror.txt\",\n          //   `${this.specialNumber} [realtime] ws event: error ${error} \\n`\n          // )\n\n          this.clearHeartbeat()\n          this._virtualWSClient.forEach(client =>\n            client.closeWithError(\n              new CloudSDKError({\n                errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_WEBSOCKET_CONNECTION_ERROR as string,\n                errMsg: event\n              })\n            )\n          )\n        }\n      }\n\n      // TODO: reconnect\n      this._ws.onclose = closeEvent => {\n        // this._ws.on(\"close\", (closeEvent, closereason) => {\n        // this._ws.onClose(closeEvent => {\n        // if (process.env.DEBUG) {\n        console.warn('[realtime] ws event: close', closeEvent)\n        // }\n\n        // writeToFile(\n        //   \"wsclose.txt\",\n        //   `${\n        //     this.specialNumber\n        //   } [realtime] ws event: close ${closeEvent} ${closereason} \\n`\n        // )\n\n        // all logins are invalid after disconnection\n        this._logins = new Map()\n\n        this.clearHeartbeat()\n        switch (closeEvent.code) {\n          case CLOSE_EVENT_CODE.ReconnectWebSocket: {\n            // just ignore\n            break\n          }\n          case CLOSE_EVENT_CODE.NoRealtimeListeners: {\n            // quit\n            break\n          }\n          case CLOSE_EVENT_CODE.HeartbeatPingError:\n          case CLOSE_EVENT_CODE.HeartbeatPongTimeoutError:\n          case CLOSE_EVENT_CODE.NormalClosure:\n          case CLOSE_EVENT_CODE.AbnormalClosure: {\n            // Normal Closure and Abnormal Closure:\n            //   expected closure, most likely dispatched by wechat client,\n            //   since this is the status code dispatched in case of network failure,\n            //   we should retry\n\n            if (this._maxReconnect > 0) {\n              // if (this._availableRetries > 0) {\n              this.initWebSocketConnection(true, this._maxReconnect)\n            } else {\n              this.closeAllClients(getWSCloseError(closeEvent.code))\n            }\n            break\n          }\n          case CLOSE_EVENT_CODE.NoAuthentication: {\n            this.closeAllClients(\n              getWSCloseError(closeEvent.code, closeEvent.reason)\n            )\n            break\n          }\n          default: {\n            // we should retry by default\n            if (this._maxReconnect > 0) {\n              // if (this._availableRetries > 0) {\n              this.initWebSocketConnection(true, this._maxReconnect)\n            } else {\n              this.closeAllClients(getWSCloseError(closeEvent.code))\n            }\n            // console.warn(`[realtime] unrecognize ws close event`, closeEvent)\n            // this.closeAllClients(getWSCloseError(closeEvent.code))\n          }\n        }\n      }\n\n      this._ws.onmessage = res => {\n        // this._ws.on(\"message\", res => {\n        // this._ws.onMessage(res => {\n        // const rawMsg = res.data\n        const rawMsg = res.data\n\n        // reset & restart heartbeat\n        this.heartbeat()\n\n        let msg: IResponseMessage\n\n        try {\n          msg = JSON.parse(rawMsg as string)\n        } catch (e) {\n          throw new Error(`[realtime] onMessage parse res.data error: ${e}`)\n        }\n\n        // console.log(\n        //   `[realtime] onMessage ${\n        //     msg.msgType\n        //   } (${new Date().toLocaleString()})`,\n        //   msg\n        // )\n\n        if (msg.msgType === 'ERROR') {\n          // 找到当前监听，并将error返回\n          let virtualWatch = null\n          this._virtualWSClient.forEach(item => {\n            if (item.watchId === msg.watchId) {\n              virtualWatch = item\n            }\n          })\n\n          if (virtualWatch) {\n            virtualWatch.listener.onError(msg)\n          }\n        }\n\n        const responseWaitSpec = this._wsResponseWait.get(msg.requestId)\n        if (responseWaitSpec) {\n          try {\n            if (msg.msgType === 'ERROR') {\n              responseWaitSpec.reject(new RealtimeErrorMessageError(msg))\n            } else {\n              responseWaitSpec.resolve(msg)\n            }\n          } catch (e) {\n            // this._context.debug &&\n            console.error(\n              'ws onMessage responseWaitSpec.resolve(msg) errored:',\n              e\n            )\n          } finally {\n            this._wsResponseWait.delete(msg.requestId)\n          }\n          if (responseWaitSpec.skipOnMessage) {\n            return\n          }\n        }\n\n        if (msg.msgType === 'PONG') {\n          if (this._lastPingSendTS) {\n            const rtt = Date.now() - this._lastPingSendTS\n            if (rtt > DEFAULT_UNTRUSTED_RTT_THRESHOLD) {\n              // this._context.debug &&\n              console.warn(`[realtime] untrusted rtt observed: ${rtt}`)\n              return\n            }\n            if (this._rttObserved.length >= MAX_RTT_OBSERVED) {\n              this._rttObserved.splice(\n                0,\n                this._rttObserved.length - MAX_RTT_OBSERVED + 1\n              )\n            }\n            this._rttObserved.push(rtt)\n          }\n          return\n        }\n\n        let client = msg.watchId && this._watchIdClientMap.get(msg.watchId)\n        if (client) {\n          client.onMessage(msg)\n        } else {\n          // TODO, this is a temporary fix done for server\n          // if (process.env.DEBUG) {\n          console.error(\n            `[realtime] no realtime listener found responsible for watchId ${msg.watchId}: `,\n            msg\n          )\n          // }\n          switch (msg.msgType) {\n            case 'INIT_EVENT':\n            case 'NEXT_EVENT':\n            case 'CHECK_EVENT': {\n              client = this._queryIdClientMap.get(msg.msgData.queryID)\n              if (client) {\n                client.onMessage(msg)\n              }\n              break\n            }\n            default: {\n              for (const [,client] of Array.from(this._watchIdClientMap.entries())) {\n                // console.log('watchid*****', watchId)\n                client.onMessage(msg)\n                break\n              }\n            }\n          }\n        }\n      }\n\n      this.heartbeat()\n    })\n\n  private isWSConnected = (): boolean => {\n    return Boolean(this._ws && this._ws.readyState === WS_READY_STATE.OPEN)\n  }\n\n  private onceWSConnected = async (): Promise<void> => {\n    if (this.isWSConnected()) {\n      return\n    }\n\n    if (this._wsInitPromise) {\n      return this._wsInitPromise\n    }\n\n    return new Promise<void>((resolve, reject) => {\n      this._wsReadySubsribers.push({\n        resolve,\n        reject\n      })\n    })\n  }\n\n  private webLogin = async (\n    envId?: string,\n    refresh?: boolean\n  ): Promise<any> => {\n    if (!refresh) {\n      // let loginInfo = this._loginInfo\n      if (envId) {\n        const loginInfo = this._logins.get(envId)\n        if (loginInfo) {\n          if (loginInfo.loggedIn && loginInfo.loginResult) {\n            // if (process.env.DEBUG) {\n            // console.log('[realtime] login: already logged in')\n            // }\n            return loginInfo.loginResult\n          } else if (loginInfo.loggingInPromise) {\n            return loginInfo.loggingInPromise\n          }\n        }\n      } else {\n        const emptyEnvLoginInfo = this._logins.get('')\n        if (emptyEnvLoginInfo?.loggingInPromise) {\n          return emptyEnvLoginInfo.loggingInPromise\n        }\n      }\n    }\n    // console.log('[realtime] login: logging in')\n\n    const promise = new Promise<ILoginResult>(async (resolve, reject) => {\n      try {\n        // const signature = await this.getSignature(envId, refresh)\n\n        const wsSign = await this.getWsSign()\n\n        // const wxVersion = getWXVersion()\n        const msgData: IRequestMessageLoginData = {\n          envId: wsSign.envId || '',\n          accessToken: '', // 已废弃字段\n          // signStr: signature.signStr,\n          // secretVersion: signature.secretVersion,\n          referrer: 'web',\n          sdkVersion: '',\n          dataVersion: ''\n        }\n        const loginMsg: IRequestMessageLoginMsg = {\n          watchId: undefined,\n          requestId: genRequestId(),\n          msgType: 'LOGIN',\n          msgData,\n          exMsgData: {\n            runtime: getRuntime(),\n            signStr: wsSign.signStr,\n            secretVersion: wsSign.secretVersion\n          }\n        }\n        const loginResMsg = await this.send<IResponseMessageLoginResMsg>({\n          msg: loginMsg,\n          waitResponse: true,\n          skipOnMessage: true,\n          timeout: DEFAULT_LOGIN_TIMEOUT\n        })\n\n        if (!loginResMsg.msgData.code) {\n          // login success\n          resolve({\n            envId: wsSign.envId\n          })\n        } else {\n          // login failed\n          reject(\n            new Error(\n              `${loginResMsg.msgData.code} ${loginResMsg.msgData.message}`\n            )\n          )\n        }\n      } catch (e) {\n        reject(e)\n      }\n    })\n\n    // let loginInfo = this._loginInfo\n    let loginInfo = envId && this._logins.get(envId)\n\n    const loginStartTS = Date.now()\n\n    if (loginInfo) {\n      loginInfo.loggedIn = false\n      loginInfo.loggingInPromise = promise\n      loginInfo.loginStartTS = loginStartTS\n    } else {\n      loginInfo = {\n        loggedIn: false,\n        loggingInPromise: promise,\n        loginStartTS\n      }\n      // this._loginInfo = loginInfo\n      this._logins.set(envId || '', loginInfo)\n    }\n\n    // try {\n    //   const loginResult = await promise\n    //   loginInfo.loggedIn = true\n    //   loginInfo.loggingInPromise = undefined\n    //   loginInfo.loginStartTS = undefined\n    //   loginInfo.loginResult = loginResult\n    //   return loginResult\n    // } catch (e) {\n    //   loginInfo.loggedIn = false\n    //   loginInfo.loggingInPromise = undefined\n    //   loginInfo.loginStartTS = undefined\n    //   loginInfo.loginResult = undefined\n    //   throw e\n    // }\n\n    try {\n      const loginResult = await promise\n      const curLoginInfo = envId && this._logins.get(envId)\n      if (\n        curLoginInfo &&\n        curLoginInfo === loginInfo &&\n        curLoginInfo.loginStartTS === loginStartTS\n      ) {\n        loginInfo.loggedIn = true\n        loginInfo.loggingInPromise = undefined\n        loginInfo.loginStartTS = undefined\n        loginInfo.loginResult = loginResult\n        return loginResult\n      } else if (curLoginInfo) {\n        if (curLoginInfo.loggedIn && curLoginInfo.loginResult) {\n          return curLoginInfo.loginResult\n        } else if (curLoginInfo.loggingInPromise) {\n          return curLoginInfo.loggingInPromise\n        } else {\n          throw new Error('ws unexpected login info')\n        }\n      } else {\n        throw new Error('ws login info reset')\n      }\n    } catch (e) {\n      loginInfo.loggedIn = false\n      loginInfo.loggingInPromise = undefined\n      loginInfo.loginStartTS = undefined\n      loginInfo.loginResult = undefined\n      throw e\n    }\n  }\n\n  private getWsSign = async (): Promise<IWsSign> => {\n    if (this._wsSign && this._wsSign.expiredTs > Date.now()) {\n      return this._wsSign\n    }\n    const expiredTs = Date.now() + 60000\n    const res = await this._context.appConfig.request.send('auth.wsWebSign', {runtime: getRuntime()})\n\n    if (res.code) {\n      throw new Error(`[tcb-js-sdk] 获取实时数据推送登录票据失败: ${res.code}`)\n    }\n\n    if (res.data) {\n      const {signStr, wsUrl, secretVersion, envId} = res.data\n      return {\n        signStr,\n        wsUrl,\n        secretVersion,\n        envId,\n        expiredTs\n      }\n    } else {\n      throw new Error('[tcb-js-sdk] 获取实时数据推送登录票据失败')\n    }\n  }\n\n  private getWaitExpectedTimeoutLength = () => {\n    if (!this._rttObserved.length) {\n      return DEFAULT_EXPECTED_EVENT_WAIT_TIME\n    }\n\n    // 1.5 * RTT\n    return (\n      (this._rttObserved.reduce((acc, cur) => acc + cur) /\n        this._rttObserved.length) *\n      1.5\n    )\n  }\n\n  private heartbeat(immediate?: boolean) {\n    this.clearHeartbeat()\n    // @ts-ignore\n    this._pingTimeoutId = setTimeout(\n      async () => {\n        try {\n          if (!this._ws || this._ws.readyState !== WS_READY_STATE.OPEN) {\n            // no need to ping\n            return\n          }\n\n          this._lastPingSendTS = Date.now()\n          await this.ping()\n          this._pingFailed = 0\n\n          // @ts-ignore\n          this._pongTimeoutId = setTimeout(() => {\n            console.error('pong timed out')\n            if (this._pongMissed < DEFAULT_PONG_MISS_TOLERANCE) {\n              this._pongMissed++\n              this.heartbeat(true)\n            } else {\n              // logical perceived connection lost, even though websocket did not receive error or close event\n              this.initWebSocketConnection(true)\n            }\n          }, this._context.appConfig.realtimePongWaitTimeout)\n        } catch (e) {\n          if (this._pingFailed < DEFAULT_PING_FAIL_TOLERANCE) {\n            this._pingFailed++\n            this.heartbeat()\n          } else {\n            this.close(CLOSE_EVENT_CODE.HeartbeatPingError)\n          }\n        }\n      },\n      immediate ? 0 : this._context.appConfig.realtimePingInterval\n    )\n  }\n\n  private ping = async () => {\n    const msg: IRequestMessagePingMsg = {\n      watchId: undefined,\n      requestId: genRequestId(),\n      msgType: 'PING',\n      msgData: null\n    }\n    await this.send({\n      msg\n    })\n    // console.log('ping sent')\n  }\n\n  private onWatchStart = (client: VirtualWebSocketClient, queryID: string) => {\n    this._queryIdClientMap.set(queryID, client)\n  }\n\n  private onWatchClose = (client: VirtualWebSocketClient, queryID: string) => {\n    if (queryID) {\n      this._queryIdClientMap.delete(queryID)\n    }\n    this._watchIdClientMap.delete(client.watchId)\n    this._virtualWSClient.delete(client)\n\n    if (!this._virtualWSClient.size) {\n      // no more existing watch, we should release the websocket connection\n      this.close(CLOSE_EVENT_CODE.NoRealtimeListeners)\n    }\n  }\n}\n"]}
|
|
738
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-client.js","sourceRoot":"","sources":["../../src/websocket-client.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAIA,uEAAoE;AACpE,qCAAyC;AAczC,uCAIoB;AAEpB,iCAA2F;AAC3F,mCAAkD;AAClD,iCAAgC;AA4DhC,IAAM,cAAc,GAAG;IACrB,UAAU,EAAE,CAAC;IACb,IAAI,EAAE,CAAC;IACP,OAAO,EAAE,CAAC;IACV,MAAM,EAAE,CAAC;CACV,CAAC;AAEF,IAAM,gBAAgB,GAAG,CAAC,CAAC;AAC3B,IAAM,gCAAgC,GAAG,IAAI,CAAC;AAC9C,IAAM,+BAA+B,GAAG,KAAK,CAAC;AAC9C,IAAM,qBAAqB,GAAG,CAAC,CAAC;AAChC,IAAM,6BAA6B,GAAG,KAAK,CAAC;AAE5C,IAAM,2BAA2B,GAAG,CAAC,CAAC;AACtC,IAAM,2BAA2B,GAAG,CAAC,CAAC;AACtC,IAAM,qBAAqB,GAAG,IAAI,CAAC;AAEnC;IA+BE,iCAAY,OAAmD;QAA/D,iBAKC;QAnCO,qBAAgB,GAAgC,IAAI,GAAG,EAAE,CAAC;QAE1D,sBAAiB,GAAwC,IAAI,GAAG,EAAE,CAAC;QACnE,sBAAiB,GAAwC,IAAI,GAAG,EAAE,CAAC;QAQnE,gBAAW,GAAG,CAAC,CAAC;QAChB,gBAAW,GAAG,CAAC,CAAC;QAGhB,YAAO,GAAwC,IAAI,GAAG,EAAE,CAAC;QAIzD,uBAAkB,GAAqB,EAAE,CAAC;QAC1C,oBAAe,GAGnB,IAAI,GAAG,EAAE,CAAC;QACN,iBAAY,GAAa,EAAE,CAAC;QAkBpC,SAAI,GAAG,UAAgB,IAAoB;;;gBAAiB,WAAA,IAAI,OAAO,CAAI,UAAO,QAAQ,EAAE,OAAO;;;;;;oCAE7F,YAAY,GAAG,KAAK,CAAC;oCACrB,YAAY,GAAG,KAAK,CAAC;oCAEnB,OAAO,GAAoB,UAAC,KAAsC;wCACtE,YAAY,GAAG,IAAI,CAAC;wCACpB,SAAS,IAAI,YAAY,CAAC,SAAS,CAAC,CAAC;wCACrC,QAAQ,CAAC,KAAK,CAAC,CAAC;oCAClB,CAAC,CAAC;oCAEI,MAAM,GAAmB,UAAC,KAAU;wCACxC,YAAY,GAAG,IAAI,CAAC;wCACpB,SAAS,IAAI,YAAY,CAAC,SAAS,CAAC,CAAC;wCACrC,OAAO,CAAC,KAAK,CAAC,CAAC;oCACjB,CAAC,CAAC;oCAEF,IAAI,IAAI,CAAC,OAAO,EAAE;wCAEhB,SAAS,GAAG,UAAU,CAAC;;;;6DACjB,CAAA,CAAC,YAAY,IAAI,CAAC,YAAY,CAAA,EAA9B,cAA8B;wDAGhC,WAAM,IAAA,aAAK,EAAC,CAAC,CAAC,EAAA;;wDAAd,SAAc,CAAC;wDACf,IAAI,CAAC,YAAY,IAAI,CAAC,YAAY,EAAE;4DAClC,MAAM,CAAC,IAAI,oBAAY,CAAC,wBAAwB,CAAC,CAAC,CAAC;yDACpD;;;;;6CAEJ,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;qCAClB;;;;yCAaK,IAAI,CAAC,cAAc,EAAnB,cAAmB;oCACrB,WAAM,IAAI,CAAC,cAAc,EAAA;;oCAAzB,SAAyB,CAAC;;;oCAG5B,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE;wCACb,MAAM,CAAC,IAAI,KAAK,CAAC,+DAA+D,CAAC,CAAC,CAAC;wCACnF,WAAO;qCACR;oCAED,IAAI,IAAI,CAAC,GAAG,CAAC,UAAU,KAAK,cAAc,CAAC,IAAI,EAAE;wCAC/C,MAAM,CAAC,IAAI,KAAK,CAAC,iCAA0B,IAAI,CAAC,GAAG,CAAC,UAAU,2BAAwB,CAAC,CAAC,CAAC;wCACzF,WAAO;qCACR;oCAED,IAAI,IAAI,CAAC,YAAY,EAAE;wCACrB,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,SAAS,EAAE;4CAC3C,OAAO,SAAA;4CACP,MAAM,QAAA;4CACN,aAAa,EAAE,IAAI,CAAC,aAAa;yCACb,CAAC,CAAC;qCACzB;;;;oCAIC,WAAM,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAA;;oCAA7C,SAA6C,CAAC;oCAC9C,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE;wCACtB,OAAO,CAAC,SAAS,CAAC,CAAC;qCACpB;;;;oCAED,IAAI,KAAG,EAAE;wCACP,MAAM,CAAC,KAAG,CAAC,CAAC;wCACZ,IAAI,IAAI,CAAC,YAAY,EAAE;4CACrB,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;yCACjD;qCACF;;;;;oCA+BH,MAAM,CAAC,GAAC,CAAC,CAAC;;;;;yBAEb,CAAC,EAAA;;aAAA,CAAC;QAWH,oBAAe,GAAG,UAAC,KAAU;YAC3B,KAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,UAAC,MAAM;gBACnC,MAAM,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;YAC/B,CAAC,CAAC,CAAC;QACL,CAAC,CAAC;QAEF,iBAAY,GAAG,UAAC,OAAqC;YACnD,CAAC,OAAO,IAAI,KAAI,CAAC,gBAAgB,CAAC,CAAC,OAAO,CAAC,UAAC,MAAM;gBAChD,MAAM,CAAC,KAAK,EAAE,CAAC;YACjB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC;QAEF,kBAAa,GAAG,UAAC,OAAqC;YACpD,CAAC,OAAO,IAAI,KAAI,CAAC,gBAAgB,CAAC,CAAC,OAAO,CAAC,UAAC,MAAM;gBAChD,MAAM,CAAC,MAAM,EAAE,CAAC;YAClB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC;QAuBM,4BAAuB,GAAG,UAChC,SAAkB,EAClB,gBAA6C;YAA7C,iCAAA,EAAA,mBAA2B,KAAI,CAAC,aAAa;;;;;;;4BAG7C,IAAI,SAAS,IAAI,IAAI,CAAC,eAAe,EAAE;gCACrC,WAAO;6BACR;4BAED,IAAI,SAAS,EAAE;gCACb,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;6BAC7B;4BAED,IAAI,IAAI,CAAC,cAAc,EAAE;gCAEvB,WAAO,IAAI,CAAC,cAAc,EAAC;6BAC5B;4BAQD,IAAI,SAAS,EAAE;gCACb,IAAI,CAAC,YAAY,EAAE,CAAC;6BACrB;4BAED,IAAI,CAAC,KAAK,CAAC,2BAAgB,CAAC,kBAAkB,CAAC,CAAC;4BAEhD,IAAI,CAAC,cAAc,GAAG,IAAI,OAAO,CAAO,UAAO,OAAO,EAAE,MAAM;;;;;;;4CAgB3C,WAAM,IAAI,CAAC,SAAS,EAAE,EAAA;;4CAA/B,WAAS,SAAsB;4CAOrC,WAAM,IAAI,OAAO,CAAC,UAAC,OAAO;oDAYxB,IAAM,GAAG,GAAG,QAAM,CAAC,KAAK,IAAI,kCAAkC,CAAC;oDAC/D,IAAM,OAAO,GAAG,IAAA,mBAAU,GAAE,CAAC;oDAC7B,KAAI,CAAC,GAAG,GAAG,OAAO,CAAC,CAAC,CAAC,IAAI,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,SAAS,CAAC,GAAG,CAAC,CAAC;oDAC3D,OAAO,CAAC,SAAS,CAAC,CAAC;gDACrB,CAAC,CAAC,EAAA;;4CAhBF,SAgBE,CAAC;iDAEC,IAAI,CAAC,GAAG,CAAC,OAAO,EAAhB,cAAgB;4CAClB,WAAM,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,EAAA;;4CAAxB,SAAwB,CAAC;;gDAS3B,WAAM,IAAI,CAAC,kBAAkB,EAAE,EAAA;;4CAA/B,SAA+B,CAAC;4CAChC,OAAO,EAAE,CAAC;4CAEV,IAAI,SAAS,EAAE;gDACb,IAAI,CAAC,aAAa,EAAE,CAAC;gDACrB,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;6CAC9B;;;;4CAGD,OAAO,CAAC,KAAK,CAAC,iDAAiD,EAAE,GAAC,CAAC,CAAC;iDAGhE,CAAA,gBAAgB,GAAG,CAAC,CAAA,EAApB,cAAoB;4CAIhB,WAAW,GAAG,IAAI,CAAC;4CAqBzB,IAAI,CAAC,cAAc,GAAG,SAAS,CAAC;iDAE5B,WAAW,EAAX,cAAW;4CAMb,WAAM,IAAA,aAAK,EAAC,IAAI,CAAC,kBAAkB,CAAC,EAAA;;4CAApC,SAAoC,CAAC;4CACrC,IAAI,SAAS,EAAE;gDACb,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;6CAC9B;;;4CAGH,OAAO,CAAC,IAAI,CAAC,uBAAuB,CAAC,SAAS,EAAE,gBAAgB,GAAG,CAAC,CAAC,CAAC,CAAC;;;4CAEvE,MAAM,CAAC,GAAC,CAAC,CAAC;4CAEV,IAAI,SAAS,EAAE;gDACb,IAAI,CAAC,eAAe,CAAC,IAAI,qBAAa,CAAC;oDACrC,OAAO,EAAE,gBAAQ,CAAC,mDAA6D;oDAC/E,MAAM,EAAE,GAAC;iDACV,CAAC,CAAC,CAAC;6CACL;;;;;;iCAGN,CAAC,CAAC;;;;4BAKD,WAAM,IAAI,CAAC,cAAc,EAAA;;4BAAzB,SAAyB,CAAC;4BAE1B,IAAI,CAAC,kBAAkB,CAAC,OAAO,CAAC,UAAC,EAAW;oCAAT,OAAO,aAAA;gCAAO,OAAA,OAAO,EAAE;4BAAT,CAAS,CAAC,CAAC;;;;4BAE5D,IAAI,CAAC,kBAAkB,CAAC,OAAO,CAAC,UAAC,EAAU;oCAAR,MAAM,YAAA;gCAAO,OAAA,MAAM,EAAE;4BAAR,CAAQ,CAAC,CAAC;;;4BAE1D,IAAI,CAAC,cAAc,GAAG,SAAS,CAAC;4BAChC,IAAI,CAAC,kBAAkB,GAAG,EAAE,CAAC;;;;;;SAQhC,CAAC;QAEM,uBAAkB,GAAG,cAAM,OAAA,IAAI,OAAO,CAAO,UAAC,OAAO,EAAE,MAAM;YACnE,IAAI,CAAC,KAAI,CAAC,GAAG,EAAE;gBACb,MAAM,IAAI,KAAK,CAAC,2CAA2C,CAAC,CAAC;aAC9D;YAED,IAAI,QAAQ,GAAG,KAAK,CAAC;YAErB,KAAI,CAAC,GAAG,CAAC,MAAM,GAAG,UAAC,KAAK;gBAItB,OAAO,CAAC,IAAI,CAAC,2BAA2B,EAAE,KAAK,CAAC,CAAC;gBACjD,QAAQ,GAAG,IAAI,CAAC;gBAChB,OAAO,EAAE,CAAC;YACZ,CAAC,CAAC;YAEF,KAAI,CAAC,GAAG,CAAC,OAAO,GAAG,UAAC,KAAK;gBAIvB,KAAI,CAAC,OAAO,GAAG,IAAI,GAAG,EAAE,CAAC;gBAIzB,IAAI,CAAC,QAAQ,EAAE;oBAEb,OAAO,CAAC,KAAK,CAAC,gDAAgD,EAAE,KAAK,CAAC,CAAC;oBAQvE,MAAM,CAAC,KAAK,CAAC,CAAC;iBACf;qBAAM;oBAEL,OAAO,CAAC,KAAK,CAAC,4BAA4B,EAAE,KAAK,CAAC,CAAC;oBAOnD,KAAI,CAAC,cAAc,EAAE,CAAC;oBACtB,KAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,UAAA,MAAM,IAAI,OAAA,MAAM,CAAC,cAAc,CAAC,IAAI,qBAAa,CAAC;wBAC9E,OAAO,EAAE,gBAAQ,CAAC,yDAAmE;wBACrF,MAAM,EAAE,KAAK;qBACd,CAAC,CAAC,EAHqC,CAGrC,CAAC,CAAC;iBACN;YACH,CAAC,CAAC;YAGF,KAAI,CAAC,GAAG,CAAC,OAAO,GAAG,UAAC,UAAU;gBAI5B,OAAO,CAAC,IAAI,CAAC,4BAA4B,EAAE,UAAU,CAAC,CAAC;gBAWvD,KAAI,CAAC,OAAO,GAAG,IAAI,GAAG,EAAE,CAAC;gBAEzB,KAAI,CAAC,cAAc,EAAE,CAAC;gBACtB,QAAQ,UAAU,CAAC,IAAI,EAAE;oBACvB,KAAK,2BAAgB,CAAC,kBAAkB,CAAC,CAAC;wBAExC,MAAM;qBACP;oBACD,KAAK,2BAAgB,CAAC,mBAAmB,CAAC,CAAC;wBAEzC,MAAM;qBACP;oBACD,KAAK,2BAAgB,CAAC,kBAAkB,CAAC;oBACzC,KAAK,2BAAgB,CAAC,yBAAyB,CAAC;oBAChD,KAAK,2BAAgB,CAAC,aAAa,CAAC;oBACpC,KAAK,2BAAgB,CAAC,eAAe,CAAC,CAAC;wBAMrC,IAAI,KAAI,CAAC,aAAa,GAAG,CAAC,EAAE;4BAE1B,KAAI,CAAC,uBAAuB,CAAC,IAAI,EAAE,KAAI,CAAC,aAAa,CAAC,CAAC;yBACxD;6BAAM;4BACL,KAAI,CAAC,eAAe,CAAC,IAAA,0BAAe,EAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC;yBACxD;wBACD,MAAM;qBACP;oBACD,KAAK,2BAAgB,CAAC,gBAAgB,CAAC,CAAC;wBACtC,KAAI,CAAC,eAAe,CAAC,IAAA,0BAAe,EAAC,UAAU,CAAC,IAAI,EAAE,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC;wBAC1E,MAAM;qBACP;oBACD,OAAO,CAAC,CAAC;wBAEP,IAAI,KAAI,CAAC,aAAa,GAAG,CAAC,EAAE;4BAE1B,KAAI,CAAC,uBAAuB,CAAC,IAAI,EAAE,KAAI,CAAC,aAAa,CAAC,CAAC;yBACxD;6BAAM;4BACL,KAAI,CAAC,eAAe,CAAC,IAAA,0BAAe,EAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC;yBACxD;qBAGF;iBACF;YACH,CAAC,CAAC;YAEF,KAAI,CAAC,GAAG,CAAC,SAAS,GAAG,UAAC,GAAG;gBAIvB,IAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC;gBAGxB,KAAI,CAAC,SAAS,EAAE,CAAC;gBAEjB,IAAI,GAAqB,CAAC;gBAE1B,IAAI;oBACF,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,MAAgB,CAAC,CAAC;iBACpC;gBAAC,OAAO,CAAC,EAAE;oBACV,MAAM,IAAI,KAAK,CAAC,qDAA8C,CAAC,CAAE,CAAC,CAAC;iBACpE;gBASD,IAAI,GAAG,CAAC,OAAO,KAAK,OAAO,EAAE;oBAE3B,IAAI,cAAY,GAAG,IAAI,CAAC;oBACxB,KAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,UAAC,IAAI;wBACjC,IAAI,IAAI,CAAC,OAAO,KAAK,GAAG,CAAC,OAAO,EAAE;4BAChC,cAAY,GAAG,IAAI,CAAC;yBACrB;oBACH,CAAC,CAAC,CAAC;oBAEH,IAAI,cAAY,EAAE;wBAChB,cAAY,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;qBACpC;iBACF;gBAED,IAAM,gBAAgB,GAAG,KAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;gBACjE,IAAI,gBAAgB,EAAE;oBACpB,IAAI;wBACF,IAAI,GAAG,CAAC,OAAO,KAAK,OAAO,EAAE;4BAC3B,gBAAgB,CAAC,MAAM,CAAC,IAAI,iCAAyB,CAAC,GAAG,CAAC,CAAC,CAAC;yBAC7D;6BAAM;4BACL,gBAAgB,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;yBAC/B;qBACF;oBAAC,OAAO,CAAC,EAAE;wBAEV,OAAO,CAAC,KAAK,CACX,qDAAqD,EACrD,CAAC,CACF,CAAC;qBACH;4BAAS;wBACR,KAAI,CAAC,eAAe,CAAC,MAAM,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;qBAC5C;oBACD,IAAI,gBAAgB,CAAC,aAAa,EAAE;wBAClC,OAAO;qBACR;iBACF;gBAED,IAAI,GAAG,CAAC,OAAO,KAAK,MAAM,EAAE;oBAC1B,IAAI,KAAI,CAAC,eAAe,EAAE;wBACxB,IAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,KAAI,CAAC,eAAe,CAAC;wBAC9C,IAAI,GAAG,GAAG,+BAA+B,EAAE;4BAEzC,OAAO,CAAC,IAAI,CAAC,6CAAsC,GAAG,CAAE,CAAC,CAAC;4BAC1D,OAAO;yBACR;wBACD,IAAI,KAAI,CAAC,YAAY,CAAC,MAAM,IAAI,gBAAgB,EAAE;4BAChD,KAAI,CAAC,YAAY,CAAC,MAAM,CACtB,CAAC,EACD,KAAI,CAAC,YAAY,CAAC,MAAM,GAAG,gBAAgB,GAAG,CAAC,CAChD,CAAC;yBACH;wBACD,KAAI,CAAC,YAAY,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;qBAC7B;oBACD,OAAO;iBACR;gBAED,IAAI,MAAM,GAAG,GAAG,CAAC,OAAO,IAAI,KAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;gBACpE,IAAI,MAAM,EAAE;oBACV,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;iBACvB;qBAAM;oBAGL,OAAO,CAAC,KAAK,CACX,wEAAiE,GAAG,CAAC,OAAO,OAAI,EAChF,GAAG,CACJ,CAAC;oBAEF,QAAQ,GAAG,CAAC,OAAO,EAAE;wBACnB,KAAK,YAAY,CAAC;wBAClB,KAAK,YAAY,CAAC;wBAClB,KAAK,aAAa,CAAC,CAAC;4BAClB,MAAM,GAAG,KAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAG,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;4BACzD,IAAI,MAAM,EAAE;gCACV,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;6BACvB;4BACD,MAAM;yBACP;wBACD,OAAO,CAAC,CAAC;4BACP,KAAwB,UAA4C,EAA5C,KAAA,KAAK,CAAC,IAAI,CAAC,KAAI,CAAC,iBAAiB,CAAC,OAAO,EAAE,CAAC,EAA5C,cAA4C,EAA5C,IAA4C,EAAE;gCAA3D,IAAA,WAAS,EAAP,QAAM,QAAA;gCAEjB,QAAM,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gCACtB,MAAM;6BACP;yBACF;qBACF;iBACF;YACH,CAAC,CAAC;YAEF,KAAI,CAAC,SAAS,EAAE,CAAC;QACnB,CAAC,CAAC,EAnOiC,CAmOjC,CAAC;QAEK,kBAAa,GAAG,cAAe,OAAA,OAAO,CAAC,KAAI,CAAC,GAAG,IAAI,KAAI,CAAC,GAAG,CAAC,UAAU,KAAK,cAAc,CAAC,IAAI,CAAC,EAAhE,CAAgE,CAAC;QAEhG,oBAAe,GAAG;;;gBACxB,IAAI,IAAI,CAAC,aAAa,EAAE,EAAE;oBACxB,WAAO;iBACR;gBAED,IAAI,IAAI,CAAC,cAAc,EAAE;oBACvB,WAAO,IAAI,CAAC,cAAc,EAAC;iBAC5B;gBAED,WAAO,IAAI,OAAO,CAAO,UAAC,OAAO,EAAE,MAAM;wBACvC,KAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC;4BAC3B,OAAO,SAAA;4BACP,MAAM,QAAA;yBACP,CAAC,CAAC;oBACL,CAAC,CAAC,EAAC;;aACJ,CAAC;QAEM,aAAQ,GAAG,UACjB,KAAc,EACd,OAAiB;;;;;;wBAEjB,IAAI,CAAC,OAAO,EAAE;4BAEZ,IAAI,KAAK,EAAE;gCACH,cAAY,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;gCAC1C,IAAI,WAAS,EAAE;oCACb,IAAI,WAAS,CAAC,QAAQ,IAAI,WAAS,CAAC,WAAW,EAAE;wCAI/C,WAAO,WAAS,CAAC,WAAW,EAAC;qCAC9B;oCAAC,IAAI,WAAS,CAAC,gBAAgB,EAAE;wCAChC,WAAO,WAAS,CAAC,gBAAgB,EAAC;qCACnC;iCACF;6BACF;iCAAM;gCACC,iBAAiB,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;gCAC/C,IAAI,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,gBAAgB,EAAE;oCACvC,WAAO,iBAAiB,CAAC,gBAAgB,EAAC;iCAC3C;6BACF;yBACF;wBAGK,OAAO,GAAG,IAAI,OAAO,CAAe,UAAO,OAAO,EAAE,MAAM;;;;;;wCAI7C,WAAM,IAAI,CAAC,SAAS,EAAE,EAAA;;wCAA/B,MAAM,GAAG,SAAsB;wCAG/B,OAAO,GAA6B;4CACxC,KAAK,EAAE,MAAM,CAAC,KAAK,IAAI,EAAE;4CACzB,WAAW,EAAE,EAAE;4CAGf,QAAQ,EAAE,KAAK;4CACf,UAAU,EAAE,EAAE;4CACd,WAAW,EAAE,EAAE;yCAChB,CAAC;wCACI,QAAQ,GAA4B;4CACxC,OAAO,EAAE,SAAS;4CAClB,SAAS,EAAE,IAAA,sBAAY,GAAE;4CACzB,OAAO,EAAE,OAAO;4CAChB,OAAO,SAAA;4CACP,SAAS,EAAE;gDACT,OAAO,EAAE,IAAA,mBAAU,GAAE;gDACrB,OAAO,EAAE,MAAM,CAAC,OAAO;gDACvB,aAAa,EAAE,MAAM,CAAC,aAAa;6CACpC;yCACF,CAAC;wCACkB,WAAM,IAAI,CAAC,IAAI,CAA8B;gDAC/D,GAAG,EAAE,QAAQ;gDACb,YAAY,EAAE,IAAI;gDAClB,aAAa,EAAE,IAAI;gDACnB,OAAO,EAAE,qBAAqB;6CAC/B,CAAC,EAAA;;wCALI,WAAW,GAAG,SAKlB;wCAEF,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,IAAI,EAAE;4CAE7B,OAAO,CAAC;gDACN,KAAK,EAAE,MAAM,CAAC,KAAK;6CACpB,CAAC,CAAC;yCACJ;6CAAM;4CAEL,MAAM,CAAC,IAAI,KAAK,CAAC,UAAG,WAAW,CAAC,OAAO,CAAC,IAAI,cAAI,WAAW,CAAC,OAAO,CAAC,OAAO,CAAE,CAAC,CAAC,CAAC;yCACjF;;;;wCAED,MAAM,CAAC,GAAC,CAAC,CAAC;;;;;6BAEb,CAAC,CAAC;wBAGC,SAAS,GAAG,KAAK,IAAI,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;wBAE3C,YAAY,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;wBAEhC,IAAI,SAAS,EAAE;4BACb,SAAS,CAAC,QAAQ,GAAG,KAAK,CAAC;4BAC3B,SAAS,CAAC,gBAAgB,GAAG,OAAO,CAAC;4BACrC,SAAS,CAAC,YAAY,GAAG,YAAY,CAAC;yBACvC;6BAAM;4BACL,SAAS,GAAG;gCACV,QAAQ,EAAE,KAAK;gCACf,gBAAgB,EAAE,OAAO;gCACzB,YAAY,cAAA;6BACb,CAAC;4BAEF,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,IAAI,EAAE,EAAE,SAAS,CAAC,CAAC;yBAC1C;;;;wBAkBqB,WAAM,OAAO,EAAA;;wBAA3B,WAAW,GAAG,SAAa;wBAC3B,YAAY,GAAG,KAAK,IAAI,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;wBACtD,IACE,YAAY;+BACT,YAAY,KAAK,SAAS;+BAC1B,YAAY,CAAC,YAAY,KAAK,YAAY,EAC7C;4BACA,SAAS,CAAC,QAAQ,GAAG,IAAI,CAAC;4BAC1B,SAAS,CAAC,gBAAgB,GAAG,SAAS,CAAC;4BACvC,SAAS,CAAC,YAAY,GAAG,SAAS,CAAC;4BACnC,SAAS,CAAC,WAAW,GAAG,WAAW,CAAC;4BACpC,WAAO,WAAW,EAAC;yBACpB;wBAAC,IAAI,YAAY,EAAE;4BAClB,IAAI,YAAY,CAAC,QAAQ,IAAI,YAAY,CAAC,WAAW,EAAE;gCACrD,WAAO,YAAY,CAAC,WAAW,EAAC;6BACjC;4BAAC,IAAI,YAAY,CAAC,gBAAgB,EAAE;gCACnC,WAAO,YAAY,CAAC,gBAAgB,EAAC;6BACtC;4BACD,MAAM,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC;yBAC7C;6BAAM;4BACL,MAAM,IAAI,KAAK,CAAC,qBAAqB,CAAC,CAAC;yBACxC;;;;wBAED,SAAS,CAAC,QAAQ,GAAG,KAAK,CAAC;wBAC3B,SAAS,CAAC,gBAAgB,GAAG,SAAS,CAAC;wBACvC,SAAS,CAAC,YAAY,GAAG,SAAS,CAAC;wBACnC,SAAS,CAAC,WAAW,GAAG,SAAS,CAAC;wBAClC,MAAM,GAAC,CAAC;;;;aAEX,CAAC;QAEM,cAAS,GAAG;;;;;wBAClB,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,OAAO,CAAC,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,EAAE;4BACvD,WAAO,IAAI,CAAC,OAAO,EAAC;yBACrB;wBACK,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC;wBACzB,WAAM,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,CAAC,gBAAgB,EAAE,EAAE,OAAO,EAAE,IAAA,mBAAU,GAAE,EAAE,CAAC,EAAA;;wBAA7F,GAAG,GAAG,SAAuF;wBAEnG,IAAI,GAAG,CAAC,IAAI,EAAE;4BACZ,MAAM,IAAI,KAAK,CAAC,6GAAgC,GAAG,CAAC,IAAI,CAAE,CAAC,CAAC;yBAC7D;wBAED,IAAI,GAAG,CAAC,IAAI,EAAE;4BACN,KAA2C,GAAG,CAAC,IAAI,EAAjD,OAAO,aAAA,EAAE,KAAK,WAAA,EAAE,aAAa,mBAAA,EAAE,KAAK,WAAA,CAAc;4BAC1D,WAAO;oCACL,OAAO,SAAA;oCACP,KAAK,OAAA;oCACL,aAAa,eAAA;oCACb,KAAK,OAAA;oCACL,SAAS,WAAA;iCACV,EAAC;yBACH;wBACD,MAAM,IAAI,KAAK,CAAC,6BAA6B,CAAC,CAAC;;;aAChD,CAAC;QAEM,iCAA4B,GAAG;YACrC,IAAI,CAAC,KAAI,CAAC,YAAY,CAAC,MAAM,EAAE;gBAC7B,OAAO,gCAAgC,CAAC;aACzC;YAGD,OAAO,CACL,CAAC,KAAI,CAAC,YAAY,CAAC,MAAM,CAAC,UAAC,GAAG,EAAE,GAAG,IAAK,OAAA,GAAG,GAAG,GAAG,EAAT,CAAS,CAAC;kBAC9C,KAAI,CAAC,YAAY,CAAC,MAAM,CAAC;kBAC3B,GAAG,CACN,CAAC;QACJ,CAAC,CAAC;QAyCM,SAAI,GAAG;;;;;wBACP,GAAG,GAA2B;4BAClC,OAAO,EAAE,SAAS;4BAClB,SAAS,EAAE,IAAA,sBAAY,GAAE;4BACzB,OAAO,EAAE,MAAM;4BACf,OAAO,EAAE,IAAI;yBACd,CAAC;wBACF,WAAM,IAAI,CAAC,IAAI,CAAC;gCACd,GAAG,KAAA;6BACJ,CAAC,EAAA;;wBAFF,SAEE,CAAC;;;;aAEJ,CAAC;QAEM,iBAAY,GAAG,UAAC,MAA8B,EAAE,OAAe;YACrE,KAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QAC9C,CAAC,CAAC;QAEM,iBAAY,GAAG,UAAC,MAA8B,EAAE,OAAe;YACrE,IAAI,OAAO,EAAE;gBACX,KAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;aACxC;YACD,KAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAC9C,KAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YAErC,IAAI,CAAC,KAAI,CAAC,gBAAgB,CAAC,IAAI,EAAE;gBAE/B,KAAI,CAAC,KAAK,CAAC,2BAAgB,CAAC,mBAAmB,CAAC,CAAC;aAClD;QACH,CAAC,CAAC;QA7zBA,IAAI,CAAC,aAAa,GAAG,OAAO,CAAC,YAAY,IAAI,qBAAqB,CAAC;QAEnE,IAAI,CAAC,kBAAkB,GAAQ,OAAO,CAAC,iBAAiB,IAAI,6BAA6B,CAAC;QAC1F,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC,OAAO,CAAC;IAClC,CAAC;IAED,gDAAc,GAAd;QACE,IAAI,CAAC,cAAc,IAAI,YAAY,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QACzD,IAAI,CAAC,cAAc,IAAI,YAAY,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;IAC3D,CAAC;IAiHD,uCAAK,GAAL,UAAM,IAAsB;QAC1B,IAAI,CAAC,cAAc,EAAE,CAAC;QAEtB,IAAI,IAAI,CAAC,GAAG,EAAE;YACZ,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,EAAE,gCAAqB,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC;YACvD,IAAI,CAAC,GAAG,GAAG,SAAS,CAAC;SACtB;IACH,CAAC;IAoBD,uCAAK,GAAL,UAAM,OAAwB;QAC5B,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;YACrC,IAAI,CAAC,uBAAuB,CAAC,KAAK,CAAC,CAAC;SACrC;QAED,IAAM,aAAa,GAAG,IAAI,iDAAsB,uBAC3C,OAAO,KACV,IAAI,EAAE,IAAI,CAAC,IAAI,EACf,KAAK,EAAE,IAAI,CAAC,QAAQ,EACpB,aAAa,EAAE,IAAI,CAAC,aAAa,EACjC,eAAe,EAAE,IAAI,CAAC,eAAe,EACrC,4BAA4B,EAAE,IAAI,CAAC,4BAA4B,EAC/D,YAAY,EAAE,IAAI,CAAC,YAAY,EAC/B,YAAY,EAAE,IAAI,CAAC,YAAY,EAC/B,KAAK,EAAE,IAAI,IACX,CAAC;QACH,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;QACzC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,aAAa,CAAC,OAAO,EAAE,aAAa,CAAC,CAAC;QACjE,OAAO,aAAa,CAAC,QAAQ,CAAC;IAChC,CAAC;IAklBO,2CAAS,GAAjB,UAAkB,SAAmB;QAArC,iBAqCC;QApCC,IAAI,CAAC,cAAc,EAAE,CAAC;QAEtB,IAAI,CAAC,cAAc,GAAG,UAAU,CAC9B;;;;;;;wBAEI,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,IAAI,CAAC,GAAG,CAAC,UAAU,KAAK,cAAc,CAAC,IAAI,EAAE;4BAE5D,WAAO;yBACR;wBAED,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;wBAClC,WAAM,IAAI,CAAC,IAAI,EAAE,EAAA;;wBAAjB,SAAiB,CAAC;wBAClB,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;wBAGrB,IAAI,CAAC,cAAc,GAAG,UAAU,CAAC;4BAC/B,OAAO,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC;4BAChC,IAAI,KAAI,CAAC,WAAW,GAAG,2BAA2B,EAAE;gCAClD,KAAI,CAAC,WAAW,EAAE,CAAC;gCACnB,KAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;6BACtB;iCAAM;gCAEL,KAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,CAAC;6BACpC;wBACH,CAAC,EAAE,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,uBAAuB,CAAC,CAAC;;;;wBAEpD,IAAI,IAAI,CAAC,WAAW,GAAG,2BAA2B,EAAE;4BAClD,IAAI,CAAC,WAAW,EAAE,CAAC;4BACnB,IAAI,CAAC,SAAS,EAAE,CAAC;yBAClB;6BAAM;4BACL,IAAI,CAAC,KAAK,CAAC,2BAAgB,CAAC,kBAAkB,CAAC,CAAC;yBACjD;;;;;aAEJ,EACD,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,oBAAoB,CAC7D,CAAC;IACJ,CAAC;IA+BH,8BAAC;AAAD,CAAC,AA91BD,IA81BC;AA91BY,0DAAuB","sourcesContent":["/* eslint-disable no-plusplus */\n/* eslint-disable @typescript-eslint/consistent-type-assertions */\n/* eslint-disable new-cap */\n/* eslint-disable @typescript-eslint/no-misused-promises */\nimport { VirtualWebSocketClient } from './virtual-websocket-client';\nimport { genRequestId } from './message';\nimport {\n  IDatabaseServiceContext,\n} from '@cloudbase/types/database';\nimport {\n  IWatchOptions,\n  DBRealtimeListener,\n  IRequestMessage,\n  IResponseMessage,\n  IRequestMessagePingMsg,\n  IRequestMessageLoginMsg,\n  IResponseMessageLoginResMsg,\n  IRequestMessageLoginData,\n} from '@cloudbase/types/realtime';\nimport {\n  CLOSE_EVENT_CODE,\n  CLOSE_EVENT_CODE_INFO,\n  getWSCloseError,\n} from './ws-event';\n\nimport { ERR_CODE, TimeoutError, RealtimeErrorMessageError, CloudSDKError } from './error';\nimport { getWsClass, getRuntime } from './common';\nimport { sleep } from './utils';\n\nexport interface IRealtimeWebSocketClientConstructorOptions {\n  maxReconnect?: number\n  reconnectInterval?: number\n  context: IDatabaseServiceContext\n}\n\nexport interface ISignature {\n  envId: string\n  secretVersion: number\n  signStr: string\n  wsUrl: string\n  expireTS: number\n}\n\nexport interface ILoginInfo {\n  loggedIn: boolean\n  loggingInPromise?: Promise<ILoginResult>\n  loginStartTS?: number\n  loginResult?: ILoginResult\n}\n\nexport interface ILoginResult {\n  envId: string\n}\n\nexport interface IWSSendOptions {\n  msg: IRequestMessage\n  waitResponse?: boolean\n  // when waitResponse is set to true, if skipOnMessage is true, general onMessage handler will be skipped\n  skipOnMessage?: boolean\n  timeout?: number\n}\n\nexport interface IWSWatchOptions extends IWatchOptions {\n  envId?: string\n  collectionName: string\n  query: string\n  limit?: number\n  orderBy?: Record<string, string>\n}\n\ninterface IResolveReject {\n  resolve: (value?: any | PromiseLike<any> | undefined) => void\n  reject: (reason?: any) => void\n}\n\ninterface IResponseWaitSpec extends IResolveReject {\n  skipOnMessage?: boolean\n}\n\ninterface IWsSign {\n  signStr: string,\n  wsUrl: string,\n  secretVersion: string\n  envId: string\n  expiredTs: number\n}\n\nconst WS_READY_STATE = {\n  CONNECTING: 0,\n  OPEN: 1,\n  CLOSING: 2,\n  CLOSED: 3,\n};\n\nconst MAX_RTT_OBSERVED = 3;\nconst DEFAULT_EXPECTED_EVENT_WAIT_TIME = 5000;\nconst DEFAULT_UNTRUSTED_RTT_THRESHOLD = 10000;\nconst DEFAULT_MAX_RECONNECT = 5;\nconst DEFAULT_WS_RECONNECT_INTERVAL = 10000;\n// const DEFAULT_WS_RECONNECT_MAX_VALID_INTERVAL = 3 * 60 * 1000\nconst DEFAULT_PING_FAIL_TOLERANCE = 2;\nconst DEFAULT_PONG_MISS_TOLERANCE = 2;\nconst DEFAULT_LOGIN_TIMEOUT = 5000;\n\nexport class RealtimeWebSocketClient {\n  private _virtualWSClient: Set<VirtualWebSocketClient> = new Set();\n  // after listener initWatch, the listener has the queryID and can store it here\n  private _queryIdClientMap: Map<string, VirtualWebSocketClient> = new Map();\n  private _watchIdClientMap: Map<string, VirtualWebSocketClient> = new Map();\n  private _maxReconnect: number;\n  // private _availableRetries: number\n  private _reconnectInterval: number;\n  private _context: IDatabaseServiceContext;\n  // private _ws?: WXNS.Socket.ISocketTask\n  private _ws?: any;\n  private _lastPingSendTS?: number;\n  private _pingFailed = 0;\n  private _pongMissed = 0;\n  private _pingTimeoutId?: number;\n  private _pongTimeoutId?: number;\n  private _logins: Map<string /* envId */, ILoginInfo> = new Map();\n  // private _loginInfo: ILoginInfo\n  // private _signatures: Map<string /* envId */, ISignature> = new Map()\n  private _wsInitPromise?: Promise<void>;\n  private _wsReadySubsribers: IResolveReject[] = [];\n  private _wsResponseWait: Map<\n  string /* requestId */,\n  IResponseWaitSpec\n  > = new Map();\n  private _rttObserved: number[] = [];\n  private _reconnectState: boolean;\n  // obtained from the first getSignature with no envId provided\n  // private _defaultEnvId?: string\n  private _wsSign: IWsSign;\n\n  constructor(options: IRealtimeWebSocketClientConstructorOptions) {\n    this._maxReconnect = options.maxReconnect || DEFAULT_MAX_RECONNECT;\n    // this._availableRetries = this._maxReconnect\n    this._reconnectInterval =      options.reconnectInterval || DEFAULT_WS_RECONNECT_INTERVAL;\n    this._context = options.context;\n  }\n\n  clearHeartbeat() {\n    this._pingTimeoutId && clearTimeout(this._pingTimeoutId);\n    this._pongTimeoutId && clearTimeout(this._pongTimeoutId);\n  }\n\n  send = async <T = any>(opts: IWSSendOptions): Promise<T> => new Promise<T>(async (_resolve, _reject) => {\n    let timeoutId: number;\n    let _hasResolved = false;\n    let _hasRejected = false;\n\n    const resolve: typeof _resolve = (value?: T | PromiseLike<T> | undefined) => {\n      _hasResolved = true;\n      timeoutId && clearTimeout(timeoutId);\n      _resolve(value);\n    };\n\n    const reject: typeof _reject = (error: any) => {\n      _hasRejected = true;\n      timeoutId && clearTimeout(timeoutId);\n      _reject(error);\n    };\n\n    if (opts.timeout) {\n      // @ts-ignore\n      timeoutId = setTimeout(async () => {\n        if (!_hasResolved || !_hasRejected) {\n          // wait another immediate timeout to allow the success/fail callback to be invoked if ws has already got the result,\n          // this is because the timer is registered before ws.send\n          await sleep(0);\n          if (!_hasResolved || !_hasRejected) {\n            reject(new TimeoutError('wsclient.send timedout'));\n          }\n        }\n      }, opts.timeout);\n    }\n\n    try {\n      // if (this._context.debug) {\n      // console.log(`[realtime] ws send (${new Date()}): `, opts)\n      // console.log(\n      //   `[realtime] ws send ${\n      //     opts.msg.msgType\n      //   } (${new Date().toLocaleString()}): `,\n      //   opts\n      // )\n      // }\n\n      if (this._wsInitPromise) {\n        await this._wsInitPromise;\n      }\n\n      if (!this._ws) {\n        reject(new Error('invalid state: ws connection not exists, can not send message'));\n        return;\n      }\n\n      if (this._ws.readyState !== WS_READY_STATE.OPEN) {\n        reject(new Error(`ws readyState invalid: ${this._ws.readyState}, can not send message`));\n        return;\n      }\n\n      if (opts.waitResponse) {\n        this._wsResponseWait.set(opts.msg.requestId, {\n          resolve,\n          reject,\n          skipOnMessage: opts.skipOnMessage,\n        } as IResponseWaitSpec);\n      }\n\n      // console.log('send msg:', opts.msg)\n      try {\n        await this._ws.send(JSON.stringify(opts.msg));\n        if (!opts.waitResponse) {\n          resolve(undefined);\n        }\n      } catch (err) {\n        if (err) {\n          reject(err);\n          if (opts.waitResponse) {\n            this._wsResponseWait.delete(opts.msg.requestId);\n          }\n        }\n      }\n      // this._ws.send(JSON.stringify(opts.msg), err => {\n      //   if (err) {\n      //     reject(err)\n      //     if (opts.waitResponse) {\n      //       this._wsResponseWait.delete(opts.msg.requestId)\n      //     }\n      //     return\n      //   }\n\n      //   if (!opts.waitResponse) {\n      //     resolve()\n      //   }\n      // })\n\n      // this._ws.send({\n      //   data: JSON.stringify(opts.msg),\n      //   success: res => {\n      //     if (!opts.waitResponse) {\n      //       resolve(res)\n      //     }\n      //   },\n      //   fail: e => {\n      //     reject(e)\n      //     if (opts.waitResponse) {\n      //       this._wsResponseWait.delete(opts.msg.requestId)\n      //     }\n      //   }\n      // })\n    } catch (e) {\n      reject(e);\n    }\n  });\n\n  close(code: CLOSE_EVENT_CODE) {\n    this.clearHeartbeat();\n\n    if (this._ws) {\n      this._ws.close(code, CLOSE_EVENT_CODE_INFO[code].name);\n      this._ws = undefined;\n    }\n  }\n\n  closeAllClients = (error: any) => {\n    this._virtualWSClient.forEach((client) => {\n      client.closeWithError(error);\n    });\n  };\n\n  pauseClients = (clients?: Set<VirtualWebSocketClient>) => {\n    (clients || this._virtualWSClient).forEach((client) => {\n      client.pause();\n    });\n  };\n\n  resumeClients = (clients?: Set<VirtualWebSocketClient>) => {\n    (clients || this._virtualWSClient).forEach((client) => {\n      client.resume();\n    });\n  };\n\n  watch(options: IWSWatchOptions): DBRealtimeListener {\n    if (!this._ws && !this._wsInitPromise) {\n      this.initWebSocketConnection(false);\n    }\n\n    const virtualClient = new VirtualWebSocketClient({\n      ...options,\n      send: this.send,\n      login: this.webLogin,\n      isWSConnected: this.isWSConnected,\n      onceWSConnected: this.onceWSConnected,\n      getWaitExpectedTimeoutLength: this.getWaitExpectedTimeoutLength,\n      onWatchStart: this.onWatchStart,\n      onWatchClose: this.onWatchClose,\n      debug: true,\n    });\n    this._virtualWSClient.add(virtualClient);\n    this._watchIdClientMap.set(virtualClient.watchId, virtualClient);\n    return virtualClient.listener;\n  }\n\n  private initWebSocketConnection = async (\n    reconnect: boolean,\n    availableRetries: number = this._maxReconnect\n  ): Promise<void> => {\n    // 当前处于正在重连中的状态\n    if (reconnect && this._reconnectState) {\n      return; // 忽略\n    }\n\n    if (reconnect) {\n      this._reconnectState = true; // 重连状态开始\n    }\n\n    if (this._wsInitPromise) {\n      // there already exists a websocket initiation, just wait for it\n      return this._wsInitPromise;\n    }\n\n    // if (process.env.DEBUG) {\n    // console.log(\n    //   `[realtime] initWebSocketConnection reconnect ${reconnect} availableRetries ${availableRetries}`\n    // )\n    // }\n\n    if (reconnect) {\n      this.pauseClients();\n    }\n\n    this.close(CLOSE_EVENT_CODE.ReconnectWebSocket);\n\n    this._wsInitPromise = new Promise<void>(async (resolve, reject) => {\n      try {\n        // if (process.env.DEBUG) {\n        // console.log(\n        //   '[realtime] initWebSocketConnection start throwErrorIfNetworkOffline'\n        // )\n        // }\n\n        // 暂不检查网络态\n        // await throwErrorIfNetworkOffline()\n\n        // if (process.env.DEBUG) {\n        // console.log('[realtime] initWebSocketConnection start getSignature')\n        // }\n\n        // const signature = await this.getSignature()\n        const wsSign = await this.getWsSign();\n\n        // if (process.env.DEBUG) {\n        // console.log('[realtime] initWebSocketConnection getSignature success')\n        // console.log('[realtime] initWebSocketConnection start connectSocket')\n        // }\n\n        await new Promise((success) => {\n          // this._ws = getSDK(this._context.identifiers)\n          //   ._socketSkipCheckDomainFactory()\n          //   .connectSocket({\n          //     url: signature.wsUrl,\n          //     header: {\n          //       \"content-type\": \"application/json\"\n          //     },\n          //     success: () => success(),\n          //     fail\n          //   })\n\n          const url = wsSign.wsUrl || 'wss://tcb-ws.tencentcloudapi.com';\n          const wsClass = getWsClass();\n          this._ws = wsClass ? new wsClass(url) : new WebSocket(url);\n          success(undefined);\n        });\n\n        if (this._ws.connect) {\n          await this._ws.connect();\n        }\n\n        // if (process.env.DEBUG) {\n        // console.log(\n        //   '[realtime] initWebSocketConnection connectSocket successfully fired'\n        // )\n        // }\n\n        await this.initWebSocketEvent();\n        resolve();\n\n        if (reconnect) {\n          this.resumeClients();\n          this._reconnectState = false; // 重连状态结束\n        }\n      } catch (e) {\n        // if (process.env.DEBUG) {\n        console.error('[realtime] initWebSocketConnection connect fail', e);\n        // }\n\n        if (availableRetries > 0) {\n          // this is an optimization, in case of network offline, we don't need to stubbornly sleep for sometime,\n          // we only need to wait for the network to be back online, this ensures minimum downtime\n          // const { isConnected } = await getNetworkStatus()\n          const isConnected = true;\n\n          // if (process.env.DEBUG) {\n          // console.log(\n          //   '[realtime] initWebSocketConnection waiting for network online'\n          // )\n          // }\n\n          // auto wait until network online, cause' it would be meaningless to reconnect while network is offline\n\n          // await onceNetworkOnline()\n\n          // COMPATIBILITY: wait for ide state update\n          // if (isDevTools()) {\n          //   await sleep(0)\n          // }\n\n          // if (process.env.DEBUG) {\n          // console.log('[realtime] initWebSocketConnection network online')\n          // }\n\n          this._wsInitPromise = undefined;\n\n          if (isConnected) {\n            // if (process.env.DEBUG) {\n            // console.log(\n            //   `[realtime] initWebSocketConnection sleep ${this._reconnectInterval}ms`\n            // )\n            // }\n            await sleep(this._reconnectInterval);\n            if (reconnect) {\n              this._reconnectState = false; // 重连异常也算重连状态结束\n            }\n          }\n\n          resolve(this.initWebSocketConnection(reconnect, availableRetries - 1));\n        } else {\n          reject(e);\n\n          if (reconnect) {\n            this.closeAllClients(new CloudSDKError({\n              errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_RECONNECT_WATCH_FAIL as string,\n              errMsg: e,\n            }));\n          }\n        }\n      }\n    });\n\n    // let success = false\n\n    try {\n      await this._wsInitPromise;\n      // success = true\n      this._wsReadySubsribers.forEach(({ resolve }) => resolve());\n    } catch (e) {\n      this._wsReadySubsribers.forEach(({ reject }) => reject());\n    } finally {\n      this._wsInitPromise = undefined;\n      this._wsReadySubsribers = [];\n    }\n\n    // if (process.env.DEBUG) {\n    // console.log(\n    //   `[realtime] initWebSocketConnection ${success ? 'success' : 'fail'}`\n    // )\n    // }\n  };\n\n  private initWebSocketEvent = () => new Promise<void>((resolve, reject) => {\n    if (!this._ws) {\n      throw new Error('can not initWebSocketEvent, ws not exists');\n    }\n\n    let wsOpened = false;\n\n    this._ws.onopen = (event) => {\n      // this._ws.onOpen(() => {\n      // this._ws.on(\"open\", () => {\n      // this._context.debug &&\n      console.warn('[realtime] ws event: open', event);\n      wsOpened = true;\n      resolve();\n    };\n\n    this._ws.onerror = (event) => {\n      // this._ws.on(\"error\", error => {\n      // this._ws.onError(error => {\n      // all logins are invalid after disconnection\n      this._logins = new Map();\n\n      // error写进file\n\n      if (!wsOpened) {\n        // this._context.debug &&\n        console.error('[realtime] ws open failed with ws event: error', event);\n        // writeToFile(\n        //   \"wserror.txt\",\n        //   `${\n        //     this.specialNumber\n        //   } [realtime] ws open failed with ws event: error ${error} \\n`\n        // )\n\n        reject(event);\n      } else {\n        // this._context.debug &&\n        console.error('[realtime] ws event: error', event);\n\n        // writeToFile(\n        //   \"wserror.txt\",\n        //   `${this.specialNumber} [realtime] ws event: error ${error} \\n`\n        // )\n\n        this.clearHeartbeat();\n        this._virtualWSClient.forEach(client => client.closeWithError(new CloudSDKError({\n          errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_WEBSOCKET_CONNECTION_ERROR as string,\n          errMsg: event,\n        })));\n      }\n    };\n\n    // TODO: reconnect\n    this._ws.onclose = (closeEvent) => {\n      // this._ws.on(\"close\", (closeEvent, closereason) => {\n      // this._ws.onClose(closeEvent => {\n      // if (process.env.DEBUG) {\n      console.warn('[realtime] ws event: close', closeEvent);\n      // }\n\n      // writeToFile(\n      //   \"wsclose.txt\",\n      //   `${\n      //     this.specialNumber\n      //   } [realtime] ws event: close ${closeEvent} ${closereason} \\n`\n      // )\n\n      // all logins are invalid after disconnection\n      this._logins = new Map();\n\n      this.clearHeartbeat();\n      switch (closeEvent.code) {\n        case CLOSE_EVENT_CODE.ReconnectWebSocket: {\n          // just ignore\n          break;\n        }\n        case CLOSE_EVENT_CODE.NoRealtimeListeners: {\n          // quit\n          break;\n        }\n        case CLOSE_EVENT_CODE.HeartbeatPingError:\n        case CLOSE_EVENT_CODE.HeartbeatPongTimeoutError:\n        case CLOSE_EVENT_CODE.NormalClosure:\n        case CLOSE_EVENT_CODE.AbnormalClosure: {\n          // Normal Closure and Abnormal Closure:\n          //   expected closure, most likely dispatched by wechat client,\n          //   since this is the status code dispatched in case of network failure,\n          //   we should retry\n\n          if (this._maxReconnect > 0) {\n            // if (this._availableRetries > 0) {\n            this.initWebSocketConnection(true, this._maxReconnect);\n          } else {\n            this.closeAllClients(getWSCloseError(closeEvent.code));\n          }\n          break;\n        }\n        case CLOSE_EVENT_CODE.NoAuthentication: {\n          this.closeAllClients(getWSCloseError(closeEvent.code, closeEvent.reason));\n          break;\n        }\n        default: {\n          // we should retry by default\n          if (this._maxReconnect > 0) {\n            // if (this._availableRetries > 0) {\n            this.initWebSocketConnection(true, this._maxReconnect);\n          } else {\n            this.closeAllClients(getWSCloseError(closeEvent.code));\n          }\n          // console.warn(`[realtime] unrecognize ws close event`, closeEvent)\n          // this.closeAllClients(getWSCloseError(closeEvent.code))\n        }\n      }\n    };\n\n    this._ws.onmessage = (res) => {\n      // this._ws.on(\"message\", res => {\n      // this._ws.onMessage(res => {\n      // const rawMsg = res.data\n      const rawMsg = res.data;\n\n      // reset & restart heartbeat\n      this.heartbeat();\n\n      let msg: IResponseMessage;\n\n      try {\n        msg = JSON.parse(rawMsg as string);\n      } catch (e) {\n        throw new Error(`[realtime] onMessage parse res.data error: ${e}`);\n      }\n\n      // console.log(\n      //   `[realtime] onMessage ${\n      //     msg.msgType\n      //   } (${new Date().toLocaleString()})`,\n      //   msg\n      // )\n\n      if (msg.msgType === 'ERROR') {\n        // 找到当前监听，并将error返回\n        let virtualWatch = null;\n        this._virtualWSClient.forEach((item) => {\n          if (item.watchId === msg.watchId) {\n            virtualWatch = item;\n          }\n        });\n\n        if (virtualWatch) {\n          virtualWatch.listener.onError(msg);\n        }\n      }\n\n      const responseWaitSpec = this._wsResponseWait.get(msg.requestId);\n      if (responseWaitSpec) {\n        try {\n          if (msg.msgType === 'ERROR') {\n            responseWaitSpec.reject(new RealtimeErrorMessageError(msg));\n          } else {\n            responseWaitSpec.resolve(msg);\n          }\n        } catch (e) {\n          // this._context.debug &&\n          console.error(\n            'ws onMessage responseWaitSpec.resolve(msg) errored:',\n            e\n          );\n        } finally {\n          this._wsResponseWait.delete(msg.requestId);\n        }\n        if (responseWaitSpec.skipOnMessage) {\n          return;\n        }\n      }\n\n      if (msg.msgType === 'PONG') {\n        if (this._lastPingSendTS) {\n          const rtt = Date.now() - this._lastPingSendTS;\n          if (rtt > DEFAULT_UNTRUSTED_RTT_THRESHOLD) {\n            // this._context.debug &&\n            console.warn(`[realtime] untrusted rtt observed: ${rtt}`);\n            return;\n          }\n          if (this._rttObserved.length >= MAX_RTT_OBSERVED) {\n            this._rttObserved.splice(\n              0,\n              this._rttObserved.length - MAX_RTT_OBSERVED + 1\n            );\n          }\n          this._rttObserved.push(rtt);\n        }\n        return;\n      }\n\n      let client = msg.watchId && this._watchIdClientMap.get(msg.watchId);\n      if (client) {\n        client.onMessage(msg);\n      } else {\n        // TODO, this is a temporary fix done for server\n        // if (process.env.DEBUG) {\n        console.error(\n          `[realtime] no realtime listener found responsible for watchId ${msg.watchId}: `,\n          msg\n        );\n        // }\n        switch (msg.msgType) {\n          case 'INIT_EVENT':\n          case 'NEXT_EVENT':\n          case 'CHECK_EVENT': {\n            client = this._queryIdClientMap.get(msg.msgData.queryID);\n            if (client) {\n              client.onMessage(msg);\n            }\n            break;\n          }\n          default: {\n            for (const [,client] of Array.from(this._watchIdClientMap.entries())) {\n              // console.log('watchid*****', watchId)\n              client.onMessage(msg);\n              break;\n            }\n          }\n        }\n      }\n    };\n\n    this.heartbeat();\n  });\n\n  private isWSConnected = (): boolean => Boolean(this._ws && this._ws.readyState === WS_READY_STATE.OPEN);\n\n  private onceWSConnected = async (): Promise<void> => {\n    if (this.isWSConnected()) {\n      return;\n    }\n\n    if (this._wsInitPromise) {\n      return this._wsInitPromise;\n    }\n\n    return new Promise<void>((resolve, reject) => {\n      this._wsReadySubsribers.push({\n        resolve,\n        reject,\n      });\n    });\n  };\n\n  private webLogin = async (\n    envId?: string,\n    refresh?: boolean\n  ): Promise<any> => {\n    if (!refresh) {\n      // let loginInfo = this._loginInfo\n      if (envId) {\n        const loginInfo = this._logins.get(envId);\n        if (loginInfo) {\n          if (loginInfo.loggedIn && loginInfo.loginResult) {\n            // if (process.env.DEBUG) {\n            // console.log('[realtime] login: already logged in')\n            // }\n            return loginInfo.loginResult;\n          } if (loginInfo.loggingInPromise) {\n            return loginInfo.loggingInPromise;\n          }\n        }\n      } else {\n        const emptyEnvLoginInfo = this._logins.get('');\n        if (emptyEnvLoginInfo?.loggingInPromise) {\n          return emptyEnvLoginInfo.loggingInPromise;\n        }\n      }\n    }\n    // console.log('[realtime] login: logging in')\n\n    const promise = new Promise<ILoginResult>(async (resolve, reject) => {\n      try {\n        // const signature = await this.getSignature(envId, refresh)\n\n        const wsSign = await this.getWsSign();\n\n        // const wxVersion = getWXVersion()\n        const msgData: IRequestMessageLoginData = {\n          envId: wsSign.envId || '',\n          accessToken: '', // 已废弃字段\n          // signStr: signature.signStr,\n          // secretVersion: signature.secretVersion,\n          referrer: 'web',\n          sdkVersion: '',\n          dataVersion: '',\n        };\n        const loginMsg: IRequestMessageLoginMsg = {\n          watchId: undefined,\n          requestId: genRequestId(),\n          msgType: 'LOGIN',\n          msgData,\n          exMsgData: {\n            runtime: getRuntime(),\n            signStr: wsSign.signStr,\n            secretVersion: wsSign.secretVersion,\n          },\n        };\n        const loginResMsg = await this.send<IResponseMessageLoginResMsg>({\n          msg: loginMsg,\n          waitResponse: true,\n          skipOnMessage: true,\n          timeout: DEFAULT_LOGIN_TIMEOUT,\n        });\n\n        if (!loginResMsg.msgData.code) {\n          // login success\n          resolve({\n            envId: wsSign.envId,\n          });\n        } else {\n          // login failed\n          reject(new Error(`${loginResMsg.msgData.code} ${loginResMsg.msgData.message}`));\n        }\n      } catch (e) {\n        reject(e);\n      }\n    });\n\n    // let loginInfo = this._loginInfo\n    let loginInfo = envId && this._logins.get(envId);\n\n    const loginStartTS = Date.now();\n\n    if (loginInfo) {\n      loginInfo.loggedIn = false;\n      loginInfo.loggingInPromise = promise;\n      loginInfo.loginStartTS = loginStartTS;\n    } else {\n      loginInfo = {\n        loggedIn: false,\n        loggingInPromise: promise,\n        loginStartTS,\n      };\n      // this._loginInfo = loginInfo\n      this._logins.set(envId || '', loginInfo);\n    }\n\n    // try {\n    //   const loginResult = await promise\n    //   loginInfo.loggedIn = true\n    //   loginInfo.loggingInPromise = undefined\n    //   loginInfo.loginStartTS = undefined\n    //   loginInfo.loginResult = loginResult\n    //   return loginResult\n    // } catch (e) {\n    //   loginInfo.loggedIn = false\n    //   loginInfo.loggingInPromise = undefined\n    //   loginInfo.loginStartTS = undefined\n    //   loginInfo.loginResult = undefined\n    //   throw e\n    // }\n\n    try {\n      const loginResult = await promise;\n      const curLoginInfo = envId && this._logins.get(envId);\n      if (\n        curLoginInfo\n        && curLoginInfo === loginInfo\n        && curLoginInfo.loginStartTS === loginStartTS\n      ) {\n        loginInfo.loggedIn = true;\n        loginInfo.loggingInPromise = undefined;\n        loginInfo.loginStartTS = undefined;\n        loginInfo.loginResult = loginResult;\n        return loginResult;\n      } if (curLoginInfo) {\n        if (curLoginInfo.loggedIn && curLoginInfo.loginResult) {\n          return curLoginInfo.loginResult;\n        } if (curLoginInfo.loggingInPromise) {\n          return curLoginInfo.loggingInPromise;\n        }\n        throw new Error('ws unexpected login info');\n      } else {\n        throw new Error('ws login info reset');\n      }\n    } catch (e) {\n      loginInfo.loggedIn = false;\n      loginInfo.loggingInPromise = undefined;\n      loginInfo.loginStartTS = undefined;\n      loginInfo.loginResult = undefined;\n      throw e;\n    }\n  };\n\n  private getWsSign = async (): Promise<IWsSign> => {\n    if (this._wsSign && this._wsSign.expiredTs > Date.now()) {\n      return this._wsSign;\n    }\n    const expiredTs = Date.now() + 60000;\n    const res = await this._context.appConfig.request.send('auth.wsWebSign', { runtime: getRuntime() });\n\n    if (res.code) {\n      throw new Error(`[tcb-js-sdk] 获取实时数据推送登录票据失败: ${res.code}`);\n    }\n\n    if (res.data) {\n      const { signStr, wsUrl, secretVersion, envId } = res.data;\n      return {\n        signStr,\n        wsUrl,\n        secretVersion,\n        envId,\n        expiredTs,\n      };\n    }\n    throw new Error('[tcb-js-sdk] 获取实时数据推送登录票据失败');\n  };\n\n  private getWaitExpectedTimeoutLength = () => {\n    if (!this._rttObserved.length) {\n      return DEFAULT_EXPECTED_EVENT_WAIT_TIME;\n    }\n\n    // 1.5 * RTT\n    return (\n      (this._rttObserved.reduce((acc, cur) => acc + cur)\n        / this._rttObserved.length)\n      * 1.5\n    );\n  };\n\n  private heartbeat(immediate?: boolean) {\n    this.clearHeartbeat();\n    // @ts-ignore\n    this._pingTimeoutId = setTimeout(\n      async () => {\n        try {\n          if (!this._ws || this._ws.readyState !== WS_READY_STATE.OPEN) {\n            // no need to ping\n            return;\n          }\n\n          this._lastPingSendTS = Date.now();\n          await this.ping();\n          this._pingFailed = 0;\n\n          // @ts-ignore\n          this._pongTimeoutId = setTimeout(() => {\n            console.error('pong timed out');\n            if (this._pongMissed < DEFAULT_PONG_MISS_TOLERANCE) {\n              this._pongMissed++;\n              this.heartbeat(true);\n            } else {\n              // logical perceived connection lost, even though websocket did not receive error or close event\n              this.initWebSocketConnection(true);\n            }\n          }, this._context.appConfig.realtimePongWaitTimeout);\n        } catch (e) {\n          if (this._pingFailed < DEFAULT_PING_FAIL_TOLERANCE) {\n            this._pingFailed++;\n            this.heartbeat();\n          } else {\n            this.close(CLOSE_EVENT_CODE.HeartbeatPingError);\n          }\n        }\n      },\n      immediate ? 0 : this._context.appConfig.realtimePingInterval\n    );\n  }\n\n  private ping = async () => {\n    const msg: IRequestMessagePingMsg = {\n      watchId: undefined,\n      requestId: genRequestId(),\n      msgType: 'PING',\n      msgData: null,\n    };\n    await this.send({\n      msg,\n    });\n    // console.log('ping sent')\n  };\n\n  private onWatchStart = (client: VirtualWebSocketClient, queryID: string) => {\n    this._queryIdClientMap.set(queryID, client);\n  };\n\n  private onWatchClose = (client: VirtualWebSocketClient, queryID: string) => {\n    if (queryID) {\n      this._queryIdClientMap.delete(queryID);\n    }\n    this._watchIdClientMap.delete(client.watchId);\n    this._virtualWSClient.delete(client);\n\n    if (!this._virtualWSClient.size) {\n      // no more existing watch, we should release the websocket connection\n      this.close(CLOSE_EVENT_CODE.NoRealtimeListeners);\n    }\n  };\n}\n"]}
|