@cloudbase/realtime 2.0.2-alpha.0 → 2.5.0-beta.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -47,20 +47,20 @@ import cloneDeep from 'lodash.clonedeep';
47
47
  import { genRequestId } from './message';
48
48
  import { RealtimeListener } from './listener';
49
49
  import { Snapshot } from './snapshot';
50
- import { ERR_CODE, CloudSDKError, isTimeoutError, CancelledError, isCancelledError, isRealtimeErrorMessageError } from './error';
50
+ import { ERR_CODE, CloudSDKError, isTimeoutError, CancelledError, isCancelledError, isRealtimeErrorMessageError, } from './error';
51
51
  import { sleep } from './utils';
52
- var WATCH_STATUS;
53
- (function (WATCH_STATUS) {
54
- WATCH_STATUS["LOGGINGIN"] = "LOGGINGIN";
55
- WATCH_STATUS["INITING"] = "INITING";
56
- WATCH_STATUS["REBUILDING"] = "REBUILDING";
57
- WATCH_STATUS["ACTIVE"] = "ACTIVE";
58
- WATCH_STATUS["ERRORED"] = "ERRORED";
59
- WATCH_STATUS["CLOSING"] = "CLOSING";
60
- WATCH_STATUS["CLOSED"] = "CLOSED";
61
- WATCH_STATUS["PAUSED"] = "PAUSED";
62
- WATCH_STATUS["RESUMING"] = "RESUMING";
63
- })(WATCH_STATUS || (WATCH_STATUS = {}));
52
+ var WatchStatus;
53
+ (function (WatchStatus) {
54
+ WatchStatus["LOGGINGIN"] = "LOGGINGIN";
55
+ WatchStatus["INITING"] = "INITING";
56
+ WatchStatus["REBUILDING"] = "REBUILDING";
57
+ WatchStatus["ACTIVE"] = "ACTIVE";
58
+ WatchStatus["ERRORED"] = "ERRORED";
59
+ WatchStatus["CLOSING"] = "CLOSING";
60
+ WatchStatus["CLOSED"] = "CLOSED";
61
+ WatchStatus["PAUSED"] = "PAUSED";
62
+ WatchStatus["RESUMING"] = "RESUMING";
63
+ })(WatchStatus || (WatchStatus = {}));
64
64
  var DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR = 100;
65
65
  var DEFAULT_MAX_AUTO_RETRY_ON_ERROR = 2;
66
66
  var DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR = 2;
