@splitsoftware/splitio-commons 1.17.1-rc.4 → 1.17.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGES.txt +28 -29
- package/LICENSE +1 -1
- package/README.md +4 -3
- package/cjs/consent/sdkUserConsent.js +4 -2
- package/cjs/evaluator/matchers/index.js +1 -3
- package/cjs/evaluator/matchers/matcherTypes.js +0 -1
- package/cjs/evaluator/matchers/segment.js +0 -6
- package/cjs/evaluator/matchersTransform/index.js +1 -4
- package/cjs/evaluator/matchersTransform/segment.js +1 -3
- package/cjs/logger/constants.js +2 -2
- package/cjs/logger/messages/info.js +1 -1
- package/cjs/logger/messages/warn.js +1 -1
- package/cjs/readiness/readinessManager.js +8 -18
- package/cjs/readiness/sdkReadinessManager.js +6 -5
- package/cjs/sdkClient/sdkClient.js +5 -5
- package/cjs/sdkClient/sdkClientMethod.js +1 -3
- package/cjs/sdkClient/sdkClientMethodCS.js +15 -9
- package/cjs/sdkClient/sdkClientMethodCSWithTT.js +15 -9
- package/cjs/sdkFactory/index.js +10 -32
- package/cjs/services/decorateHeaders.js +6 -1
- package/cjs/services/splitApi.js +5 -5
- package/cjs/services/splitHttpClient.js +5 -2
- package/cjs/storages/AbstractSegmentsCacheSync.js +33 -0
- package/cjs/storages/AbstractSplitsCacheSync.js +1 -2
- package/cjs/storages/KeyBuilderCS.js +5 -23
- package/cjs/storages/dataLoader.js +1 -1
- package/cjs/storages/inLocalStorage/MySegmentsCacheInLocal.js +56 -33
- package/cjs/storages/inLocalStorage/index.js +2 -6
- package/cjs/storages/inMemory/InMemoryStorageCS.js +1 -6
- package/cjs/storages/inMemory/MySegmentsCacheInMemory.js +44 -13
- package/cjs/storages/inMemory/SegmentsCacheInMemory.js +28 -14
- package/cjs/storages/inMemory/SplitsCacheInMemory.js +9 -8
- package/cjs/storages/inMemory/TelemetryCacheInMemory.js +10 -7
- package/cjs/storages/inRedis/SegmentsCacheInRedis.js +21 -15
- package/cjs/storages/inRedis/index.js +11 -5
- package/cjs/storages/pluggable/SegmentsCachePluggable.js +34 -13
- package/cjs/storages/pluggable/inMemoryWrapper.js +1 -1
- package/cjs/sync/offline/syncManagerOffline.js +11 -18
- package/cjs/sync/polling/fetchers/mySegmentsFetcher.js +8 -5
- package/cjs/sync/polling/fetchers/segmentChangesFetcher.js +1 -1
- package/cjs/sync/polling/pollingManagerCS.js +1 -1
- package/cjs/sync/polling/syncTasks/mySegmentsSyncTask.js +2 -2
- package/cjs/sync/polling/updaters/mySegmentsUpdater.js +21 -15
- package/cjs/sync/polling/updaters/segmentChangesUpdater.js +28 -12
- package/cjs/sync/polling/updaters/splitChangesUpdater.js +1 -1
- package/cjs/sync/streaming/AuthClient/index.js +1 -1
- package/cjs/sync/streaming/SSEHandler/index.js +5 -3
- package/cjs/sync/streaming/UpdateWorkers/MySegmentsUpdateWorker.js +48 -107
- package/cjs/sync/streaming/UpdateWorkers/SegmentsUpdateWorker.js +3 -3
- package/cjs/sync/streaming/constants.js +3 -3
- package/cjs/sync/streaming/parseUtils.js +9 -14
- package/cjs/sync/streaming/pushManager.js +67 -69
- package/cjs/sync/syncManagerOnline.js +21 -20
- package/cjs/sync/syncTask.js +2 -2
- package/cjs/trackers/eventTracker.js +10 -12
- package/cjs/trackers/impressionsTracker.js +14 -16
- package/cjs/trackers/uniqueKeysTracker.js +3 -5
- package/cjs/utils/constants/index.js +4 -5
- package/cjs/utils/settingsValidation/index.js +1 -2
- package/esm/consent/sdkUserConsent.js +4 -2
- package/esm/evaluator/matchers/index.js +1 -3
- package/esm/evaluator/matchers/matcherTypes.js +0 -1
- package/esm/evaluator/matchers/segment.js +0 -6
- package/esm/evaluator/matchersTransform/index.js +1 -4
- package/esm/evaluator/matchersTransform/segment.js +1 -3
- package/esm/logger/constants.js +1 -1
- package/esm/logger/messages/info.js +1 -1
- package/esm/logger/messages/warn.js +1 -1
- package/esm/readiness/readinessManager.js +8 -18
- package/esm/readiness/sdkReadinessManager.js +6 -5
- package/esm/sdkClient/sdkClient.js +5 -5
- package/esm/sdkClient/sdkClientMethod.js +1 -3
- package/esm/sdkClient/sdkClientMethodCS.js +13 -7
- package/esm/sdkClient/sdkClientMethodCSWithTT.js +13 -7
- package/esm/sdkFactory/index.js +10 -32
- package/esm/services/decorateHeaders.js +4 -0
- package/esm/services/splitApi.js +6 -6
- package/esm/services/splitHttpClient.js +6 -3
- package/esm/storages/AbstractSegmentsCacheSync.js +30 -0
- package/esm/storages/AbstractSplitsCacheSync.js +2 -3
- package/esm/storages/KeyBuilderCS.js +4 -21
- package/esm/storages/dataLoader.js +1 -1
- package/esm/storages/inLocalStorage/MySegmentsCacheInLocal.js +56 -33
- package/esm/storages/inLocalStorage/index.js +3 -7
- package/esm/storages/inMemory/InMemoryStorageCS.js +1 -6
- package/esm/storages/inMemory/MySegmentsCacheInMemory.js +44 -13
- package/esm/storages/inMemory/SegmentsCacheInMemory.js +28 -14
- package/esm/storages/inMemory/SplitsCacheInMemory.js +9 -8
- package/esm/storages/inMemory/TelemetryCacheInMemory.js +10 -7
- package/esm/storages/inRedis/SegmentsCacheInRedis.js +21 -15
- package/esm/storages/inRedis/index.js +11 -5
- package/esm/storages/pluggable/SegmentsCachePluggable.js +34 -13
- package/esm/storages/pluggable/inMemoryWrapper.js +1 -1
- package/esm/sync/offline/syncManagerOffline.js +11 -18
- package/esm/sync/polling/fetchers/mySegmentsFetcher.js +8 -5
- package/esm/sync/polling/fetchers/segmentChangesFetcher.js +1 -1
- package/esm/sync/polling/pollingManagerCS.js +1 -1
- package/esm/sync/polling/syncTasks/mySegmentsSyncTask.js +2 -2
- package/esm/sync/polling/updaters/mySegmentsUpdater.js +21 -15
- package/esm/sync/polling/updaters/segmentChangesUpdater.js +28 -12
- package/esm/sync/polling/updaters/splitChangesUpdater.js +1 -1
- package/esm/sync/streaming/AuthClient/index.js +1 -1
- package/esm/sync/streaming/SSEHandler/index.js +6 -4
- package/esm/sync/streaming/UpdateWorkers/MySegmentsUpdateWorker.js +49 -108
- package/esm/sync/streaming/UpdateWorkers/SegmentsUpdateWorker.js +3 -3
- package/esm/sync/streaming/constants.js +2 -2
- package/esm/sync/streaming/parseUtils.js +8 -12
- package/esm/sync/streaming/pushManager.js +70 -72
- package/esm/sync/syncManagerOnline.js +21 -20
- package/esm/sync/syncTask.js +2 -2
- package/esm/trackers/eventTracker.js +10 -12
- package/esm/trackers/impressionsTracker.js +14 -16
- package/esm/trackers/uniqueKeysTracker.js +3 -5
- package/esm/utils/constants/index.js +2 -3
- package/esm/utils/settingsValidation/index.js +1 -2
- package/package.json +2 -2
- package/src/consent/sdkUserConsent.ts +3 -2
- package/src/dtos/types.ts +7 -21
- package/src/evaluator/matchers/index.ts +0 -2
- package/src/evaluator/matchers/matcherTypes.ts +0 -1
- package/src/evaluator/matchers/segment.ts +0 -7
- package/src/evaluator/matchersTransform/index.ts +1 -4
- package/src/evaluator/matchersTransform/segment.ts +3 -5
- package/src/logger/constants.ts +1 -1
- package/src/logger/messages/info.ts +1 -1
- package/src/logger/messages/warn.ts +1 -1
- package/src/readiness/readinessManager.ts +8 -19
- package/src/readiness/sdkReadinessManager.ts +7 -7
- package/src/readiness/types.ts +2 -5
- package/src/sdkClient/sdkClient.ts +5 -5
- package/src/sdkClient/sdkClientMethod.ts +1 -4
- package/src/sdkClient/sdkClientMethodCS.ts +15 -7
- package/src/sdkClient/sdkClientMethodCSWithTT.ts +15 -7
- package/src/sdkFactory/index.ts +12 -35
- package/src/sdkFactory/types.ts +1 -4
- package/src/services/decorateHeaders.ts +5 -0
- package/src/services/splitApi.ts +7 -7
- package/src/services/splitHttpClient.ts +7 -4
- package/src/services/types.ts +2 -2
- package/src/storages/AbstractSegmentsCacheSync.ts +68 -0
- package/src/storages/AbstractSplitsCacheSync.ts +3 -4
- package/src/storages/KeyBuilderCS.ts +5 -34
- package/src/storages/dataLoader.ts +1 -1
- package/src/storages/inLocalStorage/MySegmentsCacheInLocal.ts +63 -33
- package/src/storages/inLocalStorage/index.ts +4 -8
- package/src/storages/inMemory/InMemoryStorageCS.ts +1 -6
- package/src/storages/inMemory/MySegmentsCacheInMemory.ts +47 -13
- package/src/storages/inMemory/SegmentsCacheInMemory.ts +27 -13
- package/src/storages/inMemory/SplitsCacheInMemory.ts +9 -7
- package/src/storages/inMemory/TelemetryCacheInMemory.ts +11 -7
- package/src/storages/inRedis/SegmentsCacheInRedis.ts +24 -15
- package/src/storages/inRedis/index.ts +12 -6
- package/src/storages/pluggable/SegmentsCachePluggable.ts +37 -13
- package/src/storages/pluggable/inMemoryWrapper.ts +1 -1
- package/src/storages/types.ts +17 -15
- package/src/sync/offline/syncManagerOffline.ts +13 -21
- package/src/sync/polling/fetchers/mySegmentsFetcher.ts +10 -8
- package/src/sync/polling/fetchers/segmentChangesFetcher.ts +1 -1
- package/src/sync/polling/fetchers/types.ts +2 -3
- package/src/sync/polling/pollingManagerCS.ts +4 -4
- package/src/sync/polling/syncTasks/mySegmentsSyncTask.ts +5 -4
- package/src/sync/polling/types.ts +6 -7
- package/src/sync/polling/updaters/mySegmentsUpdater.ts +22 -19
- package/src/sync/polling/updaters/segmentChangesUpdater.ts +29 -13
- package/src/sync/polling/updaters/splitChangesUpdater.ts +1 -1
- package/src/sync/streaming/AuthClient/index.ts +1 -1
- package/src/sync/streaming/SSEClient/index.ts +6 -4
- package/src/sync/streaming/SSEHandler/index.ts +8 -5
- package/src/sync/streaming/SSEHandler/types.ts +15 -15
- package/src/sync/streaming/UpdateWorkers/MySegmentsUpdateWorker.ts +49 -116
- package/src/sync/streaming/UpdateWorkers/SegmentsUpdateWorker.ts +4 -4
- package/src/sync/streaming/UpdateWorkers/SplitsUpdateWorker.ts +1 -1
- package/src/sync/streaming/UpdateWorkers/types.ts +2 -2
- package/src/sync/streaming/constants.ts +2 -2
- package/src/sync/streaming/parseUtils.ts +11 -19
- package/src/sync/streaming/pushManager.ts +72 -73
- package/src/sync/streaming/types.ts +10 -10
- package/src/sync/submitters/types.ts +5 -8
- package/src/sync/syncManagerOnline.ts +17 -17
- package/src/sync/syncTask.ts +2 -2
- package/src/sync/types.ts +1 -1
- package/src/trackers/eventTracker.ts +8 -11
- package/src/trackers/impressionsTracker.ts +10 -13
- package/src/trackers/types.ts +0 -1
- package/src/trackers/uniqueKeysTracker.ts +4 -6
- package/src/types.ts +1 -7
- package/src/utils/constants/index.ts +2 -3
- package/src/utils/settingsValidation/index.ts +2 -3
- package/src/utils/settingsValidation/types.ts +1 -1
- package/types/dtos/types.d.ts +7 -18
- package/types/evaluator/matchersTransform/segment.d.ts +2 -2
- package/types/logger/constants.d.ts +1 -1
- package/types/readiness/readinessManager.d.ts +2 -2
- package/types/readiness/sdkReadinessManager.d.ts +3 -2
- package/types/readiness/types.d.ts +2 -5
- package/types/sdkClient/sdkClientMethod.d.ts +1 -1
- package/types/sdkFactory/types.d.ts +1 -3
- package/types/services/decorateHeaders.d.ts +1 -0
- package/types/services/splitApi.d.ts +1 -1
- package/types/services/splitHttpClient.d.ts +1 -1
- package/types/services/types.d.ts +2 -2
- package/types/storages/AbstractSegmentsCacheSync.d.ts +11 -9
- package/types/storages/AbstractSplitsCacheSync.d.ts +1 -1
- package/types/storages/KeyBuilderCS.d.ts +2 -9
- package/types/storages/inLocalStorage/MySegmentsCacheInLocal.d.ts +18 -8
- package/types/storages/inMemory/MySegmentsCacheInMemory.d.ts +13 -7
- package/types/storages/inMemory/SegmentsCacheInMemory.d.ts +8 -6
- package/types/storages/inMemory/SplitsCacheInMemory.d.ts +2 -1
- package/types/storages/inMemory/TelemetryCacheInMemory.d.ts +6 -4
- package/types/storages/inRedis/SegmentsCacheInRedis.d.ts +4 -7
- package/types/storages/inRedis/index.d.ts +1 -1
- package/types/storages/pluggable/SegmentsCachePluggable.d.ts +17 -5
- package/types/storages/pluggable/inMemoryWrapper.d.ts +1 -1
- package/types/storages/types.d.ts +15 -11
- package/types/sync/polling/fetchers/mySegmentsFetcher.d.ts +2 -2
- package/types/sync/polling/fetchers/types.d.ts +2 -2
- package/types/sync/polling/syncTasks/mySegmentsSyncTask.d.ts +2 -2
- package/types/sync/polling/types.d.ts +4 -7
- package/types/sync/polling/updaters/mySegmentsUpdater.d.ts +3 -4
- package/types/sync/streaming/SSEHandler/types.d.ts +14 -16
- package/types/sync/streaming/UpdateWorkers/MySegmentsUpdateWorker.d.ts +2 -4
- package/types/sync/streaming/UpdateWorkers/SegmentsUpdateWorker.d.ts +1 -2
- package/types/sync/streaming/UpdateWorkers/SplitsUpdateWorker.d.ts +2 -3
- package/types/sync/streaming/UpdateWorkers/types.d.ts +2 -2
- package/types/sync/streaming/constants.d.ts +2 -2
- package/types/sync/streaming/parseUtils.d.ts +5 -4
- package/types/sync/streaming/types.d.ts +8 -8
- package/types/sync/submitters/types.d.ts +4 -7
- package/types/sync/types.d.ts +1 -1
- package/types/trackers/eventTracker.d.ts +1 -1
- package/types/trackers/impressionsTracker.d.ts +1 -1
- package/types/trackers/types.d.ts +0 -1
- package/types/types.d.ts +1 -7
- package/types/utils/constants/index.d.ts +2 -3
- package/types/utils/settingsValidation/types.d.ts +1 -1
- package/cjs/evaluator/matchers/large_segment.js +0 -16
- package/cjs/sdkClient/identity.js +0 -7
- package/cjs/storages/AbstractMySegmentsCacheSync.js +0 -60
- package/esm/evaluator/matchers/large_segment.js +0 -12
- package/esm/sdkClient/identity.js +0 -3
- package/esm/storages/AbstractMySegmentsCacheSync.js +0 -57
- package/src/evaluator/matchers/large_segment.ts +0 -18
- package/src/sdkClient/identity.ts +0 -5
- package/src/storages/AbstractMySegmentsCacheSync.ts +0 -94
- package/types/evaluator/matchers/large_segment.d.ts +0 -5
- package/types/evaluator/matchers/sember_inlist.d.ts +0 -3
- package/types/evaluator/matchersTransform/set.d.ts +0 -5
- package/types/evaluator/matchersTransform/string.d.ts +0 -7
- package/types/sdkClient/identity.d.ts +0 -2
- package/types/storages/AbstractMySegmentsCacheSync.d.ts +0 -39
- package/types/storages/AbstractSplitsCache.d.ts +0 -46
- package/types/sync/streaming/mySegmentsV2utils.d.ts +0 -27
- package/types/sync/streaming/pushManagerCS_Spec1_3.d.ts +0 -9
- package/types/sync/streaming/pushManager_Spec1_3.d.ts +0 -9
- package/types/trackers/impressionObserver/utils.d.ts +0 -5
- package/types/utils/inputValidation/sdkKey.d.ts +0 -7
- package/types/utils/inputValidation/splitExistance.d.ts +0 -7
- package/types/utils/inputValidation/trafficTypeExistance.d.ts +0 -9
- package/types/utils/redis/RedisMock.d.ts +0 -4
- package/types/utils/settingsValidation/logger/globalLogLevel.d.ts +0 -8
- /package/types/utils/{semVer.d.ts → Semver.d.ts} +0 -0
|
@@ -12,19 +12,32 @@ var SegmentsCachePluggable = /** @class */ (function () {
|
|
|
12
12
|
this.wrapper = wrapper;
|
|
13
13
|
}
|
|
14
14
|
/**
|
|
15
|
-
*
|
|
16
|
-
* The returned promise is resolved
|
|
17
|
-
* or rejected if
|
|
15
|
+
* Add a list of `segmentKeys` to the given segment `name`.
|
|
16
|
+
* The returned promise is resolved when the operation success
|
|
17
|
+
* or rejected if wrapper operation fails.
|
|
18
18
|
*/
|
|
19
|
-
SegmentsCachePluggable.prototype.
|
|
19
|
+
SegmentsCachePluggable.prototype.addToSegment = function (name, segmentKeys) {
|
|
20
20
|
var segmentKey = this.keys.buildSegmentNameKey(name);
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
21
|
+
if (segmentKeys.length) {
|
|
22
|
+
return this.wrapper.addItems(segmentKey, segmentKeys);
|
|
23
|
+
}
|
|
24
|
+
else {
|
|
25
|
+
return Promise.resolve();
|
|
26
|
+
}
|
|
27
|
+
};
|
|
28
|
+
/**
|
|
29
|
+
* Remove a list of `segmentKeys` from the given segment `name`.
|
|
30
|
+
* The returned promise is resolved when the operation success
|
|
31
|
+
* or rejected if wrapper operation fails.
|
|
32
|
+
*/
|
|
33
|
+
SegmentsCachePluggable.prototype.removeFromSegment = function (name, segmentKeys) {
|
|
34
|
+
var segmentKey = this.keys.buildSegmentNameKey(name);
|
|
35
|
+
if (segmentKeys.length) {
|
|
36
|
+
return this.wrapper.removeItems(segmentKey, segmentKeys);
|
|
37
|
+
}
|
|
38
|
+
else {
|
|
39
|
+
return Promise.resolve();
|
|
40
|
+
}
|
|
28
41
|
};
|
|
29
42
|
/**
|
|
30
43
|
* Returns a promise that resolves with a boolean value indicating if `key` is part of `name` segment.
|
|
@@ -33,6 +46,14 @@ var SegmentsCachePluggable = /** @class */ (function () {
|
|
|
33
46
|
SegmentsCachePluggable.prototype.isInSegment = function (name, key) {
|
|
34
47
|
return this.wrapper.itemContains(this.keys.buildSegmentNameKey(name), key);
|
|
35
48
|
};
|
|
49
|
+
/**
|
|
50
|
+
* Set till number for the given segment `name`.
|
|
51
|
+
* The returned promise is resolved when the operation success,
|
|
52
|
+
* or rejected if it fails (e.g., wrapper operation fails).
|
|
53
|
+
*/
|
|
54
|
+
SegmentsCachePluggable.prototype.setChangeNumber = function (name, changeNumber) {
|
|
55
|
+
return this.wrapper.set(this.keys.buildSegmentTillKey(name), changeNumber + '');
|
|
56
|
+
};
|
|
36
57
|
/**
|
|
37
58
|
* Get till number or -1 if it's not defined.
|
|
38
59
|
* The returned promise is resolved with the changeNumber or -1 if it doesn't exist or a wrapper operation fails.
|
|
@@ -42,10 +63,10 @@ var SegmentsCachePluggable = /** @class */ (function () {
|
|
|
42
63
|
var _this = this;
|
|
43
64
|
return this.wrapper.get(this.keys.buildSegmentTillKey(name)).then(function (value) {
|
|
44
65
|
var i = parseInt(value, 10);
|
|
45
|
-
return isNaNNumber(i) ?
|
|
66
|
+
return isNaNNumber(i) ? undefined : i;
|
|
46
67
|
}).catch(function (e) {
|
|
47
68
|
_this.log.error(LOG_PREFIX + 'Could not retrieve changeNumber from segments storage. Error: ' + e);
|
|
48
|
-
return
|
|
69
|
+
return undefined;
|
|
49
70
|
});
|
|
50
71
|
};
|
|
51
72
|
/**
|
|
@@ -5,7 +5,7 @@ import { setToArray, _Set } from '../../utils/lang/sets';
|
|
|
5
5
|
* The `_cache` property is the object were items are stored.
|
|
6
6
|
* Intended for testing purposes.
|
|
7
7
|
*
|
|
8
|
-
* @param connDelay delay in millis for `connect` resolve. If not provided, `connect` resolves
|
|
8
|
+
* @param connDelay delay in millis for `connect` resolve. If not provided, `connect` resolves inmediatelly.
|
|
9
9
|
*/
|
|
10
10
|
export function inMemoryWrapperFactory(connDelay) {
|
|
11
11
|
var _cache = {};
|
|
@@ -16,30 +16,23 @@ export function syncManagerOfflineFactory(splitsParserFactory) {
|
|
|
16
16
|
*/
|
|
17
17
|
return function (_a) {
|
|
18
18
|
var settings = _a.settings, readiness = _a.readiness, storage = _a.storage;
|
|
19
|
-
|
|
20
|
-
var mainStart = mainSyncManager.start;
|
|
21
|
-
var sharedStarts = [];
|
|
22
|
-
return objectAssign(mainSyncManager, {
|
|
23
|
-
start: function () {
|
|
24
|
-
mainStart();
|
|
25
|
-
sharedStarts.forEach(function (cb) { return cb(); });
|
|
26
|
-
sharedStarts.length = 0;
|
|
27
|
-
},
|
|
19
|
+
return objectAssign(fromObjectSyncTaskFactory(splitsParserFactory(), storage, readiness, settings), {
|
|
28
20
|
// fake flush, that resolves immediately
|
|
29
21
|
flush: flush,
|
|
30
22
|
// [Only used for client-side]
|
|
31
23
|
shared: function (matchingKey, readinessManager) {
|
|
32
|
-
// In LOCALHOST mode, shared clients are ready in the next event-loop cycle than created
|
|
33
|
-
// SDK_READY cannot be emitted directly because this will not update the readiness status
|
|
34
|
-
function emitSdkReady() {
|
|
35
|
-
readinessManager.segments.emit(SDK_SEGMENTS_ARRIVED); // SDK_SPLITS_ARRIVED emitted by main SyncManager
|
|
36
|
-
}
|
|
37
|
-
if (mainSyncManager.isRunning())
|
|
38
|
-
setTimeout(emitSdkReady);
|
|
39
|
-
else
|
|
40
|
-
sharedStarts.push(emitSdkReady);
|
|
41
24
|
return {
|
|
25
|
+
start: function () {
|
|
26
|
+
// In LOCALHOST mode, shared clients are ready in the next event-loop cycle than created
|
|
27
|
+
// SDK_READY cannot be emitted directly because this will not update the readiness status
|
|
28
|
+
setTimeout(function () {
|
|
29
|
+
readinessManager.segments.emit(SDK_SEGMENTS_ARRIVED); // SDK_SPLITS_ARRIVED emitted by main SyncManager
|
|
30
|
+
}, 0);
|
|
31
|
+
},
|
|
42
32
|
stop: function () { },
|
|
33
|
+
isRunning: function () {
|
|
34
|
+
return true;
|
|
35
|
+
},
|
|
43
36
|
flush: flush,
|
|
44
37
|
};
|
|
45
38
|
}
|
|
@@ -2,13 +2,16 @@
|
|
|
2
2
|
* Factory of MySegments fetcher.
|
|
3
3
|
* MySegments fetcher is a wrapper around `mySegments` API service that parses the response and handle errors.
|
|
4
4
|
*/
|
|
5
|
-
export function mySegmentsFetcherFactory(
|
|
6
|
-
return function mySegmentsFetcher(userMatchingKey, noCache,
|
|
7
|
-
// Optional decorator for `
|
|
5
|
+
export function mySegmentsFetcherFactory(fetchMySegments) {
|
|
6
|
+
return function mySegmentsFetcher(userMatchingKey, noCache,
|
|
7
|
+
// Optional decorator for `fetchMySegments` promise, such as timeout or time tracker
|
|
8
8
|
decorator) {
|
|
9
|
-
var mySegmentsPromise =
|
|
9
|
+
var mySegmentsPromise = fetchMySegments(userMatchingKey, noCache);
|
|
10
10
|
if (decorator)
|
|
11
11
|
mySegmentsPromise = decorator(mySegmentsPromise);
|
|
12
|
-
|
|
12
|
+
// Extract segment names
|
|
13
|
+
return mySegmentsPromise
|
|
14
|
+
.then(function (resp) { return resp.json(); })
|
|
15
|
+
.then(function (json) { return json.mySegments.map(function (segment) { return segment.name; }); });
|
|
13
16
|
};
|
|
14
17
|
}
|
|
@@ -20,7 +20,7 @@ function greedyFetch(fetchSegmentChanges, since, segmentName, noCache, targetTil
|
|
|
20
20
|
*/
|
|
21
21
|
export function segmentChangesFetcherFactory(fetchSegmentChanges) {
|
|
22
22
|
return function segmentChangesFetcher(since, segmentName, noCache, till,
|
|
23
|
-
// Optional decorator for `
|
|
23
|
+
// Optional decorator for `fetchMySegments` promise, such as timeout or time tracker
|
|
24
24
|
decorator) {
|
|
25
25
|
var segmentsPromise = greedyFetch(fetchSegmentChanges, since, segmentName, noCache, till);
|
|
26
26
|
if (decorator)
|
|
@@ -43,7 +43,7 @@ export function pollingManagerCSFactory(params) {
|
|
|
43
43
|
}
|
|
44
44
|
});
|
|
45
45
|
function add(matchingKey, readiness, storage) {
|
|
46
|
-
var mySegmentsSyncTask = mySegmentsSyncTaskFactory(splitApi.
|
|
46
|
+
var mySegmentsSyncTask = mySegmentsSyncTaskFactory(splitApi.fetchMySegments, storage, readiness, settings, matchingKey);
|
|
47
47
|
// smart ready
|
|
48
48
|
function smartReady() {
|
|
49
49
|
if (!readiness.isReady() && !storage.splits.usesSegments())
|
|
@@ -4,6 +4,6 @@ import { mySegmentsUpdaterFactory } from '../updaters/mySegmentsUpdater';
|
|
|
4
4
|
/**
|
|
5
5
|
* Creates a sync task that periodically executes a `mySegmentsUpdater` task
|
|
6
6
|
*/
|
|
7
|
-
export function mySegmentsSyncTaskFactory(
|
|
8
|
-
return syncTaskFactory(settings.log, mySegmentsUpdaterFactory(settings.log, mySegmentsFetcherFactory(
|
|
7
|
+
export function mySegmentsSyncTaskFactory(fetchMySegments, storage, readiness, settings, matchingKey) {
|
|
8
|
+
return syncTaskFactory(settings.log, mySegmentsUpdaterFactory(settings.log, mySegmentsFetcherFactory(fetchMySegments), storage.splits, storage.segments, readiness.segments, settings.startup.requestTimeoutBeforeReady, settings.startup.retriesOnFailureBeforeReady, matchingKey), settings.scheduler.segmentsRefreshRate, 'mySegmentsUpdater');
|
|
9
9
|
}
|
|
@@ -1,15 +1,13 @@
|
|
|
1
1
|
import { timeout } from '../../../utils/promise/timeout';
|
|
2
2
|
import { SDK_SEGMENTS_ARRIVED } from '../../../readiness/constants';
|
|
3
3
|
import { SYNC_MYSEGMENTS_FETCH_RETRY } from '../../../logger/constants';
|
|
4
|
-
import { MEMBERSHIPS_LS_UPDATE } from '../../streaming/constants';
|
|
5
4
|
/**
|
|
6
5
|
* factory of MySegments updater, a task that:
|
|
7
6
|
* - fetches mySegments using `mySegmentsFetcher`
|
|
8
7
|
* - updates `mySegmentsCache`
|
|
9
8
|
* - uses `segmentsEventEmitter` to emit events related to segments data updates
|
|
10
9
|
*/
|
|
11
|
-
export function mySegmentsUpdaterFactory(log, mySegmentsFetcher,
|
|
12
|
-
var splits = storage.splits, segments = storage.segments, largeSegments = storage.largeSegments;
|
|
10
|
+
export function mySegmentsUpdaterFactory(log, mySegmentsFetcher, splitsCache, mySegmentsCache, segmentsEventEmitter, requestTimeoutBeforeReady, retriesOnFailureBeforeReady, matchingKey) {
|
|
13
11
|
var readyOnAlreadyExistentState = true;
|
|
14
12
|
var startingUp = true;
|
|
15
13
|
/** timeout and telemetry decorator for `splitChangesFetcher` promise */
|
|
@@ -21,27 +19,36 @@ export function mySegmentsUpdaterFactory(log, mySegmentsFetcher, storage, segmen
|
|
|
21
19
|
// @TODO if allowing pluggable storages, handle async execution
|
|
22
20
|
function updateSegments(segmentsData) {
|
|
23
21
|
var shouldNotifyUpdate;
|
|
24
|
-
if (segmentsData
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
segments.resetSegments(segmentsData);
|
|
22
|
+
if (Array.isArray(segmentsData)) {
|
|
23
|
+
// Update the list of segment names available
|
|
24
|
+
shouldNotifyUpdate = mySegmentsCache.resetSegments(segmentsData);
|
|
28
25
|
}
|
|
29
26
|
else {
|
|
30
|
-
|
|
31
|
-
|
|
27
|
+
// Add/Delete the segment
|
|
28
|
+
var name_1 = segmentsData.name, add = segmentsData.add;
|
|
29
|
+
if (mySegmentsCache.isInSegment(name_1) !== add) {
|
|
30
|
+
shouldNotifyUpdate = true;
|
|
31
|
+
if (add)
|
|
32
|
+
mySegmentsCache.addToSegment(name_1);
|
|
33
|
+
else
|
|
34
|
+
mySegmentsCache.removeFromSegment(name_1);
|
|
35
|
+
}
|
|
36
|
+
else {
|
|
37
|
+
shouldNotifyUpdate = false;
|
|
38
|
+
}
|
|
32
39
|
}
|
|
33
40
|
// Notify update if required
|
|
34
|
-
if (
|
|
41
|
+
if (splitsCache.usesSegments() && (shouldNotifyUpdate || readyOnAlreadyExistentState)) {
|
|
35
42
|
readyOnAlreadyExistentState = false;
|
|
36
43
|
segmentsEventEmitter.emit(SDK_SEGMENTS_ARRIVED);
|
|
37
44
|
}
|
|
38
45
|
}
|
|
39
|
-
function _mySegmentsUpdater(retry, segmentsData, noCache
|
|
46
|
+
function _mySegmentsUpdater(retry, segmentsData, noCache) {
|
|
40
47
|
var updaterPromise = segmentsData ?
|
|
41
48
|
// If segmentsData is provided, there is no need to fetch mySegments
|
|
42
49
|
new Promise(function (res) { updateSegments(segmentsData); res(true); }) :
|
|
43
50
|
// If not provided, fetch mySegments
|
|
44
|
-
mySegmentsFetcher(matchingKey, noCache,
|
|
51
|
+
mySegmentsFetcher(matchingKey, noCache, _promiseDecorator).then(function (segments) {
|
|
45
52
|
// Only when we have downloaded segments completely, we should not keep retrying anymore
|
|
46
53
|
startingUp = false;
|
|
47
54
|
updateSegments(segments);
|
|
@@ -68,9 +75,8 @@ export function mySegmentsUpdaterFactory(log, mySegmentsFetcher, storage, segmen
|
|
|
68
75
|
* (2) an object with a segment name and action (true: add, or false: delete) to update the storage,
|
|
69
76
|
* (3) or `undefined`, for which the updater will fetch mySegments in order to sync the storage.
|
|
70
77
|
* @param {boolean | undefined} noCache true to revalidate data to fetch
|
|
71
|
-
* @param {boolean | undefined} till query param to bypass CDN requests
|
|
72
78
|
*/
|
|
73
|
-
return function mySegmentsUpdater(segmentsData, noCache
|
|
74
|
-
return _mySegmentsUpdater(0, segmentsData, noCache
|
|
79
|
+
return function mySegmentsUpdater(segmentsData, noCache) {
|
|
80
|
+
return _mySegmentsUpdater(0, segmentsData, noCache);
|
|
75
81
|
};
|
|
76
82
|
}
|
|
@@ -1,5 +1,7 @@
|
|
|
1
|
+
import { findIndex } from '../../../utils/lang';
|
|
1
2
|
import { SDK_SEGMENTS_ARRIVED } from '../../../readiness/constants';
|
|
2
3
|
import { LOG_PREFIX_INSTANTIATION, LOG_PREFIX_SYNC_SEGMENTS } from '../../../logger/constants';
|
|
4
|
+
import { thenable } from '../../../utils/promise/thenable';
|
|
3
5
|
/**
|
|
4
6
|
* Factory of SegmentChanges updater, a task that:
|
|
5
7
|
* - fetches segment changes using `segmentChangesFetcher`
|
|
@@ -18,16 +20,27 @@ export function segmentChangesUpdaterFactory(log, segmentChangesFetcher, segment
|
|
|
18
20
|
var sincePromise = Promise.resolve(segments.getChangeNumber(segmentName));
|
|
19
21
|
return sincePromise.then(function (since) {
|
|
20
22
|
// if fetchOnlyNew flag, avoid processing already fetched segments
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
23
|
+
if (fetchOnlyNew && since !== undefined)
|
|
24
|
+
return -1;
|
|
25
|
+
return segmentChangesFetcher(since || -1, segmentName, noCache, till).then(function (changes) {
|
|
26
|
+
var changeNumber = -1;
|
|
27
|
+
var results = [];
|
|
28
|
+
changes.forEach(function (x) {
|
|
29
|
+
if (x.added.length > 0)
|
|
30
|
+
results.push(segments.addToSegment(segmentName, x.added));
|
|
31
|
+
if (x.removed.length > 0)
|
|
32
|
+
results.push(segments.removeFromSegment(segmentName, x.removed));
|
|
33
|
+
if (x.added.length > 0 || x.removed.length > 0) {
|
|
34
|
+
results.push(segments.setChangeNumber(segmentName, x.till));
|
|
35
|
+
changeNumber = x.till;
|
|
36
|
+
}
|
|
37
|
+
log.debug(LOG_PREFIX_SYNC_SEGMENTS + "Processed " + segmentName + " with till = " + x.till + ". Added: " + x.added.length + ". Removed: " + x.removed.length);
|
|
30
38
|
});
|
|
39
|
+
// If at least one storage operation result is a promise, join all in a single promise.
|
|
40
|
+
if (results.some(function (result) { return thenable(result); }))
|
|
41
|
+
return Promise.all(results).then(function () { return changeNumber; });
|
|
42
|
+
return changeNumber;
|
|
43
|
+
});
|
|
31
44
|
});
|
|
32
45
|
}
|
|
33
46
|
/**
|
|
@@ -46,11 +59,14 @@ export function segmentChangesUpdaterFactory(log, segmentChangesFetcher, segment
|
|
|
46
59
|
// If not a segment name provided, read list of available segments names to be updated.
|
|
47
60
|
var segmentsPromise = Promise.resolve(segmentName ? [segmentName] : segments.getRegisteredSegments());
|
|
48
61
|
return segmentsPromise.then(function (segmentNames) {
|
|
49
|
-
// Async fetchers
|
|
50
|
-
var updaters =
|
|
62
|
+
// Async fetchers are collected here.
|
|
63
|
+
var updaters = [];
|
|
64
|
+
for (var index = 0; index < segmentNames.length; index++) {
|
|
65
|
+
updaters.push(updateSegment(segmentNames[index], noCache, till, fetchOnlyNew));
|
|
66
|
+
}
|
|
51
67
|
return Promise.all(updaters).then(function (shouldUpdateFlags) {
|
|
52
68
|
// if at least one segment fetch succeeded, mark segments ready
|
|
53
|
-
if (shouldUpdateFlags
|
|
69
|
+
if (findIndex(shouldUpdateFlags, function (v) { return v !== -1; }) !== -1 || readyOnAlreadyExistentState) {
|
|
54
70
|
readyOnAlreadyExistentState = false;
|
|
55
71
|
if (readiness)
|
|
56
72
|
readiness.segments.emit(SDK_SEGMENTS_ARRIVED);
|
|
@@ -11,7 +11,7 @@ function checkAllSegmentsExist(segments) {
|
|
|
11
11
|
var registeredSegments = Promise.resolve(segments.getRegisteredSegments());
|
|
12
12
|
return registeredSegments.then(function (segmentNames) {
|
|
13
13
|
return Promise.all(segmentNames.map(function (segmentName) { return segments.getChangeNumber(segmentName); }))
|
|
14
|
-
.then(function (changeNumbers) { return changeNumbers.every(function (changeNumber) { return changeNumber !==
|
|
14
|
+
.then(function (changeNumbers) { return changeNumbers.every(function (changeNumber) { return changeNumber !== undefined; }); });
|
|
15
15
|
});
|
|
16
16
|
}
|
|
17
17
|
/**
|
|
@@ -10,7 +10,7 @@ import { hash } from '../../../utils/murmur3/murmur3';
|
|
|
10
10
|
export function authenticateFactory(fetchAuth) {
|
|
11
11
|
/**
|
|
12
12
|
* Run authentication requests to Auth Server, and returns a promise that resolves with the decoded JTW token.
|
|
13
|
-
* @param {string[] | undefined} userKeys set of user Keys to track
|
|
13
|
+
* @param {string[] | undefined} userKeys set of user Keys to track MY_SEGMENTS_CHANGES. It is undefined for server-side API.
|
|
14
14
|
*/
|
|
15
15
|
return function authenticate(userKeys) {
|
|
16
16
|
return fetchAuth(userKeys)
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { errorParser, messageParser } from './NotificationParser';
|
|
2
2
|
import { notificationKeeperFactory } from './NotificationKeeper';
|
|
3
|
-
import { PUSH_RETRYABLE_ERROR, PUSH_NONRETRYABLE_ERROR, OCCUPANCY, CONTROL,
|
|
3
|
+
import { PUSH_RETRYABLE_ERROR, PUSH_NONRETRYABLE_ERROR, OCCUPANCY, CONTROL, MY_SEGMENTS_UPDATE, MY_SEGMENTS_UPDATE_V2, SEGMENT_UPDATE, SPLIT_KILL, SPLIT_UPDATE } from '../constants';
|
|
4
4
|
import { STREAMING_PARSING_ERROR_FAILS, ERROR_STREAMING_SSE, STREAMING_PARSING_MESSAGE_FAILS, STREAMING_NEW_MESSAGE } from '../../../logger/constants';
|
|
5
5
|
import { ABLY_ERROR, NON_REQUESTED, SSE_CONNECTION_ERROR } from '../../../utils/constants';
|
|
6
6
|
/**
|
|
@@ -65,18 +65,20 @@ export function SSEHandlerFactory(log, pushEmitter, telemetryTracker) {
|
|
|
65
65
|
}
|
|
66
66
|
var parsedData = messageWithParsedData.parsedData, data = messageWithParsedData.data, channel = messageWithParsedData.channel, timestamp = messageWithParsedData.timestamp;
|
|
67
67
|
log.debug(STREAMING_NEW_MESSAGE, [data]);
|
|
68
|
-
// we only handle update events if streaming is up
|
|
68
|
+
// we only handle update events if streaming is up.
|
|
69
69
|
if (!notificationKeeper.isStreamingUp() && [OCCUPANCY, CONTROL].indexOf(parsedData.type) === -1)
|
|
70
70
|
return;
|
|
71
71
|
switch (parsedData.type) {
|
|
72
72
|
/* update events */
|
|
73
73
|
case SPLIT_UPDATE:
|
|
74
74
|
case SEGMENT_UPDATE:
|
|
75
|
-
case
|
|
76
|
-
case MEMBERSHIPS_LS_UPDATE:
|
|
75
|
+
case MY_SEGMENTS_UPDATE_V2:
|
|
77
76
|
case SPLIT_KILL:
|
|
78
77
|
pushEmitter.emit(parsedData.type, parsedData);
|
|
79
78
|
break;
|
|
79
|
+
case MY_SEGMENTS_UPDATE:
|
|
80
|
+
pushEmitter.emit(parsedData.type, parsedData, channel);
|
|
81
|
+
break;
|
|
80
82
|
/* occupancy & control events, handled by NotificationManagerKeeper */
|
|
81
83
|
case OCCUPANCY:
|
|
82
84
|
notificationKeeper.handleOccupancyEvent(parsedData.metrics.publishers, channel, timestamp);
|
|
@@ -1,120 +1,61 @@
|
|
|
1
1
|
import { Backoff } from '../../../utils/Backoff';
|
|
2
|
-
import {
|
|
3
|
-
import { FETCH_BACKOFF_MAX_RETRIES } from './constants';
|
|
4
|
-
import { MEMBERSHIPS_LS_UPDATE, MEMBERSHIPS_MS_UPDATE } from '../constants';
|
|
2
|
+
import { MY_SEGMENT } from '../../../utils/constants';
|
|
5
3
|
/**
|
|
6
4
|
* MySegmentsUpdateWorker factory
|
|
7
5
|
*/
|
|
8
|
-
export function MySegmentsUpdateWorker(
|
|
9
|
-
var
|
|
10
|
-
var
|
|
11
|
-
var
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
mySegmentsSyncTask.execute(_segmentsData, true, cdnBypass ? maxChangeNumber : undefined).then(res);
|
|
31
|
-
}, _delay);
|
|
32
|
-
}) :
|
|
33
|
-
mySegmentsSyncTask.execute(_segmentsData, true, cdnBypass ? maxChangeNumber : undefined);
|
|
34
|
-
syncTask.then(function (result) {
|
|
35
|
-
if (!isHandlingEvent)
|
|
36
|
-
return; // halt if `stop` has been called
|
|
37
|
-
if (result !== false) { // Unlike `Splits|SegmentsUpdateWorker`, `mySegmentsCache.getChangeNumber` can be -1, since `/memberships` change number is optional
|
|
38
|
-
var storageChangeNumber = mySegmentsCache.getChangeNumber();
|
|
39
|
-
currentChangeNumber = storageChangeNumber > -1 ?
|
|
40
|
-
storageChangeNumber :
|
|
41
|
-
Math.max(currentChangeNumber, currentMaxChangeNumber_1); // use `currentMaxChangeNumber`, in case that `maxChangeNumber` was updated during fetch.
|
|
42
|
-
}
|
|
43
|
-
if (handleNewEvent) {
|
|
44
|
-
__handleMySegmentsUpdateCall();
|
|
45
|
-
}
|
|
46
|
-
else {
|
|
47
|
-
if (_segmentsData)
|
|
48
|
-
telemetryTracker.trackUpdatesFromSSE(MEMBERSHIPS);
|
|
49
|
-
var attempts = backoff.attempts + 1;
|
|
50
|
-
if (maxChangeNumber <= currentChangeNumber) {
|
|
51
|
-
log.debug("Refresh completed" + (cdnBypass ? ' bypassing the CDN' : '') + " in " + attempts + " attempts.");
|
|
52
|
-
isHandlingEvent = false;
|
|
53
|
-
return;
|
|
54
|
-
}
|
|
55
|
-
if (attempts < FETCH_BACKOFF_MAX_RETRIES) {
|
|
56
|
-
backoff.scheduleCall();
|
|
57
|
-
return;
|
|
58
|
-
}
|
|
59
|
-
if (cdnBypass) {
|
|
60
|
-
log.debug("No changes fetched after " + attempts + " attempts with CDN bypassed.");
|
|
61
|
-
isHandlingEvent = false;
|
|
62
|
-
}
|
|
63
|
-
else {
|
|
64
|
-
backoff.reset();
|
|
65
|
-
cdnBypass = true;
|
|
66
|
-
__handleMySegmentsUpdateCall();
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
});
|
|
70
|
-
}
|
|
71
|
-
else {
|
|
72
|
-
isHandlingEvent = false;
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
return {
|
|
76
|
-
/**
|
|
77
|
-
* Invoked by NotificationProcessor on MY_(LARGE)_SEGMENTS_UPDATE notifications
|
|
78
|
-
*
|
|
79
|
-
* @param changeNumber change number of the notification
|
|
80
|
-
* @param segmentsData data for KeyList or SegmentRemoval instant updates
|
|
81
|
-
* @param delay optional time to wait for BoundedFetchRequest or BoundedFetchRequest updates
|
|
82
|
-
*/
|
|
83
|
-
put: function (mySegmentsData, payload, delay) {
|
|
84
|
-
var type = mySegmentsData.type, cn = mySegmentsData.cn;
|
|
85
|
-
// Discard event if it is outdated or there is a pending fetch request (_delay is set), but update target change number
|
|
86
|
-
if (cn <= Math.max(currentChangeNumber, mySegmentsCache.getChangeNumber()) || cn <= maxChangeNumber)
|
|
87
|
-
return;
|
|
88
|
-
maxChangeNumber = cn;
|
|
89
|
-
if (_delay)
|
|
90
|
-
return;
|
|
91
|
-
handleNewEvent = true;
|
|
92
|
-
cdnBypass = false;
|
|
93
|
-
_segmentsData = payload && { type: type, cn: cn, added: payload.added, removed: payload.removed };
|
|
94
|
-
_delay = delay;
|
|
95
|
-
if (backoff.timeoutID || !isHandlingEvent)
|
|
6
|
+
export function MySegmentsUpdateWorker(mySegmentsSyncTask, telemetryTracker) {
|
|
7
|
+
var maxChangeNumber = 0; // keeps the maximum changeNumber among queued events
|
|
8
|
+
var currentChangeNumber = -1;
|
|
9
|
+
var handleNewEvent = false;
|
|
10
|
+
var isHandlingEvent;
|
|
11
|
+
var _segmentsData; // keeps the segmentsData (if included in notification payload) from the queued event with maximum changeNumber
|
|
12
|
+
var backoff = new Backoff(__handleMySegmentsUpdateCall);
|
|
13
|
+
function __handleMySegmentsUpdateCall() {
|
|
14
|
+
isHandlingEvent = true;
|
|
15
|
+
if (maxChangeNumber > currentChangeNumber) {
|
|
16
|
+
handleNewEvent = false;
|
|
17
|
+
var currentMaxChangeNumber_1 = maxChangeNumber;
|
|
18
|
+
// fetch mySegments revalidating data if cached
|
|
19
|
+
mySegmentsSyncTask.execute(_segmentsData, true).then(function (result) {
|
|
20
|
+
if (!isHandlingEvent)
|
|
21
|
+
return; // halt if `stop` has been called
|
|
22
|
+
if (result !== false) { // Unlike `Splits|SegmentsUpdateWorker`, we cannot use `mySegmentsCache.getChangeNumber` since `/mySegments` endpoint doesn't provide this value.
|
|
23
|
+
if (_segmentsData)
|
|
24
|
+
telemetryTracker.trackUpdatesFromSSE(MY_SEGMENT);
|
|
25
|
+
currentChangeNumber = Math.max(currentChangeNumber, currentMaxChangeNumber_1); // use `currentMaxChangeNumber`, in case that `maxChangeNumber` was updated during fetch.
|
|
26
|
+
}
|
|
27
|
+
if (handleNewEvent) {
|
|
96
28
|
__handleMySegmentsUpdateCall();
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
}
|
|
29
|
+
}
|
|
30
|
+
else {
|
|
31
|
+
backoff.scheduleCall();
|
|
32
|
+
}
|
|
33
|
+
});
|
|
34
|
+
}
|
|
35
|
+
else {
|
|
36
|
+
isHandlingEvent = false;
|
|
37
|
+
}
|
|
106
38
|
}
|
|
107
|
-
var updateWorkers = (_a = {},
|
|
108
|
-
_a[MEMBERSHIPS_MS_UPDATE] = createUpdateWorker(storage.segments),
|
|
109
|
-
_a[MEMBERSHIPS_LS_UPDATE] = createUpdateWorker(storage.largeSegments),
|
|
110
|
-
_a);
|
|
111
39
|
return {
|
|
112
|
-
|
|
113
|
-
|
|
40
|
+
/**
|
|
41
|
+
* Invoked by NotificationProcessor on MY_SEGMENTS_UPDATE event
|
|
42
|
+
*
|
|
43
|
+
* @param {number} changeNumber change number of the MY_SEGMENTS_UPDATE notification
|
|
44
|
+
* @param {SegmentsData | undefined} segmentsData might be undefined
|
|
45
|
+
*/
|
|
46
|
+
put: function (changeNumber, segmentsData) {
|
|
47
|
+
if (changeNumber <= currentChangeNumber || changeNumber <= maxChangeNumber)
|
|
48
|
+
return;
|
|
49
|
+
maxChangeNumber = changeNumber;
|
|
50
|
+
handleNewEvent = true;
|
|
51
|
+
_segmentsData = segmentsData;
|
|
52
|
+
if (backoff.timeoutID || !isHandlingEvent)
|
|
53
|
+
__handleMySegmentsUpdateCall();
|
|
54
|
+
backoff.reset();
|
|
114
55
|
},
|
|
115
56
|
stop: function () {
|
|
116
|
-
|
|
117
|
-
|
|
57
|
+
isHandlingEvent = false;
|
|
58
|
+
backoff.reset();
|
|
118
59
|
}
|
|
119
60
|
};
|
|
120
61
|
}
|
|
@@ -13,7 +13,7 @@ export function SegmentsUpdateWorker(log, segmentsSyncTask, segmentsCache) {
|
|
|
13
13
|
var backoff = new Backoff(__handleSegmentUpdateCall, FETCH_BACKOFF_BASE, FETCH_BACKOFF_MAX_WAIT);
|
|
14
14
|
function __handleSegmentUpdateCall() {
|
|
15
15
|
isHandlingEvent = true;
|
|
16
|
-
if (maxChangeNumber > segmentsCache.getChangeNumber(segment)) {
|
|
16
|
+
if (maxChangeNumber > (segmentsCache.getChangeNumber(segment) || -1)) {
|
|
17
17
|
handleNewEvent = false;
|
|
18
18
|
// fetch segments revalidating data if cached
|
|
19
19
|
segmentsSyncTask.execute(false, segment, true, cdnBypass ? maxChangeNumber : undefined).then(function () {
|
|
@@ -24,7 +24,7 @@ export function SegmentsUpdateWorker(log, segmentsSyncTask, segmentsCache) {
|
|
|
24
24
|
}
|
|
25
25
|
else {
|
|
26
26
|
var attempts = backoff.attempts + 1;
|
|
27
|
-
if (maxChangeNumber <= segmentsCache.getChangeNumber(segment)) {
|
|
27
|
+
if (maxChangeNumber <= (segmentsCache.getChangeNumber(segment) || -1)) {
|
|
28
28
|
log.debug("Refresh completed" + (cdnBypass ? ' bypassing the CDN' : '') + " in " + attempts + " attempts.");
|
|
29
29
|
isHandlingEvent = false;
|
|
30
30
|
return;
|
|
@@ -51,7 +51,7 @@ export function SegmentsUpdateWorker(log, segmentsSyncTask, segmentsCache) {
|
|
|
51
51
|
}
|
|
52
52
|
return {
|
|
53
53
|
put: function (changeNumber) {
|
|
54
|
-
var currentChangeNumber = segmentsCache.getChangeNumber(segment);
|
|
54
|
+
var currentChangeNumber = segmentsCache.getChangeNumber(segment) || -1;
|
|
55
55
|
if (changeNumber <= currentChangeNumber || changeNumber <= maxChangeNumber)
|
|
56
56
|
return;
|
|
57
57
|
maxChangeNumber = changeNumber;
|
|
@@ -22,8 +22,8 @@ export var PUSH_SUBSYSTEM_UP = 'PUSH_SUBSYSTEM_UP';
|
|
|
22
22
|
*/
|
|
23
23
|
export var PUSH_SUBSYSTEM_DOWN = 'PUSH_SUBSYSTEM_DOWN';
|
|
24
24
|
// Update-type push notifications, handled by NotificationProcessor
|
|
25
|
-
export var
|
|
26
|
-
export var
|
|
25
|
+
export var MY_SEGMENTS_UPDATE = 'MY_SEGMENTS_UPDATE';
|
|
26
|
+
export var MY_SEGMENTS_UPDATE_V2 = 'MY_SEGMENTS_UPDATE_V2';
|
|
27
27
|
export var SEGMENT_UPDATE = 'SEGMENT_UPDATE';
|
|
28
28
|
export var SPLIT_KILL = 'SPLIT_KILL';
|
|
29
29
|
export var SPLIT_UPDATE = 'SPLIT_UPDATE';
|
|
@@ -1,6 +1,5 @@
|
|
|
1
1
|
import { algorithms } from '../../utils/decompress';
|
|
2
2
|
import { decodeFromBase64 } from '../../utils/base64';
|
|
3
|
-
import { hash } from '../../utils/murmur3/murmur3';
|
|
4
3
|
var GZIP = 1;
|
|
5
4
|
var ZLIB = 2;
|
|
6
5
|
function Uint8ArrayToString(myUint8Arr) {
|
|
@@ -75,17 +74,14 @@ export function isInBitmap(bitmap, hash64hex) {
|
|
|
75
74
|
}
|
|
76
75
|
/**
|
|
77
76
|
* Parse feature flags notifications for instant feature flag updates
|
|
77
|
+
*
|
|
78
|
+
* @param {ISplitUpdateData} data
|
|
79
|
+
* @returns {KeyList}
|
|
78
80
|
*/
|
|
79
81
|
export function parseFFUpdatePayload(compression, data) {
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
export function getDelay(parsedData, matchingKey) {
|
|
86
|
-
if (parsedData.h === 0)
|
|
87
|
-
return 0;
|
|
88
|
-
var interval = parsedData.i || DEFAULT_MAX_INTERVAL;
|
|
89
|
-
var seed = parsedData.s || 0;
|
|
90
|
-
return hash(matchingKey, seed) % interval;
|
|
82
|
+
var avoidPrecisionLoss = false;
|
|
83
|
+
if (compression > 0)
|
|
84
|
+
return parseKeyList(data, compression, avoidPrecisionLoss);
|
|
85
|
+
else
|
|
86
|
+
return JSON.parse(decodeFromBase64(data));
|
|
91
87
|
}
|