@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.
@@ -55,18 +55,18 @@ var listener_1 = require("./listener");
55
55
  var snapshot_1 = require("./snapshot");
56
56
  var error_1 = require("./error");
57
57
  var utils_1 = require("./utils");
58
- var WATCH_STATUS;
59
- (function (WATCH_STATUS) {
60
- WATCH_STATUS["LOGGINGIN"] = "LOGGINGIN";
61
- WATCH_STATUS["INITING"] = "INITING";
62
- WATCH_STATUS["REBUILDING"] = "REBUILDING";
63
- WATCH_STATUS["ACTIVE"] = "ACTIVE";
64
- WATCH_STATUS["ERRORED"] = "ERRORED";
65
- WATCH_STATUS["CLOSING"] = "CLOSING";
66
- WATCH_STATUS["CLOSED"] = "CLOSED";
67
- WATCH_STATUS["PAUSED"] = "PAUSED";
68
- WATCH_STATUS["RESUMING"] = "RESUMING";
69
- })(WATCH_STATUS || (WATCH_STATUS = {}));
58
+ var WatchStatus;
59
+ (function (WatchStatus) {
60
+ WatchStatus["LOGGINGIN"] = "LOGGINGIN";
61
+ WatchStatus["INITING"] = "INITING";
62
+ WatchStatus["REBUILDING"] = "REBUILDING";
63
+ WatchStatus["ACTIVE"] = "ACTIVE";
64
+ WatchStatus["ERRORED"] = "ERRORED";
65
+ WatchStatus["CLOSING"] = "CLOSING";
66
+ WatchStatus["CLOSED"] = "CLOSED";
67
+ WatchStatus["PAUSED"] = "PAUSED";
68
+ WatchStatus["RESUMING"] = "RESUMING";
69
+ })(WatchStatus || (WatchStatus = {}));
70
70
  var DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR = 100;
71
71
  var DEFAULT_MAX_AUTO_RETRY_ON_ERROR = 2;
72
72
  var DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR = 2;
