@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.
@@ -13,7 +13,7 @@ var __generator = (this && this.__generator) || function (thisArg, body) {
13
13
  function verb(n) { return function (v) { return step([n, v]); }; }
14
14
  function step(op) {
15
15
  if (f) throw new TypeError("Generator is already executing.");
16
- while (_) try {
16
+ while (g && (g = 0, op[0] && (_ = 0)), _) try {
17
17
  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;
18
18
  if (y = 0, t) op = [op[0] & 2, t.value];
19
19
  switch (op[0]) {
@@ -34,12 +34,14 @@ var __generator = (this && this.__generator) || function (thisArg, body) {
34
34
  if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
35
35
  }
36
36
  };
37
- var __spreadArrays = (this && this.__spreadArrays) || function () {
38
- for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;
39
- for (var r = Array(s), k = 0, i = 0; i < il; i++)
40
- for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)
41
- r[k] = a[j];
42
- return r;
37
+ var __spreadArray = (this && this.__spreadArray) || function (to, from, pack) {
38
+ if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {
39
+ if (ar || !(i in from)) {
40
+ if (!ar) ar = Array.prototype.slice.call(from, 0, i);
41
+ ar[i] = from[i];
42
+ }
43
+ }
44
+ return to.concat(ar || Array.prototype.slice.call(from));
43
45
  };
44
46
  import set from 'lodash.set';
45
47
  import unset from 'lodash.unset';
@@ -47,7 +49,7 @@ import cloneDeep from 'lodash.clonedeep';
47
49
  import { genRequestId } from './message';
48
50
  import { RealtimeListener } from './listener';
49
51
  import { Snapshot } from './snapshot';
50
- import { ERR_CODE, CloudSDKError, isTimeoutError, CancelledError, isCancelledError, isRealtimeErrorMessageError } from './error';
52
+ import { ERR_CODE, CloudSDKError, isTimeoutError, CancelledError, isCancelledError, isRealtimeErrorMessageError, } from './error';
51
53
  import { sleep } from './utils';
52
54
  var WATCH_STATUS;
53
55
  (function (WATCH_STATUS) {
@@ -123,14 +125,14 @@ var VirtualWebSocketClient = (function () {
123
125
  collName: this.collectionName,
124
126
  query: this.query,
125
127
  limit: this.limit,
126
- orderBy: this.orderBy
127
- }
128
+ orderBy: this.orderBy,
129
+ },
128
130
  };
129
131
  return [4, this.send({
130
132
  msg: initWatchMsg,
131
133
  waitResponse: true,
132
134
  skipOnMessage: true,
133
- timeout: DEFAULT_INIT_WATCH_TIMEOUT
135
+ timeout: DEFAULT_INIT_WATCH_TIMEOUT,
134
136
  })];
135
137
  case 2:
136
138
  initEventMsg = _b.sent();
@@ -138,7 +140,7 @@ var VirtualWebSocketClient = (function () {
138
140
  this.sessionInfo = {
139
141
  queryID: initEventMsg.msgData.queryID,
140
142
  currentEventId: currEvent - 1,
141
- currentDocs: []
143
+ currentDocs: [],
142
144
  };
143
145
  if (events.length > 0) {
144
146
  for (_i = 0, events_1 = events; _i < events_1.length; _i++) {
@@ -153,7 +155,7 @@ var VirtualWebSocketClient = (function () {
153
155
  id: currEvent,
154
156
  docChanges: [],
155
157
  docs: [],
156
- type: 'init'
158
+ type: 'init',
157
159
  });
158
160
  this.listener.onChange(snapshot);
159
161
  this.scheduleSendACK();
@@ -168,7 +170,7 @@ var VirtualWebSocketClient = (function () {
168
170
  this.handleWatchEstablishmentError(e_1, {
169
171
  operationName: 'INIT_WATCH',
170
172
  resolve: resolve,
171
- reject: reject
173
+ reject: reject,
172
174
  });
173
175
  return [3, 4];
174
176
  case 4: return [2];
@@ -188,7 +190,7 @@ var VirtualWebSocketClient = (function () {
188
190
  this._initWatchPromise = undefined;
189
191
  return [7];
190
192
  case 4:
191
- console.log("[realtime] initWatch " + (success ? 'success' : 'fail'));
193
+ console.log("[realtime] initWatch ".concat(success ? 'success' : 'fail'));
192
194
  return [2];
193
195
  }
194
196
  });
@@ -231,14 +233,14 @@ var VirtualWebSocketClient = (function () {
231
233
  envId: envId,
232
234
  collName: this.collectionName,
233
235
  queryID: this.sessionInfo.queryID,
234
- eventID: this.sessionInfo.currentEventId
235
- }
236
+ eventID: this.sessionInfo.currentEventId,
237
+ },
236
238
  };
237
239
  return [4, this.send({
238
240
  msg: rebuildWatchMsg,
239
241
  waitResponse: true,
240
242
  skipOnMessage: false,
241
- timeout: DEFAULT_REBUILD_WATCH_TIMEOUT
243
+ timeout: DEFAULT_REBUILD_WATCH_TIMEOUT,
242
244
  })];
243
245
  case 2:
244
246
  nextEventMsg = _a.sent();
@@ -252,7 +254,7 @@ var VirtualWebSocketClient = (function () {
252
254
  this.handleWatchEstablishmentError(e_2, {
253
255
  operationName: 'REBUILD_WATCH',
254
256
  resolve: resolve,
255
- reject: reject
257
+ reject: reject,
256
258
  });
257
259
  return [3, 4];
258
260
  case 4: return [2];
@@ -272,7 +274,7 @@ var VirtualWebSocketClient = (function () {
272
274
  this._rebuildWatchPromise = undefined;
273
275
  return [7];
274
276
  case 4:
275
- console.log("[realtime] rebuildWatch " + (success ? 'success' : 'fail'));
277
+ console.log("[realtime] rebuildWatch ".concat(success ? 'success' : 'fail'));
276
278
  return [2];
277
279
  }
278
280
  });
@@ -287,7 +289,7 @@ var VirtualWebSocketClient = (function () {
287
289
  errCode: isInitWatch
288
290
  ? ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_INIT_WATCH_FAIL
289
291
  : ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_REBUILD_WATCH_FAIL,
290
- errMsg: e
292
+ errMsg: e,
291
293
  }));
292
294
  options.reject(e);
293
295
  };
@@ -340,7 +342,7 @@ var VirtualWebSocketClient = (function () {
340
342
  case 3:
341
343
  _a.sent();
342
344
  if (!(this.watchStatus === WATCH_STATUS.PAUSED)) return [3, 4];
343
- options.reject(new CancelledError(options.operationName + " cancelled due to pause after unknownError"));
345
+ options.reject(new CancelledError("".concat(options.operationName, " cancelled due to pause after unknownError")));
344
346
  return [3, 7];
345
347
  case 4:
346
348
  if (!!this.isWSConnected()) return [3, 6];
@@ -359,7 +361,7 @@ var VirtualWebSocketClient = (function () {
359
361
  case 9: return [2];
360
362
  }
361
363
  });
362
- }); }
364
+ }); },
363
365
  });
364
366
  return [2];
365
367
  });