@@ -70,13 +70,13 @@ var DEFAULT_REBUILD_WATCH_TIMEOUT = 10 * 1000;
70
70
  var VirtualWebSocketClient = (function () {
71
71
  function VirtualWebSocketClient(options) {
72
72
  var _this = this;
73
- this.watchStatus = WATCH_STATUS.INITING;
74
- this._login = function (envId, refresh) { return __awaiter(_this, void 0, void 0, function () {
73
+ this.watchStatus = WatchStatus.INITING;
74
+ this.wsLogin = function (envId, refresh) { return __awaiter(_this, void 0, void 0, function () {
75
75
  var loginResult;
76
76
  return __generator(this, function (_a) {
77
77
  switch (_a.label) {
78
78
  case 0:
79
- this.watchStatus = WATCH_STATUS.LOGGINGIN;
79
+ this.watchStatus = WatchStatus.LOGGINGIN;
80
80
  return [4, this.login(envId, refresh)];
81
81
  case 1:
82
82
  loginResult = _a.sent();
@@ -93,99 +93,101 @@ var VirtualWebSocketClient = (function () {
93
93
  return __generator(this, function (_a) {
94
94
  switch (_a.label) {
95
95
  case 0:
96
- if (this._initWatchPromise) {
97
- return [2, this._initWatchPromise];
96
+ if (this.initWatchPromise !== null && this.initWatchPromise !== undefined) {
97
+ return [2, this.initWatchPromise];
98
98
  }
99
- this._initWatchPromise = new Promise(function (resolve, reject) { return __awaiter(_this, void 0, void 0, function () {
100
- var envId, initWatchMsg, initEventMsg, _a, events, currEvent, _i, events_1, e, snapshot, e_1;
101
- return __generator(this, function (_b) {
102
- switch (_b.label) {
103
- case 0:
104
- _b.trys.push([0, 3, , 4]);
105
- if (this.watchStatus === WATCH_STATUS.PAUSED) {
106
- console.log('[realtime] initWatch cancelled on pause');
107
- return [2, resolve()];
108
- }
109
- return [4, this._login(this.envId, forceRefreshLogin)];
110
- case 1:
111
- envId = (_b.sent()).envId;
112
- if (this.watchStatus === WATCH_STATUS.PAUSED) {
113
- console.log('[realtime] initWatch cancelled on pause');
114
- return [2, resolve()];
115
- }
116
- this.watchStatus = WATCH_STATUS.INITING;
117
- initWatchMsg = {
118
- watchId: this.watchId,
119
- requestId: genRequestId(),
120
- msgType: 'INIT_WATCH',
121
- msgData: {
122
- envId: envId,
123
- collName: this.collectionName,
124
- query: this.query,
125
- limit: this.limit,
126
- orderBy: this.orderBy
99
+ this.initWatchPromise = new Promise(function (resolve, reject) {
100
+ void (function () { return __awaiter(_this, void 0, void 0, function () {
101
+ var envId, initWatchMsg, initEventMsg, _a, events, currEvent, _i, events_1, e, snapshot, e_1;
102
+ return __generator(this, function (_b) {
103
+ switch (_b.label) {
104
+ case 0:
105
+ _b.trys.push([0, 3, , 4]);
106
+ if (this.watchStatus === WatchStatus.PAUSED) {
107
+ console.log('[realtime] initWatch cancelled on pause');
108
+ return [2, resolve()];
127
109
  }
128
- };
129
- return [4, this.send({
130
- msg: initWatchMsg,
131
- waitResponse: true,
132
- skipOnMessage: true,
133
- timeout: DEFAULT_INIT_WATCH_TIMEOUT
134
- })];
135
- case 2:
136
- initEventMsg = _b.sent();
137
- _a = initEventMsg.msgData, events = _a.events, currEvent = _a.currEvent;
138
- this.sessionInfo = {
139
- queryID: initEventMsg.msgData.queryID,
140
- currentEventId: currEvent - 1,
141
- currentDocs: []
142
- };
143
- if (events.length > 0) {
144
- for (_i = 0, events_1 = events; _i < events_1.length; _i++) {
145
- e = events_1[_i];
146
- e.ID = currEvent;
110
+ return [4, this.wsLogin(this.envId, forceRefreshLogin)];
111
+ case 1:
112
+ envId = (_b.sent()).envId;
113
+ if (this.watchStatus === WatchStatus.PAUSED) {
114
+ console.log('[realtime] initWatch cancelled on pause');
115
+ return [2, resolve()];
147
116
  }
148
- this.handleServerEvents(initEventMsg);
149
- }
150
- else {
151
- this.sessionInfo.currentEventId = currEvent;
152
- snapshot = new Snapshot({
153
- id: currEvent,
154
- docChanges: [],
155
- docs: [],
156
- type: 'init'
117
+ this.watchStatus = WatchStatus.INITING;
118
+ initWatchMsg = {
119
+ watchId: this.watchId,
120
+ requestId: genRequestId(),
121
+ msgType: 'INIT_WATCH',
122
+ msgData: {
123
+ envId: envId,
124
+ collName: this.collectionName,
125
+ query: this.query,
126
+ limit: this.limit,
127
+ orderBy: this.orderBy,
128
+ },
129
+ };
130
+ return [4, this.send({
131
+ msg: initWatchMsg,
132
+ waitResponse: true,
133
+ skipOnMessage: true,
134
+ timeout: DEFAULT_INIT_WATCH_TIMEOUT,
135
+ })];
136
+ case 2:
137
+ initEventMsg = _b.sent();
138
+ _a = initEventMsg.msgData, events = _a.events, currEvent = _a.currEvent;
139
+ this.sessionInfo = {
140
+ queryID: initEventMsg.msgData.queryID,
141
+ currentEventId: currEvent - 1,
142
+ currentDocs: [],
143
+ };
144
+ if (events.length > 0) {
145
+ for (_i = 0, events_1 = events; _i < events_1.length; _i++) {
146
+ e = events_1[_i];
147
+ e.ID = currEvent;
148
+ }
149
+ this.handleServerEvents(initEventMsg);
150
+ }
151
+ else {
152
+ this.sessionInfo.currentEventId = currEvent;
153
+ snapshot = new Snapshot({
154
+ id: currEvent,
155
+ docChanges: [],
156
+ docs: [],
157
+ type: 'init',
158
+ });
159
+ this.listener.onChange(snapshot);
160
+ this.scheduleSendACK();
161
+ }
162
+ this.onWatchStart(this, this.sessionInfo.queryID);
163
+ this.watchStatus = WatchStatus.ACTIVE;
164
+ this.availableRetries.INIT_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;
165
+ resolve();
166
+ return [3, 4];
167
+ case 3:
168
+ e_1 = _b.sent();
169
+ this.handleWatchEstablishmentError(e_1, {
170
+ operationName: 'INIT_WATCH',
171
+ resolve: resolve,
172
+ reject: reject,
157
173
  });
158
- this.listener.onChange(snapshot);
159
- this.scheduleSendACK();
160
- }
161
- this.onWatchStart(this, this.sessionInfo.queryID);
162
- this.watchStatus = WATCH_STATUS.ACTIVE;
163
- this._availableRetries.INIT_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;
164
- resolve();
165
- return [3, 4];
166
- case 3:
167
- e_1 = _b.sent();
168
- this.handleWatchEstablishmentError(e_1, {
169
- operationName: 'INIT_WATCH',
170
- resolve: resolve,
171
- reject: reject
172
- });
173
- return [3, 4];
174
- case 4: return [2];
175
- }
176
- });
177
- }); });
174
+ return [3, 4];
175
+ case 4: return [2];
176
+ }
177
+ });
178
+ }); })();
179
+ });
178
180
  success = false;
179
181
  _a.label = 1;
180
182
  case 1:
181
183
  _a.trys.push([1, , 3, 4]);
182
- return [4, this._initWatchPromise];
184
+ return [4, this.initWatchPromise];
183
185
  case 2:
184
186
  _a.sent();
185
187
  success = true;
186
188
  return [3, 4];
187
189
  case 3:
188
- this._initWatchPromise = undefined;
190
+ this.initWatchPromise = undefined;
189
191
  return [7];
190
192
  case 4:
191
193
  console.log("[realtime] initWatch " + (success ? 'success' : 'fail'));
@@ -199,77 +201,79 @@ var VirtualWebSocketClient = (function () {
199
201
  return __generator(this, function (_a) {
200
202
  switch (_a.label) {
201
203
  case 0:
202
- if (this._rebuildWatchPromise) {
203
- return [2, this._rebuildWatchPromise];
204
+ if (this.rebuildWatchPromise !== null && this.rebuildWatchPromise !== undefined) {
205
+ return [2, this.rebuildWatchPromise];
204
206
  }
205
- this._rebuildWatchPromise = new Promise(function (resolve, reject) { return __awaiter(_this, void 0, void 0, function () {
206
- var envId, rebuildWatchMsg, nextEventMsg, e_2;
207
- return __generator(this, function (_a) {
208
- switch (_a.label) {
209
- case 0:
210
- _a.trys.push([0, 3, , 4]);
211
- if (this.watchStatus === WATCH_STATUS.PAUSED) {
212
- console.log('[realtime] rebuildWatch cancelled on pause');
213
- return [2, resolve()];
214
- }
215
- return [4, this._login(this.envId, forceRefreshLogin)];
216
- case 1:
217
- envId = (_a.sent()).envId;
218
- if (!this.sessionInfo) {
219
- throw new Error('can not rebuildWatch without a successful initWatch (lack of sessionInfo)');
220
- }
221
- if (this.watchStatus === WATCH_STATUS.PAUSED) {
222
- console.log('[realtime] rebuildWatch cancelled on pause');
223
- return [2, resolve()];
224
- }
225
- this.watchStatus = WATCH_STATUS.REBUILDING;
226
- rebuildWatchMsg = {
227
- watchId: this.watchId,
228
- requestId: genRequestId(),
229
- msgType: 'REBUILD_WATCH',
230
- msgData: {
231
- envId: envId,
232
- collName: this.collectionName,
233
- queryID: this.sessionInfo.queryID,
234
- eventID: this.sessionInfo.currentEventId
207
+ this.rebuildWatchPromise = new Promise(function (resolve, reject) {
208
+ void (function () { return __awaiter(_this, void 0, void 0, function () {
209
+ var envId, rebuildWatchMsg, nextEventMsg, e_2;
210
+ return __generator(this, function (_a) {
211
+ switch (_a.label) {
212
+ case 0:
213
+ _a.trys.push([0, 3, , 4]);
214
+ if (this.watchStatus === WatchStatus.PAUSED) {
215
+ console.log('[realtime] rebuildWatch cancelled on pause');
216
+ return [2, resolve()];
235
217
  }
236
- };
237
- return [4, this.send({
238
- msg: rebuildWatchMsg,
239
- waitResponse: true,
240
- skipOnMessage: false,
241
- timeout: DEFAULT_REBUILD_WATCH_TIMEOUT
242
- })];
243
- case 2:
244
- nextEventMsg = _a.sent();
245
- this.handleServerEvents(nextEventMsg);
246
- this.watchStatus = WATCH_STATUS.ACTIVE;
247
- this._availableRetries.REBUILD_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;
248
- resolve();
249
- return [3, 4];
250
- case 3:
251
- e_2 = _a.sent();
252
- this.handleWatchEstablishmentError(e_2, {
253
- operationName: 'REBUILD_WATCH',
254
- resolve: resolve,
255
- reject: reject
256
- });
257
- return [3, 4];
258
- case 4: return [2];
259
- }
260
- });
261
- }); });
218
+ return [4, this.wsLogin(this.envId, forceRefreshLogin)];
219
+ case 1:
220
+ envId = (_a.sent()).envId;
221
+ if (!this.sessionInfo) {
222
+ throw new Error('can not rebuildWatch without a successful initWatch (lack of sessionInfo)');
223
+ }
224
+ if (this.watchStatus === WatchStatus.PAUSED) {
225
+ console.log('[realtime] rebuildWatch cancelled on pause');
226
+ return [2, resolve()];
227
+ }
228
+ this.watchStatus = WatchStatus.REBUILDING;
229
+ rebuildWatchMsg = {
230
+ watchId: this.watchId,
231
+ requestId: genRequestId(),
232
+ msgType: 'REBUILD_WATCH',
233
+ msgData: {
234
+ envId: envId,
235
+ collName: this.collectionName,
236
+ queryID: this.sessionInfo.queryID,
237
+ eventID: this.sessionInfo.currentEventId,
238
+ },
239
+ };
240
+ return [4, this.send({
241
+ msg: rebuildWatchMsg,
242
+ waitResponse: true,
243
+ skipOnMessage: false,
244
+ timeout: DEFAULT_REBUILD_WATCH_TIMEOUT,
245
+ })];
246
+ case 2:
247
+ nextEventMsg = _a.sent();
248
+ this.handleServerEvents(nextEventMsg);
249
+ this.watchStatus = WatchStatus.ACTIVE;
250
+ this.availableRetries.REBUILD_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;
251
+ resolve();
252
+ return [3, 4];
253
+ case 3:
254
+ e_2 = _a.sent();
255
+ this.handleWatchEstablishmentError(e_2, {
256
+ operationName: 'REBUILD_WATCH',
257
+ resolve: resolve,
258
+ reject: reject,
259
+ });
260
+ return [3, 4];
261
+ case 4: return [2];
262
+ }
263
+ });
264
+ }); })();
265
+ });
262
266
  success = false;
263
267
  _a.label = 1;
264
268
  case 1:
265
269
  _a.trys.push([1, , 3, 4]);
266
- return [4, this._rebuildWatchPromise];
270
+ return [4, this.rebuildWatchPromise];
267
271
  case 2:
268
272
  _a.sent();
269
273
  success = true;
270
274
  return [3, 4];
271
275
  case 3:
272
- this._rebuildWatchPromise = undefined;
276
+ this.rebuildWatchPromise = undefined;
273
277
  return [7];
274
278
  case 4:
275
279
  console.log("[realtime] rebuildWatch " + (success ? 'success' : 'fail'));
@@ -287,18 +291,18 @@ var VirtualWebSocketClient = (function () {
287
291
  errCode: isInitWatch
288
292
  ? ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_INIT_WATCH_FAIL
289
293
  : ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_REBUILD_WATCH_FAIL,
290
- errMsg: e
294
+ errMsg: e,
291
295
  }));
292
296
  options.reject(e);
293
297
  };
294
298
  retry = function (refreshLogin) {
295
299
  if (_this.useRetryTicket(options.operationName)) {
296
300
  if (isInitWatch) {
297
- _this._initWatchPromise = undefined;
301
+ _this.initWatchPromise = undefined;
298
302
  options.resolve(_this.initWatch(refreshLogin));
299
303
  }
300
304
  else {
301
- _this._rebuildWatchPromise = undefined;
305
+ _this.rebuildWatchPromise = undefined;
302
306
  options.resolve(_this.rebuildWatch(refreshLogin));
303
307
  }
304
308
  }
@@ -311,55 +315,57 @@ var VirtualWebSocketClient = (function () {
311
315
  onTimeoutError: function () { return retry(false); },
312
316
  onNotRetryableError: abortWatch,
313
317
  onCancelledError: options.reject,
314
- onUnknownError: function () { return __awaiter(_this, void 0, void 0, function () {
315
- var onWSDisconnected, e_3;
316
- var _this = this;
317
- return __generator(this, function (_a) {
318
- switch (_a.label) {
319
- case 0:
320
- _a.trys.push([0, 8, , 9]);
321
- onWSDisconnected = function () { return __awaiter(_this, void 0, void 0, function () {
322
- return __generator(this, function (_a) {
323
- switch (_a.label) {
324
- case 0:
325
- this.pause();
326
- return [4, this.onceWSConnected()];
327
- case 1:
328
- _a.sent();
329
- retry(true);
330
- return [2];
331
- }
332
- });
333
- }); };
334
- if (!!this.isWSConnected()) return [3, 2];
335
- return [4, onWSDisconnected()];
336
- case 1:
337
- _a.sent();
338
- return [3, 7];
339
- case 2: return [4, sleep(DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR)];
340
- case 3:
341
- _a.sent();
342
- if (!(this.watchStatus === WATCH_STATUS.PAUSED)) return [3, 4];
343
- options.reject(new CancelledError(options.operationName + " cancelled due to pause after unknownError"));
344
- return [3, 7];
345
- case 4:
346
- if (!!this.isWSConnected()) return [3, 6];
347
- return [4, onWSDisconnected()];
348
- case 5:
349
- _a.sent();
350
- return [3, 7];
351
- case 6:
352
- retry(false);
353
- _a.label = 7;
354
- case 7: return [3, 9];
355
- case 8:
356
- e_3 = _a.sent();
357
- retry(true);
358
- return [3, 9];
359
- case 9: return [2];
360
- }
361
- });
362
- }); }
318
+ onUnknownError: function () {
319
+ (function () { return __awaiter(_this, void 0, void 0, function () {
320
+ var onWSDisconnected, e_3;
321
+ var _this = this;
322
+ return __generator(this, function (_a) {
323
+ switch (_a.label) {
324
+ case 0:
325
+ _a.trys.push([0, 8, , 9]);
326
+ onWSDisconnected = function () { return __awaiter(_this, void 0, void 0, function () {
327
+ return __generator(this, function (_a) {
328
+ switch (_a.label) {
329
+ case 0:
330
+ this.pause();
331
+ return [4, this.onceWSConnected()];
332
+ case 1:
333
+ _a.sent();
334
+ retry(true);
335
+ return [2];
336
+ }
337
+ });
338
+ }); };
339
+ if (!!this.isWSConnected()) return [3, 2];
340
+ return [4, onWSDisconnected()];
341
+ case 1:
342
+ _a.sent();
343
+ return [3, 7];
344
+ case 2: return [4, sleep(DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR)];
345
+ case 3:
346
+ _a.sent();
347
+ if (!(this.watchStatus === WatchStatus.PAUSED)) return [3, 4];
348
+ options.reject(new CancelledError(options.operationName + " cancelled due to pause after unknownError"));
349
+ return [3, 7];
350
+ case 4:
351
+ if (!!this.isWSConnected()) return [3, 6];
352
+ return [4, onWSDisconnected()];
353
+ case 5:
354
+ _a.sent();
355
+ return [3, 7];
356
+ case 6:
357
+ retry(false);
358
+ _a.label = 7;
359
+ case 7: return [3, 9];
360
+ case 8:
361
+ e_3 = _a.sent();
362
+ retry(true);
363
+ return [3, 9];
364
+ case 9: return [2];
365
+ }
366
+ });
367
+ }); })();
368
+ },
363
369
  });
364
370
  return [2];
365
371
  });
@@ -370,34 +376,34 @@ var VirtualWebSocketClient = (function () {
370
376
  switch (_a.label) {
371
377
  case 0:
372
378
  queryId = this.sessionInfo ? this.sessionInfo.queryID : '';
373
- if (this.watchStatus !== WATCH_STATUS.ACTIVE) {
374
- this.watchStatus = WATCH_STATUS.CLOSED;
379
+ if (this.watchStatus !== WatchStatus.ACTIVE) {
380
+ this.watchStatus = WatchStatus.CLOSED;
375
381
  this.onWatchClose(this, queryId);
376
382
  return [2];
377
383
  }
378
384
  _a.label = 1;
379
385
  case 1:
380
386
  _a.trys.push([1, 3, 4, 5]);
381
- this.watchStatus = WATCH_STATUS.CLOSING;
387
+ this.watchStatus = WatchStatus.CLOSING;
382
388
  closeWatchMsg = {
383
389
  watchId: this.watchId,
384
390
  requestId: genRequestId(),
385
391
  msgType: 'CLOSE_WATCH',
386
- msgData: null
392
+ msgData: null,
387
393
  };
388
394
  return [4, this.send({
389
- msg: closeWatchMsg
395
+ msg: closeWatchMsg,
390
396
  })];
391
397
  case 2:
392
398
  _a.sent();
393
399
  this.sessionInfo = undefined;
394
- this.watchStatus = WATCH_STATUS.CLOSED;
400
+ this.watchStatus = WatchStatus.CLOSED;
395
401
  return [3, 5];
396
402
  case 3:
397
403
  e_4 = _a.sent();
398
404
  this.closeWithError(new CloudSDKError({
399
405
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CLOSE_WATCH_FAIL,
400
- errMsg: e_4
406
+ errMsg: e_4,
401
407
  }));
402
408
  return [3, 5];
403
409
  case 4:
@@ -409,8 +415,8 @@ var VirtualWebSocketClient = (function () {
409
415
  }); };
410
416
  this.scheduleSendACK = function () {
411
417
  _this.clearACKSchedule();
412
- _this._ackTimeoutId = setTimeout(function () {
413
- if (_this._waitExpectedTimeoutId) {
418
+ _this.ackTimeoutId = setTimeout(function () {
419
+ if (_this.waitExpectedTimeoutId) {
414
420
  _this.scheduleSendACK();
415
421
  }
416
422
  else {
@@ -419,8 +425,8 @@ var VirtualWebSocketClient = (function () {
419
425
  }, DEFAULT_SEND_ACK_DEBOUNCE_TIMEOUT);
420
426
  };
421
427
  this.clearACKSchedule = function () {
422
- if (_this._ackTimeoutId) {
423
- clearTimeout(_this._ackTimeoutId);
428
+ if (_this.ackTimeoutId) {
429
+ clearTimeout(_this.ackTimeoutId);
424
430
  }
425
431
  };
426
432
  this.sendACK = function () { return __awaiter(_this, void 0, void 0, function () {
@@ -429,7 +435,7 @@ var VirtualWebSocketClient = (function () {
429
435
  switch (_a.label) {
430
436
  case 0:
431
437
  _a.trys.push([0, 2, , 3]);
432
- if (this.watchStatus !== WATCH_STATUS.ACTIVE) {
438
+ if (this.watchStatus !== WatchStatus.ACTIVE) {
433
439
  this.scheduleSendACK();
434
440
  return [2];
435
441
  }
@@ -443,11 +449,11 @@ var VirtualWebSocketClient = (function () {
443
449
  msgType: 'CHECK_LAST',
444
450
  msgData: {
445
451
  queryID: this.sessionInfo.queryID,
446
- eventID: this.sessionInfo.currentEventId
447
- }
452
+ eventID: this.sessionInfo.currentEventId,
453
+ },
448
454
  };
449
455
  return [4, this.send({
450
- msg: ackMsg
456
+ msg: ackMsg,
451
457
  })];
452
458
  case 1:
453
459
  _a.sent();
@@ -471,7 +477,7 @@ var VirtualWebSocketClient = (function () {
471
477
  case 'COLLECTION_PERMISSION_DENIED': {
472
478
  this.closeWithError(new CloudSDKError({
473
479
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL,
474
- errMsg: msg.msgData.code
480
+ errMsg: msg.msgData.code,
475
481
  }));
476
482
  return [2];
477
483
  }
@@ -480,15 +486,15 @@ var VirtualWebSocketClient = (function () {
480
486
  }
481
487
  }
482
488
  }
483
- if (this._availableRetries.CHECK_LAST &&
484
- this._availableRetries.CHECK_LAST > 0) {
485
- this._availableRetries.CHECK_LAST--;
489
+ if (this.availableRetries.CHECK_LAST
490
+ && this.availableRetries.CHECK_LAST > 0) {
491
+ this.availableRetries.CHECK_LAST -= 1;
486
492
  this.scheduleSendACK();
487
493
  }
488
494
  else {
489
495
  this.closeWithError(new CloudSDKError({
490
496
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL,
491
- errMsg: e_5
497
+ errMsg: e_5,
492
498
  }));
493
499
  }
494
500
  return [3, 3];
@@ -544,40 +550,42 @@ var VirtualWebSocketClient = (function () {
544
550
  this.onWatchStart = options.onWatchStart;
545
551
  this.onWatchClose = options.onWatchClose;
546
552
  this.debug = options.debug;
547
- this._availableRetries = {
553
+ this.availableRetries = {
548
554
  INIT_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,
549
555
  REBUILD_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,
550
- CHECK_LAST: DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR
556
+ CHECK_LAST: DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR,
551
557
  };
552
558
  this.listener = new RealtimeListener({
553
- close: this.closeWatch,
559
+ close: void (function () {
560
+ _this.closeWatch();
561
+ }),
554
562
  onChange: options.onChange,
555
563
  onError: options.onError,
556
564
  debug: this.debug,
557
- virtualClient: this
565
+ virtualClient: this,
558
566
  });
559
567
  this.initWatch();
560
568
  }
561
569
  VirtualWebSocketClient.prototype.onMessage = function (msg) {
562
570
  var _this = this;
563
571
  switch (this.watchStatus) {
564
- case WATCH_STATUS.PAUSED: {
572
+ case WatchStatus.PAUSED: {
565
573
  if (msg.msgType !== 'ERROR') {
566
574
  return;
567
575
  }
568
576
  break;
569
577
  }
570
- case WATCH_STATUS.LOGGINGIN:
571
- case WATCH_STATUS.INITING:
572
- case WATCH_STATUS.REBUILDING: {
578
+ case WatchStatus.LOGGINGIN:
579
+ case WatchStatus.INITING:
580
+ case WatchStatus.REBUILDING: {
573
581
  console.warn("[realtime listener] internal non-fatal error: unexpected message received while " + this.watchStatus);
574
582
  return;
575
583
  }
576
- case WATCH_STATUS.CLOSED: {
584
+ case WatchStatus.CLOSED: {
577
585
  console.warn('[realtime listener] internal non-fatal error: unexpected message received when the watch has closed');
578
586
  return;
579
587
  }
580
- case WATCH_STATUS.ERRORED: {
588
+ case WatchStatus.ERRORED: {
581
589
  console.warn('[realtime listener] internal non-fatal error: unexpected message received when the watch has ended with error');
582
590
  return;
583
591
  }
@@ -597,7 +605,7 @@ var VirtualWebSocketClient = (function () {
597
605
  if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {
598
606
  this.sessionInfo.expectEventId = msg.msgData.currEvent;
599
607
  this.clearWaitExpectedEvent();
600
- this._waitExpectedTimeoutId = setTimeout(function () {
608
+ this.waitExpectedTimeoutId = setTimeout(function () {
601
609
  _this.rebuildWatch();
602
610
  }, this.getWaitExpectedTimeoutLength());
603
611
  console.log("[realtime] waitExpectedTimeoutLength " + this.getWaitExpectedTimeoutLength());
@@ -607,7 +615,7 @@ var VirtualWebSocketClient = (function () {
607
615
  case 'ERROR': {
608
616
  this.closeWithError(new CloudSDKError({
609
617
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_SERVER_ERROR_MSG,
610
- errMsg: msg.msgData.code + " - " + msg.msgData.message
618
+ errMsg: msg.msgData.code + " - " + msg.msgData.message,
611
619
  }));
612
620
  break;
613
621
  }
@@ -618,14 +626,15 @@ var VirtualWebSocketClient = (function () {
618
626
  }
619
627
  };
620
628
  VirtualWebSocketClient.prototype.closeWithError = function (error) {
621
- this.watchStatus = WATCH_STATUS.ERRORED;
629
+ var _a;
630
+ this.watchStatus = WatchStatus.ERRORED;
622
631
  this.clearACKSchedule();
623
632
  this.listener.onError(error);
624
- this.onWatchClose(this, (this.sessionInfo && this.sessionInfo.queryID) || '');
633
+ this.onWatchClose(this, ((_a = this.sessionInfo) === null || _a === void 0 ? void 0 : _a.queryID) || '');
625
634
  console.log("[realtime] client closed (" + this.collectionName + " " + this.query + ") (watchId " + this.watchId + ")");
626
635
  };
627
636
  VirtualWebSocketClient.prototype.pause = function () {
628
- this.watchStatus = WATCH_STATUS.PAUSED;
637
+ this.watchStatus = WatchStatus.PAUSED;
629
638
  console.log("[realtime] client paused (" + this.collectionName + " " + this.query + ") (watchId " + this.watchId + ")");
630
639
  };
631
640
  VirtualWebSocketClient.prototype.resume = function () {
@@ -634,7 +643,7 @@ var VirtualWebSocketClient = (function () {
634
643
  return __generator(this, function (_a) {
635
644
  switch (_a.label) {
636
645
  case 0:
637
- this.watchStatus = WATCH_STATUS.RESUMING;
646
+ this.watchStatus = WatchStatus.RESUMING;
638
647
  console.log("[realtime] client resuming with " + (this.sessionInfo ? 'REBUILD_WATCH' : 'INIT_WATCH') + " (" + this.collectionName + " " + this.query + ") (" + this.watchId + ")");
639
648
  _a.label = 1;
640
649
  case 1:
@@ -654,10 +663,10 @@ var VirtualWebSocketClient = (function () {
654
663
  });
655
664
  };
656
665
  VirtualWebSocketClient.prototype.useRetryTicket = function (operationName) {
657
- if (this._availableRetries[operationName] &&
658
- this._availableRetries[operationName] > 0) {
659
- this._availableRetries[operationName]--;
660
- console.log("[realtime] " + operationName + " use a retry ticket, now only " + this._availableRetries[operationName] + " retry left");
666
+ if (this.availableRetries[operationName]
667
+ && this.availableRetries[operationName] > 0) {
668
+ this.availableRetries[operationName] -= 1;
669
+ console.log("[realtime] " + operationName + " use a retry ticket, now only " + this.availableRetries[operationName] + " retry left");
661
670
  return true;
662
671
  }
663
672
  return false;
@@ -670,10 +679,10 @@ var VirtualWebSocketClient = (function () {
670
679
  case 0:
671
680
  _a.trys.push([0, 2, , 3]);
672
681
  this.scheduleSendACK();
673
- return [4, this._handleServerEvents(msg)];
682
+ return [4, this.handleServerEventsInternel(msg)];
674
683
  case 1:
675
684
  _a.sent();
676
- this._postHandleServerEventsValidityCheck(msg);
685
+ this.postHandleServerEventsValidityCheck(msg);
677
686
  return [3, 3];
678
687
  case 2:
679
688
  e_7 = _a.sent();
@@ -684,7 +693,7 @@ var VirtualWebSocketClient = (function () {
684
693
  });
685
694
  });
686
695
  };
687
- VirtualWebSocketClient.prototype._handleServerEvents = function (msg) {
696
+ VirtualWebSocketClient.prototype.handleServerEventsInternel = function (msg) {
688
697
  return __awaiter(this, void 0, void 0, function () {
689
698
  var requestId, events, msgType, sessionInfo, allChangeEvents, docs, initEncountered, _loop_1, this_1, i, len, state_1;
690
699
  return __generator(this, function (_a) {
@@ -703,14 +712,14 @@ var VirtualWebSocketClient = (function () {
703
712
  catch (e) {
704
713
  this.closeWithError(new CloudSDKError({
705
714
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_RECEIVE_INVALID_SERVER_DATA,
706
- errMsg: e
715
+ errMsg: e,
707
716
  }));
708
717
  return [2];
709
718
  }
710
719
  docs = __spreadArrays(sessionInfo.currentDocs);
711
720
  initEncountered = false;
712
721
  _loop_1 = function (i, len) {
713
- var change, localDoc, doc, fieldPath, _i, _a, fieldPath, err, err, doc, doc, err, ind, ind, docsSnapshot, docChanges, snapshot;
722
+ var change, localDoc, doc_1, _i, _a, fieldPath, err, err, doc, doc, err, ind, ind, docsSnapshot, docChanges, snapshot;
714
723
  return __generator(this, function (_b) {
715
724
  switch (_b.label) {
716
725
  case 0:
@@ -733,19 +742,19 @@ var VirtualWebSocketClient = (function () {
733
742
  case 'dequeue': {
734
743
  localDoc = docs.find(function (doc) { return doc._id === change.docId; });
735
744
  if (localDoc) {
736
- doc = cloneDeep(localDoc);
745
+ doc_1 = cloneDeep(localDoc);
737
746
  if (change.updatedFields) {
738
- for (fieldPath in change.updatedFields) {
739
- set(doc, fieldPath, change.updatedFields[fieldPath]);
740
- }
747
+ Object.keys(change.updatedFields).forEach(function (fieldPath) {
748
+ set(doc_1, fieldPath, change.updatedFields[fieldPath]);
749
+ });
741
750
  }
742
751
  if (change.removedFields) {
743
752
  for (_i = 0, _a = change.removedFields; _i < _a.length; _i++) {
744
753
  fieldPath = _a[_i];
745
- unset(doc, fieldPath);
754
+ unset(doc_1, fieldPath);
746
755
  }
747
756
  }
748
- change.doc = doc;
757
+ change.doc = doc_1;
749
758
  }
750
759
  else {
751
760
  console.error('[realtime listener] internal non-fatal server error: unexpected update dataType event where no doc is associated.');
@@ -755,7 +764,7 @@ var VirtualWebSocketClient = (function () {
755
764
  case 'enqueue': {
756
765
  err = new CloudSDKError({
757
766
  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 + ")"
767
+ errMsg: "HandleServerEvents: full doc is not provided with dataType=\"update\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")",
759
768
  });
760
769
  this_1.closeWithError(err);
761
770
  throw err;
@@ -771,7 +780,7 @@ var VirtualWebSocketClient = (function () {
771
780
  if (!change.doc) {
772
781
  err = new CloudSDKError({
773
782
  errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR,
774
- errMsg: "HandleServerEvents: full doc is not provided with dataType=\"replace\" (requestId " + msg.requestId + ")"
783
+ errMsg: "HandleServerEvents: full doc is not provided with dataType=\"replace\" (requestId " + msg.requestId + ")",
775
784
  });
776
785
  this_1.closeWithError(err);
777
786
  throw err;
@@ -804,7 +813,7 @@ var VirtualWebSocketClient = (function () {
804
813
  case 'enqueue': {
805
814
  err = new CloudSDKError({
806
815
  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 + ")"
816
+ errMsg: "HandleServerEvents: full doc is not provided with dataType=\"limit\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")",
808
817
  });
809
818
  this_1.closeWithError(err);
810
819
  throw err;
@@ -853,8 +862,8 @@ var VirtualWebSocketClient = (function () {
853
862
  break;
854
863
  }
855
864
  }
856
- if (i === len - 1 ||
857
- (allChangeEvents[i + 1] && allChangeEvents[i + 1].id !== change.id)) {
865
+ if (i === len - 1
866
+ || (allChangeEvents[i + 1] && allChangeEvents[i + 1].id !== change.id)) {
858
867
  docsSnapshot = __spreadArrays(docs);
859
868
  docChanges = allChangeEvents
860
869
  .slice(0, i + 1)
@@ -865,7 +874,7 @@ var VirtualWebSocketClient = (function () {
865
874
  id: change.id,
866
875
  docChanges: docChanges,
867
876
  docs: docsSnapshot,
868
- msgType: msgType
877
+ msgType: msgType,
869
878
  });
870
879
  this_1.listener.onChange(snapshot);
871
880
  }
@@ -899,13 +908,13 @@ var VirtualWebSocketClient = (function () {
899
908
  });
900
909
  });
901
910
  };
902
- VirtualWebSocketClient.prototype._postHandleServerEventsValidityCheck = function (msg) {
911
+ VirtualWebSocketClient.prototype.postHandleServerEventsValidityCheck = function (msg) {
903
912
  if (!this.sessionInfo) {
904
913
  console.error('[realtime listener] internal non-fatal error: sessionInfo lost after server event handling, this should never occur');
905
914
  return;
906
915
  }
907
- if (this.sessionInfo.expectEventId &&
908
- this.sessionInfo.currentEventId >= this.sessionInfo.expectEventId) {
916
+ if (this.sessionInfo.expectEventId
917
+ && this.sessionInfo.currentEventId >= this.sessionInfo.expectEventId) {
909
918
  this.clearWaitExpectedEvent();
910
919
  }
911
920
  if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {
@@ -914,9 +923,9 @@ var VirtualWebSocketClient = (function () {
914
923
  }
915
924
  };
916
925
  VirtualWebSocketClient.prototype.clearWaitExpectedEvent = function () {
917
- if (this._waitExpectedTimeoutId) {
918
- clearTimeout(this._waitExpectedTimeoutId);
919
- this._waitExpectedTimeoutId = undefined;
926
+ if (this.waitExpectedTimeoutId) {
927
+ clearTimeout(this.waitExpectedTimeoutId);
928
+ this.waitExpectedTimeoutId = undefined;
920
929
  }
921
930
  };
922
931
  return VirtualWebSocketClient;
@@ -928,7 +937,7 @@ function getPublicEvent(event) {
928
937
  dataType: event.DataType,
929
938
  queueType: event.QueueType,
930
939
  docId: event.DocID,
931
- doc: event.Doc && event.Doc !== '{}' ? JSON.parse(event.Doc) : undefined
940
+ doc: event.Doc && event.Doc !== '{}' ? JSON.parse(event.Doc) : undefined,
932
941
  };
933
942
  if (event.DataType === 'update') {
934
943
  if (event.UpdatedFields) {
@@ -940,4 +949,4 @@ function getPublicEvent(event) {
940
949
  }
941
950
  return e;
942
951
  }
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"]}
952
+ //# 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'\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  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}\n\nenum WatchStatus {\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: WatchStatus = WatchStatus.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: void (() => {\n        this.closeWatch()\n      }),\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 WatchStatus.PAUSED: {\n        // ignore all but error message\n        if (msg.msgType !== 'ERROR') {\n          return\n        }\n        break\n      }\n      case WatchStatus.LOGGINGIN:\n      case WatchStatus.INITING:\n      case WatchStatus.REBUILDING: {\n        console.warn(`[realtime listener] internal non-fatal error: unexpected message received while ${this.watchStatus}`)\n        return\n      }\n      case WatchStatus.CLOSED: {\n        console.warn('[realtime listener] internal non-fatal error: unexpected message received when the watch has closed')\n        return\n      }\n      case WatchStatus.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          console.log(`[realtime] waitExpectedTimeoutLength ${this.getWaitExpectedTimeoutLength()}`)\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        console.warn(\n          `[realtime listener] virtual client receive unexpected msg ${msg.msgType}: `,\n          msg\n        )\n        break\n      }\n    }\n  }\n\n  closeWithError(error: any) {\n    this.watchStatus = WatchStatus.ERRORED\n    this.clearACKSchedule()\n    this.listener.onError(error)\n    this.onWatchClose(\n      this,\n      (this.sessionInfo?.queryID) || ''\n    )\n\n    console.log(`[realtime] client closed (${this.collectionName} ${this.query}) (watchId ${this.watchId})`)\n  }\n\n  pause() {\n    this.watchStatus = WatchStatus.PAUSED\n    console.log(`[realtime] client paused (${this.collectionName} ${this.query}) (watchId ${this.watchId})`)\n  }\n\n\n  async resume(): Promise<void> {\n    this.watchStatus = WatchStatus.RESUMING\n\n    console.log(`[realtime] client resuming with ${\n      this.sessionInfo ? 'REBUILD_WATCH' : 'INIT_WATCH'\n    } (${this.collectionName} ${this.query}) (${this.watchId})`)\n\n    try {\n      await (this.sessionInfo ? this.rebuildWatch() : this.initWatch())\n\n      console.log(`[realtime] client successfully resumed (${this.collectionName} ${this.query}) (${this.watchId})`)\n    } catch (e) {\n      console.error(\n        `[realtime] client resume failed (${this.collectionName} ${this.query}) (${this.watchId})`,\n        e\n      )\n    }\n  }\n\n  private wsLogin = async (\n    envId?: string,\n    refresh?: boolean\n  ): Promise<ILoginResult> => {\n    this.watchStatus = WatchStatus.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 !== null && this.initWatchPromise !== undefined) {\n      return this.initWatchPromise\n    }\n\n    this.initWatchPromise = new Promise<void>((resolve, reject) => {\n      void (async () => {\n        try {\n          if (this.watchStatus === WatchStatus.PAUSED) {\n            console.log('[realtime] initWatch cancelled on pause')\n            return resolve()\n          }\n\n          const { envId } = await this.wsLogin(this.envId, forceRefreshLogin)\n          if ((this.watchStatus as WatchStatus) === WatchStatus.PAUSED) {\n            console.log('[realtime] initWatch cancelled on pause')\n            return resolve()\n          }\n\n          this.watchStatus = WatchStatus.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 = WatchStatus.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    console.log(`[realtime] initWatch ${success ? 'success' : 'fail'}`)\n  }\n\n  private rebuildWatch = async (forceRefreshLogin?: boolean): Promise<void> => {\n    if (this.rebuildWatchPromise !== null && this.rebuildWatchPromise !== undefined) {\n      return this.rebuildWatchPromise\n    }\n\n    this.rebuildWatchPromise = new Promise<void>((resolve, reject) => {\n      void (async () => {\n        try {\n          if (this.watchStatus === WatchStatus.PAUSED) {\n            console.log('[realtime] rebuildWatch cancelled on pause')\n            return resolve()\n          }\n          const { envId } = await this.wsLogin(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 WatchStatus) === WatchStatus.PAUSED) {\n            console.log('[realtime] rebuildWatch cancelled on pause')\n            return resolve()\n          }\n\n          this.watchStatus = WatchStatus.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 = WatchStatus.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    console.log(`[realtime] rebuildWatch ${success ? 'success' : 'fail'}`)\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: () => {\n        (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 === WatchStatus.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\n  private closeWatch = async () => {\n    const queryId = this.sessionInfo ? this.sessionInfo.queryID : ''\n\n    if (this.watchStatus !== WatchStatus.ACTIVE) {\n      this.watchStatus = WatchStatus.CLOSED\n      this.onWatchClose(this, queryId)\n      return\n    }\n\n    try {\n      this.watchStatus = WatchStatus.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 = WatchStatus.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 !== WatchStatus.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 -= 1\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]! -= 1\n      console.log(`[realtime] ${operationName} use a retry ticket, now only ${this.availableRetries[operationName]} retry left`)\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.handleServerEventsInternel(msg)\n      this.postHandleServerEventsValidityCheck(msg)\n    } catch (e) {\n      // TODO: report\n      console.error(\n        '[realtime listener] internal non-fatal error: handle server events failed with error: ',\n        e\n      )\n      throw e\n    }\n  }\n\n  private async handleServerEventsInternel(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        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                      Object.keys(change.updatedFields).forEach((fieldPath) => {\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                  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            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            break\n          }\n          case 'update': {\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            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          this.listener.onChange(snapshot)\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      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"]}