@@ -76,13 +76,13 @@ var DEFAULT_REBUILD_WATCH_TIMEOUT = 10 * 1000;
76
76
  var VirtualWebSocketClient = (function () {
77
77
  function VirtualWebSocketClient(options) {
78
78
  var _this = this;
79
- this.watchStatus = WATCH_STATUS.INITING;
80
- this._login = function (envId, refresh) { return __awaiter(_this, void 0, void 0, function () {
79
+ this.watchStatus = WatchStatus.INITING;
80
+ this.wsLogin = function (envId, refresh) { return __awaiter(_this, void 0, void 0, function () {
81
81
  var loginResult;
82
82
  return __generator(this, function (_a) {
83
83
  switch (_a.label) {
84
84
  case 0:
85
- this.watchStatus = WATCH_STATUS.LOGGINGIN;
85
+ this.watchStatus = WatchStatus.LOGGINGIN;
86
86
  return [4, this.login(envId, refresh)];
87
87
  case 1:
88
88
  loginResult = _a.sent();
@@ -99,99 +99,101 @@ var VirtualWebSocketClient = (function () {
99
99
  return __generator(this, function (_a) {
100
100
  switch (_a.label) {
101
101
  case 0:
102
- if (this._initWatchPromise) {
103
- return [2, this._initWatchPromise];
102
+ if (this.initWatchPromise !== null && this.initWatchPromise !== undefined) {
103
+ return [2, this.initWatchPromise];
104
104
  }
105
- this._initWatchPromise = new Promise(function (resolve, reject) { return __awaiter(_this, void 0, void 0, function () {
106
- var envId, initWatchMsg, initEventMsg, _a, events, currEvent, _i, events_1, e, snapshot, e_1;
107
- return __generator(this, function (_b) {
108
- switch (_b.label) {
109
- case 0:
110
- _b.trys.push([0, 3, , 4]);
111
- if (this.watchStatus === WATCH_STATUS.PAUSED) {
112
- console.log('[realtime] initWatch cancelled on pause');
113
- return [2, resolve()];
114
- }
115
- return [4, this._login(this.envId, forceRefreshLogin)];
116
- case 1:
117
- envId = (_b.sent()).envId;
118
- if (this.watchStatus === WATCH_STATUS.PAUSED) {
119
- console.log('[realtime] initWatch cancelled on pause');
120
- return [2, resolve()];
121
- }
122
- this.watchStatus = WATCH_STATUS.INITING;
123
- initWatchMsg = {
124
- watchId: this.watchId,
125
- requestId: message_1.genRequestId(),
126
- msgType: 'INIT_WATCH',
127
- msgData: {
128
- envId: envId,
129
- collName: this.collectionName,
130
- query: this.query,
131
- limit: this.limit,
132
- orderBy: this.orderBy
105
+ this.initWatchPromise = new Promise(function (resolve, reject) {
106
+ void (function () { return __awaiter(_this, void 0, void 0, function () {
107
+ var envId, initWatchMsg, initEventMsg, _a, events, currEvent, _i, events_1, e, snapshot, e_1;
108
+ return __generator(this, function (_b) {
109
+ switch (_b.label) {
110
+ case 0:
111
+ _b.trys.push([0, 3, , 4]);
112
+ if (this.watchStatus === WatchStatus.PAUSED) {
113
+ console.log('[realtime] initWatch cancelled on pause');
114
+ return [2, resolve()];
133
115
  }
134
- };
135
- return [4, this.send({
136
- msg: initWatchMsg,
137
- waitResponse: true,
138
- skipOnMessage: true,
139
- timeout: DEFAULT_INIT_WATCH_TIMEOUT
140
- })];
141
- case 2:
142
- initEventMsg = _b.sent();
143
- _a = initEventMsg.msgData, events = _a.events, currEvent = _a.currEvent;
144
- this.sessionInfo = {
145
- queryID: initEventMsg.msgData.queryID,
146
- currentEventId: currEvent - 1,
147
- currentDocs: []
148
- };
149
- if (events.length > 0) {
150
- for (_i = 0, events_1 = events; _i < events_1.length; _i++) {
151
- e = events_1[_i];
152
- e.ID = currEvent;
116
+ return [4, this.wsLogin(this.envId, forceRefreshLogin)];
117
+ case 1:
118
+ envId = (_b.sent()).envId;
119
+ if (this.watchStatus === WatchStatus.PAUSED) {
120
+ console.log('[realtime] initWatch cancelled on pause');
121
+ return [2, resolve()];
153
122
  }
154
- this.handleServerEvents(initEventMsg);
155
- }
156
- else {
157
- this.sessionInfo.currentEventId = currEvent;
158
- snapshot = new snapshot_1.Snapshot({
159
- id: currEvent,
160
- docChanges: [],
161
- docs: [],
162
- type: 'init'
123
+ this.watchStatus = WatchStatus.INITING;
124
+ initWatchMsg = {
125
+ watchId: this.watchId,
126
+ requestId: message_1.genRequestId(),
127
+ msgType: 'INIT_WATCH',
128
+ msgData: {
129
+ envId: envId,
130
+ collName: this.collectionName,
131
+ query: this.query,
132
+ limit: this.limit,
133
+ orderBy: this.orderBy,
134
+ },
135
+ };
136
+ return [4, this.send({
137
+ msg: initWatchMsg,
138
+ waitResponse: true,
139
+ skipOnMessage: true,
140
+ timeout: DEFAULT_INIT_WATCH_TIMEOUT,
141
+ })];
142
+ case 2:
143
+ initEventMsg = _b.sent();
144
+ _a = initEventMsg.msgData, events = _a.events, currEvent = _a.currEvent;
145
+ this.sessionInfo = {
146
+ queryID: initEventMsg.msgData.queryID,
147
+ currentEventId: currEvent - 1,
148
+ currentDocs: [],
149
+ };
150
+ if (events.length > 0) {
151
+ for (_i = 0, events_1 = events; _i < events_1.length; _i++) {
152
+ e = events_1[_i];
153
+ e.ID = currEvent;
154
+ }
155
+ this.handleServerEvents(initEventMsg);
156
+ }
157
+ else {
158
+ this.sessionInfo.currentEventId = currEvent;
159
+ snapshot = new snapshot_1.Snapshot({
160
+ id: currEvent,
161
+ docChanges: [],
162
+ docs: [],
163
+ type: 'init',
164
+ });
165
+ this.listener.onChange(snapshot);
166
+ this.scheduleSendACK();
167
+ }
168
+ this.onWatchStart(this, this.sessionInfo.queryID);
169
+ this.watchStatus = WatchStatus.ACTIVE;
170
+ this.availableRetries.INIT_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;
171
+ resolve();
172
+ return [3, 4];
173
+ case 3:
174
+ e_1 = _b.sent();
175
+ this.handleWatchEstablishmentError(e_1, {
176
+ operationName: 'INIT_WATCH',
177
+ resolve: resolve,
178
+ reject: reject,
163
179
  });
164
- this.listener.onChange(snapshot);
165
- this.scheduleSendACK();
166
- }
167
- this.onWatchStart(this, this.sessionInfo.queryID);
168
- this.watchStatus = WATCH_STATUS.ACTIVE;
169
- this._availableRetries.INIT_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;
170
- resolve();
171
- return [3, 4];
172
- case 3:
173
- e_1 = _b.sent();
174
- this.handleWatchEstablishmentError(e_1, {
175
- operationName: 'INIT_WATCH',
176
- resolve: resolve,
177
- reject: reject
178
- });
179
- return [3, 4];
180
- case 4: return [2];
181
- }
182
- });
183
- }); });
180
+ return [3, 4];
181
+ case 4: return [2];
182
+ }
183
+ });
184
+ }); })();
185
+ });
184
186
  success = false;
185
187
  _a.label = 1;
186
188
  case 1:
187
189
  _a.trys.push([1, , 3, 4]);
188
- return [4, this._initWatchPromise];
190
+ return [4, this.initWatchPromise];
189
191
  case 2:
190
192
  _a.sent();
191
193
  success = true;
192
194
  return [3, 4];
193
195
  case 3:
194
- this._initWatchPromise = undefined;
196
+ this.initWatchPromise = undefined;
195
197
  return [7];
196
198
  case 4:
197
199
  console.log("[realtime] initWatch " + (success ? 'success' : 'fail'));
@@ -205,77 +207,79 @@ var VirtualWebSocketClient = (function () {
205
207
  return __generator(this, function (_a) {
206
208
  switch (_a.label) {
207
209
  case 0:
208
- if (this._rebuildWatchPromise) {
209
- return [2, this._rebuildWatchPromise];
210
+ if (this.rebuildWatchPromise !== null && this.rebuildWatchPromise !== undefined) {
211
+ return [2, this.rebuildWatchPromise];
210
212
  }
211
- this._rebuildWatchPromise = new Promise(function (resolve, reject) { return __awaiter(_this, void 0, void 0, function () {
212
- var envId, rebuildWatchMsg, nextEventMsg, e_2;
213
- return __generator(this, function (_a) {
214
- switch (_a.label) {
215
- case 0:
216
- _a.trys.push([0, 3, , 4]);
217
- if (this.watchStatus === WATCH_STATUS.PAUSED) {
218
- console.log('[realtime] rebuildWatch cancelled on pause');
219
- return [2, resolve()];
220
- }
221
- return [4, this._login(this.envId, forceRefreshLogin)];
222
- case 1:
223
- envId = (_a.sent()).envId;
224
- if (!this.sessionInfo) {
225
- throw new Error('can not rebuildWatch without a successful initWatch (lack of sessionInfo)');
226
- }
227
- if (this.watchStatus === WATCH_STATUS.PAUSED) {
228
- console.log('[realtime] rebuildWatch cancelled on pause');
229
- return [2, resolve()];
230
- }
231
- this.watchStatus = WATCH_STATUS.REBUILDING;
232
- rebuildWatchMsg = {
233
- watchId: this.watchId,
234
- requestId: message_1.genRequestId(),
235
- msgType: 'REBUILD_WATCH',
236
- msgData: {
237
- envId: envId,
238
- collName: this.collectionName,
239
- queryID: this.sessionInfo.queryID,
240
- eventID: this.sessionInfo.currentEventId
213
+ this.rebuildWatchPromise = new Promise(function (resolve, reject) {
214
+ void (function () { return __awaiter(_this, void 0, void 0, function () {
215
+ var envId, rebuildWatchMsg, nextEventMsg, e_2;
216
+ return __generator(this, function (_a) {
217
+ switch (_a.label) {
218
+ case 0:
219
+ _a.trys.push([0, 3, , 4]);
220
+ if (this.watchStatus === WatchStatus.PAUSED) {
221
+ console.log('[realtime] rebuildWatch cancelled on pause');
222
+ return [2, resolve()];
241
223
  }
242
- };
243
- return [4, this.send({
244
- msg: rebuildWatchMsg,
245
- waitResponse: true,
246
- skipOnMessage: false,
247
- timeout: DEFAULT_REBUILD_WATCH_TIMEOUT
248
- })];
249
- case 2:
250
- nextEventMsg = _a.sent();
251
- this.handleServerEvents(nextEventMsg);
252
- this.watchStatus = WATCH_STATUS.ACTIVE;
253
- this._availableRetries.REBUILD_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;
254
- resolve();
255
- return [3, 4];
256
- case 3:
257
- e_2 = _a.sent();
258
- this.handleWatchEstablishmentError(e_2, {
259
- operationName: 'REBUILD_WATCH',
260
- resolve: resolve,
261
- reject: reject
262
- });
263
- return [3, 4];
264
- case 4: return [2];
265
- }
266
- });
267
- }); });
224
+ return [4, this.wsLogin(this.envId, forceRefreshLogin)];
225
+ case 1:
226
+ envId = (_a.sent()).envId;
227
+ if (!this.sessionInfo) {
228
+ throw new Error('can not rebuildWatch without a successful initWatch (lack of sessionInfo)');
229
+ }
230
+ if (this.watchStatus === WatchStatus.PAUSED) {
231
+ console.log('[realtime] rebuildWatch cancelled on pause');
232
+ return [2, resolve()];
233
+ }
234
+ this.watchStatus = WatchStatus.REBUILDING;
235
+ rebuildWatchMsg = {
236
+ watchId: this.watchId,
237
+ requestId: message_1.genRequestId(),
238
+ msgType: 'REBUILD_WATCH',
239
+ msgData: {
240
+ envId: envId,
241
+ collName: this.collectionName,
242
+ queryID: this.sessionInfo.queryID,
243
+ eventID: this.sessionInfo.currentEventId,
244
+ },
245
+ };
246
+ return [4, this.send({
247
+ msg: rebuildWatchMsg,
248
+ waitResponse: true,
249
+ skipOnMessage: false,
250
+ timeout: DEFAULT_REBUILD_WATCH_TIMEOUT,
251
+ })];
252
+ case 2:
253
+ nextEventMsg = _a.sent();
254
+ this.handleServerEvents(nextEventMsg);
255
+ this.watchStatus = WatchStatus.ACTIVE;
256
+ this.availableRetries.REBUILD_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;
257
+ resolve();
258
+ return [3, 4];
259
+ case 3:
260
+ e_2 = _a.sent();
261
+ this.handleWatchEstablishmentError(e_2, {
262
+ operationName: 'REBUILD_WATCH',
263
+ resolve: resolve,
264
+ reject: reject,
265
+ });
266
+ return [3, 4];
267
+ case 4: return [2];
268
+ }
269
+ });
270
+ }); })();
271
+ });
268
272
  success = false;
269
273
  _a.label = 1;
270
274
  case 1:
271
275
  _a.trys.push([1, , 3, 4]);
272
- return [4, this._rebuildWatchPromise];
276
+ return [4, this.rebuildWatchPromise];
273
277
  case 2:
274
278
  _a.sent();
275
279
  success = true;
276
280
  return [3, 4];
277
281
  case 3:
278
- this._rebuildWatchPromise = undefined;
282
+ this.rebuildWatchPromise = undefined;
279
283
  return [7];
280
284
  case 4:
281
285
  console.log("[realtime] rebuildWatch " + (success ? 'success' : 'fail'));
@@ -293,18 +297,18 @@ var VirtualWebSocketClient = (function () {
293
297
  errCode: isInitWatch
294
298
  ? error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_INIT_WATCH_FAIL
295
299
  : error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_REBUILD_WATCH_FAIL,
296
- errMsg: e
300
+ errMsg: e,
297
301
  }));
298
302
  options.reject(e);
299
303
  };
300
304
  retry = function (refreshLogin) {
301
305
  if (_this.useRetryTicket(options.operationName)) {
302
306
  if (isInitWatch) {
303
- _this._initWatchPromise = undefined;
307
+ _this.initWatchPromise = undefined;
304
308
  options.resolve(_this.initWatch(refreshLogin));
305
309
  }
306
310
  else {
307
- _this._rebuildWatchPromise = undefined;
311
+ _this.rebuildWatchPromise = undefined;
308
312
  options.resolve(_this.rebuildWatch(refreshLogin));
309
313
  }
310
314
  }
@@ -317,55 +321,57 @@ var VirtualWebSocketClient = (function () {
317
321
  onTimeoutError: function () { return retry(false); },
318
322
  onNotRetryableError: abortWatch,
319
323
  onCancelledError: options.reject,
320
- onUnknownError: function () { return __awaiter(_this, void 0, void 0, function () {
321
- var onWSDisconnected, e_3;
322
- var _this = this;
323
- return __generator(this, function (_a) {
324
- switch (_a.label) {
325
- case 0:
326
- _a.trys.push([0, 8, , 9]);
327
- onWSDisconnected = function () { return __awaiter(_this, void 0, void 0, function () {
328
- return __generator(this, function (_a) {
329
- switch (_a.label) {
330
- case 0:
331
- this.pause();
332
- return [4, this.onceWSConnected()];
333
- case 1:
334
- _a.sent();
335
- retry(true);
336
- return [2];
337
- }
338
- });
339
- }); };
340
- if (!!this.isWSConnected()) return [3, 2];
341
- return [4, onWSDisconnected()];
342
- case 1:
343
- _a.sent();
344
- return [3, 7];
345
- case 2: return [4, utils_1.sleep(DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR)];
346
- case 3:
347
- _a.sent();
348
- if (!(this.watchStatus === WATCH_STATUS.PAUSED)) return [3, 4];
349
- options.reject(new error_1.CancelledError(options.operationName + " cancelled due to pause after unknownError"));
350
- return [3, 7];
351
- case 4:
352
- if (!!this.isWSConnected()) return [3, 6];
353
- return [4, onWSDisconnected()];
354
- case 5:
355
- _a.sent();
356
- return [3, 7];
357
- case 6:
358
- retry(false);
359
- _a.label = 7;
360
- case 7: return [3, 9];
361
- case 8:
362
- e_3 = _a.sent();
363
- retry(true);
364
- return [3, 9];
365
- case 9: return [2];
366
- }
367
- });
368
- }); }
324
+ onUnknownError: function () {
325
+ (function () { return __awaiter(_this, void 0, void 0, function () {
326
+ var onWSDisconnected, e_3;
327
+ var _this = this;
328
+ return __generator(this, function (_a) {
329
+ switch (_a.label) {
330
+ case 0:
331
+ _a.trys.push([0, 8, , 9]);
332
+ onWSDisconnected = function () { return __awaiter(_this, void 0, void 0, function () {
333
+ return __generator(this, function (_a) {
334
+ switch (_a.label) {
335
+ case 0:
336
+ this.pause();
337
+ return [4, this.onceWSConnected()];
338
+ case 1:
339
+ _a.sent();
340
+ retry(true);
341
+ return [2];
342
+ }
343
+ });
344
+ }); };
345
+ if (!!this.isWSConnected()) return [3, 2];
346
+ return [4, onWSDisconnected()];
347
+ case 1:
348
+ _a.sent();
349
+ return [3, 7];
350
+ case 2: return [4, utils_1.sleep(DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR)];
351
+ case 3:
352
+ _a.sent();
353
+ if (!(this.watchStatus === WatchStatus.PAUSED)) return [3, 4];
354
+ options.reject(new error_1.CancelledError(options.operationName + " cancelled due to pause after unknownError"));
355
+ return [3, 7];
356
+ case 4:
357
+ if (!!this.isWSConnected()) return [3, 6];
358
+ return [4, onWSDisconnected()];
359
+ case 5:
360
+ _a.sent();
361
+ return [3, 7];
362
+ case 6:
363
+ retry(false);
364
+ _a.label = 7;
365
+ case 7: return [3, 9];
366
+ case 8:
367
+ e_3 = _a.sent();
368
+ retry(true);
369
+ return [3, 9];
370
+ case 9: return [2];
371
+ }
372
+ });
373
+ }); })();
374
+ },
369
375
  });
370
376
  return [2];
371
377
  });
@@ -376,34 +382,34 @@ var VirtualWebSocketClient = (function () {
376
382
  switch (_a.label) {
377
383
  case 0:
378
384
  queryId = this.sessionInfo ? this.sessionInfo.queryID : '';
379
- if (this.watchStatus !== WATCH_STATUS.ACTIVE) {
380
- this.watchStatus = WATCH_STATUS.CLOSED;
385
+ if (this.watchStatus !== WatchStatus.ACTIVE) {
386
+ this.watchStatus = WatchStatus.CLOSED;
381
387
  this.onWatchClose(this, queryId);
382
388
  return [2];
383
389
  }
384
390
  _a.label = 1;
385
391
  case 1:
386
392
  _a.trys.push([1, 3, 4, 5]);
387
- this.watchStatus = WATCH_STATUS.CLOSING;
393
+ this.watchStatus = WatchStatus.CLOSING;
388
394
  closeWatchMsg = {
389
395
  watchId: this.watchId,
390
396
  requestId: message_1.genRequestId(),
391
397
  msgType: 'CLOSE_WATCH',
392
- msgData: null
398
+ msgData: null,
393
399
  };
394
400
  return [4, this.send({
395
- msg: closeWatchMsg
401
+ msg: closeWatchMsg,
396
402
  })];
397
403
  case 2:
398
404
  _a.sent();
399
405
  this.sessionInfo = undefined;
400
- this.watchStatus = WATCH_STATUS.CLOSED;
406
+ this.watchStatus = WatchStatus.CLOSED;
401
407
  return [3, 5];
402
408
  case 3:
403
409
  e_4 = _a.sent();
404
410
  this.closeWithError(new error_1.CloudSDKError({
405
411
  errCode: error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CLOSE_WATCH_FAIL,
406
- errMsg: e_4
412
+ errMsg: e_4,
407
413
  }));
408
414
  return [3, 5];
409
415
  case 4:
@@ -415,8 +421,8 @@ var VirtualWebSocketClient = (function () {
415
421
  }); };
416
422
  this.scheduleSendACK = function () {
417
423
  _this.clearACKSchedule();
418
- _this._ackTimeoutId = setTimeout(function () {
419
- if (_this._waitExpectedTimeoutId) {
424
+ _this.ackTimeoutId = setTimeout(function () {
425
+ if (_this.waitExpectedTimeoutId) {
420
426
  _this.scheduleSendACK();
421
427
  }
422
428
  else {
@@ -425,8 +431,8 @@ var VirtualWebSocketClient = (function () {
425
431
  }, DEFAULT_SEND_ACK_DEBOUNCE_TIMEOUT);
426
432
  };
427
433
  this.clearACKSchedule = function () {
428
- if (_this._ackTimeoutId) {
429
- clearTimeout(_this._ackTimeoutId);
434
+ if (_this.ackTimeoutId) {
435
+ clearTimeout(_this.ackTimeoutId);
430
436
  }
431
437
  };
432
438
  this.sendACK = function () { return __awaiter(_this, void 0, void 0, function () {
@@ -435,7 +441,7 @@ var VirtualWebSocketClient = (function () {
435
441
  switch (_a.label) {
436
442
  case 0:
437
443
  _a.trys.push([0, 2, , 3]);
438
- if (this.watchStatus !== WATCH_STATUS.ACTIVE) {
444
+ if (this.watchStatus !== WatchStatus.ACTIVE) {
439
445
  this.scheduleSendACK();
440
446
  return [2];
441
447
  }
@@ -449,11 +455,11 @@ var VirtualWebSocketClient = (function () {
449
455
  msgType: 'CHECK_LAST',
450
456
  msgData: {
451
457
  queryID: this.sessionInfo.queryID,
452
- eventID: this.sessionInfo.currentEventId
453
- }
458
+ eventID: this.sessionInfo.currentEventId,
459
+ },
454
460
  };
455
461
  return [4, this.send({
456
- msg: ackMsg
462
+ msg: ackMsg,
457
463
  })];
458
464
  case 1:
459
465
  _a.sent();
@@ -477,7 +483,7 @@ var VirtualWebSocketClient = (function () {
477
483
  case 'COLLECTION_PERMISSION_DENIED': {
478
484
  this.closeWithError(new error_1.CloudSDKError({
479
485
  errCode: error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL,
480
- errMsg: msg.msgData.code
486
+ errMsg: msg.msgData.code,
481
487
  }));
482
488
  return [2];
483
489
  }
@@ -486,15 +492,15 @@ var VirtualWebSocketClient = (function () {
486
492
  }
487
493
  }
488
494
  }
489
- if (this._availableRetries.CHECK_LAST &&
490
- this._availableRetries.CHECK_LAST > 0) {
491
- this._availableRetries.CHECK_LAST--;
495
+ if (this.availableRetries.CHECK_LAST
496
+ && this.availableRetries.CHECK_LAST > 0) {
497
+ this.availableRetries.CHECK_LAST -= 1;
492
498
  this.scheduleSendACK();
493
499
  }
494
500
  else {
495
501
  this.closeWithError(new error_1.CloudSDKError({
496
502
  errCode: error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL,
497
- errMsg: e_5
503
+ errMsg: e_5,
498
504
  }));
499
505
  }
500
506
  return [3, 3];
@@ -550,40 +556,42 @@ var VirtualWebSocketClient = (function () {
550
556
  this.onWatchStart = options.onWatchStart;
551
557
  this.onWatchClose = options.onWatchClose;
552
558
  this.debug = options.debug;
553
- this._availableRetries = {
559
+ this.availableRetries = {
554
560
  INIT_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,
555
561
  REBUILD_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,
556
- CHECK_LAST: DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR
562
+ CHECK_LAST: DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR,
557
563
  };
558
564
  this.listener = new listener_1.RealtimeListener({
559
- close: this.closeWatch,
565
+ close: void (function () {
566
+ _this.closeWatch();
567
+ }),
560
568
  onChange: options.onChange,
561
569
  onError: options.onError,
562
570
  debug: this.debug,
563
- virtualClient: this
571
+ virtualClient: this,
564
572
  });
565
573
  this.initWatch();
566
574
  }
567
575
  VirtualWebSocketClient.prototype.onMessage = function (msg) {
568
576
  var _this = this;
569
577
  switch (this.watchStatus) {
570
- case WATCH_STATUS.PAUSED: {
578
+ case WatchStatus.PAUSED: {
571
579
  if (msg.msgType !== 'ERROR') {
572
580
  return;
573
581
  }
574
582
  break;
575
583
  }
576
- case WATCH_STATUS.LOGGINGIN:
577
- case WATCH_STATUS.INITING:
578
- case WATCH_STATUS.REBUILDING: {
584
+ case WatchStatus.LOGGINGIN:
585
+ case WatchStatus.INITING:
586
+ case WatchStatus.REBUILDING: {
579
587
  console.warn("[realtime listener] internal non-fatal error: unexpected message received while " + this.watchStatus);
580
588
  return;
581
589
  }
582
- case WATCH_STATUS.CLOSED: {
590
+ case WatchStatus.CLOSED: {
583
591
  console.warn('[realtime listener] internal non-fatal error: unexpected message received when the watch has closed');
584
592
  return;
585
593
  }
586
- case WATCH_STATUS.ERRORED: {
594
+ case WatchStatus.ERRORED: {
587
595
  console.warn('[realtime listener] internal non-fatal error: unexpected message received when the watch has ended with error');
588
596
  return;
589
597
  }
@@ -603,7 +611,7 @@ var VirtualWebSocketClient = (function () {
603
611
  if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {
604
612
  this.sessionInfo.expectEventId = msg.msgData.currEvent;
605
613
  this.clearWaitExpectedEvent();
606
- this._waitExpectedTimeoutId = setTimeout(function () {
614
+ this.waitExpectedTimeoutId = setTimeout(function () {
607
615
  _this.rebuildWatch();
608
616
  }, this.getWaitExpectedTimeoutLength());
609
617
  console.log("[realtime] waitExpectedTimeoutLength " + this.getWaitExpectedTimeoutLength());
@@ -613,7 +621,7 @@ var VirtualWebSocketClient = (function () {
613
621
  case 'ERROR': {
614
622
  this.closeWithError(new error_1.CloudSDKError({
615
623
  errCode: error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_SERVER_ERROR_MSG,
616
- errMsg: msg.msgData.code + " - " + msg.msgData.message
624
+ errMsg: msg.msgData.code + " - " + msg.msgData.message,
617
625
  }));
618
626
  break;
619
627
  }
@@ -624,14 +632,15 @@ var VirtualWebSocketClient = (function () {
624
632
  }
625
633
  };
626
634
  VirtualWebSocketClient.prototype.closeWithError = function (error) {
627
- this.watchStatus = WATCH_STATUS.ERRORED;
635
+ var _a;
636
+ this.watchStatus = WatchStatus.ERRORED;
628
637
  this.clearACKSchedule();
629
638
  this.listener.onError(error);
630
- this.onWatchClose(this, (this.sessionInfo && this.sessionInfo.queryID) || '');
639
+ this.onWatchClose(this, ((_a = this.sessionInfo) === null || _a === void 0 ? void 0 : _a.queryID) || '');
631
640
  console.log("[realtime] client closed (" + this.collectionName + " " + this.query + ") (watchId " + this.watchId + ")");
632
641
  };
633
642
  VirtualWebSocketClient.prototype.pause = function () {
634
- this.watchStatus = WATCH_STATUS.PAUSED;
643
+ this.watchStatus = WatchStatus.PAUSED;
635
644
  console.log("[realtime] client paused (" + this.collectionName + " " + this.query + ") (watchId " + this.watchId + ")");
636
645
  };
637
646
  VirtualWebSocketClient.prototype.resume = function () {
@@ -640,7 +649,7 @@ var VirtualWebSocketClient = (function () {
640
649
  return __generator(this, function (_a) {
641
650
  switch (_a.label) {
642
651
  case 0:
643
- this.watchStatus = WATCH_STATUS.RESUMING;
652
+ this.watchStatus = WatchStatus.RESUMING;
644
653
  console.log("[realtime] client resuming with " + (this.sessionInfo ? 'REBUILD_WATCH' : 'INIT_WATCH') + " (" + this.collectionName + " " + this.query + ") (" + this.watchId + ")");
645
654
  _a.label = 1;
646
655
  case 1:
@@ -660,10 +669,10 @@ var VirtualWebSocketClient = (function () {
660
669
  });
661
670
  };
662
671
  VirtualWebSocketClient.prototype.useRetryTicket = function (operationName) {
663
- if (this._availableRetries[operationName] &&
664
- this._availableRetries[operationName] > 0) {
665
- this._availableRetries[operationName]--;
666
- console.log("[realtime] " + operationName + " use a retry ticket, now only " + this._availableRetries[operationName] + " retry left");
672
+ if (this.availableRetries[operationName]
673
+ && this.availableRetries[operationName] > 0) {
674
+ this.availableRetries[operationName] -= 1;
675
+ console.log("[realtime] " + operationName + " use a retry ticket, now only " + this.availableRetries[operationName] + " retry left");
667
676
  return true;
668
677
  }
669
678
  return false;
@@ -676,10 +685,10 @@ var VirtualWebSocketClient = (function () {
676
685
  case 0:
677
686
  _a.trys.push([0, 2, , 3]);
678
687
  this.scheduleSendACK();
679
- return [4, this._handleServerEvents(msg)];
688
+ return [4, this.handleServerEventsInternel(msg)];
680
689
  case 1:
681
690
  _a.sent();
682
- this._postHandleServerEventsValidityCheck(msg);
691
+ this.postHandleServerEventsValidityCheck(msg);
683
692
  return [3, 3];
684
693
  case 2:
685
694
  e_7 = _a.sent();
@@ -690,7 +699,7 @@ var VirtualWebSocketClient = (function () {
690
699
  });
691
700
  });
692
701
  };
693
- VirtualWebSocketClient.prototype._handleServerEvents = function (msg) {
702
+ VirtualWebSocketClient.prototype.handleServerEventsInternel = function (msg) {
694
703
  return __awaiter(this, void 0, void 0, function () {
695
704
  var requestId, events, msgType, sessionInfo, allChangeEvents, docs, initEncountered, _loop_1, this_1, i, len, state_1;
696
705
  return __generator(this, function (_a) {
@@ -709,14 +718,14 @@ var VirtualWebSocketClient = (function () {
709
718
  catch (e) {
710
719
  this.closeWithError(new error_1.CloudSDKError({
711
720
  errCode: error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_RECEIVE_INVALID_SERVER_DATA,
712
- errMsg: e
721
+ errMsg: e,
713
722
  }));
714
723
  return [2];
715
724
  }
716
725
  docs = __spreadArrays(sessionInfo.currentDocs);
717
726
  initEncountered = false;
718
727
  _loop_1 = function (i, len) {
719
- var change, localDoc, doc, fieldPath, _i, _a, fieldPath, err, err, doc, doc, err, ind, ind, docsSnapshot, docChanges, snapshot;
728
+ var change, localDoc, doc_1, _i, _a, fieldPath, err, err, doc, doc, err, ind, ind, docsSnapshot, docChanges, snapshot;
720
729
  return __generator(this, function (_b) {
721
730
  switch (_b.label) {
722
731
  case 0:
@@ -739,19 +748,19 @@ var VirtualWebSocketClient = (function () {
739
748
  case 'dequeue': {
740
749
  localDoc = docs.find(function (doc) { return doc._id === change.docId; });
741
750
  if (localDoc) {
742
- doc = lodash_clonedeep_1.default(localDoc);
751
+ doc_1 = lodash_clonedeep_1.default(localDoc);
743
752
  if (change.updatedFields) {
744
- for (fieldPath in change.updatedFields) {
745
- lodash_set_1.default(doc, fieldPath, change.updatedFields[fieldPath]);
746
- }
753
+ Object.keys(change.updatedFields).forEach(function (fieldPath) {
754
+ lodash_set_1.default(doc_1, fieldPath, change.updatedFields[fieldPath]);
755
+ });
747
756
  }
748
757
  if (change.removedFields) {
749
758
  for (_i = 0, _a = change.removedFields; _i < _a.length; _i++) {
750
759
  fieldPath = _a[_i];
751
- lodash_unset_1.default(doc, fieldPath);
760
+ lodash_unset_1.default(doc_1, fieldPath);
752
761
  }
753
762
  }
754
- change.doc = doc;
763
+ change.doc = doc_1;
755
764
  }
756
765
  else {
757
766
  console.error('[realtime listener] internal non-fatal server error: unexpected update dataType event where no doc is associated.');
@@ -761,7 +770,7 @@ var VirtualWebSocketClient = (function () {
761
770
  case 'enqueue': {
762
771
  err = new error_1.CloudSDKError({
763
772
  errCode: error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR,
764
- errMsg: "HandleServerEvents: full doc is not provided with dataType=\"update\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")"
773
+ errMsg: "HandleServerEvents: full doc is not provided with dataType=\"update\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")",
765
774
  });
766
775
  this_1.closeWithError(err);
767
776
  throw err;
@@ -777,7 +786,7 @@ var VirtualWebSocketClient = (function () {
777
786
  if (!change.doc) {
778
787
  err = new error_1.CloudSDKError({
779
788
  errCode: error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR,
780
- errMsg: "HandleServerEvents: full doc is not provided with dataType=\"replace\" (requestId " + msg.requestId + ")"
789
+ errMsg: "HandleServerEvents: full doc is not provided with dataType=\"replace\" (requestId " + msg.requestId + ")",
781
790
  });
782
791
  this_1.closeWithError(err);
783
792
  throw err;
@@ -810,7 +819,7 @@ var VirtualWebSocketClient = (function () {
810
819
  case 'enqueue': {
811
820
  err = new error_1.CloudSDKError({
812
821
  errCode: error_1.ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR,
813
- errMsg: "HandleServerEvents: full doc is not provided with dataType=\"limit\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")"
822
+ errMsg: "HandleServerEvents: full doc is not provided with dataType=\"limit\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")",
814
823
  });
815
824
  this_1.closeWithError(err);
816
825
  throw err;
@@ -859,8 +868,8 @@ var VirtualWebSocketClient = (function () {
859
868
  break;
860
869
  }
861
870
  }
862
- if (i === len - 1 ||
863
- (allChangeEvents[i + 1] && allChangeEvents[i + 1].id !== change.id)) {
871
+ if (i === len - 1
872
+ || (allChangeEvents[i + 1] && allChangeEvents[i + 1].id !== change.id)) {
864
873
  docsSnapshot = __spreadArrays(docs);
865
874
  docChanges = allChangeEvents
866
875
  .slice(0, i + 1)
@@ -871,7 +880,7 @@ var VirtualWebSocketClient = (function () {
871
880
  id: change.id,
872
881
  docChanges: docChanges,
873
882
  docs: docsSnapshot,
874
- msgType: msgType
883
+ msgType: msgType,
875
884
  });
876
885
  this_1.listener.onChange(snapshot);
877
886
  }
@@ -905,13 +914,13 @@ var VirtualWebSocketClient = (function () {
905
914
  });
906
915
  });
907
916
  };
908
- VirtualWebSocketClient.prototype._postHandleServerEventsValidityCheck = function (msg) {
917
+ VirtualWebSocketClient.prototype.postHandleServerEventsValidityCheck = function (msg) {
909
918
  if (!this.sessionInfo) {
910
919
  console.error('[realtime listener] internal non-fatal error: sessionInfo lost after server event handling, this should never occur');
911
920
  return;
912
921
  }
913
- if (this.sessionInfo.expectEventId &&
914
- this.sessionInfo.currentEventId >= this.sessionInfo.expectEventId) {
922
+ if (this.sessionInfo.expectEventId
923
+ && this.sessionInfo.currentEventId >= this.sessionInfo.expectEventId) {
915
924
  this.clearWaitExpectedEvent();
916
925
  }
917
926
  if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {
@@ -920,9 +929,9 @@ var VirtualWebSocketClient = (function () {
920
929
  }
921
930
  };
922
931
  VirtualWebSocketClient.prototype.clearWaitExpectedEvent = function () {
923
- if (this._waitExpectedTimeoutId) {
924
- clearTimeout(this._waitExpectedTimeoutId);
925
- this._waitExpectedTimeoutId = undefined;
932
+ if (this.waitExpectedTimeoutId) {
933
+ clearTimeout(this.waitExpectedTimeoutId);
934
+ this.waitExpectedTimeoutId = undefined;
926
935
  }
927
936
  };
928
937
  return VirtualWebSocketClient;
@@ -934,7 +943,7 @@ function getPublicEvent(event) {
934
943
  dataType: event.DataType,
935
944
  queueType: event.QueueType,
936
945
  docId: event.DocID,
937
- doc: event.Doc && event.Doc !== '{}' ? JSON.parse(event.Doc) : undefined
946
+ doc: event.Doc && event.Doc !== '{}' ? JSON.parse(event.Doc) : undefined,
938
947
  };
939
948
  if (event.DataType === 'update') {
940
949
  if (event.UpdatedFields) {
@@ -946,4 +955,4 @@ function getPublicEvent(event) {
946
955
  }
947
956
  return e;
948
957
  }
949
- //# 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"]}
958
+ //# 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"]}