@@ -383,10 +385,10 @@ var VirtualWebSocketClient = (function () {
383
385
  watchId: this.watchId,
384
386
  requestId: genRequestId(),
385
387
  msgType: 'CLOSE_WATCH',
386
- msgData: null
388
+ msgData: null,
387
389
  };
388
390
  return [4, this.send({
389
- msg: closeWatchMsg
391
+ msg: closeWatchMsg,
390
392
  })];
391
393
  case 2:
392
394
  _a.sent();
@@ -397,7 +399,7 @@ var VirtualWebSocketClient = (function () {
397
399
  e_4 = _a.sent();
398
400
  this.closeWithError(new CloudSDKError({
399
401
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CLOSE_WATCH_FAIL,
400
- errMsg: e_4
402
+ errMsg: e_4,
401
403
  }));
402
404
  return [3, 5];
403
405
  case 4:
@@ -443,11 +445,11 @@ var VirtualWebSocketClient = (function () {
443
445
  msgType: 'CHECK_LAST',
444
446
  msgData: {
445
447
  queryID: this.sessionInfo.queryID,
446
- eventID: this.sessionInfo.currentEventId
447
- }
448
+ eventID: this.sessionInfo.currentEventId,
449
+ },
448
450
  };
449
451
  return [4, this.send({
450
- msg: ackMsg
452
+ msg: ackMsg,
451
453
  })];
452
454
  case 1:
453
455
  _a.sent();
@@ -471,7 +473,7 @@ var VirtualWebSocketClient = (function () {
471
473
  case 'COLLECTION_PERMISSION_DENIED': {
472
474
  this.closeWithError(new CloudSDKError({
473
475
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL,
474
- errMsg: msg.msgData.code
476
+ errMsg: msg.msgData.code,
475
477
  }));
476
478
  return [2];
477
479
  }
@@ -480,15 +482,15 @@ var VirtualWebSocketClient = (function () {
480
482
  }
481
483
  }
482
484
  }
483
- if (this._availableRetries.CHECK_LAST &&
484
- this._availableRetries.CHECK_LAST > 0) {
485
+ if (this._availableRetries.CHECK_LAST
486
+ && this._availableRetries.CHECK_LAST > 0) {
485
487
  this._availableRetries.CHECK_LAST--;
486
488
  this.scheduleSendACK();
487
489
  }
488
490
  else {
489
491
  this.closeWithError(new CloudSDKError({
490
492
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL,
491
- errMsg: e_5
493
+ errMsg: e_5,
492
494
  }));
493
495
  }
494
496
  return [3, 3];
@@ -530,7 +532,7 @@ var VirtualWebSocketClient = (function () {
530
532
  }
531
533
  options.onUnknownError(e);
532
534
  };
533
- this.watchId = "watchid_" + +new Date() + "_" + Math.random();
535
+ this.watchId = "watchid_".concat(+new Date(), "_").concat(Math.random());
534
536
  this.envId = options.envId;
535
537
  this.collectionName = options.collectionName;
536
538
  this.query = options.query;
@@ -547,14 +549,14 @@ var VirtualWebSocketClient = (function () {
547
549
  this._availableRetries = {
548
550
  INIT_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,
549
551
  REBUILD_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,
550
- CHECK_LAST: DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR
552
+ CHECK_LAST: DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR,
551
553
  };
552
554
  this.listener = new RealtimeListener({
553
555
  close: this.closeWatch,
554
556
  onChange: options.onChange,
555
557
  onError: options.onError,
556
558
  debug: this.debug,
557
- virtualClient: this
559
+ virtualClient: this,
558
560
  });
559
561
  this.initWatch();
560
562
  }
@@ -570,7 +572,7 @@ var VirtualWebSocketClient = (function () {
570
572
  case WATCH_STATUS.LOGGINGIN:
571
573
  case WATCH_STATUS.INITING:
572
574
  case WATCH_STATUS.REBUILDING: {
573
- console.warn("[realtime listener] internal non-fatal error: unexpected message received while " + this.watchStatus);
575
+ console.warn("[realtime listener] internal non-fatal error: unexpected message received while ".concat(this.watchStatus));
574
576
  return;
575
577
  }
576
578
  case WATCH_STATUS.CLOSED: {
@@ -589,7 +591,7 @@ var VirtualWebSocketClient = (function () {
589
591
  this.scheduleSendACK();
590
592
  switch (msg.msgType) {
591
593
  case 'NEXT_EVENT': {
592
- console.warn("nextevent " + msg.msgData.currEvent + " ignored", msg);
594
+ console.warn("nextevent ".concat(msg.msgData.currEvent, " ignored"), msg);
593
595
  this.handleServerEvents(msg);
594
596
  break;
595
597
  }
@@ -600,33 +602,34 @@ var VirtualWebSocketClient = (function () {
600
602
  this._waitExpectedTimeoutId = setTimeout(function () {
601
603
  _this.rebuildWatch();
602
604
  }, this.getWaitExpectedTimeoutLength());
603
- console.log("[realtime] waitExpectedTimeoutLength " + this.getWaitExpectedTimeoutLength());
605
+ console.log("[realtime] waitExpectedTimeoutLength ".concat(this.getWaitExpectedTimeoutLength()));
604
606
  }
605
607
  break;
606
608
  }
607
609
  case 'ERROR': {
608
610
  this.closeWithError(new CloudSDKError({
609
611
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_SERVER_ERROR_MSG,
610
- errMsg: msg.msgData.code + " - " + msg.msgData.message
612
+ errMsg: "".concat(msg.msgData.code, " - ").concat(msg.msgData.message),
611
613
  }));
612
614
  break;
613
615
  }
614
616
  default: {
615
- console.warn("[realtime listener] virtual client receive unexpected msg " + msg.msgType + ": ", msg);
617
+ console.warn("[realtime listener] virtual client receive unexpected msg ".concat(msg.msgType, ": "), msg);
616
618
  break;
617
619
  }
618
620
  }
619
621
  };
620
622
  VirtualWebSocketClient.prototype.closeWithError = function (error) {
623
+ var _a;
621
624
  this.watchStatus = WATCH_STATUS.ERRORED;
622
625
  this.clearACKSchedule();
623
626
  this.listener.onError(error);
624
- this.onWatchClose(this, (this.sessionInfo && this.sessionInfo.queryID) || '');
625
- console.log("[realtime] client closed (" + this.collectionName + " " + this.query + ") (watchId " + this.watchId + ")");
627
+ this.onWatchClose(this, ((_a = this.sessionInfo) === null || _a === void 0 ? void 0 : _a.queryID) || '');
628
+ console.log("[realtime] client closed (".concat(this.collectionName, " ").concat(this.query, ") (watchId ").concat(this.watchId, ")"));
626
629
  };
627
630
  VirtualWebSocketClient.prototype.pause = function () {
628
631
  this.watchStatus = WATCH_STATUS.PAUSED;
629
- console.log("[realtime] client paused (" + this.collectionName + " " + this.query + ") (watchId " + this.watchId + ")");
632
+ console.log("[realtime] client paused (".concat(this.collectionName, " ").concat(this.query, ") (watchId ").concat(this.watchId, ")"));
630
633
  };
631
634
  VirtualWebSocketClient.prototype.resume = function () {
632
635
  return __awaiter(this, void 0, void 0, function () {
@@ -635,18 +638,18 @@ var VirtualWebSocketClient = (function () {
635
638
  switch (_a.label) {
636
639
  case 0:
637
640
  this.watchStatus = WATCH_STATUS.RESUMING;
638
- console.log("[realtime] client resuming with " + (this.sessionInfo ? 'REBUILD_WATCH' : 'INIT_WATCH') + " (" + this.collectionName + " " + this.query + ") (" + this.watchId + ")");
641
+ console.log("[realtime] client resuming with ".concat(this.sessionInfo ? 'REBUILD_WATCH' : 'INIT_WATCH', " (").concat(this.collectionName, " ").concat(this.query, ") (").concat(this.watchId, ")"));
639
642
  _a.label = 1;
640
643
  case 1:
641
644
  _a.trys.push([1, 3, , 4]);
642
645
  return [4, (this.sessionInfo ? this.rebuildWatch() : this.initWatch())];
643
646
  case 2:
644
647
  _a.sent();
645
- console.log("[realtime] client successfully resumed (" + this.collectionName + " " + this.query + ") (" + this.watchId + ")");
648
+ console.log("[realtime] client successfully resumed (".concat(this.collectionName, " ").concat(this.query, ") (").concat(this.watchId, ")"));
646
649
  return [3, 4];
647
650
  case 3:
648
651
  e_6 = _a.sent();
649
- console.error("[realtime] client resume failed (" + this.collectionName + " " + this.query + ") (" + this.watchId + ")", e_6);
652
+ console.error("[realtime] client resume failed (".concat(this.collectionName, " ").concat(this.query, ") (").concat(this.watchId, ")"), e_6);
650
653
  return [3, 4];
651
654
  case 4: return [2];
652
655
  }
@@ -654,10 +657,10 @@ var VirtualWebSocketClient = (function () {
654
657
  });
655
658
  };
656
659
  VirtualWebSocketClient.prototype.useRetryTicket = function (operationName) {
657
- if (this._availableRetries[operationName] &&
658
- this._availableRetries[operationName] > 0) {
660
+ if (this._availableRetries[operationName]
661
+ && this._availableRetries[operationName] > 0) {
659
662
  this._availableRetries[operationName]--;
660
- console.log("[realtime] " + operationName + " use a retry ticket, now only " + this._availableRetries[operationName] + " retry left");
663
+ console.log("[realtime] ".concat(operationName, " use a retry ticket, now only ").concat(this._availableRetries[operationName], " retry left"));
661
664
  return true;
662
665
  }
663
666
  return false;
@@ -703,24 +706,24 @@ var VirtualWebSocketClient = (function () {
703
706
  catch (e) {
704
707
  this.closeWithError(new CloudSDKError({
705
708
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_RECEIVE_INVALID_SERVER_DATA,
706
- errMsg: e
709
+ errMsg: e,
707
710
  }));
708
711
  return [2];
709
712
  }
710
- docs = __spreadArrays(sessionInfo.currentDocs);
713
+ docs = __spreadArray([], sessionInfo.currentDocs, true);
711
714
  initEncountered = false;
712
715
  _loop_1 = function (i, len) {
713
- var change, localDoc, doc, fieldPath, _i, _a, fieldPath, err, err, doc, doc, err, ind, ind, docsSnapshot, docChanges, snapshot;
714
- return __generator(this, function (_b) {
715
- switch (_b.label) {
716
+ var change, localDoc, doc, fieldPath, _i, _b, fieldPath, err, err, doc, doc, err, ind, ind, docsSnapshot, docChanges, snapshot;
717
+ return __generator(this, function (_c) {
718
+ switch (_c.label) {
716
719
  case 0:
717
720
  change = allChangeEvents[i];
718
721
  if (!(sessionInfo.currentEventId >= change.id)) return [3, 1];
719
722
  if (!allChangeEvents[i - 1] || change.id > allChangeEvents[i - 1].id) {
720
- console.warn("[realtime] duplicate event received, cur " + sessionInfo.currentEventId + " but got " + change.id);
723
+ console.warn("[realtime] duplicate event received, cur ".concat(sessionInfo.currentEventId, " but got ").concat(change.id));
721
724
  }
722
725
  else {
723
- console.error("[realtime listener] server non-fatal error: events out of order (the latter event's id is smaller than that of the former) (requestId " + requestId + ")");
726
+ console.error("[realtime listener] server non-fatal error: events out of order (the latter event's id is smaller than that of the former) (requestId ".concat(requestId, ")"));
724
727
  }
725
728
  return [2, "continue"];
726
729
  case 1:
@@ -740,8 +743,8 @@ var VirtualWebSocketClient = (function () {
740
743
  }
741
744
  }
742
745
  if (change.removedFields) {
743
- for (_i = 0, _a = change.removedFields; _i < _a.length; _i++) {
744
- fieldPath = _a[_i];
746
+ for (_i = 0, _b = change.removedFields; _i < _b.length; _i++) {
747
+ fieldPath = _b[_i];
745
748
  unset(doc, fieldPath);
746
749
  }
747
750
  }
@@ -755,7 +758,7 @@ var VirtualWebSocketClient = (function () {
755
758
  case 'enqueue': {
756
759
  err = new CloudSDKError({
757
760
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR,
758
- errMsg: "HandleServerEvents: full doc is not provided with dataType=\"update\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")"
761
+ errMsg: "HandleServerEvents: full doc is not provided with dataType=\"update\" and queueType=\"enqueue\" (requestId ".concat(msg.requestId, ")"),
759
762
  });
760
763
  this_1.closeWithError(err);
761
764
  throw err;
@@ -771,7 +774,7 @@ var VirtualWebSocketClient = (function () {
771
774
  if (!change.doc) {
772
775
  err = new CloudSDKError({
773
776
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR,
774
- errMsg: "HandleServerEvents: full doc is not provided with dataType=\"replace\" (requestId " + msg.requestId + ")"
777
+ errMsg: "HandleServerEvents: full doc is not provided with dataType=\"replace\" (requestId ".concat(msg.requestId, ")"),
775
778
  });
776
779
  this_1.closeWithError(err);
777
780
  throw err;
@@ -804,7 +807,7 @@ var VirtualWebSocketClient = (function () {
804
807
  case 'enqueue': {
805
808
  err = new CloudSDKError({
806
809
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR,
807
- errMsg: "HandleServerEvents: full doc is not provided with dataType=\"limit\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")"
810
+ errMsg: "HandleServerEvents: full doc is not provided with dataType=\"limit\" and queueType=\"enqueue\" (requestId ".concat(msg.requestId, ")"),
808
811
  });
809
812
  this_1.closeWithError(err);
810
813
  throw err;
@@ -853,9 +856,9 @@ var VirtualWebSocketClient = (function () {
853
856
  break;
854
857
  }
855
858
  }
856
- if (i === len - 1 ||
857
- (allChangeEvents[i + 1] && allChangeEvents[i + 1].id !== change.id)) {
858
- docsSnapshot = __spreadArrays(docs);
859
+ if (i === len - 1
860
+ || (allChangeEvents[i + 1] && allChangeEvents[i + 1].id !== change.id)) {
861
+ docsSnapshot = __spreadArray([], docs, true);
859
862
  docChanges = allChangeEvents
860
863
  .slice(0, i + 1)
861
864
  .filter(function (c) { return c.id === change.id; });
@@ -865,16 +868,16 @@ var VirtualWebSocketClient = (function () {
865
868
  id: change.id,
866
869
  docChanges: docChanges,
867
870
  docs: docsSnapshot,
868
- msgType: msgType
871
+ msgType: msgType,
869
872
  });
870
873
  this_1.listener.onChange(snapshot);
871
874
  }
872
875
  return [3, 4];
873
876
  case 2:
874
- console.warn("[realtime listener] event received is out of order, cur " + this_1.sessionInfo.currentEventId + " but got " + change.id);
877
+ console.warn("[realtime listener] event received is out of order, cur ".concat(this_1.sessionInfo.currentEventId, " but got ").concat(change.id));
875
878
  return [4, this_1.rebuildWatch()];
876
879
  case 3:
877
- _b.sent();
880
+ _c.sent();
878
881
  return [2, { value: void 0 }];
879
882
  case 4: return [2];
880
883
  }
@@ -904,8 +907,8 @@ var VirtualWebSocketClient = (function () {
904
907
  console.error('[realtime listener] internal non-fatal error: sessionInfo lost after server event handling, this should never occur');
905
908
  return;
906
909
  }
907
- if (this.sessionInfo.expectEventId &&
908
- this.sessionInfo.currentEventId >= this.sessionInfo.expectEventId) {
910
+ if (this.sessionInfo.expectEventId
911
+ && this.sessionInfo.currentEventId >= this.sessionInfo.expectEventId) {
909
912
  this.clearWaitExpectedEvent();
910
913
  }
911
914
  if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {
@@ -928,7 +931,7 @@ function getPublicEvent(event) {
928
931
  dataType: event.DataType,
929
932
  queueType: event.QueueType,
930
933
  docId: event.DocID,
931
- doc: event.Doc && event.Doc !== '{}' ? JSON.parse(event.Doc) : undefined
934
+ doc: event.Doc && event.Doc !== '{}' ? JSON.parse(event.Doc) : undefined,
932
935
  };
933
936
  if (event.DataType === 'update') {
934
937
  if (event.UpdatedFields) {
@@ -940,4 +943,4 @@ function getPublicEvent(event) {
940
943
  }
941
944
  return e;
942
945
  }
943
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"virtual-websocket-client.js","sourceRoot":"","sources":["../../src/virtual-websocket-client.ts"],"names":[],"mappings":"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","sourcesContent":["import set from 'lodash.set'\nimport unset from 'lodash.unset'\nimport cloneDeep from 'lodash.clonedeep'\nimport { genRequestId } from './message'\nimport {\n  IResponseMessage,\n  IRequestMessageInitWatchMsg,\n  IResponseMessageInitEventMsg,\n  IDBEvent,\n  IRequestMessageRebuildWatchMsg,\n  IRequestMessageCloseWatchMsg,\n  IRequestMsgType,\n  IResponseMessageNextEventMsg,\n  IRequestMessageCheckLastMsg,\n  IWatchOptions\n} from '@cloudbase/types/realtime'\nimport { \n  ISingleDBEvent\n} from '@cloudbase/types/database'\n// import Reporter from \"./externals/public-lib/reporter\"\nimport { RealtimeListener } from './listener'\nimport { Snapshot } from './snapshot'\nimport { IWSSendOptions, ILoginResult } from './websocket-client'\nimport { \n  ERR_CODE,\n  CloudSDKError,\n  isTimeoutError,\n  CancelledError,\n  isCancelledError,\n  isRealtimeErrorMessageError, \n  RealtimeErrorMessageError, \n  TimeoutError \n} from './error'\nimport { sleep } from './utils'\n\n// =============== Realtime Virtual WebSocket Client (Internal) ====================\n\ninterface IVirtualWebSocketClientConstructorOptions extends IWatchOptions {\n  // ws: RealtimeWebSocketClient\n  envId?: string\n  collectionName: string\n  query: string\n  limit?: number\n  orderBy?: Record<string, string>\n  send: <T = any>(opts: IWSSendOptions) => Promise<T>\n  login: (envId?: string, refresh?: boolean) => Promise<any>\n  isWSConnected: () => boolean\n  onceWSConnected: () => Promise<void>\n  getWaitExpectedTimeoutLength: () => number\n  onWatchStart: (client: VirtualWebSocketClient, queryID: string) => void\n  onWatchClose: (client: VirtualWebSocketClient, queryID: string) => void\n  debug?: boolean\n}\n\ninterface IWatchSessionInfo {\n  queryID: string\n  currentEventId: number\n  currentDocs: Record<string, any>[]\n  expectEventId?: number\n}\n\ninterface IHandleCommonErrorOptions {\n  onSignError: (e: RealtimeErrorMessageError) => void\n  onTimeoutError: (e: TimeoutError) => void\n  onCancelledError: (e: CancelledError) => void\n  onNotRetryableError: (e: RealtimeErrorMessageError) => void\n  onUnknownError: (e: any) => void\n}\n\ninterface IHandleWatchEstablishmentErrorOptions {\n  operationName: 'INIT_WATCH' | 'REBUILD_WATCH'\n  resolve: (value?: PromiseLike<void> | undefined) => void\n  reject: (e: any) => void\n  // retry: (refreshLogin?: boolean) => void\n  // abortWatch: (e: any) => void\n}\n\nenum WATCH_STATUS {\n  LOGGINGIN = 'LOGGINGIN',\n  INITING = 'INITING',\n  REBUILDING = 'REBUILDING',\n  ACTIVE = 'ACTIVE',\n  ERRORED = 'ERRORED',\n  CLOSING = 'CLOSING',\n  CLOSED = 'CLOSED',\n  PAUSED = 'PAUSED',\n  RESUMING = 'RESUMING'\n}\n\nconst DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR = 100\nconst DEFAULT_MAX_AUTO_RETRY_ON_ERROR = 2\nconst DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR = 2\nconst DEFAULT_SEND_ACK_DEBOUNCE_TIMEOUT = 10 * 1000\nconst DEFAULT_INIT_WATCH_TIMEOUT = 10 * 1000\nconst DEFAULT_REBUILD_WATCH_TIMEOUT = 10 * 1000\n\nexport class VirtualWebSocketClient {\n  // passed over\n  watchId: string\n  // own\n  listener: RealtimeListener\n  private envId?: string\n  private collectionName: string\n  private query: string\n  private limit: number\n  private orderBy: Record<string, string>\n  private send: <T = any>(opts: IWSSendOptions) => Promise<T>\n  private login: (envId?: string, refresh?: boolean) => Promise<any>\n  private isWSConnected: () => boolean\n  private onceWSConnected: () => Promise<void>\n  private getWaitExpectedTimeoutLength: () => number\n  private onWatchStart: (\n    client: VirtualWebSocketClient,\n    queryID: string\n  ) => void\n  private onWatchClose: (\n    client: VirtualWebSocketClient,\n    queryID: string\n  ) => void\n  private debug?: boolean\n\n  private watchStatus: WATCH_STATUS = WATCH_STATUS.INITING\n  private _availableRetries: Partial<Record<IRequestMsgType, number>>\n  private _ackTimeoutId?: number\n  private _initWatchPromise?: Promise<void>\n  private _rebuildWatchPromise?: Promise<void>\n\n  // obtained\n  private sessionInfo?: IWatchSessionInfo\n\n  // internal\n  private _waitExpectedTimeoutId?: number\n\n  constructor(options: IVirtualWebSocketClientConstructorOptions) {\n    this.watchId = `watchid_${+new Date()}_${Math.random()}`\n    this.envId = options.envId\n    this.collectionName = options.collectionName\n    this.query = options.query\n    this.limit = options.limit\n    this.orderBy = options.orderBy\n    this.send = options.send\n    this.login = options.login\n    this.isWSConnected = options.isWSConnected\n    this.onceWSConnected = options.onceWSConnected\n    this.getWaitExpectedTimeoutLength = options.getWaitExpectedTimeoutLength\n    this.onWatchStart = options.onWatchStart\n    this.onWatchClose = options.onWatchClose\n    this.debug = options.debug\n\n    this._availableRetries = {\n      INIT_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,\n      REBUILD_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,\n      CHECK_LAST: DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR\n    }\n\n    this.listener = new RealtimeListener({\n      close: this.closeWatch,\n      onChange: options.onChange,\n      onError: options.onError,\n      debug: this.debug,\n      virtualClient: this\n    })\n\n    this.initWatch()\n  }\n\n  onMessage(msg: IResponseMessage) {\n    // watchStatus sanity check\n    switch (this.watchStatus) {\n      case WATCH_STATUS.PAUSED: {\n        // ignore all but error message\n        if (msg.msgType !== 'ERROR') {\n          return\n        }\n        break\n      }\n      case WATCH_STATUS.LOGGINGIN:\n      case WATCH_STATUS.INITING:\n      case WATCH_STATUS.REBUILDING: {\n        console.warn(\n          `[realtime listener] internal non-fatal error: unexpected message received while ${this.watchStatus}`\n        )\n        return\n      }\n      case WATCH_STATUS.CLOSED: {\n        console.warn(\n          '[realtime listener] internal non-fatal error: unexpected message received when the watch has closed'\n        )\n        return\n      }\n      case WATCH_STATUS.ERRORED: {\n        console.warn(\n          '[realtime listener] internal non-fatal error: unexpected message received when the watch has ended with error'\n        )\n        return\n      }\n    }\n\n    if (!this.sessionInfo) {\n      console.warn(\n        '[realtime listener] internal non-fatal error: sessionInfo not found while message is received.'\n      )\n      return\n    }\n\n    this.scheduleSendACK()\n\n    switch (msg.msgType) {\n      case 'NEXT_EVENT': {\n        // if (process.env.DEBUG) {\n        // @ts-ignore\n        // if (wx._ignore) {\n        console.warn(`nextevent ${msg.msgData.currEvent} ignored`, msg)\n        // @ts-ignore\n        // wx._ignore = false\n        // return\n        // }\n        // }\n\n        this.handleServerEvents(msg)\n        break\n      }\n      case 'CHECK_EVENT': {\n        if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {\n          // client eventID < server eventID:\n          // there might be one or more pending events not yet received but sent by the server\n          this.sessionInfo.expectEventId = msg.msgData.currEvent\n          this.clearWaitExpectedEvent()\n          // @ts-ignore\n          this._waitExpectedTimeoutId = setTimeout(() => {\n            // must rebuild watch\n            this.rebuildWatch()\n          }, this.getWaitExpectedTimeoutLength())\n\n          // if (process.env.DEBUG) {\n          console.log(\n            `[realtime] waitExpectedTimeoutLength ${this.getWaitExpectedTimeoutLength()}`\n          )\n          // }\n        }\n        break\n      }\n      case 'ERROR': {\n        // receive server error\n        this.closeWithError(\n          new CloudSDKError({\n            errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_SERVER_ERROR_MSG as string,\n            errMsg: `${msg.msgData.code} - ${msg.msgData.message}`\n          })\n        )\n        break\n      }\n      default: {\n        // if (process.env.DEBUG) {\n        console.warn(\n          `[realtime listener] virtual client receive unexpected msg ${msg.msgType}: `,\n          msg\n        )\n        // }\n        break\n      }\n    }\n  }\n\n  closeWithError(error: any) {\n    this.watchStatus = WATCH_STATUS.ERRORED\n    this.clearACKSchedule()\n    this.listener.onError(error)\n    // Reporter.surroundThirdByTryCatch(() => this.listener.onError(error))\n    this.onWatchClose(\n      this,\n      (this.sessionInfo && this.sessionInfo.queryID) || ''\n    )\n\n    // if (process.env.DEBUG) {\n    console.log(\n      `[realtime] client closed (${this.collectionName} ${this.query}) (watchId ${this.watchId})`\n    )\n    // }\n  }\n\n  pause() {\n    this.watchStatus = WATCH_STATUS.PAUSED\n    // if (process.env.DEBUG) {\n    console.log(\n      `[realtime] client paused (${this.collectionName} ${this.query}) (watchId ${this.watchId})`\n    )\n    // }\n  }\n\n  // resume() {\n  //   return this.sessionInfo ? this.rebuildWatch() : this.initWatch()\n  // }\n\n  async resume(): Promise<void> {\n    this.watchStatus = WATCH_STATUS.RESUMING\n\n    // if (process.env.DEBUG) {\n    console.log(\n      `[realtime] client resuming with ${\n        this.sessionInfo ? 'REBUILD_WATCH' : 'INIT_WATCH'\n      } (${this.collectionName} ${this.query}) (${this.watchId})`\n    )\n    // }\n\n    try {\n      await (this.sessionInfo ? this.rebuildWatch() : this.initWatch())\n\n      // if (process.env.DEBUG) {\n      console.log(\n        `[realtime] client successfully resumed (${this.collectionName} ${this.query}) (${this.watchId})`\n      )\n      // }\n    } catch (e) {\n      // if (process.env.DEBUG) {\n      console.error(\n        `[realtime] client resume failed (${this.collectionName} ${this.query}) (${this.watchId})`,\n        e\n      )\n      // }\n    }\n  }\n\n  private _login = async (\n    envId?: string,\n    refresh?: boolean\n  ): Promise<ILoginResult> => {\n    this.watchStatus = WATCH_STATUS.LOGGINGIN\n    const loginResult = await this.login(envId, refresh)\n    if (!this.envId) {\n      this.envId = loginResult.envId\n    }\n    return loginResult\n  }\n\n  private initWatch = async (forceRefreshLogin?: boolean): Promise<void> => {\n    if (this._initWatchPromise) {\n      return this._initWatchPromise\n    }\n\n    this._initWatchPromise = new Promise<void>(\n      async (resolve, reject): Promise<void> => {\n        try {\n          if (this.watchStatus === WATCH_STATUS.PAUSED) {\n            // if (process.env.DEBUG) {\n            console.log('[realtime] initWatch cancelled on pause')\n            // }\n            return resolve()\n          }\n\n          const { envId } = await this._login(this.envId, forceRefreshLogin)\n\n          // if (!this.sessionInfo) {\n          //   throw new Error(`can not rebuildWatch without a successful initWatch (lack of sessionInfo)`)\n          // }\n\n          if ((this.watchStatus as WATCH_STATUS) === WATCH_STATUS.PAUSED) {\n            console.log('[realtime] initWatch cancelled on pause')\n            return resolve()\n          }\n\n          this.watchStatus = WATCH_STATUS.INITING\n\n          const initWatchMsg: IRequestMessageInitWatchMsg = {\n            watchId: this.watchId,\n            requestId: genRequestId(),\n            msgType: 'INIT_WATCH',\n            msgData: {\n              envId,\n              collName: this.collectionName,\n              query: this.query,\n              limit: this.limit,\n              orderBy: this.orderBy\n            }\n          }\n\n          const initEventMsg = await this.send<IResponseMessageInitEventMsg>({\n            msg: initWatchMsg,\n            waitResponse: true,\n            skipOnMessage: true,\n            timeout: DEFAULT_INIT_WATCH_TIMEOUT\n          })\n\n          const { events, currEvent } = initEventMsg.msgData\n\n          this.sessionInfo = {\n            queryID: initEventMsg.msgData.queryID,\n            currentEventId: currEvent - 1,\n            currentDocs: []\n          }\n\n          // FIX: in initEvent message, all events have id 0, which is inconsistent with currEvent\n          if (events.length > 0) {\n            for (const e of events) {\n              e.ID = currEvent\n            }\n            this.handleServerEvents(initEventMsg)\n          } else {\n            this.sessionInfo.currentEventId = currEvent\n            const snapshot = new Snapshot({\n              id: currEvent,\n              docChanges: [],\n              docs: [],\n              type: 'init'\n            })\n            this.listener.onChange(snapshot)\n            this.scheduleSendACK()\n          }\n          this.onWatchStart(this, this.sessionInfo.queryID)\n          this.watchStatus = WATCH_STATUS.ACTIVE\n          this._availableRetries.INIT_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR\n          resolve()\n        } catch (e) {\n          this.handleWatchEstablishmentError(e, {\n            operationName: 'INIT_WATCH',\n            resolve,\n            reject\n          })\n        }\n      }\n    )\n\n    let success = false\n\n    try {\n      await this._initWatchPromise\n      success = true\n    } finally {\n      this._initWatchPromise = undefined\n    }\n\n    // if (process.env.DEBUG) {\n    console.log(`[realtime] initWatch ${success ? 'success' : 'fail'}`)\n    // }\n  }\n\n  private rebuildWatch = async (forceRefreshLogin?: boolean): Promise<void> => {\n    if (this._rebuildWatchPromise) {\n      return this._rebuildWatchPromise\n    }\n\n    this._rebuildWatchPromise = new Promise<void>(\n      async (resolve, reject): Promise<void> => {\n        try {\n          if (this.watchStatus === WATCH_STATUS.PAUSED) {\n            // if (process.env.DEBUG) {\n            console.log('[realtime] rebuildWatch cancelled on pause')\n            // }\n            return resolve()\n          }\n          const { envId } = await this._login(this.envId, forceRefreshLogin)\n\n          if (!this.sessionInfo) {\n            throw new Error(\n              'can not rebuildWatch without a successful initWatch (lack of sessionInfo)'\n            )\n          }\n\n          if ((this.watchStatus as WATCH_STATUS) === WATCH_STATUS.PAUSED) {\n            console.log('[realtime] rebuildWatch cancelled on pause')\n            return resolve()\n          }\n\n          this.watchStatus = WATCH_STATUS.REBUILDING\n\n          const rebuildWatchMsg: IRequestMessageRebuildWatchMsg = {\n            watchId: this.watchId,\n            requestId: genRequestId(),\n            msgType: 'REBUILD_WATCH',\n            msgData: {\n              envId,\n              collName: this.collectionName,\n              queryID: this.sessionInfo.queryID,\n              eventID: this.sessionInfo.currentEventId\n            }\n          }\n\n          const nextEventMsg = await this.send<IResponseMessageNextEventMsg>({\n            msg: rebuildWatchMsg,\n            waitResponse: true,\n            skipOnMessage: false,\n            timeout: DEFAULT_REBUILD_WATCH_TIMEOUT\n          })\n\n          this.handleServerEvents(nextEventMsg)\n\n          this.watchStatus = WATCH_STATUS.ACTIVE\n          this._availableRetries.REBUILD_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR\n          resolve()\n        } catch (e) {\n          this.handleWatchEstablishmentError(e, {\n            operationName: 'REBUILD_WATCH',\n            resolve,\n            reject\n          })\n        }\n      }\n    )\n\n    let success = false\n\n    try {\n      await this._rebuildWatchPromise\n      success = true\n    } finally {\n      this._rebuildWatchPromise = undefined\n    }\n\n    // if (process.env.DEBUG) {\n    console.log(`[realtime] rebuildWatch ${success ? 'success' : 'fail'}`)\n    // }\n  }\n\n  private handleWatchEstablishmentError = async (\n    e: any,\n    options: IHandleWatchEstablishmentErrorOptions\n  ) => {\n    const isInitWatch = options.operationName === 'INIT_WATCH'\n\n    const abortWatch = () => {\n      // mock temp comment\n      this.closeWithError(\n        new CloudSDKError({\n          errCode: isInitWatch\n            ? (ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_INIT_WATCH_FAIL as string)\n            : (ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_REBUILD_WATCH_FAIL as string),\n          errMsg: e\n        })\n      )\n      options.reject(e)\n    }\n\n    const retry = (refreshLogin?: boolean) => {\n      if (this.useRetryTicket(options.operationName)) {\n        if (isInitWatch) {\n          this._initWatchPromise = undefined\n          options.resolve(this.initWatch(refreshLogin))\n        } else {\n          this._rebuildWatchPromise = undefined\n          options.resolve(this.rebuildWatch(refreshLogin))\n        }\n      } else {\n        abortWatch()\n      }\n    }\n\n    this.handleCommonError(e, {\n      onSignError: () => retry(true),\n      onTimeoutError: () => retry(false),\n      onNotRetryableError: abortWatch,\n      onCancelledError: options.reject,\n      onUnknownError: async () => {\n        try {\n          const onWSDisconnected = async () => {\n            this.pause()\n            await this.onceWSConnected()\n            retry(true)\n          }\n\n          if (!this.isWSConnected()) {\n            await onWSDisconnected()\n          } else {\n            await sleep(DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR)\n            if (this.watchStatus === WATCH_STATUS.PAUSED) {\n              // cancel\n              options.reject(\n                new CancelledError(\n                  `${options.operationName} cancelled due to pause after unknownError`\n                )\n              )\n            } else if (!this.isWSConnected()) {\n              await onWSDisconnected()\n            } else {\n              retry(false)\n            }\n          }\n        } catch (e) {\n          // unexpected error while handling error, in order to provide maximum effort on SEAMINGLESS FAULT TOLERANCE, just retry\n          retry(true)\n        }\n      }\n    })\n  }\n\n  private closeWatch = async () => {\n    const queryId = this.sessionInfo ? this.sessionInfo.queryID : ''\n\n    if (this.watchStatus !== WATCH_STATUS.ACTIVE) {\n      this.watchStatus = WATCH_STATUS.CLOSED\n      this.onWatchClose(this, queryId)\n      return\n    }\n\n    try {\n      this.watchStatus = WATCH_STATUS.CLOSING\n\n      const closeWatchMsg: IRequestMessageCloseWatchMsg = {\n        watchId: this.watchId,\n        requestId: genRequestId(),\n        msgType: 'CLOSE_WATCH',\n        msgData: null\n      }\n\n      await this.send({\n        msg: closeWatchMsg\n      })\n\n      this.sessionInfo = undefined\n      this.watchStatus = WATCH_STATUS.CLOSED\n    } catch (e) {\n      this.closeWithError(\n        new CloudSDKError({\n          errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CLOSE_WATCH_FAIL as string,\n          errMsg: e\n        })\n      )\n    } finally {\n      this.onWatchClose(this, queryId)\n    }\n  }\n\n  private scheduleSendACK = () => {\n    this.clearACKSchedule()\n\n    // TODO: should we check status after timeout\n    // @ts-ignore\n    this._ackTimeoutId = setTimeout(() => {\n      if (this._waitExpectedTimeoutId) {\n        this.scheduleSendACK()\n      } else {\n        this.sendACK()\n      }\n    }, DEFAULT_SEND_ACK_DEBOUNCE_TIMEOUT)\n  }\n\n  private clearACKSchedule = () => {\n    if (this._ackTimeoutId) {\n      clearTimeout(this._ackTimeoutId)\n    }\n  }\n\n  private sendACK = async (): Promise<void> => {\n    try {\n      if (this.watchStatus !== WATCH_STATUS.ACTIVE) {\n        this.scheduleSendACK()\n        return\n      }\n\n      if (!this.sessionInfo) {\n        console.warn(\n          '[realtime listener] can not send ack without a successful initWatch (lack of sessionInfo)'\n        )\n        return\n      }\n\n      const ackMsg: IRequestMessageCheckLastMsg = {\n        watchId: this.watchId,\n        requestId: genRequestId(),\n        msgType: 'CHECK_LAST',\n        msgData: {\n          queryID: this.sessionInfo.queryID,\n          eventID: this.sessionInfo.currentEventId\n        }\n      }\n\n      await this.send({\n        msg: ackMsg\n      })\n\n      this.scheduleSendACK()\n    } catch (e) {\n      // TODO: refactor\n      if (isRealtimeErrorMessageError(e)) {\n        const msg = e.payload\n        switch (msg.msgData.code) {\n          // signature error -> retry with refreshed signature\n          case 'CHECK_LOGIN_FAILED':\n          case 'SIGN_EXPIRED_ERROR':\n          case 'SIGN_INVALID_ERROR':\n          case 'SIGN_PARAM_INVALID': {\n            this.rebuildWatch()\n            return\n          }\n          // other -> throw\n          case 'QUERYID_INVALID_ERROR':\n          case 'SYS_ERR':\n          case 'INVALIID_ENV':\n          case 'COLLECTION_PERMISSION_DENIED': {\n            // must throw\n            this.closeWithError(\n              new CloudSDKError({\n                errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL as string,\n                errMsg: msg.msgData.code\n              })\n            )\n            return\n          }\n          default: {\n            break\n          }\n        }\n      }\n\n      // maybe retryable\n      if (\n        this._availableRetries.CHECK_LAST &&\n        this._availableRetries.CHECK_LAST > 0\n      ) {\n        this._availableRetries.CHECK_LAST--\n        this.scheduleSendACK()\n      } else {\n        this.closeWithError(\n          new CloudSDKError({\n            errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL as string,\n            errMsg: e\n          })\n        )\n      }\n    }\n  }\n\n  private handleCommonError = (\n    e: any,\n    options: IHandleCommonErrorOptions\n  ): void => {\n    if (isRealtimeErrorMessageError(e)) {\n      const msg = e.payload\n      switch (msg.msgData.code) {\n        // signature error -> retry with refreshed signature\n        case 'CHECK_LOGIN_FAILED':\n        case 'SIGN_EXPIRED_ERROR':\n        case 'SIGN_INVALID_ERROR':\n        case 'SIGN_PARAM_INVALID': {\n          options.onSignError(e)\n          return\n        }\n        // not-retryable error -> throw\n        case 'QUERYID_INVALID_ERROR':\n        case 'SYS_ERR':\n        case 'INVALIID_ENV':\n        case 'COLLECTION_PERMISSION_DENIED': {\n          options.onNotRetryableError(e)\n          return\n        }\n        default: {\n          options.onNotRetryableError(e)\n          return\n        }\n      }\n    } else if (isTimeoutError(e)) {\n      // timeout error\n      options.onTimeoutError(e)\n      return\n    } else if (isCancelledError(e)) {\n      // cancelled error\n      options.onCancelledError(e)\n      return\n    }\n\n    // unknown error\n    options.onUnknownError(e)\n  }\n\n  // credit a retry chance from availableRetries\n  private useRetryTicket(operationName: IRequestMsgType): boolean {\n    if (\n      this._availableRetries[operationName] &&\n      this._availableRetries[operationName]! > 0\n    ) {\n      this._availableRetries[operationName]!--\n\n      // if (process.env.DEBUG) {\n      console.log(\n        `[realtime] ${operationName} use a retry ticket, now only ${this._availableRetries[operationName]} retry left`\n      )\n      // }\n\n      return true\n    }\n    return false\n  }\n\n  private async handleServerEvents(\n    msg: IResponseMessageInitEventMsg | IResponseMessageNextEventMsg\n  ) {\n    try {\n      this.scheduleSendACK()\n      await this._handleServerEvents(msg)\n      this._postHandleServerEventsValidityCheck(msg)\n    } catch (e) {\n      // if (process.env.DEBUG) {\n      // TODO: report\n      console.error(\n        '[realtime listener] internal non-fatal error: handle server events failed with error: ',\n        e\n      )\n\n      // writeToFile(\n      //   \"wserror.txt\",\n      //   `[realtime listener] internal non-fatal error: handle server events failed with error:  ${JSON.stringify(\n      //     Object.assign({}, e, {\n      //       requestId: msg.requestId,\n      //       watchId: msg.watchId\n      //     })\n      //   )} \\n`\n      // )\n      // }\n\n      throw e\n    }\n  }\n\n  private async _handleServerEvents(\n    msg: IResponseMessageInitEventMsg | IResponseMessageNextEventMsg\n  ) {\n    const { requestId } = msg\n\n    const { events } = msg.msgData\n    const { msgType } = msg\n\n    if (!events.length || !this.sessionInfo) {\n      return\n    }\n\n    const sessionInfo = this.sessionInfo\n\n    let allChangeEvents: ISingleDBEvent[]\n    try {\n      allChangeEvents = events.map(getPublicEvent)\n    } catch (e) {\n      this.closeWithError(\n        new CloudSDKError({\n          errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_RECEIVE_INVALID_SERVER_DATA as string,\n          errMsg: e\n        })\n      )\n      return\n    }\n\n    // aggregate docs\n    let docs = [...sessionInfo.currentDocs]\n    let initEncountered = false\n    for (let i = 0, len = allChangeEvents.length; i < len; i++) {\n      const change = allChangeEvents[i]\n\n      if (sessionInfo.currentEventId >= change.id) {\n        if (!allChangeEvents[i - 1] || change.id > allChangeEvents[i - 1].id) {\n          // duplicate event, dropable\n          // TODO: report\n          // if (process.env.DEBUG) {\n          console.warn(\n            `[realtime] duplicate event received, cur ${sessionInfo.currentEventId} but got ${change.id}`\n          )\n          // }\n        } else {\n          // allChangeEvents should be in ascending order according to eventId, this should never happens, must report a non-fatal error\n          console.error(\n            `[realtime listener] server non-fatal error: events out of order (the latter event's id is smaller than that of the former) (requestId ${requestId})`\n          )\n\n          // writeToFile(\n          //   \"wserror.txt\",\n          //   `[realtime listener] server non-fatal error: events out of order (the latter event's id is smaller than that of the former)  ${JSON.stringify(\n          //     Object.assign(\n          //       {},\n          //       {\n          //         requestId: msg.requestId,\n          //         watchId: msg.watchId\n          //       }\n          //     )\n          //   )} \\n`\n          // )\n        }\n        continue\n      } else if (sessionInfo.currentEventId === change.id - 1) {\n        // correct sequence\n        // first handle dataType then queueType:\n        // 1. dataType: we ONLY populate change.doc if neccessary\n        // 2. queueType: we build the data snapshot\n\n        switch (change.dataType) {\n          case 'update': {\n            // only need to populate change.doc when it is not provided\n            if (!change.doc) {\n              switch (change.queueType) {\n                case 'update':\n                case 'dequeue': {\n                  const localDoc = docs.find(doc => doc._id === change.docId)\n                  if (localDoc) {\n                    // a partial update\n                    const doc = cloneDeep(localDoc)\n\n                    if (change.updatedFields) {\n                      for (const fieldPath in change.updatedFields) {\n                        set(doc, fieldPath, change.updatedFields[fieldPath])\n                      }\n                    }\n\n                    if (change.removedFields) {\n                      for (const fieldPath of change.removedFields) {\n                        unset(doc, fieldPath)\n                      }\n                    }\n\n                    change.doc = doc\n                  } else {\n                    // TODO report\n                    console.error(\n                      '[realtime listener] internal non-fatal server error: unexpected update dataType event where no doc is associated.'\n                    )\n\n                    // writeToFile(\n                    //   \"wserror.txt\",\n                    //   `[realtime listener] internal non-fatal server error: unexpected update dataType event where no doc is associated.  ${JSON.stringify(\n                    //     Object.assign(\n                    //       {},\n                    //       {\n                    //         requestId: msg.requestId,\n                    //         watchId: msg.watchId\n                    //       }\n                    //     )\n                    //   )} \\n`\n                    // )\n                  }\n                  break\n                }\n                case 'enqueue': {\n                  // doc is provided by server, this should never occur\n                  const err = new CloudSDKError({\n                    errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR as string,\n                    errMsg: `HandleServerEvents: full doc is not provided with dataType=\"update\" and queueType=\"enqueue\" (requestId ${msg.requestId})`\n                  })\n                  this.closeWithError(err)\n                  throw err\n                }\n                default: {\n                  break\n                }\n              }\n            }\n            break\n          }\n          case 'replace': {\n            // validation\n            if (!change.doc) {\n              // doc is provided by server, this should never occur\n              const err = new CloudSDKError({\n                errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR as string,\n                errMsg: `HandleServerEvents: full doc is not provided with dataType=\"replace\" (requestId ${msg.requestId})`\n              })\n              this.closeWithError(err)\n              throw err\n            }\n            break\n          }\n          case 'remove': {\n            const doc = docs.find(doc => doc._id === change.docId)\n            if (doc) {\n              change.doc = doc\n            } else {\n              // TODO report\n              console.error(\n                '[realtime listener] internal non-fatal server error: unexpected remove event where no doc is associated.'\n              )\n\n              // writeToFile(\n              //   \"wserror.txt\",\n              //   `[realtime listener] internal non-fatal server error: unexpected remove event where no doc is associated. ${JSON.stringify(\n              //     Object.assign(\n              //       {},\n              //       {\n              //         requestId: msg.requestId,\n              //         watchId: msg.watchId\n              //       }\n              //     )\n              //   )} \\n`\n              // )\n            }\n            break\n          }\n          case 'limit': {\n            if (!change.doc) {\n              switch(change.queueType) {\n                case 'dequeue': {\n                  const doc = docs.find(doc => doc._id === change.docId)\n                  if (doc) {\n                    change.doc = doc\n                  } else {\n                    console.error(\n                      '[realtime listener] internal non-fatal server error: unexpected limit dataType event where no doc is associated.'\n                    )\n                  }\n                  break\n                }\n                case 'enqueue': {\n                  // doc is provided by server, this should never occur\n                  const err = new CloudSDKError({\n                    errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR as string,\n                    errMsg: `HandleServerEvents: full doc is not provided with dataType=\"limit\" and queueType=\"enqueue\" (requestId ${msg.requestId})`\n                  })\n                  this.closeWithError(err)\n                  throw err\n                }\n                default: {\n                  break\n                }\n              }\n            }\n            break\n          }\n        }\n\n        switch (change.queueType) {\n          case 'init': {\n            if (!initEncountered) {\n              initEncountered = true\n              docs = [change.doc]\n            } else {\n              docs.push(change.doc)\n            }\n            break\n          }\n          case 'enqueue': {\n            docs.push(change.doc)\n            break\n          }\n          case 'dequeue': {\n            const ind = docs.findIndex(doc => doc._id === change.docId)\n            if (ind > -1) {\n              docs.splice(ind, 1)\n            } else {\n              // TODO report\n              console.error(\n                '[realtime listener] internal non-fatal server error: unexpected dequeue event where no doc is associated.'\n              )\n\n              // writeToFile(\n              //   \"wserror.txt\",\n              //   `[realtime listener] internal non-fatal server error: unexpected dequeue event where no doc is associated. ${JSON.stringify(\n              //     Object.assign(\n              //       {},\n              //       {\n              //         requestId: msg.requestId,\n              //         watchId: msg.watchId\n              //       }\n              //     )\n              //   )} \\n`\n              // )\n            }\n            break\n          }\n          case 'update': {\n            // writeToFile(\n            //   \"wserror.txt\",\n            //   `[realtime listener] docs ${JSON.stringify(\n            //     docs\n            //   )} change doc ${JSON.stringify(change)} \\n`\n            // )\n            const ind = docs.findIndex(doc => doc._id === change.docId)\n            if (ind > -1) {\n              docs[ind] = change.doc\n            } else {\n              // TODO report\n              console.error(\n                '[realtime listener] internal non-fatal server error: unexpected queueType update event where no doc is associated.'\n              )\n\n              // writeToFile(\n              //   \"wserror.txt\",\n              //   `[realtime listener] internal non-fatal server error: unexpected queueType update event where no doc is associated. ${JSON.stringify(\n              //     Object.assign(\n              //       {},\n              //       {\n              //         requestId: msg.requestId,\n              //         watchId: msg.watchId\n              //       }\n              //     )\n              //   )} \\n`\n              // )\n            }\n            break\n          }\n        }\n\n        if (\n          i === len - 1 ||\n          (allChangeEvents[i + 1] && allChangeEvents[i + 1].id !== change.id)\n        ) {\n          // a shallow slice creates a shallow snapshot\n          const docsSnapshot = [...docs]\n\n          // we slice first cause' if there're allChangeEvents that are of the same id after this change, we don't want to involve it for it is unexpected invalid order\n          const docChanges = allChangeEvents\n            .slice(0, i + 1)\n            .filter(c => c.id === change.id)\n\n          // all changes of this event has been handle, we could dispatch the event now\n          this.sessionInfo.currentEventId = change.id\n          this.sessionInfo.currentDocs = docs\n\n          const snapshot = new Snapshot({\n            id: change.id,\n            docChanges,\n            docs: docsSnapshot,\n            msgType\n          })\n\n          // Reporter.surroundThirdByTryCatch(() =>\n          this.listener.onChange(snapshot)\n          // )()\n        }\n      } else {\n        // out-of-order event\n        // if (process.env.DEBUG) {\n        // TODO: report\n        console.warn(\n          `[realtime listener] event received is out of order, cur ${this.sessionInfo.currentEventId} but got ${change.id}`\n        )\n        // }\n        // rebuild watch\n        await this.rebuildWatch()\n        return\n      }\n    }\n  }\n\n  private _postHandleServerEventsValidityCheck(\n    msg: IResponseMessageInitEventMsg | IResponseMessageNextEventMsg\n  ) {\n    if (!this.sessionInfo) {\n      console.error(\n        '[realtime listener] internal non-fatal error: sessionInfo lost after server event handling, this should never occur'\n      )\n\n      // writeToFile(\n      //   \"wserror.txt\",\n      //   `[realtime listener] internal non-fatal error: sessionInfo lost after server event handling, this should never occur ${JSON.stringify(\n      //     Object.assign(\n      //       {},\n      //       {\n      //         requestId: msg.requestId,\n      //         watchId: msg.watchId\n      //       }\n      //     )\n      //   )} \\n`\n      // )\n      return\n    }\n\n    if (\n      this.sessionInfo.expectEventId &&\n      this.sessionInfo.currentEventId >= this.sessionInfo.expectEventId\n    ) {\n      this.clearWaitExpectedEvent()\n    }\n\n    if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {\n      console.warn(\n        '[realtime listener] internal non-fatal error: client eventId does not match with server event id after server event handling'\n      )\n      return\n    }\n  }\n\n  private clearWaitExpectedEvent() {\n    if (this._waitExpectedTimeoutId) {\n      clearTimeout(this._waitExpectedTimeoutId)\n      this._waitExpectedTimeoutId = undefined\n    }\n  }\n}\n\nfunction getPublicEvent(event: IDBEvent): ISingleDBEvent {\n  const e: ISingleDBEvent = {\n    id: event.ID,\n    dataType: event.DataType,\n    queueType: event.QueueType,\n    docId: event.DocID,\n    doc: event.Doc && event.Doc !== '{}' ? JSON.parse(event.Doc) : undefined\n  }\n\n  if (event.DataType === 'update') {\n    // @ts-ignore\n    if (event.UpdatedFields) {\n      e.updatedFields = JSON.parse(event.UpdatedFields)\n    }\n    // TODO: wait for tcb to change removedFields to RemovedFields\n    // @ts-ignore\n    if (event.removedFields || event.RemovedFields) {\n      // @ts-ignore\n      // e.removedFields = event.removedFields\n      //   ? JSON.parse(event.removedFields)\n      //   : JSON.parse(event.RemovedFields)\n      e.removedFields = JSON.parse(event.removedFields)\n    }\n  }\n\n  return e\n}\n"]}
946
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"virtual-websocket-client.js","sourceRoot":"","sources":["../../src/virtual-websocket-client.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-plusplus */\n/* eslint-disable @typescript-eslint/no-misused-promises */\nimport set from 'lodash.set';\nimport unset from 'lodash.unset';\nimport cloneDeep from 'lodash.clonedeep';\nimport { genRequestId } from './message';\nimport {\n  IResponseMessage,\n  IRequestMessageInitWatchMsg,\n  IResponseMessageInitEventMsg,\n  IDBEvent,\n  IRequestMessageRebuildWatchMsg,\n  IRequestMessageCloseWatchMsg,\n  IRequestMsgType,\n  IResponseMessageNextEventMsg,\n  IRequestMessageCheckLastMsg,\n  IWatchOptions,\n} from '@cloudbase/types/realtime';\nimport {\n  ISingleDBEvent,\n} from '@cloudbase/types/database';\n// import Reporter from \"./externals/public-lib/reporter\"\nimport { RealtimeListener } from './listener';\nimport { Snapshot } from './snapshot';\nimport { IWSSendOptions, ILoginResult } from './websocket-client';\nimport {\n  ERR_CODE,\n  CloudSDKError,\n  isTimeoutError,\n  CancelledError,\n  isCancelledError,\n  isRealtimeErrorMessageError,\n  RealtimeErrorMessageError,\n  TimeoutError,\n} from './error';\nimport { sleep } from './utils';\n\n// =============== Realtime Virtual WebSocket Client (Internal) ====================\n\ninterface IVirtualWebSocketClientConstructorOptions extends IWatchOptions {\n  // ws: RealtimeWebSocketClient\n  envId?: string\n  collectionName: string\n  query: string\n  limit?: number\n  orderBy?: Record<string, string>\n  send: <T = any>(opts: IWSSendOptions) => Promise<T>\n  login: (envId?: string, refresh?: boolean) => Promise<any>\n  isWSConnected: () => boolean\n  onceWSConnected: () => Promise<void>\n  getWaitExpectedTimeoutLength: () => number\n  onWatchStart: (client: VirtualWebSocketClient, queryID: string) => void\n  onWatchClose: (client: VirtualWebSocketClient, queryID: string) => void\n  debug?: boolean\n}\n\ninterface IWatchSessionInfo {\n  queryID: string\n  currentEventId: number\n  currentDocs: Record<string, any>[]\n  expectEventId?: number\n}\n\ninterface IHandleCommonErrorOptions {\n  onSignError: (e: RealtimeErrorMessageError) => void\n  onTimeoutError: (e: TimeoutError) => void\n  onCancelledError: (e: CancelledError) => void\n  onNotRetryableError: (e: RealtimeErrorMessageError) => void\n  onUnknownError: (e: any) => void\n}\n\ninterface IHandleWatchEstablishmentErrorOptions {\n  operationName: 'INIT_WATCH' | 'REBUILD_WATCH'\n  resolve: (value?: PromiseLike<void> | undefined) => void\n  reject: (e: any) => void\n  // retry: (refreshLogin?: boolean) => void\n  // abortWatch: (e: any) => void\n}\n\nenum WATCH_STATUS {\n  LOGGINGIN = 'LOGGINGIN',\n  INITING = 'INITING',\n  REBUILDING = 'REBUILDING',\n  ACTIVE = 'ACTIVE',\n  ERRORED = 'ERRORED',\n  CLOSING = 'CLOSING',\n  CLOSED = 'CLOSED',\n  PAUSED = 'PAUSED',\n  RESUMING = 'RESUMING'\n}\n\nconst DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR = 100;\nconst DEFAULT_MAX_AUTO_RETRY_ON_ERROR = 2;\nconst DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR = 2;\nconst DEFAULT_SEND_ACK_DEBOUNCE_TIMEOUT = 10 * 1000;\nconst DEFAULT_INIT_WATCH_TIMEOUT = 10 * 1000;\nconst DEFAULT_REBUILD_WATCH_TIMEOUT = 10 * 1000;\n\nexport class VirtualWebSocketClient {\n  // passed over\n  watchId: string;\n  // own\n  listener: RealtimeListener;\n  private envId?: string;\n  private collectionName: string;\n  private query: string;\n  private limit: number;\n  private orderBy: Record<string, string>;\n  private send: <T = any>(opts: IWSSendOptions) => Promise<T>;\n  private login: (envId?: string, refresh?: boolean) => Promise<any>;\n  private isWSConnected: () => boolean;\n  private onceWSConnected: () => Promise<void>;\n  private getWaitExpectedTimeoutLength: () => number;\n  private onWatchStart: (\n    client: VirtualWebSocketClient,\n    queryID: string\n  ) => void;\n  private onWatchClose: (\n    client: VirtualWebSocketClient,\n    queryID: string\n  ) => void;\n  private debug?: boolean;\n\n  private watchStatus: WATCH_STATUS = WATCH_STATUS.INITING;\n  private _availableRetries: Partial<Record<IRequestMsgType, number>>;\n  private _ackTimeoutId?: number;\n  private _initWatchPromise?: Promise<void>;\n  private _rebuildWatchPromise?: Promise<void>;\n\n  // obtained\n  private sessionInfo?: IWatchSessionInfo;\n\n  // internal\n  private _waitExpectedTimeoutId?: number;\n\n  constructor(options: IVirtualWebSocketClientConstructorOptions) {\n    this.watchId = `watchid_${+new Date()}_${Math.random()}`;\n    this.envId = options.envId;\n    this.collectionName = options.collectionName;\n    this.query = options.query;\n    this.limit = options.limit;\n    this.orderBy = options.orderBy;\n    this.send = options.send;\n    this.login = options.login;\n    this.isWSConnected = options.isWSConnected;\n    this.onceWSConnected = options.onceWSConnected;\n    this.getWaitExpectedTimeoutLength = options.getWaitExpectedTimeoutLength;\n    this.onWatchStart = options.onWatchStart;\n    this.onWatchClose = options.onWatchClose;\n    this.debug = options.debug;\n\n    this._availableRetries = {\n      INIT_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,\n      REBUILD_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,\n      CHECK_LAST: DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR,\n    };\n\n    this.listener = new RealtimeListener({\n      close: this.closeWatch,\n      onChange: options.onChange,\n      onError: options.onError,\n      debug: this.debug,\n      virtualClient: this,\n    });\n\n    this.initWatch();\n  }\n\n  onMessage(msg: IResponseMessage) {\n    // watchStatus sanity check\n    switch (this.watchStatus) {\n      case WATCH_STATUS.PAUSED: {\n        // ignore all but error message\n        if (msg.msgType !== 'ERROR') {\n          return;\n        }\n        break;\n      }\n      case WATCH_STATUS.LOGGINGIN:\n      case WATCH_STATUS.INITING:\n      case WATCH_STATUS.REBUILDING: {\n        console.warn(`[realtime listener] internal non-fatal error: unexpected message received while ${this.watchStatus}`);\n        return;\n      }\n      case WATCH_STATUS.CLOSED: {\n        console.warn('[realtime listener] internal non-fatal error: unexpected message received when the watch has closed');\n        return;\n      }\n      case WATCH_STATUS.ERRORED: {\n        console.warn('[realtime listener] internal non-fatal error: unexpected message received when the watch has ended with error');\n        return;\n      }\n    }\n\n    if (!this.sessionInfo) {\n      console.warn('[realtime listener] internal non-fatal error: sessionInfo not found while message is received.');\n      return;\n    }\n\n    this.scheduleSendACK();\n\n    switch (msg.msgType) {\n      case 'NEXT_EVENT': {\n        // if (process.env.DEBUG) {\n        // @ts-ignore\n        // if (wx._ignore) {\n        console.warn(`nextevent ${msg.msgData.currEvent} ignored`, msg);\n        // @ts-ignore\n        // wx._ignore = false\n        // return\n        // }\n        // }\n\n        this.handleServerEvents(msg);\n        break;\n      }\n      case 'CHECK_EVENT': {\n        if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {\n          // client eventID < server eventID:\n          // there might be one or more pending events not yet received but sent by the server\n          this.sessionInfo.expectEventId = msg.msgData.currEvent;\n          this.clearWaitExpectedEvent();\n          // @ts-ignore\n          this._waitExpectedTimeoutId = setTimeout(() => {\n            // must rebuild watch\n            this.rebuildWatch();\n          }, this.getWaitExpectedTimeoutLength());\n\n          // if (process.env.DEBUG) {\n          console.log(`[realtime] waitExpectedTimeoutLength ${this.getWaitExpectedTimeoutLength()}`);\n          // }\n        }\n        break;\n      }\n      case 'ERROR': {\n        // receive server error\n        this.closeWithError(new CloudSDKError({\n          errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_SERVER_ERROR_MSG as string,\n          errMsg: `${msg.msgData.code} - ${msg.msgData.message}`,\n        }));\n        break;\n      }\n      default: {\n        // if (process.env.DEBUG) {\n        console.warn(\n          `[realtime listener] virtual client receive unexpected msg ${msg.msgType}: `,\n          msg\n        );\n        // }\n        break;\n      }\n    }\n  }\n\n  closeWithError(error: any) {\n    this.watchStatus = WATCH_STATUS.ERRORED;\n    this.clearACKSchedule();\n    this.listener.onError(error);\n    // Reporter.surroundThirdByTryCatch(() => this.listener.onError(error))\n    this.onWatchClose(\n      this,\n      this.sessionInfo?.queryID || ''\n    );\n\n    // if (process.env.DEBUG) {\n    console.log(`[realtime] client closed (${this.collectionName} ${this.query}) (watchId ${this.watchId})`);\n    // }\n  }\n\n  pause() {\n    this.watchStatus = WATCH_STATUS.PAUSED;\n    // if (process.env.DEBUG) {\n    console.log(`[realtime] client paused (${this.collectionName} ${this.query}) (watchId ${this.watchId})`);\n    // }\n  }\n\n  // resume() {\n  //   return this.sessionInfo ? this.rebuildWatch() : this.initWatch()\n  // }\n\n  async resume(): Promise<void> {\n    this.watchStatus = WATCH_STATUS.RESUMING;\n\n    // if (process.env.DEBUG) {\n    console.log(`[realtime] client resuming with ${\n      this.sessionInfo ? 'REBUILD_WATCH' : 'INIT_WATCH'\n    } (${this.collectionName} ${this.query}) (${this.watchId})`);\n    // }\n\n    try {\n      await (this.sessionInfo ? this.rebuildWatch() : this.initWatch());\n\n      // if (process.env.DEBUG) {\n      console.log(`[realtime] client successfully resumed (${this.collectionName} ${this.query}) (${this.watchId})`);\n      // }\n    } catch (e) {\n      // if (process.env.DEBUG) {\n      console.error(\n        `[realtime] client resume failed (${this.collectionName} ${this.query}) (${this.watchId})`,\n        e\n      );\n      // }\n    }\n  }\n\n  private _login = async (\n    envId?: string,\n    refresh?: boolean\n  ): Promise<ILoginResult> => {\n    this.watchStatus = WATCH_STATUS.LOGGINGIN;\n    const loginResult = await this.login(envId, refresh);\n    if (!this.envId) {\n      this.envId = loginResult.envId;\n    }\n    return loginResult;\n  };\n\n  private initWatch = async (forceRefreshLogin?: boolean): Promise<void> => {\n    if (this._initWatchPromise) {\n      return this._initWatchPromise;\n    }\n\n    this._initWatchPromise = new Promise<void>(async (resolve, reject): Promise<void> => {\n      try {\n        if (this.watchStatus === WATCH_STATUS.PAUSED) {\n          // if (process.env.DEBUG) {\n          console.log('[realtime] initWatch cancelled on pause');\n          // }\n          return resolve();\n        }\n\n        const { envId } = await this._login(this.envId, forceRefreshLogin);\n\n        // if (!this.sessionInfo) {\n        //   throw new Error(`can not rebuildWatch without a successful initWatch (lack of sessionInfo)`)\n        // }\n\n        if ((this.watchStatus as WATCH_STATUS) === WATCH_STATUS.PAUSED) {\n          console.log('[realtime] initWatch cancelled on pause');\n          return resolve();\n        }\n\n        this.watchStatus = WATCH_STATUS.INITING;\n\n        const initWatchMsg: IRequestMessageInitWatchMsg = {\n          watchId: this.watchId,\n          requestId: genRequestId(),\n          msgType: 'INIT_WATCH',\n          msgData: {\n            envId,\n            collName: this.collectionName,\n            query: this.query,\n            limit: this.limit,\n            orderBy: this.orderBy,\n          },\n        };\n\n        const initEventMsg = await this.send<IResponseMessageInitEventMsg>({\n          msg: initWatchMsg,\n          waitResponse: true,\n          skipOnMessage: true,\n          timeout: DEFAULT_INIT_WATCH_TIMEOUT,\n        });\n\n        const { events, currEvent } = initEventMsg.msgData;\n\n        this.sessionInfo = {\n          queryID: initEventMsg.msgData.queryID,\n          currentEventId: currEvent - 1,\n          currentDocs: [],\n        };\n\n        // FIX: in initEvent message, all events have id 0, which is inconsistent with currEvent\n        if (events.length > 0) {\n          for (const e of events) {\n            e.ID = currEvent;\n          }\n          this.handleServerEvents(initEventMsg);\n        } else {\n          this.sessionInfo.currentEventId = currEvent;\n          const snapshot = new Snapshot({\n            id: currEvent,\n            docChanges: [],\n            docs: [],\n            type: 'init',\n          });\n          this.listener.onChange(snapshot);\n          this.scheduleSendACK();\n        }\n        this.onWatchStart(this, this.sessionInfo.queryID);\n        this.watchStatus = WATCH_STATUS.ACTIVE;\n        this._availableRetries.INIT_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;\n        resolve();\n      } catch (e) {\n        this.handleWatchEstablishmentError(e, {\n          operationName: 'INIT_WATCH',\n          resolve,\n          reject,\n        });\n      }\n    });\n\n    let success = false;\n\n    try {\n      await this._initWatchPromise;\n      success = true;\n    } finally {\n      this._initWatchPromise = undefined;\n    }\n\n    // if (process.env.DEBUG) {\n    console.log(`[realtime] initWatch ${success ? 'success' : 'fail'}`);\n    // }\n  };\n\n  private rebuildWatch = async (forceRefreshLogin?: boolean): Promise<void> => {\n    if (this._rebuildWatchPromise) {\n      return this._rebuildWatchPromise;\n    }\n\n    this._rebuildWatchPromise = new Promise<void>(async (resolve, reject): Promise<void> => {\n      try {\n        if (this.watchStatus === WATCH_STATUS.PAUSED) {\n          // if (process.env.DEBUG) {\n          console.log('[realtime] rebuildWatch cancelled on pause');\n          // }\n          return resolve();\n        }\n        const { envId } = await this._login(this.envId, forceRefreshLogin);\n\n        if (!this.sessionInfo) {\n          throw new Error('can not rebuildWatch without a successful initWatch (lack of sessionInfo)');\n        }\n\n        if ((this.watchStatus as WATCH_STATUS) === WATCH_STATUS.PAUSED) {\n          console.log('[realtime] rebuildWatch cancelled on pause');\n          return resolve();\n        }\n\n        this.watchStatus = WATCH_STATUS.REBUILDING;\n\n        const rebuildWatchMsg: IRequestMessageRebuildWatchMsg = {\n          watchId: this.watchId,\n          requestId: genRequestId(),\n          msgType: 'REBUILD_WATCH',\n          msgData: {\n            envId,\n            collName: this.collectionName,\n            queryID: this.sessionInfo.queryID,\n            eventID: this.sessionInfo.currentEventId,\n          },\n        };\n\n        const nextEventMsg = await this.send<IResponseMessageNextEventMsg>({\n          msg: rebuildWatchMsg,\n          waitResponse: true,\n          skipOnMessage: false,\n          timeout: DEFAULT_REBUILD_WATCH_TIMEOUT,\n        });\n\n        this.handleServerEvents(nextEventMsg);\n\n        this.watchStatus = WATCH_STATUS.ACTIVE;\n        this._availableRetries.REBUILD_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;\n        resolve();\n      } catch (e) {\n        this.handleWatchEstablishmentError(e, {\n          operationName: 'REBUILD_WATCH',\n          resolve,\n          reject,\n        });\n      }\n    });\n\n    let success = false;\n\n    try {\n      await this._rebuildWatchPromise;\n      success = true;\n    } finally {\n      this._rebuildWatchPromise = undefined;\n    }\n\n    // if (process.env.DEBUG) {\n    console.log(`[realtime] rebuildWatch ${success ? 'success' : 'fail'}`);\n    // }\n  };\n\n  private handleWatchEstablishmentError = async (\n    e: any,\n    options: IHandleWatchEstablishmentErrorOptions\n  ) => {\n    const isInitWatch = options.operationName === 'INIT_WATCH';\n\n    const abortWatch = () => {\n      // mock temp comment\n      this.closeWithError(new CloudSDKError({\n        errCode: isInitWatch\n          ? (ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_INIT_WATCH_FAIL as string)\n          : (ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_REBUILD_WATCH_FAIL as string),\n        errMsg: e,\n      }));\n      options.reject(e);\n    };\n\n    const retry = (refreshLogin?: boolean) => {\n      if (this.useRetryTicket(options.operationName)) {\n        if (isInitWatch) {\n          this._initWatchPromise = undefined;\n          options.resolve(this.initWatch(refreshLogin));\n        } else {\n          this._rebuildWatchPromise = undefined;\n          options.resolve(this.rebuildWatch(refreshLogin));\n        }\n      } else {\n        abortWatch();\n      }\n    };\n\n    this.handleCommonError(e, {\n      onSignError: () => retry(true),\n      onTimeoutError: () => retry(false),\n      onNotRetryableError: abortWatch,\n      onCancelledError: options.reject,\n      onUnknownError: async () => {\n        try {\n          const onWSDisconnected = async () => {\n            this.pause();\n            await this.onceWSConnected();\n            retry(true);\n          };\n\n          if (!this.isWSConnected()) {\n            await onWSDisconnected();\n          } else {\n            await sleep(DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR);\n            if (this.watchStatus === WATCH_STATUS.PAUSED) {\n              // cancel\n              options.reject(new CancelledError(`${options.operationName} cancelled due to pause after unknownError`));\n            } else if (!this.isWSConnected()) {\n              await onWSDisconnected();\n            } else {\n              retry(false);\n            }\n          }\n        } catch (e) {\n          // unexpected error while handling error, in order to provide maximum effort on SEAMINGLESS FAULT TOLERANCE, just retry\n          retry(true);\n        }\n      },\n    });\n  };\n\n  private closeWatch = async () => {\n    const queryId = this.sessionInfo ? this.sessionInfo.queryID : '';\n\n    if (this.watchStatus !== WATCH_STATUS.ACTIVE) {\n      this.watchStatus = WATCH_STATUS.CLOSED;\n      this.onWatchClose(this, queryId);\n      return;\n    }\n\n    try {\n      this.watchStatus = WATCH_STATUS.CLOSING;\n\n      const closeWatchMsg: IRequestMessageCloseWatchMsg = {\n        watchId: this.watchId,\n        requestId: genRequestId(),\n        msgType: 'CLOSE_WATCH',\n        msgData: null,\n      };\n\n      await this.send({\n        msg: closeWatchMsg,\n      });\n\n      this.sessionInfo = undefined;\n      this.watchStatus = WATCH_STATUS.CLOSED;\n    } catch (e) {\n      this.closeWithError(new CloudSDKError({\n        errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CLOSE_WATCH_FAIL as string,\n        errMsg: e,\n      }));\n    } finally {\n      this.onWatchClose(this, queryId);\n    }\n  };\n\n  private scheduleSendACK = () => {\n    this.clearACKSchedule();\n\n    // TODO: should we check status after timeout\n    // @ts-ignore\n    this._ackTimeoutId = setTimeout(() => {\n      if (this._waitExpectedTimeoutId) {\n        this.scheduleSendACK();\n      } else {\n        this.sendACK();\n      }\n    }, DEFAULT_SEND_ACK_DEBOUNCE_TIMEOUT);\n  };\n\n  private clearACKSchedule = () => {\n    if (this._ackTimeoutId) {\n      clearTimeout(this._ackTimeoutId);\n    }\n  };\n\n  private sendACK = async (): Promise<void> => {\n    try {\n      if (this.watchStatus !== WATCH_STATUS.ACTIVE) {\n        this.scheduleSendACK();\n        return;\n      }\n\n      if (!this.sessionInfo) {\n        console.warn('[realtime listener] can not send ack without a successful initWatch (lack of sessionInfo)');\n        return;\n      }\n\n      const ackMsg: IRequestMessageCheckLastMsg = {\n        watchId: this.watchId,\n        requestId: genRequestId(),\n        msgType: 'CHECK_LAST',\n        msgData: {\n          queryID: this.sessionInfo.queryID,\n          eventID: this.sessionInfo.currentEventId,\n        },\n      };\n\n      await this.send({\n        msg: ackMsg,\n      });\n\n      this.scheduleSendACK();\n    } catch (e) {\n      // TODO: refactor\n      if (isRealtimeErrorMessageError(e)) {\n        const msg = e.payload;\n        switch (msg.msgData.code) {\n          // signature error -> retry with refreshed signature\n          case 'CHECK_LOGIN_FAILED':\n          case 'SIGN_EXPIRED_ERROR':\n          case 'SIGN_INVALID_ERROR':\n          case 'SIGN_PARAM_INVALID': {\n            this.rebuildWatch();\n            return;\n          }\n          // other -> throw\n          case 'QUERYID_INVALID_ERROR':\n          case 'SYS_ERR':\n          case 'INVALIID_ENV':\n          case 'COLLECTION_PERMISSION_DENIED': {\n            // must throw\n            this.closeWithError(new CloudSDKError({\n              errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL as string,\n              errMsg: msg.msgData.code,\n            }));\n            return;\n          }\n          default: {\n            break;\n          }\n        }\n      }\n\n      // maybe retryable\n      if (\n        this._availableRetries.CHECK_LAST\n        && this._availableRetries.CHECK_LAST > 0\n      ) {\n        this._availableRetries.CHECK_LAST--;\n        this.scheduleSendACK();\n      } else {\n        this.closeWithError(new CloudSDKError({\n          errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL as string,\n          errMsg: e,\n        }));\n      }\n    }\n  };\n\n  private handleCommonError = (\n    e: any,\n    options: IHandleCommonErrorOptions\n  ): void => {\n    if (isRealtimeErrorMessageError(e)) {\n      const msg = e.payload;\n      switch (msg.msgData.code) {\n        // signature error -> retry with refreshed signature\n        case 'CHECK_LOGIN_FAILED':\n        case 'SIGN_EXPIRED_ERROR':\n        case 'SIGN_INVALID_ERROR':\n        case 'SIGN_PARAM_INVALID': {\n          options.onSignError(e);\n          return;\n        }\n        // not-retryable error -> throw\n        case 'QUERYID_INVALID_ERROR':\n        case 'SYS_ERR':\n        case 'INVALIID_ENV':\n        case 'COLLECTION_PERMISSION_DENIED': {\n          options.onNotRetryableError(e);\n          return;\n        }\n        default: {\n          options.onNotRetryableError(e);\n          return;\n        }\n      }\n    } else if (isTimeoutError(e)) {\n      // timeout error\n      options.onTimeoutError(e);\n      return;\n    } else if (isCancelledError(e)) {\n      // cancelled error\n      options.onCancelledError(e);\n      return;\n    }\n\n    // unknown error\n    options.onUnknownError(e);\n  };\n\n  // credit a retry chance from availableRetries\n  private useRetryTicket(operationName: IRequestMsgType): boolean {\n    if (\n      this._availableRetries[operationName]\n      && this._availableRetries[operationName]! > 0\n    ) {\n      this._availableRetries[operationName]!--;\n\n      // if (process.env.DEBUG) {\n      console.log(`[realtime] ${operationName} use a retry ticket, now only ${this._availableRetries[operationName]} retry left`);\n      // }\n\n      return true;\n    }\n    return false;\n  }\n\n  private async handleServerEvents(msg: IResponseMessageInitEventMsg | IResponseMessageNextEventMsg) {\n    try {\n      this.scheduleSendACK();\n      await this._handleServerEvents(msg);\n      this._postHandleServerEventsValidityCheck(msg);\n    } catch (e) {\n      // if (process.env.DEBUG) {\n      // TODO: report\n      console.error(\n        '[realtime listener] internal non-fatal error: handle server events failed with error: ',\n        e\n      );\n\n      // writeToFile(\n      //   \"wserror.txt\",\n      //   `[realtime listener] internal non-fatal error: handle server events failed with error:  ${JSON.stringify(\n      //     Object.assign({}, e, {\n      //       requestId: msg.requestId,\n      //       watchId: msg.watchId\n      //     })\n      //   )} \\n`\n      // )\n      // }\n\n      throw e;\n    }\n  }\n\n  private async _handleServerEvents(msg: IResponseMessageInitEventMsg | IResponseMessageNextEventMsg) {\n    const { requestId } = msg;\n\n    const { events } = msg.msgData;\n    const { msgType } = msg;\n\n    if (!events.length || !this.sessionInfo) {\n      return;\n    }\n\n    const { sessionInfo } = this;\n\n    let allChangeEvents: ISingleDBEvent[];\n    try {\n      allChangeEvents = events.map(getPublicEvent);\n    } catch (e) {\n      this.closeWithError(new CloudSDKError({\n        errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_RECEIVE_INVALID_SERVER_DATA as string,\n        errMsg: e,\n      }));\n      return;\n    }\n\n    // aggregate docs\n    let docs = [...sessionInfo.currentDocs];\n    let initEncountered = false;\n    for (let i = 0, len = allChangeEvents.length; i < len; i++) {\n      const change = allChangeEvents[i];\n\n      if (sessionInfo.currentEventId >= change.id) {\n        if (!allChangeEvents[i - 1] || change.id > allChangeEvents[i - 1].id) {\n          // duplicate event, dropable\n          // TODO: report\n          // if (process.env.DEBUG) {\n          console.warn(`[realtime] duplicate event received, cur ${sessionInfo.currentEventId} but got ${change.id}`);\n          // }\n        } else {\n          // allChangeEvents should be in ascending order according to eventId, this should never happens, must report a non-fatal error\n          console.error(`[realtime listener] server non-fatal error: events out of order (the latter event's id is smaller than that of the former) (requestId ${requestId})`);\n\n          // writeToFile(\n          //   \"wserror.txt\",\n          //   `[realtime listener] server non-fatal error: events out of order (the latter event's id is smaller than that of the former)  ${JSON.stringify(\n          //     Object.assign(\n          //       {},\n          //       {\n          //         requestId: msg.requestId,\n          //         watchId: msg.watchId\n          //       }\n          //     )\n          //   )} \\n`\n          // )\n        }\n        continue;\n      } else if (sessionInfo.currentEventId === change.id - 1) {\n        // correct sequence\n        // first handle dataType then queueType:\n        // 1. dataType: we ONLY populate change.doc if neccessary\n        // 2. queueType: we build the data snapshot\n\n        switch (change.dataType) {\n          case 'update': {\n            // only need to populate change.doc when it is not provided\n            if (!change.doc) {\n              switch (change.queueType) {\n                case 'update':\n                case 'dequeue': {\n                  const localDoc = docs.find(doc => doc._id === change.docId);\n                  if (localDoc) {\n                    // a partial update\n                    const doc = cloneDeep(localDoc);\n\n                    if (change.updatedFields) {\n                      for (const fieldPath in change.updatedFields) {\n                        set(doc, fieldPath, change.updatedFields[fieldPath]);\n                      }\n                    }\n\n                    if (change.removedFields) {\n                      for (const fieldPath of change.removedFields) {\n                        unset(doc, fieldPath);\n                      }\n                    }\n\n                    change.doc = doc;\n                  } else {\n                    // TODO report\n                    console.error('[realtime listener] internal non-fatal server error: unexpected update dataType event where no doc is associated.');\n\n                    // writeToFile(\n                    //   \"wserror.txt\",\n                    //   `[realtime listener] internal non-fatal server error: unexpected update dataType event where no doc is associated.  ${JSON.stringify(\n                    //     Object.assign(\n                    //       {},\n                    //       {\n                    //         requestId: msg.requestId,\n                    //         watchId: msg.watchId\n                    //       }\n                    //     )\n                    //   )} \\n`\n                    // )\n                  }\n                  break;\n                }\n                case 'enqueue': {\n                  // doc is provided by server, this should never occur\n                  const err = new CloudSDKError({\n                    errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR as string,\n                    errMsg: `HandleServerEvents: full doc is not provided with dataType=\"update\" and queueType=\"enqueue\" (requestId ${msg.requestId})`,\n                  });\n                  this.closeWithError(err);\n                  throw err;\n                }\n                default: {\n                  break;\n                }\n              }\n            }\n            break;\n          }\n          case 'replace': {\n            // validation\n            if (!change.doc) {\n              // doc is provided by server, this should never occur\n              const err = new CloudSDKError({\n                errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR as string,\n                errMsg: `HandleServerEvents: full doc is not provided with dataType=\"replace\" (requestId ${msg.requestId})`,\n              });\n              this.closeWithError(err);\n              throw err;\n            }\n            break;\n          }\n          case 'remove': {\n            const doc = docs.find(doc => doc._id === change.docId);\n            if (doc) {\n              change.doc = doc;\n            } else {\n              // TODO report\n              console.error('[realtime listener] internal non-fatal server error: unexpected remove event where no doc is associated.');\n\n              // writeToFile(\n              //   \"wserror.txt\",\n              //   `[realtime listener] internal non-fatal server error: unexpected remove event where no doc is associated. ${JSON.stringify(\n              //     Object.assign(\n              //       {},\n              //       {\n              //         requestId: msg.requestId,\n              //         watchId: msg.watchId\n              //       }\n              //     )\n              //   )} \\n`\n              // )\n            }\n            break;\n          }\n          case 'limit': {\n            if (!change.doc) {\n              switch (change.queueType) {\n                case 'dequeue': {\n                  const doc = docs.find(doc => doc._id === change.docId);\n                  if (doc) {\n                    change.doc = doc;\n                  } else {\n                    console.error('[realtime listener] internal non-fatal server error: unexpected limit dataType event where no doc is associated.');\n                  }\n                  break;\n                }\n                case 'enqueue': {\n                  // doc is provided by server, this should never occur\n                  const err = new CloudSDKError({\n                    errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR as string,\n                    errMsg: `HandleServerEvents: full doc is not provided with dataType=\"limit\" and queueType=\"enqueue\" (requestId ${msg.requestId})`,\n                  });\n                  this.closeWithError(err);\n                  throw err;\n                }\n                default: {\n                  break;\n                }\n              }\n            }\n            break;\n          }\n        }\n\n        switch (change.queueType) {\n          case 'init': {\n            if (!initEncountered) {\n              initEncountered = true;\n              docs = [change.doc];\n            } else {\n              docs.push(change.doc);\n            }\n            break;\n          }\n          case 'enqueue': {\n            docs.push(change.doc);\n            break;\n          }\n          case 'dequeue': {\n            const ind = docs.findIndex(doc => doc._id === change.docId);\n            if (ind > -1) {\n              docs.splice(ind, 1);\n            } else {\n              // TODO report\n              console.error('[realtime listener] internal non-fatal server error: unexpected dequeue event where no doc is associated.');\n\n              // writeToFile(\n              //   \"wserror.txt\",\n              //   `[realtime listener] internal non-fatal server error: unexpected dequeue event where no doc is associated. ${JSON.stringify(\n              //     Object.assign(\n              //       {},\n              //       {\n              //         requestId: msg.requestId,\n              //         watchId: msg.watchId\n              //       }\n              //     )\n              //   )} \\n`\n              // )\n            }\n            break;\n          }\n          case 'update': {\n            // writeToFile(\n            //   \"wserror.txt\",\n            //   `[realtime listener] docs ${JSON.stringify(\n            //     docs\n            //   )} change doc ${JSON.stringify(change)} \\n`\n            // )\n            const ind = docs.findIndex(doc => doc._id === change.docId);\n            if (ind > -1) {\n              docs[ind] = change.doc;\n            } else {\n              // TODO report\n              console.error('[realtime listener] internal non-fatal server error: unexpected queueType update event where no doc is associated.');\n\n              // writeToFile(\n              //   \"wserror.txt\",\n              //   `[realtime listener] internal non-fatal server error: unexpected queueType update event where no doc is associated. ${JSON.stringify(\n              //     Object.assign(\n              //       {},\n              //       {\n              //         requestId: msg.requestId,\n              //         watchId: msg.watchId\n              //       }\n              //     )\n              //   )} \\n`\n              // )\n            }\n            break;\n          }\n        }\n\n        if (\n          i === len - 1\n          || (allChangeEvents[i + 1] && allChangeEvents[i + 1].id !== change.id)\n        ) {\n          // a shallow slice creates a shallow snapshot\n          const docsSnapshot = [...docs];\n\n          // we slice first cause' if there're allChangeEvents that are of the same id after this change, we don't want to involve it for it is unexpected invalid order\n          const docChanges = allChangeEvents\n            .slice(0, i + 1)\n            .filter(c => c.id === change.id);\n\n          // all changes of this event has been handle, we could dispatch the event now\n          this.sessionInfo.currentEventId = change.id;\n          this.sessionInfo.currentDocs = docs;\n\n          const snapshot = new Snapshot({\n            id: change.id,\n            docChanges,\n            docs: docsSnapshot,\n            msgType,\n          });\n\n          // Reporter.surroundThirdByTryCatch(() =>\n          this.listener.onChange(snapshot);\n          // )()\n        }\n      } else {\n        // out-of-order event\n        // if (process.env.DEBUG) {\n        // TODO: report\n        console.warn(`[realtime listener] event received is out of order, cur ${this.sessionInfo.currentEventId} but got ${change.id}`);\n        // }\n        // rebuild watch\n        await this.rebuildWatch();\n        return;\n      }\n    }\n  }\n\n  private _postHandleServerEventsValidityCheck(msg: IResponseMessageInitEventMsg | IResponseMessageNextEventMsg) {\n    if (!this.sessionInfo) {\n      console.error('[realtime listener] internal non-fatal error: sessionInfo lost after server event handling, this should never occur');\n\n      // writeToFile(\n      //   \"wserror.txt\",\n      //   `[realtime listener] internal non-fatal error: sessionInfo lost after server event handling, this should never occur ${JSON.stringify(\n      //     Object.assign(\n      //       {},\n      //       {\n      //         requestId: msg.requestId,\n      //         watchId: msg.watchId\n      //       }\n      //     )\n      //   )} \\n`\n      // )\n      return;\n    }\n\n    if (\n      this.sessionInfo.expectEventId\n      && this.sessionInfo.currentEventId >= this.sessionInfo.expectEventId\n    ) {\n      this.clearWaitExpectedEvent();\n    }\n\n    if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {\n      console.warn('[realtime listener] internal non-fatal error: client eventId does not match with server event id after server event handling');\n      return;\n    }\n  }\n\n  private clearWaitExpectedEvent() {\n    if (this._waitExpectedTimeoutId) {\n      clearTimeout(this._waitExpectedTimeoutId);\n      this._waitExpectedTimeoutId = undefined;\n    }\n  }\n}\n\nfunction getPublicEvent(event: IDBEvent): ISingleDBEvent {\n  const e: ISingleDBEvent = {\n    id: event.ID,\n    dataType: event.DataType,\n    queueType: event.QueueType,\n    docId: event.DocID,\n    doc: event.Doc && event.Doc !== '{}' ? JSON.parse(event.Doc) : undefined,\n  };\n\n  if (event.DataType === 'update') {\n    // @ts-ignore\n    if (event.UpdatedFields) {\n      e.updatedFields = JSON.parse(event.UpdatedFields);\n    }\n    // TODO: wait for tcb to change removedFields to RemovedFields\n    // @ts-ignore\n    if (event.removedFields || event.RemovedFields) {\n      // @ts-ignore\n      // e.removedFields = event.removedFields\n      //   ? JSON.parse(event.removedFields)\n      //   : JSON.parse(event.RemovedFields)\n      e.removedFields = JSON.parse(event.removedFields);\n    }\n  }\n\n  return e;\n}\n"]}