@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.
- package/.eslintrc.js +46 -0
- package/dist/cjs/common.js +1 -1
- package/dist/cjs/error.js +8 -12
- package/dist/cjs/index.js +3 -3
- package/dist/cjs/listener.js +2 -4
- package/dist/cjs/message.js +1 -1
- package/dist/cjs/snapshot.js +7 -7
- package/dist/cjs/utils.js +1 -1
- package/dist/cjs/virtual-websocket-client.d.ts +8 -8
- package/dist/cjs/virtual-websocket-client.js +281 -272
- package/dist/cjs/websocket-client.d.ts +21 -21
- package/dist/cjs/websocket-client.js +441 -444
- package/dist/cjs/ws-event.d.ts +2 -2
- package/dist/cjs/ws-event.js +47 -47
- package/dist/esm/common.js +1 -1
- package/dist/esm/error.js +8 -12
- package/dist/esm/index.js +3 -3
- package/dist/esm/listener.js +2 -4
- package/dist/esm/message.js +1 -1
- package/dist/esm/snapshot.js +7 -7
- package/dist/esm/utils.js +1 -1
- package/dist/esm/virtual-websocket-client.d.ts +8 -8
- package/dist/esm/virtual-websocket-client.js +282 -273
- package/dist/esm/websocket-client.d.ts +21 -21
- package/dist/esm/websocket-client.js +442 -445
- package/dist/esm/ws-event.d.ts +2 -2
- package/dist/esm/ws-event.js +46 -46
- package/package.json +9 -13
- package/src/common.ts +11 -11
- package/src/error.ts +13 -19
- package/src/index.ts +26 -26
- package/src/listener.ts +2 -4
- package/src/message.ts +2 -4
- package/src/snapshot.ts +6 -6
- package/src/utils.ts +1 -1
- package/src/virtual-websocket-client.ts +187 -387
- package/src/websocket-client.ts +385 -600
- package/src/ws-event.ts +24 -24
- package/.eslintrc +0 -30
|
@@ -47,20 +47,20 @@ import cloneDeep from 'lodash.clonedeep';
|
|
|
47
47
|
import { genRequestId } from './message';
|
|
48
48
|
import { RealtimeListener } from './listener';
|
|
49
49
|
import { Snapshot } from './snapshot';
|
|
50
|
-
import { ERR_CODE, CloudSDKError, isTimeoutError, CancelledError, isCancelledError, isRealtimeErrorMessageError } from './error';
|
|
50
|
+
import { ERR_CODE, CloudSDKError, isTimeoutError, CancelledError, isCancelledError, isRealtimeErrorMessageError, } from './error';
|
|
51
51
|
import { sleep } from './utils';
|
|
52
|
-
var
|
|
53
|
-
(function (
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
})(
|
|
52
|
+
var WatchStatus;
|
|
53
|
+
(function (WatchStatus) {
|
|
54
|
+
WatchStatus["LOGGINGIN"] = "LOGGINGIN";
|
|
55
|
+
WatchStatus["INITING"] = "INITING";
|
|
56
|
+
WatchStatus["REBUILDING"] = "REBUILDING";
|
|
57
|
+
WatchStatus["ACTIVE"] = "ACTIVE";
|
|
58
|
+
WatchStatus["ERRORED"] = "ERRORED";
|
|
59
|
+
WatchStatus["CLOSING"] = "CLOSING";
|
|
60
|
+
WatchStatus["CLOSED"] = "CLOSED";
|
|
61
|
+
WatchStatus["PAUSED"] = "PAUSED";
|
|
62
|
+
WatchStatus["RESUMING"] = "RESUMING";
|
|
63
|
+
})(WatchStatus || (WatchStatus = {}));
|
|
64
64
|
var DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR = 100;
|
|
65
65
|
var DEFAULT_MAX_AUTO_RETRY_ON_ERROR = 2;
|
|
66
66
|
var DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR = 2;
|
|
@@ -70,13 +70,13 @@ var DEFAULT_REBUILD_WATCH_TIMEOUT = 10 * 1000;
|
|
|
70
70
|
var VirtualWebSocketClient = (function () {
|
|
71
71
|
function VirtualWebSocketClient(options) {
|
|
72
72
|
var _this = this;
|
|
73
|
-
this.watchStatus =
|
|
74
|
-
this.
|
|
73
|
+
this.watchStatus = WatchStatus.INITING;
|
|
74
|
+
this.wsLogin = function (envId, refresh) { return __awaiter(_this, void 0, void 0, function () {
|
|
75
75
|
var loginResult;
|
|
76
76
|
return __generator(this, function (_a) {
|
|
77
77
|
switch (_a.label) {
|
|
78
78
|
case 0:
|
|
79
|
-
this.watchStatus =
|
|
79
|
+
this.watchStatus = WatchStatus.LOGGINGIN;
|
|
80
80
|
return [4, this.login(envId, refresh)];
|
|
81
81
|
case 1:
|
|
82
82
|
loginResult = _a.sent();
|
|
@@ -93,99 +93,101 @@ var VirtualWebSocketClient = (function () {
|
|
|
93
93
|
return __generator(this, function (_a) {
|
|
94
94
|
switch (_a.label) {
|
|
95
95
|
case 0:
|
|
96
|
-
if (this.
|
|
97
|
-
return [2, this.
|
|
96
|
+
if (this.initWatchPromise !== null && this.initWatchPromise !== undefined) {
|
|
97
|
+
return [2, this.initWatchPromise];
|
|
98
98
|
}
|
|
99
|
-
this.
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
return [4, this._login(this.envId, forceRefreshLogin)];
|
|
110
|
-
case 1:
|
|
111
|
-
envId = (_b.sent()).envId;
|
|
112
|
-
if (this.watchStatus === WATCH_STATUS.PAUSED) {
|
|
113
|
-
console.log('[realtime] initWatch cancelled on pause');
|
|
114
|
-
return [2, resolve()];
|
|
115
|
-
}
|
|
116
|
-
this.watchStatus = WATCH_STATUS.INITING;
|
|
117
|
-
initWatchMsg = {
|
|
118
|
-
watchId: this.watchId,
|
|
119
|
-
requestId: genRequestId(),
|
|
120
|
-
msgType: 'INIT_WATCH',
|
|
121
|
-
msgData: {
|
|
122
|
-
envId: envId,
|
|
123
|
-
collName: this.collectionName,
|
|
124
|
-
query: this.query,
|
|
125
|
-
limit: this.limit,
|
|
126
|
-
orderBy: this.orderBy
|
|
99
|
+
this.initWatchPromise = new Promise(function (resolve, reject) {
|
|
100
|
+
void (function () { return __awaiter(_this, void 0, void 0, function () {
|
|
101
|
+
var envId, initWatchMsg, initEventMsg, _a, events, currEvent, _i, events_1, e, snapshot, e_1;
|
|
102
|
+
return __generator(this, function (_b) {
|
|
103
|
+
switch (_b.label) {
|
|
104
|
+
case 0:
|
|
105
|
+
_b.trys.push([0, 3, , 4]);
|
|
106
|
+
if (this.watchStatus === WatchStatus.PAUSED) {
|
|
107
|
+
console.log('[realtime] initWatch cancelled on pause');
|
|
108
|
+
return [2, resolve()];
|
|
127
109
|
}
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
})];
|
|
135
|
-
case 2:
|
|
136
|
-
initEventMsg = _b.sent();
|
|
137
|
-
_a = initEventMsg.msgData, events = _a.events, currEvent = _a.currEvent;
|
|
138
|
-
this.sessionInfo = {
|
|
139
|
-
queryID: initEventMsg.msgData.queryID,
|
|
140
|
-
currentEventId: currEvent - 1,
|
|
141
|
-
currentDocs: []
|
|
142
|
-
};
|
|
143
|
-
if (events.length > 0) {
|
|
144
|
-
for (_i = 0, events_1 = events; _i < events_1.length; _i++) {
|
|
145
|
-
e = events_1[_i];
|
|
146
|
-
e.ID = currEvent;
|
|
110
|
+
return [4, this.wsLogin(this.envId, forceRefreshLogin)];
|
|
111
|
+
case 1:
|
|
112
|
+
envId = (_b.sent()).envId;
|
|
113
|
+
if (this.watchStatus === WatchStatus.PAUSED) {
|
|
114
|
+
console.log('[realtime] initWatch cancelled on pause');
|
|
115
|
+
return [2, resolve()];
|
|
147
116
|
}
|
|
148
|
-
this.
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
117
|
+
this.watchStatus = WatchStatus.INITING;
|
|
118
|
+
initWatchMsg = {
|
|
119
|
+
watchId: this.watchId,
|
|
120
|
+
requestId: genRequestId(),
|
|
121
|
+
msgType: 'INIT_WATCH',
|
|
122
|
+
msgData: {
|
|
123
|
+
envId: envId,
|
|
124
|
+
collName: this.collectionName,
|
|
125
|
+
query: this.query,
|
|
126
|
+
limit: this.limit,
|
|
127
|
+
orderBy: this.orderBy,
|
|
128
|
+
},
|
|
129
|
+
};
|
|
130
|
+
return [4, this.send({
|
|
131
|
+
msg: initWatchMsg,
|
|
132
|
+
waitResponse: true,
|
|
133
|
+
skipOnMessage: true,
|
|
134
|
+
timeout: DEFAULT_INIT_WATCH_TIMEOUT,
|
|
135
|
+
})];
|
|
136
|
+
case 2:
|
|
137
|
+
initEventMsg = _b.sent();
|
|
138
|
+
_a = initEventMsg.msgData, events = _a.events, currEvent = _a.currEvent;
|
|
139
|
+
this.sessionInfo = {
|
|
140
|
+
queryID: initEventMsg.msgData.queryID,
|
|
141
|
+
currentEventId: currEvent - 1,
|
|
142
|
+
currentDocs: [],
|
|
143
|
+
};
|
|
144
|
+
if (events.length > 0) {
|
|
145
|
+
for (_i = 0, events_1 = events; _i < events_1.length; _i++) {
|
|
146
|
+
e = events_1[_i];
|
|
147
|
+
e.ID = currEvent;
|
|
148
|
+
}
|
|
149
|
+
this.handleServerEvents(initEventMsg);
|
|
150
|
+
}
|
|
151
|
+
else {
|
|
152
|
+
this.sessionInfo.currentEventId = currEvent;
|
|
153
|
+
snapshot = new Snapshot({
|
|
154
|
+
id: currEvent,
|
|
155
|
+
docChanges: [],
|
|
156
|
+
docs: [],
|
|
157
|
+
type: 'init',
|
|
158
|
+
});
|
|
159
|
+
this.listener.onChange(snapshot);
|
|
160
|
+
this.scheduleSendACK();
|
|
161
|
+
}
|
|
162
|
+
this.onWatchStart(this, this.sessionInfo.queryID);
|
|
163
|
+
this.watchStatus = WatchStatus.ACTIVE;
|
|
164
|
+
this.availableRetries.INIT_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;
|
|
165
|
+
resolve();
|
|
166
|
+
return [3, 4];
|
|
167
|
+
case 3:
|
|
168
|
+
e_1 = _b.sent();
|
|
169
|
+
this.handleWatchEstablishmentError(e_1, {
|
|
170
|
+
operationName: 'INIT_WATCH',
|
|
171
|
+
resolve: resolve,
|
|
172
|
+
reject: reject,
|
|
157
173
|
});
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
resolve();
|
|
165
|
-
return [3, 4];
|
|
166
|
-
case 3:
|
|
167
|
-
e_1 = _b.sent();
|
|
168
|
-
this.handleWatchEstablishmentError(e_1, {
|
|
169
|
-
operationName: 'INIT_WATCH',
|
|
170
|
-
resolve: resolve,
|
|
171
|
-
reject: reject
|
|
172
|
-
});
|
|
173
|
-
return [3, 4];
|
|
174
|
-
case 4: return [2];
|
|
175
|
-
}
|
|
176
|
-
});
|
|
177
|
-
}); });
|
|
174
|
+
return [3, 4];
|
|
175
|
+
case 4: return [2];
|
|
176
|
+
}
|
|
177
|
+
});
|
|
178
|
+
}); })();
|
|
179
|
+
});
|
|
178
180
|
success = false;
|
|
179
181
|
_a.label = 1;
|
|
180
182
|
case 1:
|
|
181
183
|
_a.trys.push([1, , 3, 4]);
|
|
182
|
-
return [4, this.
|
|
184
|
+
return [4, this.initWatchPromise];
|
|
183
185
|
case 2:
|
|
184
186
|
_a.sent();
|
|
185
187
|
success = true;
|
|
186
188
|
return [3, 4];
|
|
187
189
|
case 3:
|
|
188
|
-
this.
|
|
190
|
+
this.initWatchPromise = undefined;
|
|
189
191
|
return [7];
|
|
190
192
|
case 4:
|
|
191
193
|
console.log("[realtime] initWatch " + (success ? 'success' : 'fail'));
|
|
@@ -199,77 +201,79 @@ var VirtualWebSocketClient = (function () {
|
|
|
199
201
|
return __generator(this, function (_a) {
|
|
200
202
|
switch (_a.label) {
|
|
201
203
|
case 0:
|
|
202
|
-
if (this.
|
|
203
|
-
return [2, this.
|
|
204
|
+
if (this.rebuildWatchPromise !== null && this.rebuildWatchPromise !== undefined) {
|
|
205
|
+
return [2, this.rebuildWatchPromise];
|
|
204
206
|
}
|
|
205
|
-
this.
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
return [4, this._login(this.envId, forceRefreshLogin)];
|
|
216
|
-
case 1:
|
|
217
|
-
envId = (_a.sent()).envId;
|
|
218
|
-
if (!this.sessionInfo) {
|
|
219
|
-
throw new Error('can not rebuildWatch without a successful initWatch (lack of sessionInfo)');
|
|
220
|
-
}
|
|
221
|
-
if (this.watchStatus === WATCH_STATUS.PAUSED) {
|
|
222
|
-
console.log('[realtime] rebuildWatch cancelled on pause');
|
|
223
|
-
return [2, resolve()];
|
|
224
|
-
}
|
|
225
|
-
this.watchStatus = WATCH_STATUS.REBUILDING;
|
|
226
|
-
rebuildWatchMsg = {
|
|
227
|
-
watchId: this.watchId,
|
|
228
|
-
requestId: genRequestId(),
|
|
229
|
-
msgType: 'REBUILD_WATCH',
|
|
230
|
-
msgData: {
|
|
231
|
-
envId: envId,
|
|
232
|
-
collName: this.collectionName,
|
|
233
|
-
queryID: this.sessionInfo.queryID,
|
|
234
|
-
eventID: this.sessionInfo.currentEventId
|
|
207
|
+
this.rebuildWatchPromise = new Promise(function (resolve, reject) {
|
|
208
|
+
void (function () { return __awaiter(_this, void 0, void 0, function () {
|
|
209
|
+
var envId, rebuildWatchMsg, nextEventMsg, e_2;
|
|
210
|
+
return __generator(this, function (_a) {
|
|
211
|
+
switch (_a.label) {
|
|
212
|
+
case 0:
|
|
213
|
+
_a.trys.push([0, 3, , 4]);
|
|
214
|
+
if (this.watchStatus === WatchStatus.PAUSED) {
|
|
215
|
+
console.log('[realtime] rebuildWatch cancelled on pause');
|
|
216
|
+
return [2, resolve()];
|
|
235
217
|
}
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
218
|
+
return [4, this.wsLogin(this.envId, forceRefreshLogin)];
|
|
219
|
+
case 1:
|
|
220
|
+
envId = (_a.sent()).envId;
|
|
221
|
+
if (!this.sessionInfo) {
|
|
222
|
+
throw new Error('can not rebuildWatch without a successful initWatch (lack of sessionInfo)');
|
|
223
|
+
}
|
|
224
|
+
if (this.watchStatus === WatchStatus.PAUSED) {
|
|
225
|
+
console.log('[realtime] rebuildWatch cancelled on pause');
|
|
226
|
+
return [2, resolve()];
|
|
227
|
+
}
|
|
228
|
+
this.watchStatus = WatchStatus.REBUILDING;
|
|
229
|
+
rebuildWatchMsg = {
|
|
230
|
+
watchId: this.watchId,
|
|
231
|
+
requestId: genRequestId(),
|
|
232
|
+
msgType: 'REBUILD_WATCH',
|
|
233
|
+
msgData: {
|
|
234
|
+
envId: envId,
|
|
235
|
+
collName: this.collectionName,
|
|
236
|
+
queryID: this.sessionInfo.queryID,
|
|
237
|
+
eventID: this.sessionInfo.currentEventId,
|
|
238
|
+
},
|
|
239
|
+
};
|
|
240
|
+
return [4, this.send({
|
|
241
|
+
msg: rebuildWatchMsg,
|
|
242
|
+
waitResponse: true,
|
|
243
|
+
skipOnMessage: false,
|
|
244
|
+
timeout: DEFAULT_REBUILD_WATCH_TIMEOUT,
|
|
245
|
+
})];
|
|
246
|
+
case 2:
|
|
247
|
+
nextEventMsg = _a.sent();
|
|
248
|
+
this.handleServerEvents(nextEventMsg);
|
|
249
|
+
this.watchStatus = WatchStatus.ACTIVE;
|
|
250
|
+
this.availableRetries.REBUILD_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR;
|
|
251
|
+
resolve();
|
|
252
|
+
return [3, 4];
|
|
253
|
+
case 3:
|
|
254
|
+
e_2 = _a.sent();
|
|
255
|
+
this.handleWatchEstablishmentError(e_2, {
|
|
256
|
+
operationName: 'REBUILD_WATCH',
|
|
257
|
+
resolve: resolve,
|
|
258
|
+
reject: reject,
|
|
259
|
+
});
|
|
260
|
+
return [3, 4];
|
|
261
|
+
case 4: return [2];
|
|
262
|
+
}
|
|
263
|
+
});
|
|
264
|
+
}); })();
|
|
265
|
+
});
|
|
262
266
|
success = false;
|
|
263
267
|
_a.label = 1;
|
|
264
268
|
case 1:
|
|
265
269
|
_a.trys.push([1, , 3, 4]);
|
|
266
|
-
return [4, this.
|
|
270
|
+
return [4, this.rebuildWatchPromise];
|
|
267
271
|
case 2:
|
|
268
272
|
_a.sent();
|
|
269
273
|
success = true;
|
|
270
274
|
return [3, 4];
|
|
271
275
|
case 3:
|
|
272
|
-
this.
|
|
276
|
+
this.rebuildWatchPromise = undefined;
|
|
273
277
|
return [7];
|
|
274
278
|
case 4:
|
|
275
279
|
console.log("[realtime] rebuildWatch " + (success ? 'success' : 'fail'));
|
|
@@ -287,18 +291,18 @@ var VirtualWebSocketClient = (function () {
|
|
|
287
291
|
errCode: isInitWatch
|
|
288
292
|
? ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_INIT_WATCH_FAIL
|
|
289
293
|
: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_REBUILD_WATCH_FAIL,
|
|
290
|
-
errMsg: e
|
|
294
|
+
errMsg: e,
|
|
291
295
|
}));
|
|
292
296
|
options.reject(e);
|
|
293
297
|
};
|
|
294
298
|
retry = function (refreshLogin) {
|
|
295
299
|
if (_this.useRetryTicket(options.operationName)) {
|
|
296
300
|
if (isInitWatch) {
|
|
297
|
-
_this.
|
|
301
|
+
_this.initWatchPromise = undefined;
|
|
298
302
|
options.resolve(_this.initWatch(refreshLogin));
|
|
299
303
|
}
|
|
300
304
|
else {
|
|
301
|
-
_this.
|
|
305
|
+
_this.rebuildWatchPromise = undefined;
|
|
302
306
|
options.resolve(_this.rebuildWatch(refreshLogin));
|
|
303
307
|
}
|
|
304
308
|
}
|
|
@@ -311,55 +315,57 @@ var VirtualWebSocketClient = (function () {
|
|
|
311
315
|
onTimeoutError: function () { return retry(false); },
|
|
312
316
|
onNotRetryableError: abortWatch,
|
|
313
317
|
onCancelledError: options.reject,
|
|
314
|
-
onUnknownError: function () {
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
return
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
318
|
+
onUnknownError: function () {
|
|
319
|
+
(function () { return __awaiter(_this, void 0, void 0, function () {
|
|
320
|
+
var onWSDisconnected, e_3;
|
|
321
|
+
var _this = this;
|
|
322
|
+
return __generator(this, function (_a) {
|
|
323
|
+
switch (_a.label) {
|
|
324
|
+
case 0:
|
|
325
|
+
_a.trys.push([0, 8, , 9]);
|
|
326
|
+
onWSDisconnected = function () { return __awaiter(_this, void 0, void 0, function () {
|
|
327
|
+
return __generator(this, function (_a) {
|
|
328
|
+
switch (_a.label) {
|
|
329
|
+
case 0:
|
|
330
|
+
this.pause();
|
|
331
|
+
return [4, this.onceWSConnected()];
|
|
332
|
+
case 1:
|
|
333
|
+
_a.sent();
|
|
334
|
+
retry(true);
|
|
335
|
+
return [2];
|
|
336
|
+
}
|
|
337
|
+
});
|
|
338
|
+
}); };
|
|
339
|
+
if (!!this.isWSConnected()) return [3, 2];
|
|
340
|
+
return [4, onWSDisconnected()];
|
|
341
|
+
case 1:
|
|
342
|
+
_a.sent();
|
|
343
|
+
return [3, 7];
|
|
344
|
+
case 2: return [4, sleep(DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR)];
|
|
345
|
+
case 3:
|
|
346
|
+
_a.sent();
|
|
347
|
+
if (!(this.watchStatus === WatchStatus.PAUSED)) return [3, 4];
|
|
348
|
+
options.reject(new CancelledError(options.operationName + " cancelled due to pause after unknownError"));
|
|
349
|
+
return [3, 7];
|
|
350
|
+
case 4:
|
|
351
|
+
if (!!this.isWSConnected()) return [3, 6];
|
|
352
|
+
return [4, onWSDisconnected()];
|
|
353
|
+
case 5:
|
|
354
|
+
_a.sent();
|
|
355
|
+
return [3, 7];
|
|
356
|
+
case 6:
|
|
357
|
+
retry(false);
|
|
358
|
+
_a.label = 7;
|
|
359
|
+
case 7: return [3, 9];
|
|
360
|
+
case 8:
|
|
361
|
+
e_3 = _a.sent();
|
|
362
|
+
retry(true);
|
|
363
|
+
return [3, 9];
|
|
364
|
+
case 9: return [2];
|
|
365
|
+
}
|
|
366
|
+
});
|
|
367
|
+
}); })();
|
|
368
|
+
},
|
|
363
369
|
});
|
|
364
370
|
return [2];
|
|
365
371
|
});
|
|
@@ -370,34 +376,34 @@ var VirtualWebSocketClient = (function () {
|
|
|
370
376
|
switch (_a.label) {
|
|
371
377
|
case 0:
|
|
372
378
|
queryId = this.sessionInfo ? this.sessionInfo.queryID : '';
|
|
373
|
-
if (this.watchStatus !==
|
|
374
|
-
this.watchStatus =
|
|
379
|
+
if (this.watchStatus !== WatchStatus.ACTIVE) {
|
|
380
|
+
this.watchStatus = WatchStatus.CLOSED;
|
|
375
381
|
this.onWatchClose(this, queryId);
|
|
376
382
|
return [2];
|
|
377
383
|
}
|
|
378
384
|
_a.label = 1;
|
|
379
385
|
case 1:
|
|
380
386
|
_a.trys.push([1, 3, 4, 5]);
|
|
381
|
-
this.watchStatus =
|
|
387
|
+
this.watchStatus = WatchStatus.CLOSING;
|
|
382
388
|
closeWatchMsg = {
|
|
383
389
|
watchId: this.watchId,
|
|
384
390
|
requestId: genRequestId(),
|
|
385
391
|
msgType: 'CLOSE_WATCH',
|
|
386
|
-
msgData: null
|
|
392
|
+
msgData: null,
|
|
387
393
|
};
|
|
388
394
|
return [4, this.send({
|
|
389
|
-
msg: closeWatchMsg
|
|
395
|
+
msg: closeWatchMsg,
|
|
390
396
|
})];
|
|
391
397
|
case 2:
|
|
392
398
|
_a.sent();
|
|
393
399
|
this.sessionInfo = undefined;
|
|
394
|
-
this.watchStatus =
|
|
400
|
+
this.watchStatus = WatchStatus.CLOSED;
|
|
395
401
|
return [3, 5];
|
|
396
402
|
case 3:
|
|
397
403
|
e_4 = _a.sent();
|
|
398
404
|
this.closeWithError(new CloudSDKError({
|
|
399
405
|
errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CLOSE_WATCH_FAIL,
|
|
400
|
-
errMsg: e_4
|
|
406
|
+
errMsg: e_4,
|
|
401
407
|
}));
|
|
402
408
|
return [3, 5];
|
|
403
409
|
case 4:
|
|
@@ -409,8 +415,8 @@ var VirtualWebSocketClient = (function () {
|
|
|
409
415
|
}); };
|
|
410
416
|
this.scheduleSendACK = function () {
|
|
411
417
|
_this.clearACKSchedule();
|
|
412
|
-
_this.
|
|
413
|
-
if (_this.
|
|
418
|
+
_this.ackTimeoutId = setTimeout(function () {
|
|
419
|
+
if (_this.waitExpectedTimeoutId) {
|
|
414
420
|
_this.scheduleSendACK();
|
|
415
421
|
}
|
|
416
422
|
else {
|
|
@@ -419,8 +425,8 @@ var VirtualWebSocketClient = (function () {
|
|
|
419
425
|
}, DEFAULT_SEND_ACK_DEBOUNCE_TIMEOUT);
|
|
420
426
|
};
|
|
421
427
|
this.clearACKSchedule = function () {
|
|
422
|
-
if (_this.
|
|
423
|
-
clearTimeout(_this.
|
|
428
|
+
if (_this.ackTimeoutId) {
|
|
429
|
+
clearTimeout(_this.ackTimeoutId);
|
|
424
430
|
}
|
|
425
431
|
};
|
|
426
432
|
this.sendACK = function () { return __awaiter(_this, void 0, void 0, function () {
|
|
@@ -429,7 +435,7 @@ var VirtualWebSocketClient = (function () {
|
|
|
429
435
|
switch (_a.label) {
|
|
430
436
|
case 0:
|
|
431
437
|
_a.trys.push([0, 2, , 3]);
|
|
432
|
-
if (this.watchStatus !==
|
|
438
|
+
if (this.watchStatus !== WatchStatus.ACTIVE) {
|
|
433
439
|
this.scheduleSendACK();
|
|
434
440
|
return [2];
|
|
435
441
|
}
|
|
@@ -443,11 +449,11 @@ var VirtualWebSocketClient = (function () {
|
|
|
443
449
|
msgType: 'CHECK_LAST',
|
|
444
450
|
msgData: {
|
|
445
451
|
queryID: this.sessionInfo.queryID,
|
|
446
|
-
eventID: this.sessionInfo.currentEventId
|
|
447
|
-
}
|
|
452
|
+
eventID: this.sessionInfo.currentEventId,
|
|
453
|
+
},
|
|
448
454
|
};
|
|
449
455
|
return [4, this.send({
|
|
450
|
-
msg: ackMsg
|
|
456
|
+
msg: ackMsg,
|
|
451
457
|
})];
|
|
452
458
|
case 1:
|
|
453
459
|
_a.sent();
|
|
@@ -471,7 +477,7 @@ var VirtualWebSocketClient = (function () {
|
|
|
471
477
|
case 'COLLECTION_PERMISSION_DENIED': {
|
|
472
478
|
this.closeWithError(new CloudSDKError({
|
|
473
479
|
errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL,
|
|
474
|
-
errMsg: msg.msgData.code
|
|
480
|
+
errMsg: msg.msgData.code,
|
|
475
481
|
}));
|
|
476
482
|
return [2];
|
|
477
483
|
}
|
|
@@ -480,15 +486,15 @@ var VirtualWebSocketClient = (function () {
|
|
|
480
486
|
}
|
|
481
487
|
}
|
|
482
488
|
}
|
|
483
|
-
if (this.
|
|
484
|
-
this.
|
|
485
|
-
this.
|
|
489
|
+
if (this.availableRetries.CHECK_LAST
|
|
490
|
+
&& this.availableRetries.CHECK_LAST > 0) {
|
|
491
|
+
this.availableRetries.CHECK_LAST -= 1;
|
|
486
492
|
this.scheduleSendACK();
|
|
487
493
|
}
|
|
488
494
|
else {
|
|
489
495
|
this.closeWithError(new CloudSDKError({
|
|
490
496
|
errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL,
|
|
491
|
-
errMsg: e_5
|
|
497
|
+
errMsg: e_5,
|
|
492
498
|
}));
|
|
493
499
|
}
|
|
494
500
|
return [3, 3];
|
|
@@ -544,40 +550,42 @@ var VirtualWebSocketClient = (function () {
|
|
|
544
550
|
this.onWatchStart = options.onWatchStart;
|
|
545
551
|
this.onWatchClose = options.onWatchClose;
|
|
546
552
|
this.debug = options.debug;
|
|
547
|
-
this.
|
|
553
|
+
this.availableRetries = {
|
|
548
554
|
INIT_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,
|
|
549
555
|
REBUILD_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,
|
|
550
|
-
CHECK_LAST: DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR
|
|
556
|
+
CHECK_LAST: DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR,
|
|
551
557
|
};
|
|
552
558
|
this.listener = new RealtimeListener({
|
|
553
|
-
close:
|
|
559
|
+
close: void (function () {
|
|
560
|
+
_this.closeWatch();
|
|
561
|
+
}),
|
|
554
562
|
onChange: options.onChange,
|
|
555
563
|
onError: options.onError,
|
|
556
564
|
debug: this.debug,
|
|
557
|
-
virtualClient: this
|
|
565
|
+
virtualClient: this,
|
|
558
566
|
});
|
|
559
567
|
this.initWatch();
|
|
560
568
|
}
|
|
561
569
|
VirtualWebSocketClient.prototype.onMessage = function (msg) {
|
|
562
570
|
var _this = this;
|
|
563
571
|
switch (this.watchStatus) {
|
|
564
|
-
case
|
|
572
|
+
case WatchStatus.PAUSED: {
|
|
565
573
|
if (msg.msgType !== 'ERROR') {
|
|
566
574
|
return;
|
|
567
575
|
}
|
|
568
576
|
break;
|
|
569
577
|
}
|
|
570
|
-
case
|
|
571
|
-
case
|
|
572
|
-
case
|
|
578
|
+
case WatchStatus.LOGGINGIN:
|
|
579
|
+
case WatchStatus.INITING:
|
|
580
|
+
case WatchStatus.REBUILDING: {
|
|
573
581
|
console.warn("[realtime listener] internal non-fatal error: unexpected message received while " + this.watchStatus);
|
|
574
582
|
return;
|
|
575
583
|
}
|
|
576
|
-
case
|
|
584
|
+
case WatchStatus.CLOSED: {
|
|
577
585
|
console.warn('[realtime listener] internal non-fatal error: unexpected message received when the watch has closed');
|
|
578
586
|
return;
|
|
579
587
|
}
|
|
580
|
-
case
|
|
588
|
+
case WatchStatus.ERRORED: {
|
|
581
589
|
console.warn('[realtime listener] internal non-fatal error: unexpected message received when the watch has ended with error');
|
|
582
590
|
return;
|
|
583
591
|
}
|
|
@@ -597,7 +605,7 @@ var VirtualWebSocketClient = (function () {
|
|
|
597
605
|
if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {
|
|
598
606
|
this.sessionInfo.expectEventId = msg.msgData.currEvent;
|
|
599
607
|
this.clearWaitExpectedEvent();
|
|
600
|
-
this.
|
|
608
|
+
this.waitExpectedTimeoutId = setTimeout(function () {
|
|
601
609
|
_this.rebuildWatch();
|
|
602
610
|
}, this.getWaitExpectedTimeoutLength());
|
|
603
611
|
console.log("[realtime] waitExpectedTimeoutLength " + this.getWaitExpectedTimeoutLength());
|
|
@@ -607,7 +615,7 @@ var VirtualWebSocketClient = (function () {
|
|
|
607
615
|
case 'ERROR': {
|
|
608
616
|
this.closeWithError(new CloudSDKError({
|
|
609
617
|
errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_SERVER_ERROR_MSG,
|
|
610
|
-
errMsg: msg.msgData.code + " - " + msg.msgData.message
|
|
618
|
+
errMsg: msg.msgData.code + " - " + msg.msgData.message,
|
|
611
619
|
}));
|
|
612
620
|
break;
|
|
613
621
|
}
|
|
@@ -618,14 +626,15 @@ var VirtualWebSocketClient = (function () {
|
|
|
618
626
|
}
|
|
619
627
|
};
|
|
620
628
|
VirtualWebSocketClient.prototype.closeWithError = function (error) {
|
|
621
|
-
|
|
629
|
+
var _a;
|
|
630
|
+
this.watchStatus = WatchStatus.ERRORED;
|
|
622
631
|
this.clearACKSchedule();
|
|
623
632
|
this.listener.onError(error);
|
|
624
|
-
this.onWatchClose(this, (this.sessionInfo
|
|
633
|
+
this.onWatchClose(this, ((_a = this.sessionInfo) === null || _a === void 0 ? void 0 : _a.queryID) || '');
|
|
625
634
|
console.log("[realtime] client closed (" + this.collectionName + " " + this.query + ") (watchId " + this.watchId + ")");
|
|
626
635
|
};
|
|
627
636
|
VirtualWebSocketClient.prototype.pause = function () {
|
|
628
|
-
this.watchStatus =
|
|
637
|
+
this.watchStatus = WatchStatus.PAUSED;
|
|
629
638
|
console.log("[realtime] client paused (" + this.collectionName + " " + this.query + ") (watchId " + this.watchId + ")");
|
|
630
639
|
};
|
|
631
640
|
VirtualWebSocketClient.prototype.resume = function () {
|
|
@@ -634,7 +643,7 @@ var VirtualWebSocketClient = (function () {
|
|
|
634
643
|
return __generator(this, function (_a) {
|
|
635
644
|
switch (_a.label) {
|
|
636
645
|
case 0:
|
|
637
|
-
this.watchStatus =
|
|
646
|
+
this.watchStatus = WatchStatus.RESUMING;
|
|
638
647
|
console.log("[realtime] client resuming with " + (this.sessionInfo ? 'REBUILD_WATCH' : 'INIT_WATCH') + " (" + this.collectionName + " " + this.query + ") (" + this.watchId + ")");
|
|
639
648
|
_a.label = 1;
|
|
640
649
|
case 1:
|
|
@@ -654,10 +663,10 @@ var VirtualWebSocketClient = (function () {
|
|
|
654
663
|
});
|
|
655
664
|
};
|
|
656
665
|
VirtualWebSocketClient.prototype.useRetryTicket = function (operationName) {
|
|
657
|
-
if (this.
|
|
658
|
-
this.
|
|
659
|
-
this.
|
|
660
|
-
console.log("[realtime] " + operationName + " use a retry ticket, now only " + this.
|
|
666
|
+
if (this.availableRetries[operationName]
|
|
667
|
+
&& this.availableRetries[operationName] > 0) {
|
|
668
|
+
this.availableRetries[operationName] -= 1;
|
|
669
|
+
console.log("[realtime] " + operationName + " use a retry ticket, now only " + this.availableRetries[operationName] + " retry left");
|
|
661
670
|
return true;
|
|
662
671
|
}
|
|
663
672
|
return false;
|
|
@@ -670,10 +679,10 @@ var VirtualWebSocketClient = (function () {
|
|
|
670
679
|
case 0:
|
|
671
680
|
_a.trys.push([0, 2, , 3]);
|
|
672
681
|
this.scheduleSendACK();
|
|
673
|
-
return [4, this.
|
|
682
|
+
return [4, this.handleServerEventsInternel(msg)];
|
|
674
683
|
case 1:
|
|
675
684
|
_a.sent();
|
|
676
|
-
this.
|
|
685
|
+
this.postHandleServerEventsValidityCheck(msg);
|
|
677
686
|
return [3, 3];
|
|
678
687
|
case 2:
|
|
679
688
|
e_7 = _a.sent();
|
|
@@ -684,7 +693,7 @@ var VirtualWebSocketClient = (function () {
|
|
|
684
693
|
});
|
|
685
694
|
});
|
|
686
695
|
};
|
|
687
|
-
VirtualWebSocketClient.prototype.
|
|
696
|
+
VirtualWebSocketClient.prototype.handleServerEventsInternel = function (msg) {
|
|
688
697
|
return __awaiter(this, void 0, void 0, function () {
|
|
689
698
|
var requestId, events, msgType, sessionInfo, allChangeEvents, docs, initEncountered, _loop_1, this_1, i, len, state_1;
|
|
690
699
|
return __generator(this, function (_a) {
|
|
@@ -703,14 +712,14 @@ var VirtualWebSocketClient = (function () {
|
|
|
703
712
|
catch (e) {
|
|
704
713
|
this.closeWithError(new CloudSDKError({
|
|
705
714
|
errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_RECEIVE_INVALID_SERVER_DATA,
|
|
706
|
-
errMsg: e
|
|
715
|
+
errMsg: e,
|
|
707
716
|
}));
|
|
708
717
|
return [2];
|
|
709
718
|
}
|
|
710
719
|
docs = __spreadArrays(sessionInfo.currentDocs);
|
|
711
720
|
initEncountered = false;
|
|
712
721
|
_loop_1 = function (i, len) {
|
|
713
|
-
var change, localDoc,
|
|
722
|
+
var change, localDoc, doc_1, _i, _a, fieldPath, err, err, doc, doc, err, ind, ind, docsSnapshot, docChanges, snapshot;
|
|
714
723
|
return __generator(this, function (_b) {
|
|
715
724
|
switch (_b.label) {
|
|
716
725
|
case 0:
|
|
@@ -733,19 +742,19 @@ var VirtualWebSocketClient = (function () {
|
|
|
733
742
|
case 'dequeue': {
|
|
734
743
|
localDoc = docs.find(function (doc) { return doc._id === change.docId; });
|
|
735
744
|
if (localDoc) {
|
|
736
|
-
|
|
745
|
+
doc_1 = cloneDeep(localDoc);
|
|
737
746
|
if (change.updatedFields) {
|
|
738
|
-
|
|
739
|
-
set(
|
|
740
|
-
}
|
|
747
|
+
Object.keys(change.updatedFields).forEach(function (fieldPath) {
|
|
748
|
+
set(doc_1, fieldPath, change.updatedFields[fieldPath]);
|
|
749
|
+
});
|
|
741
750
|
}
|
|
742
751
|
if (change.removedFields) {
|
|
743
752
|
for (_i = 0, _a = change.removedFields; _i < _a.length; _i++) {
|
|
744
753
|
fieldPath = _a[_i];
|
|
745
|
-
unset(
|
|
754
|
+
unset(doc_1, fieldPath);
|
|
746
755
|
}
|
|
747
756
|
}
|
|
748
|
-
change.doc =
|
|
757
|
+
change.doc = doc_1;
|
|
749
758
|
}
|
|
750
759
|
else {
|
|
751
760
|
console.error('[realtime listener] internal non-fatal server error: unexpected update dataType event where no doc is associated.');
|
|
@@ -755,7 +764,7 @@ var VirtualWebSocketClient = (function () {
|
|
|
755
764
|
case 'enqueue': {
|
|
756
765
|
err = new CloudSDKError({
|
|
757
766
|
errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR,
|
|
758
|
-
errMsg: "HandleServerEvents: full doc is not provided with dataType=\"update\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")"
|
|
767
|
+
errMsg: "HandleServerEvents: full doc is not provided with dataType=\"update\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")",
|
|
759
768
|
});
|
|
760
769
|
this_1.closeWithError(err);
|
|
761
770
|
throw err;
|
|
@@ -771,7 +780,7 @@ var VirtualWebSocketClient = (function () {
|
|
|
771
780
|
if (!change.doc) {
|
|
772
781
|
err = new CloudSDKError({
|
|
773
782
|
errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR,
|
|
774
|
-
errMsg: "HandleServerEvents: full doc is not provided with dataType=\"replace\" (requestId " + msg.requestId + ")"
|
|
783
|
+
errMsg: "HandleServerEvents: full doc is not provided with dataType=\"replace\" (requestId " + msg.requestId + ")",
|
|
775
784
|
});
|
|
776
785
|
this_1.closeWithError(err);
|
|
777
786
|
throw err;
|
|
@@ -804,7 +813,7 @@ var VirtualWebSocketClient = (function () {
|
|
|
804
813
|
case 'enqueue': {
|
|
805
814
|
err = new CloudSDKError({
|
|
806
815
|
errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR,
|
|
807
|
-
errMsg: "HandleServerEvents: full doc is not provided with dataType=\"limit\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")"
|
|
816
|
+
errMsg: "HandleServerEvents: full doc is not provided with dataType=\"limit\" and queueType=\"enqueue\" (requestId " + msg.requestId + ")",
|
|
808
817
|
});
|
|
809
818
|
this_1.closeWithError(err);
|
|
810
819
|
throw err;
|
|
@@ -853,8 +862,8 @@ var VirtualWebSocketClient = (function () {
|
|
|
853
862
|
break;
|
|
854
863
|
}
|
|
855
864
|
}
|
|
856
|
-
if (i === len - 1
|
|
857
|
-
(allChangeEvents[i + 1] && allChangeEvents[i + 1].id !== change.id)) {
|
|
865
|
+
if (i === len - 1
|
|
866
|
+
|| (allChangeEvents[i + 1] && allChangeEvents[i + 1].id !== change.id)) {
|
|
858
867
|
docsSnapshot = __spreadArrays(docs);
|
|
859
868
|
docChanges = allChangeEvents
|
|
860
869
|
.slice(0, i + 1)
|
|
@@ -865,7 +874,7 @@ var VirtualWebSocketClient = (function () {
|
|
|
865
874
|
id: change.id,
|
|
866
875
|
docChanges: docChanges,
|
|
867
876
|
docs: docsSnapshot,
|
|
868
|
-
msgType: msgType
|
|
877
|
+
msgType: msgType,
|
|
869
878
|
});
|
|
870
879
|
this_1.listener.onChange(snapshot);
|
|
871
880
|
}
|
|
@@ -899,13 +908,13 @@ var VirtualWebSocketClient = (function () {
|
|
|
899
908
|
});
|
|
900
909
|
});
|
|
901
910
|
};
|
|
902
|
-
VirtualWebSocketClient.prototype.
|
|
911
|
+
VirtualWebSocketClient.prototype.postHandleServerEventsValidityCheck = function (msg) {
|
|
903
912
|
if (!this.sessionInfo) {
|
|
904
913
|
console.error('[realtime listener] internal non-fatal error: sessionInfo lost after server event handling, this should never occur');
|
|
905
914
|
return;
|
|
906
915
|
}
|
|
907
|
-
if (this.sessionInfo.expectEventId
|
|
908
|
-
this.sessionInfo.currentEventId >= this.sessionInfo.expectEventId) {
|
|
916
|
+
if (this.sessionInfo.expectEventId
|
|
917
|
+
&& this.sessionInfo.currentEventId >= this.sessionInfo.expectEventId) {
|
|
909
918
|
this.clearWaitExpectedEvent();
|
|
910
919
|
}
|
|
911
920
|
if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {
|
|
@@ -914,9 +923,9 @@ var VirtualWebSocketClient = (function () {
|
|
|
914
923
|
}
|
|
915
924
|
};
|
|
916
925
|
VirtualWebSocketClient.prototype.clearWaitExpectedEvent = function () {
|
|
917
|
-
if (this.
|
|
918
|
-
clearTimeout(this.
|
|
919
|
-
this.
|
|
926
|
+
if (this.waitExpectedTimeoutId) {
|
|
927
|
+
clearTimeout(this.waitExpectedTimeoutId);
|
|
928
|
+
this.waitExpectedTimeoutId = undefined;
|
|
920
929
|
}
|
|
921
930
|
};
|
|
922
931
|
return VirtualWebSocketClient;
|
|
@@ -928,7 +937,7 @@ function getPublicEvent(event) {
|
|
|
928
937
|
dataType: event.DataType,
|
|
929
938
|
queueType: event.QueueType,
|
|
930
939
|
docId: event.DocID,
|
|
931
|
-
doc: event.Doc && event.Doc !== '{}' ? JSON.parse(event.Doc) : undefined
|
|
940
|
+
doc: event.Doc && event.Doc !== '{}' ? JSON.parse(event.Doc) : undefined,
|
|
932
941
|
};
|
|
933
942
|
if (event.DataType === 'update') {
|
|
934
943
|
if (event.UpdatedFields) {
|
|
@@ -940,4 +949,4 @@ function getPublicEvent(event) {
|
|
|
940
949
|
}
|
|
941
950
|
return e;
|
|
942
951
|
}
|
|
943
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"virtual-websocket-client.js","sourceRoot":"","sources":["../../src/virtual-websocket-client.ts"],"names":[],"mappings":"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","sourcesContent":["import set from 'lodash.set'\nimport unset from 'lodash.unset'\nimport cloneDeep from 'lodash.clonedeep'\nimport { genRequestId } from './message'\nimport {\n  IResponseMessage,\n  IRequestMessageInitWatchMsg,\n  IResponseMessageInitEventMsg,\n  IDBEvent,\n  IRequestMessageRebuildWatchMsg,\n  IRequestMessageCloseWatchMsg,\n  IRequestMsgType,\n  IResponseMessageNextEventMsg,\n  IRequestMessageCheckLastMsg,\n  IWatchOptions\n} from '@cloudbase/types/realtime'\nimport { \n  ISingleDBEvent\n} from '@cloudbase/types/database'\n// import Reporter from \"./externals/public-lib/reporter\"\nimport { RealtimeListener } from './listener'\nimport { Snapshot } from './snapshot'\nimport { IWSSendOptions, ILoginResult } from './websocket-client'\nimport { \n  ERR_CODE,\n  CloudSDKError,\n  isTimeoutError,\n  CancelledError,\n  isCancelledError,\n  isRealtimeErrorMessageError, \n  RealtimeErrorMessageError, \n  TimeoutError \n} from './error'\nimport { sleep } from './utils'\n\n// =============== Realtime Virtual WebSocket Client (Internal) ====================\n\ninterface IVirtualWebSocketClientConstructorOptions extends IWatchOptions {\n  // ws: RealtimeWebSocketClient\n  envId?: string\n  collectionName: string\n  query: string\n  limit?: number\n  orderBy?: Record<string, string>\n  send: <T = any>(opts: IWSSendOptions) => Promise<T>\n  login: (envId?: string, refresh?: boolean) => Promise<any>\n  isWSConnected: () => boolean\n  onceWSConnected: () => Promise<void>\n  getWaitExpectedTimeoutLength: () => number\n  onWatchStart: (client: VirtualWebSocketClient, queryID: string) => void\n  onWatchClose: (client: VirtualWebSocketClient, queryID: string) => void\n  debug?: boolean\n}\n\ninterface IWatchSessionInfo {\n  queryID: string\n  currentEventId: number\n  currentDocs: Record<string, any>[]\n  expectEventId?: number\n}\n\ninterface IHandleCommonErrorOptions {\n  onSignError: (e: RealtimeErrorMessageError) => void\n  onTimeoutError: (e: TimeoutError) => void\n  onCancelledError: (e: CancelledError) => void\n  onNotRetryableError: (e: RealtimeErrorMessageError) => void\n  onUnknownError: (e: any) => void\n}\n\ninterface IHandleWatchEstablishmentErrorOptions {\n  operationName: 'INIT_WATCH' | 'REBUILD_WATCH'\n  resolve: (value?: PromiseLike<void> | undefined) => void\n  reject: (e: any) => void\n  // retry: (refreshLogin?: boolean) => void\n  // abortWatch: (e: any) => void\n}\n\nenum WATCH_STATUS {\n  LOGGINGIN = 'LOGGINGIN',\n  INITING = 'INITING',\n  REBUILDING = 'REBUILDING',\n  ACTIVE = 'ACTIVE',\n  ERRORED = 'ERRORED',\n  CLOSING = 'CLOSING',\n  CLOSED = 'CLOSED',\n  PAUSED = 'PAUSED',\n  RESUMING = 'RESUMING'\n}\n\nconst DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR = 100\nconst DEFAULT_MAX_AUTO_RETRY_ON_ERROR = 2\nconst DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR = 2\nconst DEFAULT_SEND_ACK_DEBOUNCE_TIMEOUT = 10 * 1000\nconst DEFAULT_INIT_WATCH_TIMEOUT = 10 * 1000\nconst DEFAULT_REBUILD_WATCH_TIMEOUT = 10 * 1000\n\nexport class VirtualWebSocketClient {\n  // passed over\n  watchId: string\n  // own\n  listener: RealtimeListener\n  private envId?: string\n  private collectionName: string\n  private query: string\n  private limit: number\n  private orderBy: Record<string, string>\n  private send: <T = any>(opts: IWSSendOptions) => Promise<T>\n  private login: (envId?: string, refresh?: boolean) => Promise<any>\n  private isWSConnected: () => boolean\n  private onceWSConnected: () => Promise<void>\n  private getWaitExpectedTimeoutLength: () => number\n  private onWatchStart: (\n    client: VirtualWebSocketClient,\n    queryID: string\n  ) => void\n  private onWatchClose: (\n    client: VirtualWebSocketClient,\n    queryID: string\n  ) => void\n  private debug?: boolean\n\n  private watchStatus: WATCH_STATUS = WATCH_STATUS.INITING\n  private _availableRetries: Partial<Record<IRequestMsgType, number>>\n  private _ackTimeoutId?: number\n  private _initWatchPromise?: Promise<void>\n  private _rebuildWatchPromise?: Promise<void>\n\n  // obtained\n  private sessionInfo?: IWatchSessionInfo\n\n  // internal\n  private _waitExpectedTimeoutId?: number\n\n  constructor(options: IVirtualWebSocketClientConstructorOptions) {\n    this.watchId = `watchid_${+new Date()}_${Math.random()}`\n    this.envId = options.envId\n    this.collectionName = options.collectionName\n    this.query = options.query\n    this.limit = options.limit\n    this.orderBy = options.orderBy\n    this.send = options.send\n    this.login = options.login\n    this.isWSConnected = options.isWSConnected\n    this.onceWSConnected = options.onceWSConnected\n    this.getWaitExpectedTimeoutLength = options.getWaitExpectedTimeoutLength\n    this.onWatchStart = options.onWatchStart\n    this.onWatchClose = options.onWatchClose\n    this.debug = options.debug\n\n    this._availableRetries = {\n      INIT_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,\n      REBUILD_WATCH: DEFAULT_MAX_AUTO_RETRY_ON_ERROR,\n      CHECK_LAST: DEFAULT_MAX_SEND_ACK_AUTO_RETRY_ON_ERROR\n    }\n\n    this.listener = new RealtimeListener({\n      close: this.closeWatch,\n      onChange: options.onChange,\n      onError: options.onError,\n      debug: this.debug,\n      virtualClient: this\n    })\n\n    this.initWatch()\n  }\n\n  onMessage(msg: IResponseMessage) {\n    // watchStatus sanity check\n    switch (this.watchStatus) {\n      case WATCH_STATUS.PAUSED: {\n        // ignore all but error message\n        if (msg.msgType !== 'ERROR') {\n          return\n        }\n        break\n      }\n      case WATCH_STATUS.LOGGINGIN:\n      case WATCH_STATUS.INITING:\n      case WATCH_STATUS.REBUILDING: {\n        console.warn(\n          `[realtime listener] internal non-fatal error: unexpected message received while ${this.watchStatus}`\n        )\n        return\n      }\n      case WATCH_STATUS.CLOSED: {\n        console.warn(\n          '[realtime listener] internal non-fatal error: unexpected message received when the watch has closed'\n        )\n        return\n      }\n      case WATCH_STATUS.ERRORED: {\n        console.warn(\n          '[realtime listener] internal non-fatal error: unexpected message received when the watch has ended with error'\n        )\n        return\n      }\n    }\n\n    if (!this.sessionInfo) {\n      console.warn(\n        '[realtime listener] internal non-fatal error: sessionInfo not found while message is received.'\n      )\n      return\n    }\n\n    this.scheduleSendACK()\n\n    switch (msg.msgType) {\n      case 'NEXT_EVENT': {\n        // if (process.env.DEBUG) {\n        // @ts-ignore\n        // if (wx._ignore) {\n        console.warn(`nextevent ${msg.msgData.currEvent} ignored`, msg)\n        // @ts-ignore\n        // wx._ignore = false\n        // return\n        // }\n        // }\n\n        this.handleServerEvents(msg)\n        break\n      }\n      case 'CHECK_EVENT': {\n        if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {\n          // client eventID < server eventID:\n          // there might be one or more pending events not yet received but sent by the server\n          this.sessionInfo.expectEventId = msg.msgData.currEvent\n          this.clearWaitExpectedEvent()\n          // @ts-ignore\n          this._waitExpectedTimeoutId = setTimeout(() => {\n            // must rebuild watch\n            this.rebuildWatch()\n          }, this.getWaitExpectedTimeoutLength())\n\n          // if (process.env.DEBUG) {\n          console.log(\n            `[realtime] waitExpectedTimeoutLength ${this.getWaitExpectedTimeoutLength()}`\n          )\n          // }\n        }\n        break\n      }\n      case 'ERROR': {\n        // receive server error\n        this.closeWithError(\n          new CloudSDKError({\n            errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_SERVER_ERROR_MSG as string,\n            errMsg: `${msg.msgData.code} - ${msg.msgData.message}`\n          })\n        )\n        break\n      }\n      default: {\n        // if (process.env.DEBUG) {\n        console.warn(\n          `[realtime listener] virtual client receive unexpected msg ${msg.msgType}: `,\n          msg\n        )\n        // }\n        break\n      }\n    }\n  }\n\n  closeWithError(error: any) {\n    this.watchStatus = WATCH_STATUS.ERRORED\n    this.clearACKSchedule()\n    this.listener.onError(error)\n    // Reporter.surroundThirdByTryCatch(() => this.listener.onError(error))\n    this.onWatchClose(\n      this,\n      (this.sessionInfo && this.sessionInfo.queryID) || ''\n    )\n\n    // if (process.env.DEBUG) {\n    console.log(\n      `[realtime] client closed (${this.collectionName} ${this.query}) (watchId ${this.watchId})`\n    )\n    // }\n  }\n\n  pause() {\n    this.watchStatus = WATCH_STATUS.PAUSED\n    // if (process.env.DEBUG) {\n    console.log(\n      `[realtime] client paused (${this.collectionName} ${this.query}) (watchId ${this.watchId})`\n    )\n    // }\n  }\n\n  // resume() {\n  //   return this.sessionInfo ? this.rebuildWatch() : this.initWatch()\n  // }\n\n  async resume(): Promise<void> {\n    this.watchStatus = WATCH_STATUS.RESUMING\n\n    // if (process.env.DEBUG) {\n    console.log(\n      `[realtime] client resuming with ${\n        this.sessionInfo ? 'REBUILD_WATCH' : 'INIT_WATCH'\n      } (${this.collectionName} ${this.query}) (${this.watchId})`\n    )\n    // }\n\n    try {\n      await (this.sessionInfo ? this.rebuildWatch() : this.initWatch())\n\n      // if (process.env.DEBUG) {\n      console.log(\n        `[realtime] client successfully resumed (${this.collectionName} ${this.query}) (${this.watchId})`\n      )\n      // }\n    } catch (e) {\n      // if (process.env.DEBUG) {\n      console.error(\n        `[realtime] client resume failed (${this.collectionName} ${this.query}) (${this.watchId})`,\n        e\n      )\n      // }\n    }\n  }\n\n  private _login = async (\n    envId?: string,\n    refresh?: boolean\n  ): Promise<ILoginResult> => {\n    this.watchStatus = WATCH_STATUS.LOGGINGIN\n    const loginResult = await this.login(envId, refresh)\n    if (!this.envId) {\n      this.envId = loginResult.envId\n    }\n    return loginResult\n  }\n\n  private initWatch = async (forceRefreshLogin?: boolean): Promise<void> => {\n    if (this._initWatchPromise) {\n      return this._initWatchPromise\n    }\n\n    this._initWatchPromise = new Promise<void>(\n      async (resolve, reject): Promise<void> => {\n        try {\n          if (this.watchStatus === WATCH_STATUS.PAUSED) {\n            // if (process.env.DEBUG) {\n            console.log('[realtime] initWatch cancelled on pause')\n            // }\n            return resolve()\n          }\n\n          const { envId } = await this._login(this.envId, forceRefreshLogin)\n\n          // if (!this.sessionInfo) {\n          //   throw new Error(`can not rebuildWatch without a successful initWatch (lack of sessionInfo)`)\n          // }\n\n          if ((this.watchStatus as WATCH_STATUS) === WATCH_STATUS.PAUSED) {\n            console.log('[realtime] initWatch cancelled on pause')\n            return resolve()\n          }\n\n          this.watchStatus = WATCH_STATUS.INITING\n\n          const initWatchMsg: IRequestMessageInitWatchMsg = {\n            watchId: this.watchId,\n            requestId: genRequestId(),\n            msgType: 'INIT_WATCH',\n            msgData: {\n              envId,\n              collName: this.collectionName,\n              query: this.query,\n              limit: this.limit,\n              orderBy: this.orderBy\n            }\n          }\n\n          const initEventMsg = await this.send<IResponseMessageInitEventMsg>({\n            msg: initWatchMsg,\n            waitResponse: true,\n            skipOnMessage: true,\n            timeout: DEFAULT_INIT_WATCH_TIMEOUT\n          })\n\n          const { events, currEvent } = initEventMsg.msgData\n\n          this.sessionInfo = {\n            queryID: initEventMsg.msgData.queryID,\n            currentEventId: currEvent - 1,\n            currentDocs: []\n          }\n\n          // FIX: in initEvent message, all events have id 0, which is inconsistent with currEvent\n          if (events.length > 0) {\n            for (const e of events) {\n              e.ID = currEvent\n            }\n            this.handleServerEvents(initEventMsg)\n          } else {\n            this.sessionInfo.currentEventId = currEvent\n            const snapshot = new Snapshot({\n              id: currEvent,\n              docChanges: [],\n              docs: [],\n              type: 'init'\n            })\n            this.listener.onChange(snapshot)\n            this.scheduleSendACK()\n          }\n          this.onWatchStart(this, this.sessionInfo.queryID)\n          this.watchStatus = WATCH_STATUS.ACTIVE\n          this._availableRetries.INIT_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR\n          resolve()\n        } catch (e) {\n          this.handleWatchEstablishmentError(e, {\n            operationName: 'INIT_WATCH',\n            resolve,\n            reject\n          })\n        }\n      }\n    )\n\n    let success = false\n\n    try {\n      await this._initWatchPromise\n      success = true\n    } finally {\n      this._initWatchPromise = undefined\n    }\n\n    // if (process.env.DEBUG) {\n    console.log(`[realtime] initWatch ${success ? 'success' : 'fail'}`)\n    // }\n  }\n\n  private rebuildWatch = async (forceRefreshLogin?: boolean): Promise<void> => {\n    if (this._rebuildWatchPromise) {\n      return this._rebuildWatchPromise\n    }\n\n    this._rebuildWatchPromise = new Promise<void>(\n      async (resolve, reject): Promise<void> => {\n        try {\n          if (this.watchStatus === WATCH_STATUS.PAUSED) {\n            // if (process.env.DEBUG) {\n            console.log('[realtime] rebuildWatch cancelled on pause')\n            // }\n            return resolve()\n          }\n          const { envId } = await this._login(this.envId, forceRefreshLogin)\n\n          if (!this.sessionInfo) {\n            throw new Error(\n              'can not rebuildWatch without a successful initWatch (lack of sessionInfo)'\n            )\n          }\n\n          if ((this.watchStatus as WATCH_STATUS) === WATCH_STATUS.PAUSED) {\n            console.log('[realtime] rebuildWatch cancelled on pause')\n            return resolve()\n          }\n\n          this.watchStatus = WATCH_STATUS.REBUILDING\n\n          const rebuildWatchMsg: IRequestMessageRebuildWatchMsg = {\n            watchId: this.watchId,\n            requestId: genRequestId(),\n            msgType: 'REBUILD_WATCH',\n            msgData: {\n              envId,\n              collName: this.collectionName,\n              queryID: this.sessionInfo.queryID,\n              eventID: this.sessionInfo.currentEventId\n            }\n          }\n\n          const nextEventMsg = await this.send<IResponseMessageNextEventMsg>({\n            msg: rebuildWatchMsg,\n            waitResponse: true,\n            skipOnMessage: false,\n            timeout: DEFAULT_REBUILD_WATCH_TIMEOUT\n          })\n\n          this.handleServerEvents(nextEventMsg)\n\n          this.watchStatus = WATCH_STATUS.ACTIVE\n          this._availableRetries.REBUILD_WATCH = DEFAULT_MAX_AUTO_RETRY_ON_ERROR\n          resolve()\n        } catch (e) {\n          this.handleWatchEstablishmentError(e, {\n            operationName: 'REBUILD_WATCH',\n            resolve,\n            reject\n          })\n        }\n      }\n    )\n\n    let success = false\n\n    try {\n      await this._rebuildWatchPromise\n      success = true\n    } finally {\n      this._rebuildWatchPromise = undefined\n    }\n\n    // if (process.env.DEBUG) {\n    console.log(`[realtime] rebuildWatch ${success ? 'success' : 'fail'}`)\n    // }\n  }\n\n  private handleWatchEstablishmentError = async (\n    e: any,\n    options: IHandleWatchEstablishmentErrorOptions\n  ) => {\n    const isInitWatch = options.operationName === 'INIT_WATCH'\n\n    const abortWatch = () => {\n      // mock temp comment\n      this.closeWithError(\n        new CloudSDKError({\n          errCode: isInitWatch\n            ? (ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_INIT_WATCH_FAIL as string)\n            : (ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_REBUILD_WATCH_FAIL as string),\n          errMsg: e\n        })\n      )\n      options.reject(e)\n    }\n\n    const retry = (refreshLogin?: boolean) => {\n      if (this.useRetryTicket(options.operationName)) {\n        if (isInitWatch) {\n          this._initWatchPromise = undefined\n          options.resolve(this.initWatch(refreshLogin))\n        } else {\n          this._rebuildWatchPromise = undefined\n          options.resolve(this.rebuildWatch(refreshLogin))\n        }\n      } else {\n        abortWatch()\n      }\n    }\n\n    this.handleCommonError(e, {\n      onSignError: () => retry(true),\n      onTimeoutError: () => retry(false),\n      onNotRetryableError: abortWatch,\n      onCancelledError: options.reject,\n      onUnknownError: async () => {\n        try {\n          const onWSDisconnected = async () => {\n            this.pause()\n            await this.onceWSConnected()\n            retry(true)\n          }\n\n          if (!this.isWSConnected()) {\n            await onWSDisconnected()\n          } else {\n            await sleep(DEFAULT_WAIT_TIME_ON_UNKNOWN_ERROR)\n            if (this.watchStatus === WATCH_STATUS.PAUSED) {\n              // cancel\n              options.reject(\n                new CancelledError(\n                  `${options.operationName} cancelled due to pause after unknownError`\n                )\n              )\n            } else if (!this.isWSConnected()) {\n              await onWSDisconnected()\n            } else {\n              retry(false)\n            }\n          }\n        } catch (e) {\n          // unexpected error while handling error, in order to provide maximum effort on SEAMINGLESS FAULT TOLERANCE, just retry\n          retry(true)\n        }\n      }\n    })\n  }\n\n  private closeWatch = async () => {\n    const queryId = this.sessionInfo ? this.sessionInfo.queryID : ''\n\n    if (this.watchStatus !== WATCH_STATUS.ACTIVE) {\n      this.watchStatus = WATCH_STATUS.CLOSED\n      this.onWatchClose(this, queryId)\n      return\n    }\n\n    try {\n      this.watchStatus = WATCH_STATUS.CLOSING\n\n      const closeWatchMsg: IRequestMessageCloseWatchMsg = {\n        watchId: this.watchId,\n        requestId: genRequestId(),\n        msgType: 'CLOSE_WATCH',\n        msgData: null\n      }\n\n      await this.send({\n        msg: closeWatchMsg\n      })\n\n      this.sessionInfo = undefined\n      this.watchStatus = WATCH_STATUS.CLOSED\n    } catch (e) {\n      this.closeWithError(\n        new CloudSDKError({\n          errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CLOSE_WATCH_FAIL as string,\n          errMsg: e\n        })\n      )\n    } finally {\n      this.onWatchClose(this, queryId)\n    }\n  }\n\n  private scheduleSendACK = () => {\n    this.clearACKSchedule()\n\n    // TODO: should we check status after timeout\n    // @ts-ignore\n    this._ackTimeoutId = setTimeout(() => {\n      if (this._waitExpectedTimeoutId) {\n        this.scheduleSendACK()\n      } else {\n        this.sendACK()\n      }\n    }, DEFAULT_SEND_ACK_DEBOUNCE_TIMEOUT)\n  }\n\n  private clearACKSchedule = () => {\n    if (this._ackTimeoutId) {\n      clearTimeout(this._ackTimeoutId)\n    }\n  }\n\n  private sendACK = async (): Promise<void> => {\n    try {\n      if (this.watchStatus !== WATCH_STATUS.ACTIVE) {\n        this.scheduleSendACK()\n        return\n      }\n\n      if (!this.sessionInfo) {\n        console.warn(\n          '[realtime listener] can not send ack without a successful initWatch (lack of sessionInfo)'\n        )\n        return\n      }\n\n      const ackMsg: IRequestMessageCheckLastMsg = {\n        watchId: this.watchId,\n        requestId: genRequestId(),\n        msgType: 'CHECK_LAST',\n        msgData: {\n          queryID: this.sessionInfo.queryID,\n          eventID: this.sessionInfo.currentEventId\n        }\n      }\n\n      await this.send({\n        msg: ackMsg\n      })\n\n      this.scheduleSendACK()\n    } catch (e) {\n      // TODO: refactor\n      if (isRealtimeErrorMessageError(e)) {\n        const msg = e.payload\n        switch (msg.msgData.code) {\n          // signature error -> retry with refreshed signature\n          case 'CHECK_LOGIN_FAILED':\n          case 'SIGN_EXPIRED_ERROR':\n          case 'SIGN_INVALID_ERROR':\n          case 'SIGN_PARAM_INVALID': {\n            this.rebuildWatch()\n            return\n          }\n          // other -> throw\n          case 'QUERYID_INVALID_ERROR':\n          case 'SYS_ERR':\n          case 'INVALIID_ENV':\n          case 'COLLECTION_PERMISSION_DENIED': {\n            // must throw\n            this.closeWithError(\n              new CloudSDKError({\n                errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL as string,\n                errMsg: msg.msgData.code\n              })\n            )\n            return\n          }\n          default: {\n            break\n          }\n        }\n      }\n\n      // maybe retryable\n      if (\n        this._availableRetries.CHECK_LAST &&\n        this._availableRetries.CHECK_LAST > 0\n      ) {\n        this._availableRetries.CHECK_LAST--\n        this.scheduleSendACK()\n      } else {\n        this.closeWithError(\n          new CloudSDKError({\n            errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_CHECK_LAST_FAIL as string,\n            errMsg: e\n          })\n        )\n      }\n    }\n  }\n\n  private handleCommonError = (\n    e: any,\n    options: IHandleCommonErrorOptions\n  ): void => {\n    if (isRealtimeErrorMessageError(e)) {\n      const msg = e.payload\n      switch (msg.msgData.code) {\n        // signature error -> retry with refreshed signature\n        case 'CHECK_LOGIN_FAILED':\n        case 'SIGN_EXPIRED_ERROR':\n        case 'SIGN_INVALID_ERROR':\n        case 'SIGN_PARAM_INVALID': {\n          options.onSignError(e)\n          return\n        }\n        // not-retryable error -> throw\n        case 'QUERYID_INVALID_ERROR':\n        case 'SYS_ERR':\n        case 'INVALIID_ENV':\n        case 'COLLECTION_PERMISSION_DENIED': {\n          options.onNotRetryableError(e)\n          return\n        }\n        default: {\n          options.onNotRetryableError(e)\n          return\n        }\n      }\n    } else if (isTimeoutError(e)) {\n      // timeout error\n      options.onTimeoutError(e)\n      return\n    } else if (isCancelledError(e)) {\n      // cancelled error\n      options.onCancelledError(e)\n      return\n    }\n\n    // unknown error\n    options.onUnknownError(e)\n  }\n\n  // credit a retry chance from availableRetries\n  private useRetryTicket(operationName: IRequestMsgType): boolean {\n    if (\n      this._availableRetries[operationName] &&\n      this._availableRetries[operationName]! > 0\n    ) {\n      this._availableRetries[operationName]!--\n\n      // if (process.env.DEBUG) {\n      console.log(\n        `[realtime] ${operationName} use a retry ticket, now only ${this._availableRetries[operationName]} retry left`\n      )\n      // }\n\n      return true\n    }\n    return false\n  }\n\n  private async handleServerEvents(\n    msg: IResponseMessageInitEventMsg | IResponseMessageNextEventMsg\n  ) {\n    try {\n      this.scheduleSendACK()\n      await this._handleServerEvents(msg)\n      this._postHandleServerEventsValidityCheck(msg)\n    } catch (e) {\n      // if (process.env.DEBUG) {\n      // TODO: report\n      console.error(\n        '[realtime listener] internal non-fatal error: handle server events failed with error: ',\n        e\n      )\n\n      // writeToFile(\n      //   \"wserror.txt\",\n      //   `[realtime listener] internal non-fatal error: handle server events failed with error:  ${JSON.stringify(\n      //     Object.assign({}, e, {\n      //       requestId: msg.requestId,\n      //       watchId: msg.watchId\n      //     })\n      //   )} \\n`\n      // )\n      // }\n\n      throw e\n    }\n  }\n\n  private async _handleServerEvents(\n    msg: IResponseMessageInitEventMsg | IResponseMessageNextEventMsg\n  ) {\n    const { requestId } = msg\n\n    const { events } = msg.msgData\n    const { msgType } = msg\n\n    if (!events.length || !this.sessionInfo) {\n      return\n    }\n\n    const sessionInfo = this.sessionInfo\n\n    let allChangeEvents: ISingleDBEvent[]\n    try {\n      allChangeEvents = events.map(getPublicEvent)\n    } catch (e) {\n      this.closeWithError(\n        new CloudSDKError({\n          errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_RECEIVE_INVALID_SERVER_DATA as string,\n          errMsg: e\n        })\n      )\n      return\n    }\n\n    // aggregate docs\n    let docs = [...sessionInfo.currentDocs]\n    let initEncountered = false\n    for (let i = 0, len = allChangeEvents.length; i < len; i++) {\n      const change = allChangeEvents[i]\n\n      if (sessionInfo.currentEventId >= change.id) {\n        if (!allChangeEvents[i - 1] || change.id > allChangeEvents[i - 1].id) {\n          // duplicate event, dropable\n          // TODO: report\n          // if (process.env.DEBUG) {\n          console.warn(\n            `[realtime] duplicate event received, cur ${sessionInfo.currentEventId} but got ${change.id}`\n          )\n          // }\n        } else {\n          // allChangeEvents should be in ascending order according to eventId, this should never happens, must report a non-fatal error\n          console.error(\n            `[realtime listener] server non-fatal error: events out of order (the latter event's id is smaller than that of the former) (requestId ${requestId})`\n          )\n\n          // writeToFile(\n          //   \"wserror.txt\",\n          //   `[realtime listener] server non-fatal error: events out of order (the latter event's id is smaller than that of the former)  ${JSON.stringify(\n          //     Object.assign(\n          //       {},\n          //       {\n          //         requestId: msg.requestId,\n          //         watchId: msg.watchId\n          //       }\n          //     )\n          //   )} \\n`\n          // )\n        }\n        continue\n      } else if (sessionInfo.currentEventId === change.id - 1) {\n        // correct sequence\n        // first handle dataType then queueType:\n        // 1. dataType: we ONLY populate change.doc if neccessary\n        // 2. queueType: we build the data snapshot\n\n        switch (change.dataType) {\n          case 'update': {\n            // only need to populate change.doc when it is not provided\n            if (!change.doc) {\n              switch (change.queueType) {\n                case 'update':\n                case 'dequeue': {\n                  const localDoc = docs.find(doc => doc._id === change.docId)\n                  if (localDoc) {\n                    // a partial update\n                    const doc = cloneDeep(localDoc)\n\n                    if (change.updatedFields) {\n                      for (const fieldPath in change.updatedFields) {\n                        set(doc, fieldPath, change.updatedFields[fieldPath])\n                      }\n                    }\n\n                    if (change.removedFields) {\n                      for (const fieldPath of change.removedFields) {\n                        unset(doc, fieldPath)\n                      }\n                    }\n\n                    change.doc = doc\n                  } else {\n                    // TODO report\n                    console.error(\n                      '[realtime listener] internal non-fatal server error: unexpected update dataType event where no doc is associated.'\n                    )\n\n                    // writeToFile(\n                    //   \"wserror.txt\",\n                    //   `[realtime listener] internal non-fatal server error: unexpected update dataType event where no doc is associated.  ${JSON.stringify(\n                    //     Object.assign(\n                    //       {},\n                    //       {\n                    //         requestId: msg.requestId,\n                    //         watchId: msg.watchId\n                    //       }\n                    //     )\n                    //   )} \\n`\n                    // )\n                  }\n                  break\n                }\n                case 'enqueue': {\n                  // doc is provided by server, this should never occur\n                  const err = new CloudSDKError({\n                    errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR as string,\n                    errMsg: `HandleServerEvents: full doc is not provided with dataType=\"update\" and queueType=\"enqueue\" (requestId ${msg.requestId})`\n                  })\n                  this.closeWithError(err)\n                  throw err\n                }\n                default: {\n                  break\n                }\n              }\n            }\n            break\n          }\n          case 'replace': {\n            // validation\n            if (!change.doc) {\n              // doc is provided by server, this should never occur\n              const err = new CloudSDKError({\n                errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR as string,\n                errMsg: `HandleServerEvents: full doc is not provided with dataType=\"replace\" (requestId ${msg.requestId})`\n              })\n              this.closeWithError(err)\n              throw err\n            }\n            break\n          }\n          case 'remove': {\n            const doc = docs.find(doc => doc._id === change.docId)\n            if (doc) {\n              change.doc = doc\n            } else {\n              // TODO report\n              console.error(\n                '[realtime listener] internal non-fatal server error: unexpected remove event where no doc is associated.'\n              )\n\n              // writeToFile(\n              //   \"wserror.txt\",\n              //   `[realtime listener] internal non-fatal server error: unexpected remove event where no doc is associated. ${JSON.stringify(\n              //     Object.assign(\n              //       {},\n              //       {\n              //         requestId: msg.requestId,\n              //         watchId: msg.watchId\n              //       }\n              //     )\n              //   )} \\n`\n              // )\n            }\n            break\n          }\n          case 'limit': {\n            if (!change.doc) {\n              switch(change.queueType) {\n                case 'dequeue': {\n                  const doc = docs.find(doc => doc._id === change.docId)\n                  if (doc) {\n                    change.doc = doc\n                  } else {\n                    console.error(\n                      '[realtime listener] internal non-fatal server error: unexpected limit dataType event where no doc is associated.'\n                    )\n                  }\n                  break\n                }\n                case 'enqueue': {\n                  // doc is provided by server, this should never occur\n                  const err = new CloudSDKError({\n                    errCode: ERR_CODE.SDK_DATABASE_REALTIME_LISTENER_UNEXPECTED_FATAL_ERROR as string,\n                    errMsg: `HandleServerEvents: full doc is not provided with dataType=\"limit\" and queueType=\"enqueue\" (requestId ${msg.requestId})`\n                  })\n                  this.closeWithError(err)\n                  throw err\n                }\n                default: {\n                  break\n                }\n              }\n            }\n            break\n          }\n        }\n\n        switch (change.queueType) {\n          case 'init': {\n            if (!initEncountered) {\n              initEncountered = true\n              docs = [change.doc]\n            } else {\n              docs.push(change.doc)\n            }\n            break\n          }\n          case 'enqueue': {\n            docs.push(change.doc)\n            break\n          }\n          case 'dequeue': {\n            const ind = docs.findIndex(doc => doc._id === change.docId)\n            if (ind > -1) {\n              docs.splice(ind, 1)\n            } else {\n              // TODO report\n              console.error(\n                '[realtime listener] internal non-fatal server error: unexpected dequeue event where no doc is associated.'\n              )\n\n              // writeToFile(\n              //   \"wserror.txt\",\n              //   `[realtime listener] internal non-fatal server error: unexpected dequeue event where no doc is associated. ${JSON.stringify(\n              //     Object.assign(\n              //       {},\n              //       {\n              //         requestId: msg.requestId,\n              //         watchId: msg.watchId\n              //       }\n              //     )\n              //   )} \\n`\n              // )\n            }\n            break\n          }\n          case 'update': {\n            // writeToFile(\n            //   \"wserror.txt\",\n            //   `[realtime listener] docs ${JSON.stringify(\n            //     docs\n            //   )} change doc ${JSON.stringify(change)} \\n`\n            // )\n            const ind = docs.findIndex(doc => doc._id === change.docId)\n            if (ind > -1) {\n              docs[ind] = change.doc\n            } else {\n              // TODO report\n              console.error(\n                '[realtime listener] internal non-fatal server error: unexpected queueType update event where no doc is associated.'\n              )\n\n              // writeToFile(\n              //   \"wserror.txt\",\n              //   `[realtime listener] internal non-fatal server error: unexpected queueType update event where no doc is associated. ${JSON.stringify(\n              //     Object.assign(\n              //       {},\n              //       {\n              //         requestId: msg.requestId,\n              //         watchId: msg.watchId\n              //       }\n              //     )\n              //   )} \\n`\n              // )\n            }\n            break\n          }\n        }\n\n        if (\n          i === len - 1 ||\n          (allChangeEvents[i + 1] && allChangeEvents[i + 1].id !== change.id)\n        ) {\n          // a shallow slice creates a shallow snapshot\n          const docsSnapshot = [...docs]\n\n          // we slice first cause' if there're allChangeEvents that are of the same id after this change, we don't want to involve it for it is unexpected invalid order\n          const docChanges = allChangeEvents\n            .slice(0, i + 1)\n            .filter(c => c.id === change.id)\n\n          // all changes of this event has been handle, we could dispatch the event now\n          this.sessionInfo.currentEventId = change.id\n          this.sessionInfo.currentDocs = docs\n\n          const snapshot = new Snapshot({\n            id: change.id,\n            docChanges,\n            docs: docsSnapshot,\n            msgType\n          })\n\n          // Reporter.surroundThirdByTryCatch(() =>\n          this.listener.onChange(snapshot)\n          // )()\n        }\n      } else {\n        // out-of-order event\n        // if (process.env.DEBUG) {\n        // TODO: report\n        console.warn(\n          `[realtime listener] event received is out of order, cur ${this.sessionInfo.currentEventId} but got ${change.id}`\n        )\n        // }\n        // rebuild watch\n        await this.rebuildWatch()\n        return\n      }\n    }\n  }\n\n  private _postHandleServerEventsValidityCheck(\n    msg: IResponseMessageInitEventMsg | IResponseMessageNextEventMsg\n  ) {\n    if (!this.sessionInfo) {\n      console.error(\n        '[realtime listener] internal non-fatal error: sessionInfo lost after server event handling, this should never occur'\n      )\n\n      // writeToFile(\n      //   \"wserror.txt\",\n      //   `[realtime listener] internal non-fatal error: sessionInfo lost after server event handling, this should never occur ${JSON.stringify(\n      //     Object.assign(\n      //       {},\n      //       {\n      //         requestId: msg.requestId,\n      //         watchId: msg.watchId\n      //       }\n      //     )\n      //   )} \\n`\n      // )\n      return\n    }\n\n    if (\n      this.sessionInfo.expectEventId &&\n      this.sessionInfo.currentEventId >= this.sessionInfo.expectEventId\n    ) {\n      this.clearWaitExpectedEvent()\n    }\n\n    if (this.sessionInfo.currentEventId < msg.msgData.currEvent) {\n      console.warn(\n        '[realtime listener] internal non-fatal error: client eventId does not match with server event id after server event handling'\n      )\n      return\n    }\n  }\n\n  private clearWaitExpectedEvent() {\n    if (this._waitExpectedTimeoutId) {\n      clearTimeout(this._waitExpectedTimeoutId)\n      this._waitExpectedTimeoutId = undefined\n    }\n  }\n}\n\nfunction getPublicEvent(event: IDBEvent): ISingleDBEvent {\n  const e: ISingleDBEvent = {\n    id: event.ID,\n    dataType: event.DataType,\n    queueType: event.QueueType,\n    docId: event.DocID,\n    doc: event.Doc && event.Doc !== '{}' ? JSON.parse(event.Doc) : undefined\n  }\n\n  if (event.DataType === 'update') {\n    // @ts-ignore\n    if (event.UpdatedFields) {\n      e.updatedFields = JSON.parse(event.UpdatedFields)\n    }\n    // TODO: wait for tcb to change removedFields to RemovedFields\n    // @ts-ignore\n    if (event.removedFields || event.RemovedFields) {\n      // @ts-ignore\n      // e.removedFields = event.removedFields\n      //   ? JSON.parse(event.removedFields)\n      //   : JSON.parse(event.RemovedFields)\n      e.removedFields = JSON.parse(event.removedFields)\n    }\n  }\n\n  return e\n}\n"]}
|
|
952
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"virtual-websocket-client.js","sourceRoot":"","sources":["../../src/virtual-websocket-client.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,OAAO,GAAG,MAAM,YAAY,CAAA;AAC5B,OAAO,KAAK,MAAM,cAAc,CAAA;AAChC,OAAO,SAAS,MAAM,kBAAkB,CAAA;AACxC,OAAO,EAAE,YAAY,EAAE,MAAM,WAAW,CAAA;AAgBxC,OAAO,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAA;AAC7C,OAAO,EAAE,QAAQ,EAAE,MAAM,YAAY,CAAA;AAErC,OAAO,EACL,QAAQ,EACR,aAAa,EACb,cAAc,EACd,cAAc,EACd,gBAAgB,EAChB,2BAA2B,GAG5B,MAAM,SAAS,CAAA;AAChB,OAAO,EAAE,KAAK,EAAE,MAAM,SAAS,CAAA;AAyC/B,IAAK,WAUJ;AAVD,WAAK,WAAW;IACd,sCAAuB,CAAA;IACvB,kCAAmB,CAAA;IACnB,wCAAyB,CAAA;IACzB,gCAAiB,CAAA;IACjB,kCAAmB,CAAA;IACnB,kCAAmB,CAAA;IACnB,gCAAiB,CAAA;IACjB,gCAAiB,CAAA;IACjB,oCAAqB,CAAA;AACvB,CAAC,EAVI,WAAW,KAAX,WAAW,QAUf;AAED,IAAM,kCAAkC,GAAG,GAAG,CAAA;AAC9C,IAAM,+BAA+B,GAAG,CAAC,CAAA;AACzC,IAAM,wCAAwC,GAAG,CAAC,CAAA;AAClD,IAAM,iCAAiC,GAAG,EAAE,GAAG,IAAI,CAAA;AACnD,IAAM,0BAA0B,GAAG,EAAE,GAAG,IAAI,CAAA;AAC5C,IAAM,6BAA6B,GAAG,EAAE,GAAG,IAAI,CAAA;AAE/C;IAqCE,gCAAY,OAAkD;QAA9D,iBAiCC;QA7CO,gBAAW,GAAgB,WAAW,CAAC,OAAO,CAAA;QAsK9C,YAAO,GAAG,UAChB,KAAc,EACd,OAAiB;;;;;wBAEjB,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,SAAS,CAAA;wBACpB,WAAM,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,OAAO,CAAC,EAAA;;wBAA9C,WAAW,GAAG,SAAgC;wBACpD,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE;4BACf,IAAI,CAAC,KAAK,GAAG,WAAW,CAAC,KAAK,CAAA;yBAC/B;wBACD,WAAO,WAAW,EAAA;;;aACnB,CAAA;QAEO,cAAS,GAAG,UAAO,iBAA2B;;;;;;wBACpD,IAAI,IAAI,CAAC,gBAAgB,KAAK,IAAI,IAAI,IAAI,CAAC,gBAAgB,KAAK,SAAS,EAAE;4BACzE,WAAO,IAAI,CAAC,gBAAgB,EAAA;yBAC7B;wBAED,IAAI,CAAC,gBAAgB,GAAG,IAAI,OAAO,CAAO,UAAC,OAAO,EAAE,MAAM;4BACxD,KAAK,CAAC;;;;;;4CAEF,IAAI,IAAI,CAAC,WAAW,KAAK,WAAW,CAAC,MAAM,EAAE;gDAC3C,OAAO,CAAC,GAAG,CAAC,yCAAyC,CAAC,CAAA;gDACtD,WAAO,OAAO,EAAE,EAAA;6CACjB;4CAEiB,WAAM,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,EAAE,iBAAiB,CAAC,EAAA;;4CAA3D,KAAK,GAAK,CAAA,SAAiD,CAAA,MAAtD;4CACb,IAAK,IAAI,CAAC,WAA2B,KAAK,WAAW,CAAC,MAAM,EAAE;gDAC5D,OAAO,CAAC,GAAG,CAAC,yCAAyC,CAAC,CAAA;gDACtD,WAAO,OAAO,EAAE,EAAA;6CACjB;4CAED,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,OAAO,CAAA;4CAEhC,YAAY,GAAgC;gDAChD,OAAO,EAAE,IAAI,CAAC,OAAO;gDACrB,SAAS,EAAE,YAAY,EAAE;gDACzB,OAAO,EAAE,YAAY;gDACrB,OAAO,EAAE;oDACP,KAAK,OAAA;oDACL,QAAQ,EAAE,IAAI,CAAC,cAAc;oDAC7B,KAAK,EAAE,IAAI,CAAC,KAAK;oDACjB,KAAK,EAAE,IAAI,CAAC,KAAK;oDACjB,OAAO,EAAE,IAAI,CAAC,OAAO;iDACtB;6CACF,CAAA;4CAEoB,WAAM,IAAI,CAAC,IAAI,CAA+B;oDACjE,GAAG,EAAE,YAAY;oDACjB,YAAY,EAAE,IAAI;oDAClB,aAAa,EAAE,IAAI;oDACnB,OAAO,EAAE,0BAA0B;iDACpC,CAAC,EAAA;;4CALI,YAAY,GAAG,SAKnB;4CAEI,KAAwB,YAAY,CAAC,OAAO,EAA1C,MAAM,YAAA,EAAE,SAAS,eAAA,CAAyB;4CAElD,IAAI,CAAC,WAAW,GAAG;gDACjB,OAAO,EAAE,YAAY,CAAC,OAAO,CAAC,OAAO;gDACrC,cAAc,EAAE,SAAS,GAAG,CAAC;gDAC7B,WAAW,EAAE,EAAE;6CAChB,CAAA;4CAGD,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE;gDACrB,WAAsB,EAAN,iBAAM,EAAN,oBAAM,EAAN,IAAM,EAAE;oDAAb,CAAC;oDACV,CAAC,CAAC,EAAE,GAAG,SAAS,CAAA;iDACjB;gDACD,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAA;6CACtC;iDAAM;gDACL,IAAI,CAAC,WAAW,CAAC,cAAc,GAAG,SAAS,CAAA;gDACrC,QAAQ,GAAG,IAAI,QAAQ,CAAC;oDAC5B,EAAE,EAAE,SAAS;oDACb,UAAU,EAAE,EAAE;oDACd,IAAI,EAAE,EAAE;oDACR,IAAI,EAAE,MAAM;iDACb,CAAC,CAAA;gDACF,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAA;gDAChC,IAAI,CAAC,eAAe,EAAE,CAAA;6CACvB;4CACD,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAA;4CACjD,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,MAAM,CAAA;4CACrC,IAAI,CAAC,gBAAgB,CAAC,UAAU,GAAG,+BAA+B,CAAA;4CAClE,OAAO,EAAE,CAAA;;;;4CAET,IAAI,CAAC,6BAA6B,CAAC,GAAC,EAAE;gDACpC,aAAa,EAAE,YAAY;gDAC3B,OAAO,SAAA;gDACP,MAAM,QAAA;6CACP,CAAC,CAAA;;;;;iCAEL,CAAC,EAAE,CAAA;wBACN,CAAC,CAAC,CAAA;wBAEE,OAAO,GAAG,KAAK,CAAA;;;;wBAGjB,WAAM,IAAI,CAAC,gBAAgB,EAAA;;wBAA3B,SAA2B,CAAA;wBAC3B,OAAO,GAAG,IAAI,CAAA;;;wBAEd,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAA;;;wBAEnC,OAAO,CAAC,GAAG,CAAC,2BAAwB,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,CAAE,CAAC,CAAA;;;;aACpE,CAAA;QAEO,iBAAY,GAAG,UAAO,iBAA2B;;;;;;wBACvD,IAAI,IAAI,CAAC,mBAAmB,KAAK,IAAI,IAAI,IAAI,CAAC,mBAAmB,KAAK,SAAS,EAAE;4BAC/E,WAAO,IAAI,CAAC,mBAAmB,EAAA;yBAChC;wBAED,IAAI,CAAC,mBAAmB,GAAG,IAAI,OAAO,CAAO,UAAC,OAAO,EAAE,MAAM;4BAC3D,KAAK,CAAC;;;;;;4CAEF,IAAI,IAAI,CAAC,WAAW,KAAK,WAAW,CAAC,MAAM,EAAE;gDAC3C,OAAO,CAAC,GAAG,CAAC,4CAA4C,CAAC,CAAA;gDACzD,WAAO,OAAO,EAAE,EAAA;6CACjB;4CACiB,WAAM,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,EAAE,iBAAiB,CAAC,EAAA;;4CAA3D,KAAK,GAAK,CAAA,SAAiD,CAAA,MAAtD;4CAEb,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE;gDACrB,MAAM,IAAI,KAAK,CAAC,2EAA2E,CAAC,CAAA;6CAC7F;4CAED,IAAK,IAAI,CAAC,WAA2B,KAAK,WAAW,CAAC,MAAM,EAAE;gDAC5D,OAAO,CAAC,GAAG,CAAC,4CAA4C,CAAC,CAAA;gDACzD,WAAO,OAAO,EAAE,EAAA;6CACjB;4CAED,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,UAAU,CAAA;4CAEnC,eAAe,GAAmC;gDACtD,OAAO,EAAE,IAAI,CAAC,OAAO;gDACrB,SAAS,EAAE,YAAY,EAAE;gDACzB,OAAO,EAAE,eAAe;gDACxB,OAAO,EAAE;oDACP,KAAK,OAAA;oDACL,QAAQ,EAAE,IAAI,CAAC,cAAc;oDAC7B,OAAO,EAAE,IAAI,CAAC,WAAW,CAAC,OAAO;oDACjC,OAAO,EAAE,IAAI,CAAC,WAAW,CAAC,cAAc;iDACzC;6CACF,CAAA;4CAEoB,WAAM,IAAI,CAAC,IAAI,CAA+B;oDACjE,GAAG,EAAE,eAAe;oDACpB,YAAY,EAAE,IAAI;oDAClB,aAAa,EAAE,KAAK;oDACpB,OAAO,EAAE,6BAA6B;iDACvC,CAAC,EAAA;;4CALI,YAAY,GAAG,SAKnB;4CAEF,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAA;4CAErC,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,MAAM,CAAA;4CACrC,IAAI,CAAC,gBAAgB,CAAC,aAAa,GAAG,+BAA+B,CAAA;4CACrE,OAAO,EAAE,CAAA;;;;4CAET,IAAI,CAAC,6BAA6B,CAAC,GAAC,EAAE;gDACpC,aAAa,EAAE,eAAe;gDAC9B,OAAO,SAAA;gDACP,MAAM,QAAA;6CACP,CAAC,CAAA;;;;;iCAEL,CAAC,EAAE,CAAA;wBACN,CAAC,CAAC,CAAA;wBAEE,OAAO,GAAG,KAAK,CAAA;;;;wBAGjB,WAAM,IAAI,CAAC,mBAAmB,EAAA;;wBAA9B,SAA8B,CAAA;wBAC9B,OAAO,GAAG,IAAI,CAAA;;;wBAEd,IAAI,CAAC,mBAAmB,GAAG,SAAS,CAAA;;;wBAGtC,OAAO,CAAC,GAAG,CAAC,8BAA2B,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,CAAE,CAAC,CAAA;;;;aACvE,CAAA;QAEO,kCAA6B,GAAG,UACtC,CAAM,EACN,OAA8C;;;;gBAExC,WAAW,GAAG,OAAO,CAAC,aAAa,KAAK,YAAY,CAAA;gBAEpD,UAAU,GAAG;oBAEjB,KAAI,CAAC,cAAc,CAAC,IAAI,aAAa,CAAC;wBACpC,OAAO,EAAE,WAAW;4BAClB,CAAC,CAAE,QAAQ,CAAC,8CAAyD;4BACrE,CAAC,CAAE,QAAQ,CAAC,iDAA4D;wBAC1E,MAAM,EAAE,CAAC;qBACV,CAAC,CAAC,CAAA;oBACH,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAA;gBACnB,CAAC,CAAA;gBAEK,KAAK,GAAG,UAAC,YAAsB;oBACnC,IAAI,KAAI,CAAC,cAAc,CAAC,OAAO,CAAC,aAAa,CAAC,EAAE;wBAC9C,IAAI,WAAW,EAAE;4BACf,KAAI,CAAC,gBAAgB,GAAG,SAAS,CAAA;4BACjC,OAAO,CAAC,OAAO,CAAC,KAAI,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC,CAAA;yBAC9C;6BAAM;4BACL,KAAI,CAAC,mBAAmB,GAAG,SAAS,CAAA;4BACpC,OAAO,CAAC,OAAO,CAAC,KAAI,CAAC,YAAY,CAAC,YAAY,CAAC,CAAC,CAAA;yBACjD;qBACF;yBAAM;wBACL,UAAU,EAAE,CAAA;qBACb;gBACH,CAAC,CAAA;gBAED,IAAI,CAAC,iBAAiB,CAAC,CAAC,EAAE;oBACxB,WAAW,EAAE,cAAM,OAAA,KAAK,CAAC,IAAI,CAAC,EAAX,CAAW;oBAC9B,cAAc,EAAE,cAAM,OAAA,KAAK,CAAC,KAAK,CAAC,EAAZ,CAAY;oBAClC,mBAAmB,EAAE,UAAU;oBAC/B,gBAAgB,EAAE,OAAO,CAAC,MAAM;oBAChC,cAAc,EAAE;wBACd,CAAC;;;;;;;wCAES,gBAAgB,GAAG;;;;wDACvB,IAAI,CAAC,KAAK,EAAE,CAAA;wDACZ,WAAM,IAAI,CAAC,eAAe,EAAE,EAAA;;wDAA5B,SAA4B,CAAA;wDAC5B,KAAK,CAAC,IAAI,CAAC,CAAA;;;;6CACZ,CAAA;6CAEG,CAAC,IAAI,CAAC,aAAa,EAAE,EAArB,cAAqB;wCACvB,WAAM,gBAAgB,EAAE,EAAA;;wCAAxB,SAAwB,CAAA;;4CAExB,WAAM,KAAK,CAAC,kCAAkC,CAAC,EAAA;;wCAA/C,SAA+C,CAAA;6CAC3C,CAAA,IAAI,CAAC,WAAW,KAAK,WAAW,CAAC,MAAM,CAAA,EAAvC,cAAuC;wCAEzC,OAAO,CAAC,MAAM,CAAC,IAAI,cAAc,CAAI,OAAO,CAAC,aAAa,+CAA4C,CAAC,CAAC,CAAA;;;6CAC/F,CAAC,IAAI,CAAC,aAAa,EAAE,EAArB,cAAqB;wCAC9B,WAAM,gBAAgB,EAAE,EAAA;;wCAAxB,SAAwB,CAAA;;;wCAExB,KAAK,CAAC,KAAK,CAAC,CAAA;;;;;wCAKhB,KAAK,CAAC,IAAI,CAAC,CAAA;;;;;6BAEd,CAAC,EAAE,CAAA;oBACN,CAAC;iBACF,CAAC,CAAA;;;aACH,CAAA;QAEO,eAAU,GAAG;;;;;wBACb,OAAO,GAAG,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAA;wBAEhE,IAAI,IAAI,CAAC,WAAW,KAAK,WAAW,CAAC,MAAM,EAAE;4BAC3C,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,MAAM,CAAA;4BACrC,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,OAAO,CAAC,CAAA;4BAChC,WAAM;yBACP;;;;wBAGC,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,OAAO,CAAA;wBAEhC,aAAa,GAAiC;4BAClD,OAAO,EAAE,IAAI,CAAC,OAAO;4BACrB,SAAS,EAAE,YAAY,EAAE;4BACzB,OAAO,EAAE,aAAa;4BACtB,OAAO,EAAE,IAAI;yBACd,CAAA;wBAED,WAAM,IAAI,CAAC,IAAI,CAAC;gCACd,GAAG,EAAE,aAAa;6BACnB,CAAC,EAAA;;wBAFF,SAEE,CAAA;wBAEF,IAAI,CAAC,WAAW,GAAG,SAAS,CAAA;wBAC5B,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,MAAM,CAAA;;;;wBAErC,IAAI,CAAC,cAAc,CAAC,IAAI,aAAa,CAAC;4BACpC,OAAO,EAAE,QAAQ,CAAC,+CAAyD;4BAC3E,MAAM,EAAE,GAAC;yBACV,CAAC,CAAC,CAAA;;;wBAEH,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,OAAO,CAAC,CAAA;;;;;aAEnC,CAAA;QAEO,oBAAe,GAAG;YACxB,KAAI,CAAC,gBAAgB,EAAE,CAAA;YAIvB,KAAI,CAAC,YAAY,GAAG,UAAU,CAAC;gBAC7B,IAAI,KAAI,CAAC,qBAAqB,EAAE;oBAC9B,KAAI,CAAC,eAAe,EAAE,CAAA;iBACvB;qBAAM;oBACL,KAAI,CAAC,OAAO,EAAE,CAAA;iBACf;YACH,CAAC,EAAE,iCAAiC,CAAC,CAAA;QACvC,CAAC,CAAA;QAEO,qBAAgB,GAAG;YACzB,IAAI,KAAI,CAAC,YAAY,EAAE;gBACrB,YAAY,CAAC,KAAI,CAAC,YAAY,CAAC,CAAA;aAChC;QACH,CAAC,CAAA;QAEO,YAAO,GAAG;;;;;;wBAEd,IAAI,IAAI,CAAC,WAAW,KAAK,WAAW,CAAC,MAAM,EAAE;4BAC3C,IAAI,CAAC,eAAe,EAAE,CAAA;4BACtB,WAAM;yBACP;wBAED,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE;4BACrB,OAAO,CAAC,IAAI,CAAC,2FAA2F,CAAC,CAAA;4BACzG,WAAM;yBACP;wBAEK,MAAM,GAAgC;4BAC1C,OAAO,EAAE,IAAI,CAAC,OAAO;4BACrB,SAAS,EAAE,YAAY,EAAE;4BACzB,OAAO,EAAE,YAAY;4BACrB,OAAO,EAAE;gCACP,OAAO,EAAE,IAAI,CAAC,WAAW,CAAC,OAAO;gCACjC,OAAO,EAAE,IAAI,CAAC,WAAW,CAAC,cAAc;6BACzC;yBACF,CAAA;wBAED,WAAM,IAAI,CAAC,IAAI,CAAC;gCACd,GAAG,EAAE,MAAM;6BACZ,CAAC,EAAA;;wBAFF,SAEE,CAAA;wBAEF,IAAI,CAAC,eAAe,EAAE,CAAA;;;;wBAGtB,IAAI,2BAA2B,CAAC,GAAC,CAAC,EAAE;4BAC5B,GAAG,GAAG,GAAC,CAAC,OAAO,CAAA;4BACrB,QAAQ,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE;gCAExB,KAAK,oBAAoB,CAAC;gCAC1B,KAAK,oBAAoB,CAAC;gCAC1B,KAAK,oBAAoB,CAAC;gCAC1B,KAAK,oBAAoB,CAAC,CAAC;oCACzB,IAAI,CAAC,YAAY,EAAE,CAAA;oCACnB,WAAM;iCACP;gCAED,KAAK,uBAAuB,CAAC;gCAC7B,KAAK,SAAS,CAAC;gCACf,KAAK,cAAc,CAAC;gCACpB,KAAK,8BAA8B,CAAC,CAAC;oCAEnC,IAAI,CAAC,cAAc,CAAC,IAAI,aAAa,CAAC;wCACpC,OAAO,EAAE,QAAQ,CAAC,8CAAwD;wCAC1E,MAAM,EAAE,GAAG,CAAC,OAAO,CAAC,IAAI;qCACzB,CAAC,CAAC,CAAA;oCACH,WAAM;iCACP;gCACD,OAAO,CAAC,CAAC;oCACP,MAAK;iCACN;6BACF;yBACF;wBAGD,IACE,IAAI,CAAC,gBAAgB,CAAC,UAAU;+BAC7B,IAAI,CAAC,gBAAgB,CAAC,UAAU,GAAG,CAAC,EACvC;4BACA,IAAI,CAAC,gBAAgB,CAAC,UAAU,IAAI,CAAC,CAAA;4BACrC,IAAI,CAAC,eAAe,EAAE,CAAA;yBACvB;6BAAM;4BACL,IAAI,CAAC,cAAc,CAAC,IAAI,aAAa,CAAC;gCACpC,OAAO,EAAE,QAAQ,CAAC,8CAAwD;gCAC1E,MAAM,EAAE,GAAC;6BACV,CAAC,CAAC,CAAA;yBACJ;;;;;aAEJ,CAAA;QAEO,sBAAiB,GAAG,UAC1B,CAAM,EACN,OAAkC;YAElC,IAAI,2BAA2B,CAAC,CAAC,CAAC,EAAE;gBAClC,IAAM,GAAG,GAAG,CAAC,CAAC,OAAO,CAAA;gBACrB,QAAQ,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE;oBAExB,KAAK,oBAAoB,CAAC;oBAC1B,KAAK,oBAAoB,CAAC;oBAC1B,KAAK,oBAAoB,CAAC;oBAC1B,KAAK,oBAAoB,CAAC,CAAC;wBACzB,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC,CAAA;wBACtB,OAAM;qBACP;oBAED,KAAK,uBAAuB,CAAC;oBAC7B,KAAK,SAAS,CAAC;oBACf,KAAK,cAAc,CAAC;oBACpB,KAAK,8BAA8B,CAAC,CAAC;wBACnC,OAAO,CAAC,mBAAmB,CAAC,CAAC,CAAC,CAAA;wBAC9B,OAAM;qBACP;oBACD,OAAO,CAAC,CAAC;wBACP,OAAO,CAAC,mBAAmB,CAAC,CAAC,CAAC,CAAA;wBAC9B,OAAM;qBACP;iBACF;aACF;iBAAM,IAAI,cAAc,CAAC,CAAC,CAAC,EAAE;gBAE5B,OAAO,CAAC,cAAc,CAAC,CAAC,CAAC,CAAA;gBACzB,OAAM;aACP;iBAAM,IAAI,gBAAgB,CAAC,CAAC,CAAC,EAAE;gBAE9B,OAAO,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAA;gBAC3B,OAAM;aACP;YAGD,OAAO,CAAC,cAAc,CAAC,CAAC,CAAC,CAAA;QAC3B,CAAC,CAAA;QAnjBC,IAAI,CAAC,OAAO,GAAG,aAAW,CAAC,IAAI,IAAI,EAAE,SAAI,IAAI,CAAC,MAAM,EAAI,CAAA;QACxD,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAA;QAC1B,IAAI,CAAC,cAAc,GAAG,OAAO,CAAC,cAAc,CAAA;QAC5C,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAA;QAC1B,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAA;QAC1B,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,CAAA;QAC9B,IAAI,CAAC,IAAI,GAAG,OAAO,CAAC,IAAI,CAAA;QACxB,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAA;QAC1B,IAAI,CAAC,aAAa,GAAG,OAAO,CAAC,aAAa,CAAA;QAC1C,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC,eAAe,CAAA;QAC9C,IAAI,CAAC,4BAA4B,GAAG,OAAO,CAAC,4BAA4B,CAAA;QACxE,IAAI,CAAC,YAAY,GAAG,OAAO,CAAC,YAAY,CAAA;QACxC,IAAI,CAAC,YAAY,GAAG,OAAO,CAAC,YAAY,CAAA;QACxC,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAA;QAE1B,IAAI,CAAC,gBAAgB,GAAG;YACtB,UAAU,EAAE,+BAA+B;YAC3C,aAAa,EAAE,+BAA+B;YAC9C,UAAU,EAAE,wCAAwC;SACrD,CAAA;QAED,IAAI,CAAC,QAAQ,GAAG,IAAI,gBAAgB,CAAC;YACnC,KAAK,EAAE,KAAK,CAAC;gBACX,KAAI,CAAC,UAAU,EAAE,CAAA;YACnB,CAAC,CAAC;YACF,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,OAAO,EAAE,OAAO,CAAC,OAAO;YACxB,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,aAAa,EAAE,IAAI;SACpB,CAAC,CAAA;QAEF,IAAI,CAAC,SAAS,EAAE,CAAA;IAClB,CAAC;IAED,0CAAS,GAAT,UAAU,GAAqB;QAA/B,iBAgFC;QA9EC,QAAQ,IAAI,CAAC,WAAW,EAAE;YACxB,KAAK,WAAW,CAAC,MAAM,CAAC,CAAC;gBAEvB,IAAI,GAAG,CAAC,OAAO,KAAK,OAAO,EAAE;oBAC3B,OAAM;iBACP;gBACD,MAAK;aACN;YACD,KAAK,WAAW,CAAC,SAAS,CAAC;YAC3B,KAAK,WAAW,CAAC,OAAO,CAAC;YACzB,KAAK,WAAW,CAAC,UAAU,CAAC,CAAC;gBAC3B,OAAO,CAAC,IAAI,CAAC,qFAAmF,IAAI,CAAC,WAAa,CAAC,CAAA;gBACnH,OAAM;aACP;YACD,KAAK,WAAW,CAAC,MAAM,CAAC,CAAC;gBACvB,OAAO,CAAC,IAAI,CAAC,qGAAqG,CAAC,CAAA;gBACnH,OAAM;aACP;YACD,KAAK,WAAW,CAAC,OAAO,CAAC,CAAC;gBACxB,OAAO,CAAC,IAAI,CAAC,+GAA+G,CAAC,CAAA;gBAC7H,OAAM;aACP;SACF;QAED,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE;YACrB,OAAO,CAAC,IAAI,CAAC,gGAAgG,CAAC,CAAA;YAC9G,OAAM;SACP;QAED,IAAI,CAAC,eAAe,EAAE,CAAA;QAEtB,QAAQ,GAAG,CAAC,OAAO,EAAE;YACnB,KAAK,YAAY,CAAC,CAAC;gBAIjB,OAAO,CAAC,IAAI,CAAC,eAAa,GAAG,CAAC,OAAO,CAAC,SAAS,aAAU,EAAE,GAAG,CAAC,CAAA;gBAO/D,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,CAAA;gBAC5B,MAAK;aACN;YACD,KAAK,aAAa,CAAC,CAAC;gBAClB,IAAI,IAAI,CAAC,WAAW,CAAC,cAAc,GAAG,GAAG,CAAC,OAAO,CAAC,SAAS,EAAE;oBAG3D,IAAI,CAAC,WAAW,CAAC,aAAa,GAAG,GAAG,CAAC,OAAO,CAAC,SAAS,CAAA;oBACtD,IAAI,CAAC,sBAAsB,EAAE,CAAA;oBAE7B,IAAI,CAAC,qBAAqB,GAAG,UAAU,CAAC;wBAEtC,KAAI,CAAC,YAAY,EAAE,CAAA;oBACrB,CAAC,EAAE,IAAI,CAAC,4BAA4B,EAAE,CAAC,CAAA;oBAEvC,OAAO,CAAC,GAAG,CAAC,0CAAwC,IAAI,CAAC,4BAA4B,EAAI,CAAC,CAAA;iBAC3F;gBACD,MAAK;aACN;YACD,KAAK,OAAO,CAAC,CAAC;gBAEZ,IAAI,CAAC,cAAc,CAAC,IAAI,aAAa,CAAC;oBACpC,OAAO,EAAE,QAAQ,CAAC,+CAAyD;oBAC3E,MAAM,EAAK,GAAG,CAAC,OAAO,CAAC,IAAI,WAAM,GAAG,CAAC,OAAO,CAAC,OAAS;iBACvD,CAAC,CAAC,CAAA;gBACH,MAAK;aACN;YACD,OAAO,CAAC,CAAC;gBACP,OAAO,CAAC,IAAI,CACV,+DAA6D,GAAG,CAAC,OAAO,OAAI,EAC5E,GAAG,CACJ,CAAA;gBACD,MAAK;aACN;SACF;IACH,CAAC;IAED,+CAAc,GAAd,UAAe,KAAU;;QACvB,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,OAAO,CAAA;QACtC,IAAI,CAAC,gBAAgB,EAAE,CAAA;QACvB,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAA;QAC5B,IAAI,CAAC,YAAY,CACf,IAAI,EACJ,OAAC,IAAI,CAAC,WAAW,0CAAE,OAAO,CAAC,IAAI,EAAE,CAClC,CAAA;QAED,OAAO,CAAC,GAAG,CAAC,+BAA6B,IAAI,CAAC,cAAc,SAAI,IAAI,CAAC,KAAK,mBAAc,IAAI,CAAC,OAAO,MAAG,CAAC,CAAA;IAC1G,CAAC;IAED,sCAAK,GAAL;QACE,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,MAAM,CAAA;QACrC,OAAO,CAAC,GAAG,CAAC,+BAA6B,IAAI,CAAC,cAAc,SAAI,IAAI,CAAC,KAAK,mBAAc,IAAI,CAAC,OAAO,MAAG,CAAC,CAAA;IAC1G,CAAC;IAGK,uCAAM,GAAZ;;;;;;wBACE,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,QAAQ,CAAA;wBAEvC,OAAO,CAAC,GAAG,CAAC,sCACV,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,YAAY,WAC9C,IAAI,CAAC,cAAc,SAAI,IAAI,CAAC,KAAK,WAAM,IAAI,CAAC,OAAO,MAAG,CAAC,CAAA;;;;wBAG1D,WAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,EAAA;;wBAAjE,SAAiE,CAAA;wBAEjE,OAAO,CAAC,GAAG,CAAC,6CAA2C,IAAI,CAAC,cAAc,SAAI,IAAI,CAAC,KAAK,WAAM,IAAI,CAAC,OAAO,MAAG,CAAC,CAAA;;;;wBAE9G,OAAO,CAAC,KAAK,CACX,sCAAoC,IAAI,CAAC,cAAc,SAAI,IAAI,CAAC,KAAK,WAAM,IAAI,CAAC,OAAO,MAAG,EAC1F,GAAC,CACF,CAAA;;;;;;KAEJ;IA+ZO,+CAAc,GAAtB,UAAuB,aAA8B;QACnD,IACE,IAAI,CAAC,gBAAgB,CAAC,aAAa,CAAC;eACjC,IAAI,CAAC,gBAAgB,CAAC,aAAa,CAAE,GAAG,CAAC,EAC5C;YACA,IAAI,CAAC,gBAAgB,CAAC,aAAa,CAAE,IAAI,CAAC,CAAA;YAC1C,OAAO,CAAC,GAAG,CAAC,gBAAc,aAAa,sCAAiC,IAAI,CAAC,gBAAgB,CAAC,aAAa,CAAC,gBAAa,CAAC,CAAA;YAE1H,OAAO,IAAI,CAAA;SACZ;QACD,OAAO,KAAK,CAAA;IACd,CAAC;IAEa,mDAAkB,GAAhC,UAAiC,GAAgE;;;;;;;wBAE7F,IAAI,CAAC,eAAe,EAAE,CAAA;wBACtB,WAAM,IAAI,CAAC,0BAA0B,CAAC,GAAG,CAAC,EAAA;;wBAA1C,SAA0C,CAAA;wBAC1C,IAAI,CAAC,mCAAmC,CAAC,GAAG,CAAC,CAAA;;;;wBAG7C,OAAO,CAAC,KAAK,CACX,wFAAwF,EACxF,GAAC,CACF,CAAA;wBACD,MAAM,GAAC,CAAA;;;;;KAEV;IAEa,2DAA0B,GAAxC,UAAyC,GAAgE;;;;;;wBAC/F,SAAS,GAAK,GAAG,UAAR,CAAQ;wBAEjB,MAAM,GAAK,GAAG,CAAC,OAAO,OAAhB,CAAgB;wBACtB,OAAO,GAAK,GAAG,QAAR,CAAQ;wBAEvB,IAAI,CAAC,MAAM,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE;4BACvC,WAAM;yBACP;wBAEO,WAAW,GAAK,IAAI,YAAT,CAAS;wBAG5B,IAAI;4BACF,eAAe,GAAG,MAAM,CAAC,GAAG,CAAC,cAAc,CAAC,CAAA;yBAC7C;wBAAC,OAAO,CAAC,EAAE;4BACV,IAAI,CAAC,cAAc,CAAC,IAAI,aAAa,CAAC;gCACpC,OAAO,EAAE,QAAQ,CAAC,0DAAoE;gCACtF,MAAM,EAAE,CAAC;6BACV,CAAC,CAAC,CAAA;4BACH,WAAM;yBACP;wBAGG,IAAI,kBAAO,WAAW,CAAC,WAAW,CAAC,CAAA;wBACnC,eAAe,GAAG,KAAK,CAAA;4CAClB,CAAC,EAAM,GAAG;;;;;wCACX,MAAM,GAAG,eAAe,CAAC,CAAC,CAAC,CAAA;6CAE7B,CAAA,WAAW,CAAC,cAAc,IAAI,MAAM,CAAC,EAAE,CAAA,EAAvC,cAAuC;wCACzC,IAAI,CAAC,eAAe,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,MAAM,CAAC,EAAE,GAAG,eAAe,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;4CAIpE,OAAO,CAAC,IAAI,CAAC,8CAA4C,WAAW,CAAC,cAAc,iBAAY,MAAM,CAAC,EAAI,CAAC,CAAA;yCAE5G;6CAAM;4CAEL,OAAO,CAAC,KAAK,CAAC,2IAAyI,SAAS,MAAG,CAAC,CAAA;yCACrK;;;6CAEQ,CAAA,WAAW,CAAC,cAAc,KAAK,MAAM,CAAC,EAAE,GAAG,CAAC,CAAA,EAA5C,cAA4C;wCAMrD,QAAQ,MAAM,CAAC,QAAQ,EAAE;4CACvB,KAAK,QAAQ,CAAC,CAAC;gDAEb,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE;oDACf,QAAQ,MAAM,CAAC,SAAS,EAAE;wDACxB,KAAK,QAAQ,CAAC;wDACd,KAAK,SAAS,CAAC,CAAC;4DACR,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,UAAA,GAAG,IAAI,OAAA,GAAG,CAAC,GAAG,KAAK,MAAM,CAAC,KAAK,EAAxB,CAAwB,CAAC,CAAA;4DAC3D,IAAI,QAAQ,EAAE;gEAEN,QAAM,SAAS,CAAC,QAAQ,CAAC,CAAA;gEAE/B,IAAI,MAAM,CAAC,aAAa,EAAE;oEACxB,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,UAAC,SAAS;wEAClD,GAAG,CAAC,KAAG,EAAE,SAAS,EAAE,MAAM,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC,CAAA;oEACtD,CAAC,CAAC,CAAA;iEACH;gEAED,IAAI,MAAM,CAAC,aAAa,EAAE;oEACxB,WAA4C,EAApB,KAAA,MAAM,CAAC,aAAa,EAApB,cAAoB,EAApB,IAAoB,EAAE;wEAAnC,SAAS;wEAClB,KAAK,CAAC,KAAG,EAAE,SAAS,CAAC,CAAA;qEACtB;iEACF;gEAED,MAAM,CAAC,GAAG,GAAG,KAAG,CAAA;6DACjB;iEAAM;gEAEL,OAAO,CAAC,KAAK,CAAC,mHAAmH,CAAC,CAAA;6DACnI;4DACD,MAAK;yDACN;wDACD,KAAK,SAAS,CAAC,CAAC;4DAER,GAAG,GAAG,IAAI,aAAa,CAAC;gEAC5B,OAAO,EAAE,QAAQ,CAAC,qDAA+D;gEACjF,MAAM,EAAE,gHAA0G,GAAG,CAAC,SAAS,MAAG;6DACnI,CAAC,CAAA;4DACF,OAAK,cAAc,CAAC,GAAG,CAAC,CAAA;4DACxB,MAAM,GAAG,CAAA;yDACV;wDACD,OAAO,CAAC,CAAC;4DACP,MAAK;yDACN;qDACF;iDACF;gDACD,MAAK;6CACN;4CACD,KAAK,SAAS,CAAC,CAAC;gDAEd,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE;oDAET,GAAG,GAAG,IAAI,aAAa,CAAC;wDAC5B,OAAO,EAAE,QAAQ,CAAC,qDAA+D;wDACjF,MAAM,EAAE,uFAAmF,GAAG,CAAC,SAAS,MAAG;qDAC5G,CAAC,CAAA;oDACF,OAAK,cAAc,CAAC,GAAG,CAAC,CAAA;oDACxB,MAAM,GAAG,CAAA;iDACV;gDACD,MAAK;6CACN;4CACD,KAAK,QAAQ,CAAC,CAAC;gDACP,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,UAAA,GAAG,IAAI,OAAA,GAAG,CAAC,GAAG,KAAK,MAAM,CAAC,KAAK,EAAxB,CAAwB,CAAC,CAAA;gDACtD,IAAI,GAAG,EAAE;oDACP,MAAM,CAAC,GAAG,GAAG,GAAG,CAAA;iDACjB;qDAAM;oDAEL,OAAO,CAAC,KAAK,CAAC,0GAA0G,CAAC,CAAA;iDAC1H;gDACD,MAAK;6CACN;4CACD,KAAK,OAAO,CAAC,CAAC;gDACZ,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE;oDACf,QAAQ,MAAM,CAAC,SAAS,EAAE;wDACxB,KAAK,SAAS,CAAC,CAAC;4DACR,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,UAAA,GAAG,IAAI,OAAA,GAAG,CAAC,GAAG,KAAK,MAAM,CAAC,KAAK,EAAxB,CAAwB,CAAC,CAAA;4DACtD,IAAI,GAAG,EAAE;gEACP,MAAM,CAAC,GAAG,GAAG,GAAG,CAAA;6DACjB;iEAAM;gEACL,OAAO,CAAC,KAAK,CAAC,kHAAkH,CAAC,CAAA;6DAClI;4DACD,MAAK;yDACN;wDACD,KAAK,SAAS,CAAC,CAAC;4DAER,GAAG,GAAG,IAAI,aAAa,CAAC;gEAC5B,OAAO,EAAE,QAAQ,CAAC,qDAA+D;gEACjF,MAAM,EAAE,+GAAyG,GAAG,CAAC,SAAS,MAAG;6DAClI,CAAC,CAAA;4DACF,OAAK,cAAc,CAAC,GAAG,CAAC,CAAA;4DACxB,MAAM,GAAG,CAAA;yDACV;wDACD,OAAO,CAAC,CAAC;4DACP,MAAK;yDACN;qDACF;iDACF;gDACD,MAAK;6CACN;yCACF;wCAED,QAAQ,MAAM,CAAC,SAAS,EAAE;4CACxB,KAAK,MAAM,CAAC,CAAC;gDACX,IAAI,CAAC,eAAe,EAAE;oDACpB,eAAe,GAAG,IAAI,CAAA;oDACtB,IAAI,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAA;iDACpB;qDAAM;oDACL,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAA;iDACtB;gDACD,MAAK;6CACN;4CACD,KAAK,SAAS,CAAC,CAAC;gDACd,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAA;gDACrB,MAAK;6CACN;4CACD,KAAK,SAAS,CAAC,CAAC;gDACR,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,UAAA,GAAG,IAAI,OAAA,GAAG,CAAC,GAAG,KAAK,MAAM,CAAC,KAAK,EAAxB,CAAwB,CAAC,CAAA;gDAC3D,IAAI,GAAG,GAAG,CAAC,CAAC,EAAE;oDACZ,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,CAAC,CAAA;iDACpB;qDAAM;oDAEL,OAAO,CAAC,KAAK,CAAC,2GAA2G,CAAC,CAAA;iDAC3H;gDACD,MAAK;6CACN;4CACD,KAAK,QAAQ,CAAC,CAAC;gDACP,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,UAAA,GAAG,IAAI,OAAA,GAAG,CAAC,GAAG,KAAK,MAAM,CAAC,KAAK,EAAxB,CAAwB,CAAC,CAAA;gDAC3D,IAAI,GAAG,GAAG,CAAC,CAAC,EAAE;oDACZ,IAAI,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,GAAG,CAAA;iDACvB;qDAAM;oDAEL,OAAO,CAAC,KAAK,CAAC,oHAAoH,CAAC,CAAA;iDACpI;gDACD,MAAK;6CACN;yCACF;wCAED,IACE,CAAC,KAAK,GAAG,GAAG,CAAC;+CACV,CAAC,eAAe,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,eAAe,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,KAAK,MAAM,CAAC,EAAE,CAAC,EACtE;4CAEM,YAAY,kBAAO,IAAI,CAAC,CAAA;4CAGxB,UAAU,GAAG,eAAe;iDAC/B,KAAK,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC;iDACf,MAAM,CAAC,UAAA,CAAC,IAAI,OAAA,CAAC,CAAC,EAAE,KAAK,MAAM,CAAC,EAAE,EAAlB,CAAkB,CAAC,CAAA;4CAGlC,OAAK,WAAW,CAAC,cAAc,GAAG,MAAM,CAAC,EAAE,CAAA;4CAC3C,OAAK,WAAW,CAAC,WAAW,GAAG,IAAI,CAAA;4CAE7B,QAAQ,GAAG,IAAI,QAAQ,CAAC;gDAC5B,EAAE,EAAE,MAAM,CAAC,EAAE;gDACb,UAAU,YAAA;gDACV,IAAI,EAAE,YAAY;gDAClB,OAAO,SAAA;6CACR,CAAC,CAAA;4CAEF,OAAK,QAAQ,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAA;yCACjC;;;wCAKD,OAAO,CAAC,IAAI,CAAC,6DAA2D,OAAK,WAAW,CAAC,cAAc,iBAAY,MAAM,CAAC,EAAI,CAAC,CAAA;wCAG/H,WAAM,OAAK,YAAY,EAAE,EAAA;;wCAAzB,SAAyB,CAAA;;;;;;;wBA7LpB,CAAC,GAAG,CAAC,EAAE,GAAG,GAAG,eAAe,CAAC,MAAM;;;6BAAE,CAAA,CAAC,GAAG,GAAG,CAAA;2CAA5C,CAAC,EAAM,GAAG;;;;;;;wBAAoC,CAAC,EAAE,CAAA;;;;;;KAiM3D;IAEO,oEAAmC,GAA3C,UAA4C,GAAgE;QAC1G,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE;YACrB,OAAO,CAAC,KAAK,CAAC,qHAAqH,CAAC,CAAA;YACpI,OAAM;SACP;QAED,IACE,IAAI,CAAC,WAAW,CAAC,aAAa;eAC3B,IAAI,CAAC,WAAW,CAAC,cAAc,IAAI,IAAI,CAAC,WAAW,CAAC,aAAa,EACpE;YACA,IAAI,CAAC,sBAAsB,EAAE,CAAA;SAC9B;QAED,IAAI,IAAI,CAAC,WAAW,CAAC,cAAc,GAAG,GAAG,CAAC,OAAO,CAAC,SAAS,EAAE;YAC3D,OAAO,CAAC,IAAI,CAAC,8HAA8H,CAAC,CAAA;YAC5I,OAAM;SACP;IACH,CAAC;IAEO,uDAAsB,GAA9B;QACE,IAAI,IAAI,CAAC,qBAAqB,EAAE;YAC9B,YAAY,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAA;YACxC,IAAI,CAAC,qBAAqB,GAAG,SAAS,CAAA;SACvC;IACH,CAAC;IACH,6BAAC;AAAD,CAAC,AA92BD,IA82BC;;AAED,SAAS,cAAc,CAAC,KAAe;IACrC,IAAM,CAAC,GAAmB;QACxB,EAAE,EAAE,KAAK,CAAC,EAAE;QACZ,QAAQ,EAAE,KAAK,CAAC,QAAQ;QACxB,SAAS,EAAE,KAAK,CAAC,SAAS;QAC1B,KAAK,EAAE,KAAK,CAAC,KAAK;QAClB,GAAG,EAAE,KAAK,CAAC,GAAG,IAAI,KAAK,CAAC,GAAG,KAAK,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS;KACzE,CAAA;IAED,IAAI,KAAK,CAAC,QAAQ,KAAK,QAAQ,EAAE;QAE/B,IAAI,KAAK,CAAC,aAAa,EAAE;YACvB,CAAC,CAAC,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,aAAa,CAAC,CAAA;SAClD;QAGD,IAAI,KAAK,CAAC,aAAa,IAAI,KAAK,CAAC,aAAa,EAAE;YAK9C,CAAC,CAAC,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,aAAa,CAAC,CAAA;SAClD;KACF;IAED,OAAO,CAAC,CAAA;AACV,CAAC","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"]}
|