@cloudbase/realtime 1.2.3-alpha.0 → 1.8.2

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.
@@ -24,7 +24,7 @@ var __generator = (this && this.__generator) || function (thisArg, body) {
24
24
  function verb(n) { return function (v) { return step([n, v]); }; }
25
25
  function step(op) {
26
26
  if (f) throw new TypeError("Generator is already executing.");
27
- while (_) try {
27
+ while (g && (g = 0, op[0] && (_ = 0)), _) try {
28
28
  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;
29
29
  if (y = 0, t) op = [op[0] & 2, t.value];
30
30
  switch (op[0]) {
@@ -47,7 +47,7 @@ var __generator = (this && this.__generator) || function (thisArg, body) {
47
47
  };
48
48
  import { VirtualWebSocketClient } from './virtual-websocket-client';
49
49
  import { genRequestId } from './message';
50
- import { CLOSE_EVENT_CODE, CLOSE_EVENT_CODE_INFO, getWSCloseError } from './ws-event';
50
+ import { CLOSE_EVENT_CODE, CLOSE_EVENT_CODE_INFO, getWSCloseError, } from './ws-event';
51
51
  import { ERR_CODE, TimeoutError, RealtimeErrorMessageError, CloudSDKError } from './error';
52
52
  import { getWsClass, getRuntime } from './common';
53
53
  import { sleep } from './utils';
@@ -55,7 +55,7 @@ var WS_READY_STATE = {
55
55
  CONNECTING: 0,
56
56
  OPEN: 1,
57
57
  CLOSING: 2,
58
- CLOSED: 3
58
+ CLOSED: 3,
59
59
  };
60
60
  var MAX_RTT_OBSERVED = 3;
61
61
  var DEFAULT_EXPECTED_EVENT_WAIT_TIME = 5000;
@@ -130,14 +130,14 @@ var RealtimeWebSocketClient = (function () {
130
130
  return [2];
131
131
  }
132
132
  if (this._ws.readyState !== WS_READY_STATE.OPEN) {
133
- reject(new Error("ws readyState invalid: " + this._ws.readyState + ", can not send message"));
133
+ reject(new Error("ws readyState invalid: ".concat(this._ws.readyState, ", can not send message")));
134
134
  return [2];
135
135
  }
136
136
  if (opts.waitResponse) {
137
137
  this._wsResponseWait.set(opts.msg.requestId, {
138
138
  resolve: resolve,
139
139
  reject: reject,
140
- skipOnMessage: opts.skipOnMessage
140
+ skipOnMessage: opts.skipOnMessage,
141
141
  });
142
142
  }
143
143
  _a.label = 4;
@@ -147,7 +147,7 @@ var RealtimeWebSocketClient = (function () {
147
147
  case 5:
148
148
  _a.sent();
149
149
  if (!opts.waitResponse) {
150
- resolve();
150
+ resolve(undefined);
151
151
  }
152
152
  return [3, 7];
153
153
  case 6:
@@ -176,13 +176,11 @@ var RealtimeWebSocketClient = (function () {
176
176
  });
177
177
  };
178
178
  this.pauseClients = function (clients) {
179
- ;
180
179
  (clients || _this._virtualWSClient).forEach(function (client) {
181
180
  client.pause();
182
181
  });
183
182
  };
184
183
  this.resumeClients = function (clients) {
185
- ;
186
184
  (clients || _this._virtualWSClient).forEach(function (client) {
187
185
  client.resume();
188
186
  });
@@ -222,7 +220,7 @@ var RealtimeWebSocketClient = (function () {
222
220
  var url = wsSign_1.wsUrl || 'wss://tcb-ws.tencentcloudapi.com';
223
221
  var wsClass = getWsClass();
224
222
  _this._ws = wsClass ? new wsClass(url) : new WebSocket(url);
225
- success();
223
+ success(undefined);
226
224
  })];
227
225
  case 2:
228
226
  _a.sent();
@@ -262,7 +260,7 @@ var RealtimeWebSocketClient = (function () {
262
260
  if (reconnect) {
263
261
  this.closeAllClients(new CloudSDKError({
264
262
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_RECONNECT_WATCH_FAIL,
265
- errMsg: e_3
263
+ errMsg: e_3,
266
264
  }));
267
265
  }
268
266
  _a.label = 10;
@@ -298,158 +296,152 @@ var RealtimeWebSocketClient = (function () {
298
296
  });
299
297
  });
300
298
  };
301
- this.initWebSocketEvent = function () {
302
- return new Promise(function (resolve, reject) {
303
- if (!_this._ws) {
304
- throw new Error('can not initWebSocketEvent, ws not exists');
299
+ this.initWebSocketEvent = function () { return new Promise(function (resolve, reject) {
300
+ if (!_this._ws) {
301
+ throw new Error('can not initWebSocketEvent, ws not exists');
302
+ }
303
+ var wsOpened = false;
304
+ _this._ws.onopen = function (event) {
305
+ console.warn('[realtime] ws event: open', event);
306
+ wsOpened = true;
307
+ resolve();
308
+ };
309
+ _this._ws.onerror = function (event) {
310
+ _this._logins = new Map();
311
+ if (!wsOpened) {
312
+ console.error('[realtime] ws open failed with ws event: error', event);
313
+ reject(event);
314
+ }
315
+ else {
316
+ console.error('[realtime] ws event: error', event);
317
+ _this.clearHeartbeat();
318
+ _this._virtualWSClient.forEach(function (client) { return client.closeWithError(new CloudSDKError({
319
+ errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_WEBSOCKET_CONNECTION_ERROR,
320
+ errMsg: event,
321
+ })); });
305
322
  }
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);
323
+ };
324
+ _this._ws.onclose = function (closeEvent) {
325
+ console.warn('[realtime] ws event: close', closeEvent);
326
+ _this._logins = new Map();
327
+ _this.clearHeartbeat();
328
+ switch (closeEvent.code) {
329
+ case CLOSE_EVENT_CODE.ReconnectWebSocket: {
330
+ break;
317
331
  }
318
- else {
319
- console.error('[realtime] ws event: error', event);
320
- _this.clearHeartbeat();
321
- _this._virtualWSClient.forEach(function (client) {
322
- return client.closeWithError(new CloudSDKError({
323
- errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_WEBSOCKET_CONNECTION_ERROR,
324
- errMsg: event
325
- }));
326
- });
332
+ case CLOSE_EVENT_CODE.NoRealtimeListeners: {
333
+ break;
327
334
  }
328
- };
329
- _this._ws.onclose = function (closeEvent) {
330
- console.warn('[realtime] ws event: close', closeEvent);
331
- _this._logins = new Map();
332
- _this.clearHeartbeat();
333
- switch (closeEvent.code) {
334
- case CLOSE_EVENT_CODE.ReconnectWebSocket: {
335
- break;
335
+ case CLOSE_EVENT_CODE.HeartbeatPingError:
336
+ case CLOSE_EVENT_CODE.HeartbeatPongTimeoutError:
337
+ case CLOSE_EVENT_CODE.NormalClosure:
338
+ case CLOSE_EVENT_CODE.AbnormalClosure: {
339
+ if (_this._maxReconnect > 0) {
340
+ _this.initWebSocketConnection(true, _this._maxReconnect);
336
341
  }
337
- case CLOSE_EVENT_CODE.NoRealtimeListeners: {
338
- break;
342
+ else {
343
+ _this.closeAllClients(getWSCloseError(closeEvent.code));
339
344
  }
340
- case CLOSE_EVENT_CODE.HeartbeatPingError:
341
- case CLOSE_EVENT_CODE.HeartbeatPongTimeoutError:
342
- case CLOSE_EVENT_CODE.NormalClosure:
343
- case CLOSE_EVENT_CODE.AbnormalClosure: {
344
- if (_this._maxReconnect > 0) {
345
- _this.initWebSocketConnection(true, _this._maxReconnect);
346
- }
347
- else {
348
- _this.closeAllClients(getWSCloseError(closeEvent.code));
349
- }
350
- break;
345
+ break;
346
+ }
347
+ case CLOSE_EVENT_CODE.NoAuthentication: {
348
+ _this.closeAllClients(getWSCloseError(closeEvent.code, closeEvent.reason));
349
+ break;
350
+ }
351
+ default: {
352
+ if (_this._maxReconnect > 0) {
353
+ _this.initWebSocketConnection(true, _this._maxReconnect);
351
354
  }
352
- case CLOSE_EVENT_CODE.NoAuthentication: {
353
- _this.closeAllClients(getWSCloseError(closeEvent.code, closeEvent.reason));
354
- break;
355
+ else {
356
+ _this.closeAllClients(getWSCloseError(closeEvent.code));
355
357
  }
356
- default: {
357
- if (_this._maxReconnect > 0) {
358
- _this.initWebSocketConnection(true, _this._maxReconnect);
359
- }
360
- else {
361
- _this.closeAllClients(getWSCloseError(closeEvent.code));
362
- }
358
+ }
359
+ }
360
+ };
361
+ _this._ws.onmessage = function (res) {
362
+ var rawMsg = res.data;
363
+ _this.heartbeat();
364
+ var msg;
365
+ try {
366
+ msg = JSON.parse(rawMsg);
367
+ }
368
+ catch (e) {
369
+ throw new Error("[realtime] onMessage parse res.data error: ".concat(e));
370
+ }
371
+ if (msg.msgType === 'ERROR') {
372
+ var virtualWatch_1 = null;
373
+ _this._virtualWSClient.forEach(function (item) {
374
+ if (item.watchId === msg.watchId) {
375
+ virtualWatch_1 = item;
363
376
  }
377
+ });
378
+ if (virtualWatch_1) {
379
+ virtualWatch_1.listener.onError(msg);
364
380
  }
365
- };
366
- _this._ws.onmessage = function (res) {
367
- var rawMsg = res.data;
368
- _this.heartbeat();
369
- var msg;
381
+ }
382
+ var responseWaitSpec = _this._wsResponseWait.get(msg.requestId);
383
+ if (responseWaitSpec) {
370
384
  try {
371
- msg = JSON.parse(rawMsg);
385
+ if (msg.msgType === 'ERROR') {
386
+ responseWaitSpec.reject(new RealtimeErrorMessageError(msg));
387
+ }
388
+ else {
389
+ responseWaitSpec.resolve(msg);
390
+ }
372
391
  }
373
392
  catch (e) {
374
- throw new Error("[realtime] onMessage parse res.data error: " + e);
393
+ console.error('ws onMessage responseWaitSpec.resolve(msg) errored:', e);
375
394
  }
376
- if (msg.msgType === 'ERROR') {
377
- var virtualWatch_1 = null;
378
- _this._virtualWSClient.forEach(function (item) {
379
- if (item.watchId === msg.watchId) {
380
- virtualWatch_1 = item;
381
- }
382
- });
383
- if (virtualWatch_1) {
384
- virtualWatch_1.listener.onError(msg);
385
- }
395
+ finally {
396
+ _this._wsResponseWait.delete(msg.requestId);
386
397
  }
387
- var responseWaitSpec = _this._wsResponseWait.get(msg.requestId);
388
- if (responseWaitSpec) {
389
- try {
390
- if (msg.msgType === 'ERROR') {
391
- responseWaitSpec.reject(new RealtimeErrorMessageError(msg));
392
- }
393
- else {
394
- responseWaitSpec.resolve(msg);
395
- }
396
- }
397
- catch (e) {
398
- console.error('ws onMessage responseWaitSpec.resolve(msg) errored:', e);
399
- }
400
- finally {
401
- _this._wsResponseWait.delete(msg.requestId);
402
- }
403
- if (responseWaitSpec.skipOnMessage) {
398
+ if (responseWaitSpec.skipOnMessage) {
399
+ return;
400
+ }
401
+ }
402
+ if (msg.msgType === 'PONG') {
403
+ if (_this._lastPingSendTS) {
404
+ var rtt = Date.now() - _this._lastPingSendTS;
405
+ if (rtt > DEFAULT_UNTRUSTED_RTT_THRESHOLD) {
406
+ console.warn("[realtime] untrusted rtt observed: ".concat(rtt));
404
407
  return;
405
408
  }
409
+ if (_this._rttObserved.length >= MAX_RTT_OBSERVED) {
410
+ _this._rttObserved.splice(0, _this._rttObserved.length - MAX_RTT_OBSERVED + 1);
411
+ }
412
+ _this._rttObserved.push(rtt);
406
413
  }
407
- if (msg.msgType === 'PONG') {
408
- if (_this._lastPingSendTS) {
409
- var rtt = Date.now() - _this._lastPingSendTS;
410
- if (rtt > DEFAULT_UNTRUSTED_RTT_THRESHOLD) {
411
- console.warn("[realtime] untrusted rtt observed: " + rtt);
412
- return;
413
- }
414
- if (_this._rttObserved.length >= MAX_RTT_OBSERVED) {
415
- _this._rttObserved.splice(0, _this._rttObserved.length - MAX_RTT_OBSERVED + 1);
414
+ return;
415
+ }
416
+ var client = msg.watchId && _this._watchIdClientMap.get(msg.watchId);
417
+ if (client) {
418
+ client.onMessage(msg);
419
+ }
420
+ else {
421
+ console.error("[realtime] no realtime listener found responsible for watchId ".concat(msg.watchId, ": "), msg);
422
+ switch (msg.msgType) {
423
+ case 'INIT_EVENT':
424
+ case 'NEXT_EVENT':
425
+ case 'CHECK_EVENT': {
426
+ client = _this._queryIdClientMap.get(msg.msgData.queryID);
427
+ if (client) {
428
+ client.onMessage(msg);
416
429
  }
417
- _this._rttObserved.push(rtt);
430
+ break;
418
431
  }
419
- return;
420
- }
421
- var client = msg.watchId && _this._watchIdClientMap.get(msg.watchId);
422
- if (client) {
423
- client.onMessage(msg);
424
- }
425
- else {
426
- console.error("[realtime] no realtime listener found responsible for watchId " + msg.watchId + ": ", msg);
427
- switch (msg.msgType) {
428
- case 'INIT_EVENT':
429
- case 'NEXT_EVENT':
430
- case 'CHECK_EVENT': {
431
- client = _this._queryIdClientMap.get(msg.msgData.queryID);
432
- if (client) {
433
- client.onMessage(msg);
434
- }
432
+ default: {
433
+ for (var _i = 0, _a = Array.from(_this._watchIdClientMap.entries()); _i < _a.length; _i++) {
434
+ var _b = _a[_i], client_1 = _b[1];
435
+ client_1.onMessage(msg);
435
436
  break;
436
437
  }
437
- default: {
438
- for (var _i = 0, _a = Array.from(_this._watchIdClientMap.entries()); _i < _a.length; _i++) {
439
- var _b = _a[_i], client_1 = _b[1];
440
- client_1.onMessage(msg);
441
- break;
442
- }
443
- }
444
438
  }
445
439
  }
446
- };
447
- _this.heartbeat();
448
- });
449
- };
450
- this.isWSConnected = function () {
451
- return Boolean(_this._ws && _this._ws.readyState === WS_READY_STATE.OPEN);
452
- };
440
+ }
441
+ };
442
+ _this.heartbeat();
443
+ }); };
444
+ this.isWSConnected = function () { return Boolean(_this._ws && _this._ws.readyState === WS_READY_STATE.OPEN); };
453
445
  this.onceWSConnected = function () { return __awaiter(_this, void 0, void 0, function () {
454
446
  var _this = this;
455
447
  return __generator(this, function (_a) {
@@ -462,7 +454,7 @@ var RealtimeWebSocketClient = (function () {
462
454
  return [2, new Promise(function (resolve, reject) {
463
455
  _this._wsReadySubsribers.push({
464
456
  resolve: resolve,
465
- reject: reject
457
+ reject: reject,
466
458
  });
467
459
  })];
468
460
  });
@@ -480,7 +472,7 @@ var RealtimeWebSocketClient = (function () {
480
472
  if (loginInfo_1.loggedIn && loginInfo_1.loginResult) {
481
473
  return [2, loginInfo_1.loginResult];
482
474
  }
483
- else if (loginInfo_1.loggingInPromise) {
475
+ if (loginInfo_1.loggingInPromise) {
484
476
  return [2, loginInfo_1.loggingInPromise];
485
477
  }
486
478
  }
@@ -506,7 +498,7 @@ var RealtimeWebSocketClient = (function () {
506
498
  accessToken: '',
507
499
  referrer: 'web',
508
500
  sdkVersion: '',
509
- dataVersion: ''
501
+ dataVersion: '',
510
502
  };
511
503
  loginMsg = {
512
504
  watchId: undefined,
@@ -516,24 +508,24 @@ var RealtimeWebSocketClient = (function () {
516
508
  exMsgData: {
517
509
  runtime: getRuntime(),
518
510
  signStr: wsSign.signStr,
519
- secretVersion: wsSign.secretVersion
520
- }
511
+ secretVersion: wsSign.secretVersion,
512
+ },
521
513
  };
522
514
  return [4, this.send({
523
515
  msg: loginMsg,
524
516
  waitResponse: true,
525
517
  skipOnMessage: true,
526
- timeout: DEFAULT_LOGIN_TIMEOUT
518
+ timeout: DEFAULT_LOGIN_TIMEOUT,
527
519
  })];
528
520
  case 2:
529
521
  loginResMsg = _a.sent();
530
522
  if (!loginResMsg.msgData.code) {
531
523
  resolve({
532
- envId: wsSign.envId
524
+ envId: wsSign.envId,
533
525
  });
534
526
  }
535
527
  else {
536
- reject(new Error(loginResMsg.msgData.code + " " + loginResMsg.msgData.message));
528
+ reject(new Error("".concat(loginResMsg.msgData.code, " ").concat(loginResMsg.msgData.message)));
537
529
  }
538
530
  return [3, 4];
539
531
  case 3:
@@ -555,7 +547,7 @@ var RealtimeWebSocketClient = (function () {
555
547
  loginInfo = {
556
548
  loggedIn: false,
557
549
  loggingInPromise: promise,
558
- loginStartTS: loginStartTS
550
+ loginStartTS: loginStartTS,
559
551
  };
560
552
  this._logins.set(envId || '', loginInfo);
561
553
  }
@@ -566,25 +558,23 @@ var RealtimeWebSocketClient = (function () {
566
558
  case 2:
567
559
  loginResult = _a.sent();
568
560
  curLoginInfo = envId && this._logins.get(envId);
569
- if (curLoginInfo &&
570
- curLoginInfo === loginInfo &&
571
- curLoginInfo.loginStartTS === loginStartTS) {
561
+ if (curLoginInfo
562
+ && curLoginInfo === loginInfo
563
+ && curLoginInfo.loginStartTS === loginStartTS) {
572
564
  loginInfo.loggedIn = true;
573
565
  loginInfo.loggingInPromise = undefined;
574
566
  loginInfo.loginStartTS = undefined;
575
567
  loginInfo.loginResult = loginResult;
576
568
  return [2, loginResult];
577
569
  }
578
- else if (curLoginInfo) {
570
+ if (curLoginInfo) {
579
571
  if (curLoginInfo.loggedIn && curLoginInfo.loginResult) {
580
572
  return [2, curLoginInfo.loginResult];
581
573
  }
582
- else if (curLoginInfo.loggingInPromise) {
574
+ if (curLoginInfo.loggingInPromise) {
583
575
  return [2, curLoginInfo.loggingInPromise];
584
576
  }
585
- else {
586
- throw new Error('ws unexpected login info');
587
- }
577
+ throw new Error('ws unexpected login info');
588
578
  }
589
579
  else {
590
580
  throw new Error('ws login info reset');
@@ -614,7 +604,7 @@ var RealtimeWebSocketClient = (function () {
614
604
  case 1:
615
605
  res = _b.sent();
616
606
  if (res.code) {
617
- throw new Error("[tcb-js-sdk] \u83B7\u53D6\u5B9E\u65F6\u6570\u636E\u63A8\u9001\u767B\u5F55\u7968\u636E\u5931\u8D25: " + res.code);
607
+ throw new Error("[tcb-js-sdk] \u83B7\u53D6\u5B9E\u65F6\u6570\u636E\u63A8\u9001\u767B\u5F55\u7968\u636E\u5931\u8D25: ".concat(res.code));
618
608
  }
619
609
  if (res.data) {
620
610
  _a = res.data, signStr = _a.signStr, wsUrl = _a.wsUrl, secretVersion = _a.secretVersion, envId = _a.envId;
@@ -623,13 +613,10 @@ var RealtimeWebSocketClient = (function () {
623
613
  wsUrl: wsUrl,
624
614
  secretVersion: secretVersion,
625
615
  envId: envId,
626
- expiredTs: expiredTs
616
+ expiredTs: expiredTs,
627
617
  }];
628
618
  }
629
- else {
630
- throw new Error('[tcb-js-sdk] 获取实时数据推送登录票据失败');
631
- }
632
- return [2];
619
+ throw new Error('[tcb-js-sdk] 获取实时数据推送登录票据失败');
633
620
  }
634
621
  });
635
622
  }); };
@@ -637,9 +624,9 @@ var RealtimeWebSocketClient = (function () {
637
624
  if (!_this._rttObserved.length) {
638
625
  return DEFAULT_EXPECTED_EVENT_WAIT_TIME;
639
626
  }
640
- return ((_this._rttObserved.reduce(function (acc, cur) { return acc + cur; }) /
641
- _this._rttObserved.length) *
642
- 1.5);
627
+ return ((_this._rttObserved.reduce(function (acc, cur) { return acc + cur; })
628
+ / _this._rttObserved.length)
629
+ * 1.5);
643
630
  };
644
631
  this.ping = function () { return __awaiter(_this, void 0, void 0, function () {
645
632
  var msg;
@@ -650,10 +637,10 @@ var RealtimeWebSocketClient = (function () {
650
637
  watchId: undefined,
651
638
  requestId: genRequestId(),
652
639
  msgType: 'PING',
653
- msgData: null
640
+ msgData: null,
654
641
  };
655
642
  return [4, this.send({
656
- msg: msg
643
+ msg: msg,
657
644
  })];
658
645
  case 1:
659
646
  _a.sent();
@@ -675,8 +662,7 @@ var RealtimeWebSocketClient = (function () {
675
662
  }
676
663
  };
677
664
  this._maxReconnect = options.maxReconnect || DEFAULT_MAX_RECONNECT;
678
- this._reconnectInterval =
679
- options.reconnectInterval || DEFAULT_WS_RECONNECT_INTERVAL;
665
+ this._reconnectInterval = options.reconnectInterval || DEFAULT_WS_RECONNECT_INTERVAL;
680
666
  this._context = options.context;
681
667
  }
682
668
  RealtimeWebSocketClient.prototype.clearHeartbeat = function () {
@@ -746,4 +732,4 @@ var RealtimeWebSocketClient = (function () {
746
732
  return RealtimeWebSocketClient;
747
733
  }());
748
734
  export { RealtimeWebSocketClient };
749
- //# 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"]}
735
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-client.js","sourceRoot":"","sources":["../../src/websocket-client.ts"],"names":[],"mappings":"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","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"]}