@splitsoftware/splitio-commons 2.1.1-rc.0 → 2.1.1-rc.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 +3 -0
- package/README.md +0 -1
- package/cjs/evaluator/combiners/and.js +6 -2
- package/cjs/evaluator/combiners/ifelseif.js +6 -6
- package/cjs/evaluator/condition/index.js +5 -6
- package/cjs/evaluator/index.js +7 -7
- package/cjs/evaluator/matchers/index.js +1 -3
- package/cjs/evaluator/matchers/matcherTypes.js +0 -1
- package/cjs/evaluator/matchersTransform/index.js +0 -4
- package/cjs/evaluator/parser/index.js +2 -2
- package/cjs/evaluator/value/sanitize.js +0 -1
- package/cjs/logger/constants.js +3 -4
- package/cjs/logger/messages/debug.js +2 -3
- package/cjs/logger/messages/error.js +1 -1
- package/cjs/logger/messages/warn.js +2 -2
- package/cjs/sdkClient/client.js +29 -19
- package/cjs/sdkClient/clientAttributesDecoration.js +19 -25
- package/cjs/sdkClient/clientInputValidation.js +28 -26
- package/cjs/services/splitApi.js +2 -2
- package/cjs/storages/AbstractSplitsCacheSync.js +2 -5
- package/cjs/storages/KeyBuilder.js +0 -9
- package/cjs/storages/KeyBuilderCS.js +1 -4
- package/cjs/storages/KeyBuilderSS.js +0 -3
- package/cjs/storages/inLocalStorage/SplitsCacheInLocal.js +14 -9
- package/cjs/storages/inLocalStorage/index.js +0 -4
- package/cjs/storages/inMemory/InMemoryStorage.js +0 -3
- package/cjs/storages/inMemory/InMemoryStorageCS.js +0 -4
- package/cjs/storages/inRedis/index.js +0 -2
- package/cjs/storages/pluggable/index.js +0 -2
- package/cjs/storages/utils.js +1 -0
- package/cjs/sync/polling/fetchers/splitChangesFetcher.js +2 -2
- package/cjs/sync/polling/pollingManagerCS.js +7 -7
- package/cjs/sync/polling/updaters/mySegmentsUpdater.js +2 -2
- package/cjs/sync/polling/updaters/segmentChangesUpdater.js +1 -1
- package/cjs/sync/polling/updaters/splitChangesUpdater.js +35 -51
- package/cjs/sync/streaming/SSEHandler/index.js +0 -1
- package/cjs/sync/streaming/UpdateWorkers/SplitsUpdateWorker.js +77 -106
- package/cjs/sync/streaming/constants.js +1 -2
- package/cjs/sync/streaming/pushManager.js +16 -3
- package/cjs/sync/submitters/impressionsSubmitter.js +3 -2
- package/cjs/sync/syncManagerOnline.js +2 -2
- package/cjs/trackers/strategy/strategyOptimized.js +3 -0
- package/cjs/utils/constants/index.js +2 -3
- package/cjs/utils/inputValidation/eventProperties.js +12 -1
- package/cjs/utils/inputValidation/index.js +3 -1
- package/esm/evaluator/combiners/and.js +6 -2
- package/esm/evaluator/combiners/ifelseif.js +7 -7
- package/esm/evaluator/condition/index.js +5 -6
- package/esm/evaluator/index.js +7 -7
- package/esm/evaluator/matchers/index.js +1 -3
- package/esm/evaluator/matchers/matcherTypes.js +0 -1
- package/esm/evaluator/matchersTransform/index.js +0 -4
- package/esm/evaluator/parser/index.js +2 -2
- package/esm/evaluator/value/sanitize.js +0 -1
- package/esm/logger/constants.js +0 -1
- package/esm/logger/messages/debug.js +2 -3
- package/esm/logger/messages/error.js +1 -1
- package/esm/logger/messages/warn.js +2 -2
- package/esm/sdkClient/client.js +29 -19
- package/esm/sdkClient/clientAttributesDecoration.js +19 -25
- package/esm/sdkClient/clientInputValidation.js +29 -27
- package/esm/services/splitApi.js +2 -2
- package/esm/storages/AbstractSplitsCacheSync.js +2 -5
- package/esm/storages/KeyBuilder.js +0 -9
- package/esm/storages/KeyBuilderCS.js +1 -4
- package/esm/storages/KeyBuilderSS.js +0 -3
- package/esm/storages/inLocalStorage/SplitsCacheInLocal.js +14 -9
- package/esm/storages/inLocalStorage/index.js +0 -4
- package/esm/storages/inMemory/InMemoryStorage.js +0 -3
- package/esm/storages/inMemory/InMemoryStorageCS.js +0 -4
- package/esm/storages/inRedis/index.js +0 -2
- package/esm/storages/pluggable/index.js +0 -2
- package/esm/storages/utils.js +1 -0
- package/esm/sync/polling/fetchers/splitChangesFetcher.js +2 -2
- package/esm/sync/polling/pollingManagerCS.js +7 -7
- package/esm/sync/polling/updaters/mySegmentsUpdater.js +2 -2
- package/esm/sync/polling/updaters/segmentChangesUpdater.js +1 -1
- package/esm/sync/polling/updaters/splitChangesUpdater.js +35 -51
- package/esm/sync/streaming/SSEHandler/index.js +1 -2
- package/esm/sync/streaming/UpdateWorkers/SplitsUpdateWorker.js +73 -102
- package/esm/sync/streaming/constants.js +0 -1
- package/esm/sync/streaming/pushManager.js +19 -6
- package/esm/sync/submitters/impressionsSubmitter.js +3 -2
- package/esm/sync/syncManagerOnline.js +2 -2
- package/esm/trackers/strategy/strategyOptimized.js +3 -0
- package/esm/utils/constants/index.js +1 -2
- package/esm/utils/inputValidation/eventProperties.js +10 -0
- package/esm/utils/inputValidation/index.js +1 -0
- package/package.json +1 -1
- package/src/dtos/types.ts +8 -32
- package/src/evaluator/Engine.ts +1 -1
- package/src/evaluator/combiners/and.ts +4 -5
- package/src/evaluator/combiners/ifelseif.ts +9 -7
- package/src/evaluator/condition/engineUtils.ts +1 -1
- package/src/evaluator/condition/index.ts +12 -12
- package/src/evaluator/index.ts +7 -7
- package/src/evaluator/matchers/index.ts +1 -3
- package/src/evaluator/matchers/matcherTypes.ts +0 -1
- package/src/evaluator/matchersTransform/index.ts +0 -3
- package/src/evaluator/parser/index.ts +3 -3
- package/src/evaluator/types.ts +2 -2
- package/src/evaluator/value/index.ts +2 -2
- package/src/evaluator/value/sanitize.ts +4 -5
- package/src/logger/constants.ts +0 -1
- package/src/logger/messages/debug.ts +2 -3
- package/src/logger/messages/error.ts +1 -1
- package/src/logger/messages/warn.ts +2 -2
- package/src/sdkClient/client.ts +31 -21
- package/src/sdkClient/clientAttributesDecoration.ts +20 -27
- package/src/sdkClient/clientInputValidation.ts +30 -27
- package/src/sdkManager/index.ts +1 -1
- package/src/services/splitApi.ts +2 -2
- package/src/services/types.ts +1 -1
- package/src/storages/AbstractSplitsCacheSync.ts +3 -6
- package/src/storages/KeyBuilder.ts +0 -12
- package/src/storages/KeyBuilderCS.ts +1 -5
- package/src/storages/KeyBuilderSS.ts +0 -4
- package/src/storages/inLocalStorage/SplitsCacheInLocal.ts +14 -10
- package/src/storages/inLocalStorage/index.ts +0 -4
- package/src/storages/inMemory/InMemoryStorage.ts +0 -3
- package/src/storages/inMemory/InMemoryStorageCS.ts +0 -4
- package/src/storages/inRedis/index.ts +0 -2
- package/src/storages/pluggable/index.ts +0 -2
- package/src/storages/types.ts +1 -33
- package/src/storages/utils.ts +1 -0
- package/src/sync/polling/fetchers/splitChangesFetcher.ts +1 -2
- package/src/sync/polling/fetchers/types.ts +0 -1
- package/src/sync/polling/pollingManagerCS.ts +7 -7
- package/src/sync/polling/types.ts +2 -2
- package/src/sync/polling/updaters/mySegmentsUpdater.ts +2 -2
- package/src/sync/polling/updaters/segmentChangesUpdater.ts +1 -1
- package/src/sync/polling/updaters/splitChangesUpdater.ts +44 -61
- package/src/sync/streaming/SSEHandler/index.ts +1 -2
- package/src/sync/streaming/SSEHandler/types.ts +2 -2
- package/src/sync/streaming/UpdateWorkers/SplitsUpdateWorker.ts +68 -98
- package/src/sync/streaming/constants.ts +0 -1
- package/src/sync/streaming/parseUtils.ts +2 -2
- package/src/sync/streaming/pushManager.ts +18 -6
- package/src/sync/streaming/types.ts +2 -3
- package/src/sync/submitters/impressionsSubmitter.ts +3 -2
- package/src/sync/submitters/types.ts +23 -33
- package/src/sync/syncManagerOnline.ts +2 -2
- package/src/trackers/strategy/strategyOptimized.ts +3 -0
- package/src/utils/constants/index.ts +1 -2
- package/src/utils/inputValidation/eventProperties.ts +10 -0
- package/src/utils/inputValidation/index.ts +1 -0
- package/src/utils/lang/index.ts +1 -1
- package/types/splitio.d.ts +100 -35
- package/cjs/evaluator/matchers/rbsegment.js +0 -43
- package/cjs/storages/inLocalStorage/RBSegmentsCacheInLocal.js +0 -117
- package/cjs/storages/inMemory/RBSegmentsCacheInMemory.js +0 -61
- package/cjs/storages/inRedis/RBSegmentsCacheInRedis.js +0 -64
- package/cjs/storages/pluggable/RBSegmentsCachePluggable.js +0 -64
- package/esm/evaluator/matchers/rbsegment.js +0 -39
- package/esm/storages/inLocalStorage/RBSegmentsCacheInLocal.js +0 -114
- package/esm/storages/inMemory/RBSegmentsCacheInMemory.js +0 -58
- package/esm/storages/inRedis/RBSegmentsCacheInRedis.js +0 -61
- package/esm/storages/pluggable/RBSegmentsCachePluggable.js +0 -61
- package/src/evaluator/matchers/rbsegment.ts +0 -61
- package/src/storages/inLocalStorage/RBSegmentsCacheInLocal.ts +0 -136
- package/src/storages/inMemory/RBSegmentsCacheInMemory.ts +0 -68
- package/src/storages/inRedis/RBSegmentsCacheInRedis.ts +0 -79
- package/src/storages/pluggable/RBSegmentsCachePluggable.ts +0 -76
|
@@ -36,7 +36,7 @@ export function segmentChangesUpdaterFactory(log, segmentChangesFetcher, segment
|
|
|
36
36
|
* Returned promise will not be rejected.
|
|
37
37
|
*
|
|
38
38
|
* @param fetchOnlyNew - if true, only fetch the segments that not exists, i.e., which `changeNumber` is equal to -1.
|
|
39
|
-
* This param is used by SplitUpdateWorker on server-side SDK, to fetch new registered segments on SPLIT_UPDATE
|
|
39
|
+
* This param is used by SplitUpdateWorker on server-side SDK, to fetch new registered segments on SPLIT_UPDATE notifications.
|
|
40
40
|
* @param segmentName - segment name to fetch. By passing `undefined` it fetches the list of segments registered at the storage
|
|
41
41
|
* @param noCache - true to revalidate data to fetch on a SEGMENT_UPDATE notifications.
|
|
42
42
|
* @param till - till target for the provided segmentName, for CDN bypass.
|
|
@@ -1,10 +1,9 @@
|
|
|
1
1
|
import { timeout } from '../../../utils/promise/timeout';
|
|
2
2
|
import { SDK_SPLITS_ARRIVED, SDK_SPLITS_CACHE_LOADED } from '../../../readiness/constants';
|
|
3
|
-
import { SYNC_SPLITS_FETCH, SYNC_SPLITS_UPDATE,
|
|
3
|
+
import { SYNC_SPLITS_FETCH, SYNC_SPLITS_UPDATE, SYNC_SPLITS_FETCH_FAILS, SYNC_SPLITS_FETCH_RETRY } from '../../../logger/constants';
|
|
4
4
|
import { startsWith } from '../../../utils/lang';
|
|
5
|
-
import {
|
|
5
|
+
import { IN_SEGMENT } from '../../../utils/constants';
|
|
6
6
|
import { setToArray } from '../../../utils/lang/sets';
|
|
7
|
-
import { SPLIT_UPDATE } from '../../streaming/constants';
|
|
8
7
|
// Checks that all registered segments have been fetched (changeNumber !== -1 for every segment).
|
|
9
8
|
// Returns a promise that could be rejected.
|
|
10
9
|
// @TODO review together with Segments and MySegments storage APIs
|
|
@@ -19,14 +18,13 @@ function checkAllSegmentsExist(segments) {
|
|
|
19
18
|
* Collect segments from a raw split definition.
|
|
20
19
|
* Exported for testing purposes.
|
|
21
20
|
*/
|
|
22
|
-
export function parseSegments(
|
|
23
|
-
|
|
24
|
-
var
|
|
25
|
-
var segments = new Set(excluded && excluded.segments);
|
|
21
|
+
export function parseSegments(_a) {
|
|
22
|
+
var conditions = _a.conditions;
|
|
23
|
+
var segments = new Set();
|
|
26
24
|
for (var i = 0; i < conditions.length; i++) {
|
|
27
25
|
var matchers = conditions[i].matcherGroup.matchers;
|
|
28
26
|
matchers.forEach(function (matcher) {
|
|
29
|
-
if (matcher.matcherType ===
|
|
27
|
+
if (matcher.matcherType === IN_SEGMENT)
|
|
30
28
|
segments.add(matcher.userDefinedSegmentMatcherData.segmentName);
|
|
31
29
|
});
|
|
32
30
|
}
|
|
@@ -56,19 +54,22 @@ function matchFilters(featureFlag, filters) {
|
|
|
56
54
|
* i.e., an object with added splits, removed splits and used segments.
|
|
57
55
|
* Exported for testing purposes.
|
|
58
56
|
*/
|
|
59
|
-
export function
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
57
|
+
export function computeSplitsMutation(entries, filters) {
|
|
58
|
+
var segments = new Set();
|
|
59
|
+
var computed = entries.reduce(function (accum, split) {
|
|
60
|
+
if (split.status === 'ACTIVE' && matchFilters(split, filters)) {
|
|
61
|
+
accum.added.push(split);
|
|
62
|
+
parseSegments(split).forEach(function (segmentName) {
|
|
64
63
|
segments.add(segmentName);
|
|
65
64
|
});
|
|
66
65
|
}
|
|
67
66
|
else {
|
|
68
|
-
accum.removed.push(
|
|
67
|
+
accum.removed.push(split);
|
|
69
68
|
}
|
|
70
69
|
return accum;
|
|
71
|
-
}, { added: [], removed: [] });
|
|
70
|
+
}, { added: [], removed: [], segments: [] });
|
|
71
|
+
computed.segments = setToArray(segments);
|
|
72
|
+
return computed;
|
|
72
73
|
}
|
|
73
74
|
/**
|
|
74
75
|
* factory of SplitChanges updater, a task that:
|
|
@@ -87,7 +88,7 @@ export function computeMutation(rules, segments, filters) {
|
|
|
87
88
|
export function splitChangesUpdaterFactory(log, splitChangesFetcher, storage, splitFiltersValidation, splitsEventEmitter, requestTimeoutBeforeReady, retriesOnFailureBeforeReady, isClientSide) {
|
|
88
89
|
if (requestTimeoutBeforeReady === void 0) { requestTimeoutBeforeReady = 0; }
|
|
89
90
|
if (retriesOnFailureBeforeReady === void 0) { retriesOnFailureBeforeReady = 0; }
|
|
90
|
-
var splits = storage.splits,
|
|
91
|
+
var splits = storage.splits, segments = storage.segments;
|
|
91
92
|
var startingUp = true;
|
|
92
93
|
/** timeout decorator for `splitChangesFetcher` promise */
|
|
93
94
|
function _promiseDecorator(promise) {
|
|
@@ -102,48 +103,31 @@ export function splitChangesUpdaterFactory(log, splitChangesFetcher, storage, sp
|
|
|
102
103
|
* @param noCache - true to revalidate data to fetch
|
|
103
104
|
* @param till - query param to bypass CDN requests
|
|
104
105
|
*/
|
|
105
|
-
return function splitChangesUpdater(noCache, till,
|
|
106
|
+
return function splitChangesUpdater(noCache, till, splitUpdateNotification) {
|
|
106
107
|
/**
|
|
107
108
|
* @param since - current changeNumber at splitsCache
|
|
108
109
|
* @param retry - current number of retry attempts
|
|
109
110
|
*/
|
|
110
|
-
function _splitChangesUpdater(
|
|
111
|
+
function _splitChangesUpdater(since, retry) {
|
|
111
112
|
if (retry === void 0) { retry = 0; }
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
// IFFU edge case: a change to a flag that adds an IN_RULE_BASED_SEGMENT matcher that is not present yet
|
|
117
|
-
Promise.resolve(rbSegments.contains(parseSegments(instantUpdate.payload, IN_RULE_BASED_SEGMENT))).then(function (contains) {
|
|
118
|
-
return contains ?
|
|
119
|
-
{ ff: { d: [instantUpdate.payload], t: instantUpdate.changeNumber } } :
|
|
120
|
-
splitChangesFetcher(since, noCache, till, rbSince, _promiseDecorator);
|
|
121
|
-
}) :
|
|
122
|
-
{ rbs: { d: [instantUpdate.payload], t: instantUpdate.changeNumber } } :
|
|
123
|
-
splitChangesFetcher(since, noCache, till, rbSince, _promiseDecorator))
|
|
113
|
+
log.debug(SYNC_SPLITS_FETCH, [since]);
|
|
114
|
+
var fetcherPromise = Promise.resolve(splitUpdateNotification ?
|
|
115
|
+
{ splits: [splitUpdateNotification.payload], till: splitUpdateNotification.changeNumber } :
|
|
116
|
+
splitChangesFetcher(since, noCache, till, _promiseDecorator))
|
|
124
117
|
.then(function (splitChanges) {
|
|
125
118
|
startingUp = false;
|
|
126
|
-
var
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
var rbsUpdate = false;
|
|
134
|
-
if (splitChanges.rbs) {
|
|
135
|
-
var _b = computeMutation(splitChanges.rbs.d, usedSegments), added = _b.added, removed = _b.removed;
|
|
136
|
-
log.debug(SYNC_RBS_UPDATE, [added.length, removed.length]);
|
|
137
|
-
rbsUpdate = rbSegments.update(added, removed, splitChanges.rbs.t);
|
|
138
|
-
}
|
|
139
|
-
return Promise.all([ffUpdate, rbsUpdate,
|
|
140
|
-
// @TODO if at least 1 segment fetch fails due to 404 and other segments are updated in the storage, SDK_UPDATE is not emitted
|
|
141
|
-
segments.registerSegments(setToArray(usedSegments))
|
|
119
|
+
var mutation = computeSplitsMutation(splitChanges.splits, splitFiltersValidation);
|
|
120
|
+
log.debug(SYNC_SPLITS_UPDATE, [mutation.added.length, mutation.removed.length, mutation.segments.length]);
|
|
121
|
+
// Write into storage
|
|
122
|
+
// @TODO call `setChangeNumber` only if the other storage operations have succeeded, in order to keep storage consistency
|
|
123
|
+
return Promise.all([
|
|
124
|
+
splits.update(mutation.added, mutation.removed, splitChanges.till),
|
|
125
|
+
segments.registerSegments(mutation.segments)
|
|
142
126
|
]).then(function (_a) {
|
|
143
|
-
var
|
|
127
|
+
var isThereUpdate = _a[0];
|
|
144
128
|
if (splitsEventEmitter) {
|
|
145
129
|
// To emit SDK_SPLITS_ARRIVED for server-side SDK, we must check that all registered segments have been fetched
|
|
146
|
-
return Promise.resolve(!splitsEventEmitter.splitsArrived || (
|
|
130
|
+
return Promise.resolve(!splitsEventEmitter.splitsArrived || (since !== splitChanges.till && isThereUpdate && (isClientSide || checkAllSegmentsExist(segments))))
|
|
147
131
|
.catch(function () { return false; } /** noop. just to handle a possible `checkAllSegmentsExist` rejection, before emitting SDK event */)
|
|
148
132
|
.then(function (emitSplitsArrivedEvent) {
|
|
149
133
|
// emit SDK events
|
|
@@ -160,7 +144,7 @@ export function splitChangesUpdaterFactory(log, splitChangesFetcher, storage, sp
|
|
|
160
144
|
if (startingUp && retriesOnFailureBeforeReady > retry) {
|
|
161
145
|
retry += 1;
|
|
162
146
|
log.info(SYNC_SPLITS_FETCH_RETRY, [retry, error]);
|
|
163
|
-
return _splitChangesUpdater(
|
|
147
|
+
return _splitChangesUpdater(since, retry);
|
|
164
148
|
}
|
|
165
149
|
else {
|
|
166
150
|
startingUp = false;
|
|
@@ -177,7 +161,7 @@ export function splitChangesUpdaterFactory(log, splitChangesFetcher, storage, sp
|
|
|
177
161
|
}
|
|
178
162
|
return fetcherPromise;
|
|
179
163
|
}
|
|
180
|
-
// `getChangeNumber` never rejects or throws error
|
|
181
|
-
return
|
|
164
|
+
var sincePromise = Promise.resolve(splits.getChangeNumber()); // `getChangeNumber` never rejects or throws error
|
|
165
|
+
return sincePromise.then(_splitChangesUpdater);
|
|
182
166
|
};
|
|
183
167
|
}
|
|
@@ -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, SEGMENT_UPDATE, SPLIT_KILL, SPLIT_UPDATE, MEMBERSHIPS_MS_UPDATE, MEMBERSHIPS_LS_UPDATE
|
|
3
|
+
import { PUSH_RETRYABLE_ERROR, PUSH_NONRETRYABLE_ERROR, OCCUPANCY, CONTROL, SEGMENT_UPDATE, SPLIT_KILL, SPLIT_UPDATE, MEMBERSHIPS_MS_UPDATE, MEMBERSHIPS_LS_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
|
/**
|
|
@@ -75,7 +75,6 @@ export function SSEHandlerFactory(log, pushEmitter, telemetryTracker) {
|
|
|
75
75
|
case MEMBERSHIPS_MS_UPDATE:
|
|
76
76
|
case MEMBERSHIPS_LS_UPDATE:
|
|
77
77
|
case SPLIT_KILL:
|
|
78
|
-
case RB_SEGMENT_UPDATE:
|
|
79
78
|
pushEmitter.emit(parsedData.type, parsedData);
|
|
80
79
|
break;
|
|
81
80
|
/* occupancy & control events, handled by NotificationManagerKeeper */
|
|
@@ -1,132 +1,103 @@
|
|
|
1
|
-
import { STREAMING_PARSING_SPLIT_UPDATE } from '../../../logger/constants';
|
|
2
1
|
import { SDK_SPLITS_ARRIVED } from '../../../readiness/constants';
|
|
3
2
|
import { Backoff } from '../../../utils/Backoff';
|
|
4
3
|
import { SPLITS } from '../../../utils/constants';
|
|
5
|
-
import { RB_SEGMENT_UPDATE } from '../constants';
|
|
6
|
-
import { parseFFUpdatePayload } from '../parseUtils';
|
|
7
4
|
import { FETCH_BACKOFF_BASE, FETCH_BACKOFF_MAX_WAIT, FETCH_BACKOFF_MAX_RETRIES } from './constants';
|
|
8
5
|
/**
|
|
9
6
|
* SplitsUpdateWorker factory
|
|
10
7
|
*/
|
|
11
|
-
export function SplitsUpdateWorker(log,
|
|
12
|
-
var
|
|
13
|
-
var
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
8
|
+
export function SplitsUpdateWorker(log, splitsCache, splitsSyncTask, splitsEventEmitter, telemetryTracker, segmentsSyncTask) {
|
|
9
|
+
var maxChangeNumber = 0;
|
|
10
|
+
var handleNewEvent = false;
|
|
11
|
+
var isHandlingEvent;
|
|
12
|
+
var cdnBypass;
|
|
13
|
+
var payload;
|
|
14
|
+
var backoff = new Backoff(__handleSplitUpdateCall, FETCH_BACKOFF_BASE, FETCH_BACKOFF_MAX_WAIT);
|
|
15
|
+
function __handleSplitUpdateCall() {
|
|
16
|
+
isHandlingEvent = true;
|
|
17
|
+
if (maxChangeNumber > splitsCache.getChangeNumber()) {
|
|
18
|
+
handleNewEvent = false;
|
|
19
|
+
var splitUpdateNotification_1 = payload ? { payload: payload, changeNumber: maxChangeNumber } : undefined;
|
|
20
|
+
// fetch splits revalidating data if cached
|
|
21
|
+
splitsSyncTask.execute(true, cdnBypass ? maxChangeNumber : undefined, splitUpdateNotification_1).then(function () {
|
|
22
|
+
if (!isHandlingEvent)
|
|
23
|
+
return; // halt if `stop` has been called
|
|
24
|
+
if (handleNewEvent) {
|
|
25
|
+
__handleSplitUpdateCall();
|
|
26
|
+
}
|
|
27
|
+
else {
|
|
28
|
+
if (splitUpdateNotification_1)
|
|
29
|
+
telemetryTracker.trackUpdatesFromSSE(SPLITS);
|
|
30
|
+
// fetch new registered segments for server-side API. Not retrying on error
|
|
31
|
+
if (segmentsSyncTask)
|
|
32
|
+
segmentsSyncTask.execute(true);
|
|
33
|
+
var attempts = backoff.attempts + 1;
|
|
34
|
+
if (maxChangeNumber <= splitsCache.getChangeNumber()) {
|
|
35
|
+
log.debug("Refresh completed" + (cdnBypass ? ' bypassing the CDN' : '') + " in " + attempts + " attempts.");
|
|
36
|
+
isHandlingEvent = false;
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
if (attempts < FETCH_BACKOFF_MAX_RETRIES) {
|
|
40
|
+
backoff.scheduleCall();
|
|
41
|
+
return;
|
|
42
|
+
}
|
|
43
|
+
if (cdnBypass) {
|
|
44
|
+
log.debug("No changes fetched after " + attempts + " attempts with CDN bypassed.");
|
|
45
|
+
isHandlingEvent = false;
|
|
31
46
|
}
|
|
32
47
|
else {
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
if (segmentsSyncTask)
|
|
37
|
-
segmentsSyncTask.execute(true);
|
|
38
|
-
var attempts = backoff.attempts + 1;
|
|
39
|
-
if (ff.isSync() && rbs.isSync()) {
|
|
40
|
-
log.debug("Refresh completed" + (cdnBypass ? ' bypassing the CDN' : '') + " in " + attempts + " attempts.");
|
|
41
|
-
isHandlingEvent = false;
|
|
42
|
-
return;
|
|
43
|
-
}
|
|
44
|
-
if (attempts < FETCH_BACKOFF_MAX_RETRIES) {
|
|
45
|
-
backoff.scheduleCall();
|
|
46
|
-
return;
|
|
47
|
-
}
|
|
48
|
-
if (cdnBypass) {
|
|
49
|
-
log.debug("No changes fetched after " + attempts + " attempts with CDN bypassed.");
|
|
50
|
-
isHandlingEvent = false;
|
|
51
|
-
}
|
|
52
|
-
else {
|
|
53
|
-
backoff.reset();
|
|
54
|
-
cdnBypass = true;
|
|
55
|
-
__handleSplitUpdateCall();
|
|
56
|
-
}
|
|
48
|
+
backoff.reset();
|
|
49
|
+
cdnBypass = true;
|
|
50
|
+
__handleSplitUpdateCall();
|
|
57
51
|
}
|
|
58
|
-
});
|
|
59
|
-
}
|
|
60
|
-
else {
|
|
61
|
-
isHandlingEvent = false;
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
return {
|
|
65
|
-
/**
|
|
66
|
-
* Invoked by NotificationProcessor on SPLIT_UPDATE or RB_SEGMENT_UPDATE event
|
|
67
|
-
*
|
|
68
|
-
* @param changeNumber - change number of the notification
|
|
69
|
-
*/
|
|
70
|
-
put: function (_a, payload) {
|
|
71
|
-
var changeNumber = _a.changeNumber, pcn = _a.pcn, type = _a.type;
|
|
72
|
-
var currentChangeNumber = cache.getChangeNumber();
|
|
73
|
-
if (changeNumber <= currentChangeNumber || changeNumber <= maxChangeNumber)
|
|
74
|
-
return;
|
|
75
|
-
maxChangeNumber = changeNumber;
|
|
76
|
-
handleNewEvent = true;
|
|
77
|
-
cdnBypass = false;
|
|
78
|
-
instantUpdate = undefined;
|
|
79
|
-
if (payload && currentChangeNumber === pcn) {
|
|
80
|
-
instantUpdate = { payload: payload, changeNumber: changeNumber, type: type };
|
|
81
52
|
}
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
else {
|
|
56
|
+
isHandlingEvent = false;
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Invoked by NotificationProcessor on SPLIT_UPDATE event
|
|
61
|
+
*
|
|
62
|
+
* @param changeNumber - change number of the SPLIT_UPDATE notification
|
|
63
|
+
*/
|
|
64
|
+
function put(_a, _payload) {
|
|
65
|
+
var changeNumber = _a.changeNumber, pcn = _a.pcn;
|
|
66
|
+
var currentChangeNumber = splitsCache.getChangeNumber();
|
|
67
|
+
if (changeNumber <= currentChangeNumber || changeNumber <= maxChangeNumber)
|
|
68
|
+
return;
|
|
69
|
+
maxChangeNumber = changeNumber;
|
|
70
|
+
handleNewEvent = true;
|
|
71
|
+
cdnBypass = false;
|
|
72
|
+
payload = undefined;
|
|
73
|
+
if (_payload && currentChangeNumber === pcn) {
|
|
74
|
+
payload = _payload;
|
|
75
|
+
}
|
|
76
|
+
if (backoff.timeoutID || !isHandlingEvent)
|
|
77
|
+
__handleSplitUpdateCall();
|
|
78
|
+
backoff.reset();
|
|
94
79
|
}
|
|
95
80
|
return {
|
|
96
|
-
put:
|
|
97
|
-
if (parsedData.d && parsedData.c !== undefined) {
|
|
98
|
-
try {
|
|
99
|
-
var payload = parseFFUpdatePayload(parsedData.c, parsedData.d);
|
|
100
|
-
if (payload) {
|
|
101
|
-
(parsedData.type === RB_SEGMENT_UPDATE ? rbs : ff).put(parsedData, payload);
|
|
102
|
-
return;
|
|
103
|
-
}
|
|
104
|
-
}
|
|
105
|
-
catch (e) {
|
|
106
|
-
log.warn(STREAMING_PARSING_SPLIT_UPDATE, [parsedData.type, e]);
|
|
107
|
-
}
|
|
108
|
-
}
|
|
109
|
-
(parsedData.type === RB_SEGMENT_UPDATE ? rbs : ff).put(parsedData);
|
|
110
|
-
},
|
|
81
|
+
put: put,
|
|
111
82
|
/**
|
|
112
83
|
* Invoked by NotificationProcessor on SPLIT_KILL event
|
|
113
84
|
*
|
|
114
|
-
* @param changeNumber - change number of the notification
|
|
85
|
+
* @param changeNumber - change number of the SPLIT_UPDATE notification
|
|
115
86
|
* @param splitName - name of split to kill
|
|
116
87
|
* @param defaultTreatment - default treatment value
|
|
117
88
|
*/
|
|
118
89
|
killSplit: function (_a) {
|
|
119
90
|
var changeNumber = _a.changeNumber, splitName = _a.splitName, defaultTreatment = _a.defaultTreatment;
|
|
120
|
-
if (
|
|
91
|
+
if (splitsCache.killLocally(splitName, defaultTreatment, changeNumber)) {
|
|
121
92
|
// trigger an SDK_UPDATE if Split was killed locally
|
|
122
93
|
splitsEventEmitter.emit(SDK_SPLITS_ARRIVED, true);
|
|
123
94
|
}
|
|
124
95
|
// queues the SplitChanges fetch (only if changeNumber is newer)
|
|
125
|
-
|
|
96
|
+
put({ changeNumber: changeNumber });
|
|
126
97
|
},
|
|
127
98
|
stop: function () {
|
|
128
|
-
|
|
129
|
-
|
|
99
|
+
isHandlingEvent = false;
|
|
100
|
+
backoff.reset();
|
|
130
101
|
}
|
|
131
102
|
};
|
|
132
103
|
}
|
|
@@ -27,7 +27,6 @@ export var MEMBERSHIPS_LS_UPDATE = 'MEMBERSHIPS_LS_UPDATE';
|
|
|
27
27
|
export var SEGMENT_UPDATE = 'SEGMENT_UPDATE';
|
|
28
28
|
export var SPLIT_KILL = 'SPLIT_KILL';
|
|
29
29
|
export var SPLIT_UPDATE = 'SPLIT_UPDATE';
|
|
30
|
-
export var RB_SEGMENT_UPDATE = 'RB_SEGMENT_UPDATE';
|
|
31
30
|
// Control-type push notifications, handled by NotificationKeeper
|
|
32
31
|
export var CONTROL = 'CONTROL';
|
|
33
32
|
export var OCCUPANCY = 'OCCUPANCY';
|
|
@@ -8,10 +8,10 @@ import { authenticateFactory, hashUserKey } from './AuthClient';
|
|
|
8
8
|
import { forOwn } from '../../utils/lang';
|
|
9
9
|
import { SSEClient } from './SSEClient';
|
|
10
10
|
import { getMatching } from '../../utils/key';
|
|
11
|
-
import { MEMBERSHIPS_MS_UPDATE, MEMBERSHIPS_LS_UPDATE, PUSH_NONRETRYABLE_ERROR, PUSH_SUBSYSTEM_DOWN, SECONDS_BEFORE_EXPIRATION, SEGMENT_UPDATE, SPLIT_KILL, SPLIT_UPDATE,
|
|
12
|
-
import { STREAMING_FALLBACK, STREAMING_REFRESH_TOKEN, STREAMING_CONNECTING, STREAMING_DISABLED, ERROR_STREAMING_AUTH, STREAMING_DISCONNECTING, STREAMING_RECONNECT, STREAMING_PARSING_MEMBERSHIPS_UPDATE } from '../../logger/constants';
|
|
11
|
+
import { MEMBERSHIPS_MS_UPDATE, MEMBERSHIPS_LS_UPDATE, PUSH_NONRETRYABLE_ERROR, PUSH_SUBSYSTEM_DOWN, SECONDS_BEFORE_EXPIRATION, SEGMENT_UPDATE, SPLIT_KILL, SPLIT_UPDATE, PUSH_RETRYABLE_ERROR, PUSH_SUBSYSTEM_UP, ControlType } from './constants';
|
|
12
|
+
import { STREAMING_FALLBACK, STREAMING_REFRESH_TOKEN, STREAMING_CONNECTING, STREAMING_DISABLED, ERROR_STREAMING_AUTH, STREAMING_DISCONNECTING, STREAMING_RECONNECT, STREAMING_PARSING_MEMBERSHIPS_UPDATE, STREAMING_PARSING_SPLIT_UPDATE } from '../../logger/constants';
|
|
13
13
|
import { UpdateStrategy } from './SSEHandler/types';
|
|
14
|
-
import { getDelay, isInBitmap, parseBitmap, parseKeyList } from './parseUtils';
|
|
14
|
+
import { getDelay, isInBitmap, parseBitmap, parseFFUpdatePayload, parseKeyList } from './parseUtils';
|
|
15
15
|
import { hash64 } from '../../utils/murmur3/murmur3_64';
|
|
16
16
|
import { TOKEN_REFRESH, AUTH_REJECTION } from '../../utils/constants';
|
|
17
17
|
/**
|
|
@@ -43,7 +43,7 @@ export function pushManagerFactory(params, pollingManager) {
|
|
|
43
43
|
// MySegmentsUpdateWorker (client-side) are initiated in `add` method
|
|
44
44
|
var segmentsUpdateWorker = userKey ? undefined : SegmentsUpdateWorker(log, pollingManager.segmentsSyncTask, storage.segments);
|
|
45
45
|
// For server-side we pass the segmentsSyncTask, used by SplitsUpdateWorker to fetch new segments
|
|
46
|
-
var splitsUpdateWorker = SplitsUpdateWorker(log, storage, pollingManager.splitsSyncTask, readiness.splits, telemetryTracker, userKey ? undefined : pollingManager.segmentsSyncTask);
|
|
46
|
+
var splitsUpdateWorker = SplitsUpdateWorker(log, storage.splits, pollingManager.splitsSyncTask, readiness.splits, telemetryTracker, userKey ? undefined : pollingManager.segmentsSyncTask);
|
|
47
47
|
// [Only for client-side] map of hashes to user keys, to dispatch membership update events to the corresponding MySegmentsUpdateWorker
|
|
48
48
|
var userKeyHashes = {};
|
|
49
49
|
// [Only for client-side] map of user keys to their corresponding hash64 and MySegmentsUpdateWorkers.
|
|
@@ -179,8 +179,21 @@ export function pushManagerFactory(params, pollingManager) {
|
|
|
179
179
|
});
|
|
180
180
|
/** Functions related to synchronization (Queues and Workers in the spec) */
|
|
181
181
|
pushEmitter.on(SPLIT_KILL, splitsUpdateWorker.killSplit);
|
|
182
|
-
pushEmitter.on(SPLIT_UPDATE,
|
|
183
|
-
|
|
182
|
+
pushEmitter.on(SPLIT_UPDATE, function (parsedData) {
|
|
183
|
+
if (parsedData.d && parsedData.c !== undefined) {
|
|
184
|
+
try {
|
|
185
|
+
var payload = parseFFUpdatePayload(parsedData.c, parsedData.d);
|
|
186
|
+
if (payload) {
|
|
187
|
+
splitsUpdateWorker.put(parsedData, payload);
|
|
188
|
+
return;
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
catch (e) {
|
|
192
|
+
log.warn(STREAMING_PARSING_SPLIT_UPDATE, [e]);
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
splitsUpdateWorker.put(parsedData);
|
|
196
|
+
});
|
|
184
197
|
function handleMySegmentsUpdate(parsedData) {
|
|
185
198
|
switch (parsedData.u) {
|
|
186
199
|
case UpdateStrategy.BoundedFetchRequest: {
|
|
@@ -19,8 +19,9 @@ export function fromImpressionsCollector(sendLabels, data) {
|
|
|
19
19
|
m: entry.time,
|
|
20
20
|
c: entry.changeNumber,
|
|
21
21
|
r: sendLabels ? entry.label : undefined,
|
|
22
|
-
b: entry.bucketingKey
|
|
23
|
-
pt: entry.pt
|
|
22
|
+
b: entry.bucketingKey,
|
|
23
|
+
pt: entry.pt,
|
|
24
|
+
properties: entry.properties // Properties
|
|
24
25
|
};
|
|
25
26
|
return keyImpression;
|
|
26
27
|
})
|
|
@@ -122,7 +122,7 @@ export function syncManagerOnlineFactory(pollingManagerFactory, pushManagerFacto
|
|
|
122
122
|
if (pushManager) {
|
|
123
123
|
if (pollingManager.isRunning()) {
|
|
124
124
|
// if doing polling, we must start the periodic fetch of data
|
|
125
|
-
if (storage.splits.usesSegments()
|
|
125
|
+
if (storage.splits.usesSegments())
|
|
126
126
|
mySegmentsSyncTask.start();
|
|
127
127
|
}
|
|
128
128
|
else {
|
|
@@ -132,7 +132,7 @@ export function syncManagerOnlineFactory(pollingManagerFactory, pushManagerFacto
|
|
|
132
132
|
}
|
|
133
133
|
}
|
|
134
134
|
else {
|
|
135
|
-
if (storage.splits.usesSegments()
|
|
135
|
+
if (storage.splits.usesSegments())
|
|
136
136
|
mySegmentsSyncTask.start();
|
|
137
137
|
}
|
|
138
138
|
}
|
|
@@ -9,6 +9,9 @@ import { truncateTimeFrame } from '../../utils/time';
|
|
|
9
9
|
export function strategyOptimizedFactory(impressionsObserver, impressionCounts) {
|
|
10
10
|
return {
|
|
11
11
|
process: function (impression) {
|
|
12
|
+
// DEBUG mode without previous time, for impressions with properties
|
|
13
|
+
if (impression.properties)
|
|
14
|
+
return true;
|
|
12
15
|
impression.pt = impressionsObserver.testAndSet(impression);
|
|
13
16
|
var now = Date.now();
|
|
14
17
|
// Increments impression counter per featureName
|
|
@@ -86,8 +86,7 @@ export var NON_REQUESTED = 1;
|
|
|
86
86
|
export var DISABLED = 0;
|
|
87
87
|
export var ENABLED = 1;
|
|
88
88
|
export var PAUSED = 2;
|
|
89
|
-
export var FLAG_SPEC_VERSION = '1.
|
|
89
|
+
export var FLAG_SPEC_VERSION = '1.2';
|
|
90
90
|
// Matcher types
|
|
91
91
|
export var IN_SEGMENT = 'IN_SEGMENT';
|
|
92
92
|
export var IN_LARGE_SEGMENT = 'IN_LARGE_SEGMENT';
|
|
93
|
-
export var IN_RULE_BASED_SEGMENT = 'IN_RULE_BASED_SEGMENT';
|
|
@@ -57,3 +57,13 @@ export function validateEventProperties(log, maybeProperties, method) {
|
|
|
57
57
|
}
|
|
58
58
|
return output;
|
|
59
59
|
}
|
|
60
|
+
export function validateEvaluationOptions(log, maybeOptions, method) {
|
|
61
|
+
if (isObject(maybeOptions)) {
|
|
62
|
+
var properties = validateEventProperties(log, maybeOptions.properties, method).properties;
|
|
63
|
+
return properties && Object.keys(properties).length > 0 ? { properties: properties } : undefined;
|
|
64
|
+
}
|
|
65
|
+
else if (maybeOptions) {
|
|
66
|
+
log.error(ERROR_NOT_PLAIN_OBJECT, [method, 'evaluation options']);
|
|
67
|
+
}
|
|
68
|
+
return undefined;
|
|
69
|
+
}
|
|
@@ -11,3 +11,4 @@ export { validateIfNotDestroyed, validateIfOperational } from './isOperational';
|
|
|
11
11
|
export { validateSplitExistence } from './splitExistence';
|
|
12
12
|
export { validateTrafficTypeExistence } from './trafficTypeExistence';
|
|
13
13
|
export { validatePreloadedData } from './preloadedData';
|
|
14
|
+
export { validateEvaluationOptions } from './eventProperties';
|
package/package.json
CHANGED
package/src/dtos/types.ts
CHANGED
|
@@ -66,11 +66,6 @@ interface IInSegmentMatcher extends ISplitMatcherBase {
|
|
|
66
66
|
userDefinedSegmentMatcherData: IInSegmentMatcherData
|
|
67
67
|
}
|
|
68
68
|
|
|
69
|
-
interface IInRBSegmentMatcher extends ISplitMatcherBase {
|
|
70
|
-
matcherType: 'IN_RULE_BASED_SEGMENT',
|
|
71
|
-
userDefinedSegmentMatcherData: IInSegmentMatcherData
|
|
72
|
-
}
|
|
73
|
-
|
|
74
69
|
interface IInLargeSegmentMatcher extends ISplitMatcherBase {
|
|
75
70
|
matcherType: 'IN_LARGE_SEGMENT',
|
|
76
71
|
userDefinedLargeSegmentMatcherData: IInLargeSegmentMatcherData
|
|
@@ -181,7 +176,7 @@ export type ISplitMatcher = IAllKeysMatcher | IInSegmentMatcher | IWhitelistMatc
|
|
|
181
176
|
ILessThanOrEqualToMatcher | IBetweenMatcher | IEqualToSetMatcher | IContainsAnyOfSetMatcher | IContainsAllOfSetMatcher | IPartOfSetMatcher |
|
|
182
177
|
IStartsWithMatcher | IEndsWithMatcher | IContainsStringMatcher | IInSplitTreatmentMatcher | IEqualToBooleanMatcher | IMatchesStringMatcher |
|
|
183
178
|
IEqualToSemverMatcher | IGreaterThanOrEqualToSemverMatcher | ILessThanOrEqualToSemverMatcher | IBetweenSemverMatcher | IInListSemverMatcher |
|
|
184
|
-
IInLargeSegmentMatcher
|
|
179
|
+
IInLargeSegmentMatcher
|
|
185
180
|
|
|
186
181
|
/** Split object */
|
|
187
182
|
export interface ISplitPartition {
|
|
@@ -194,30 +189,19 @@ export interface ISplitCondition {
|
|
|
194
189
|
combiner: 'AND',
|
|
195
190
|
matchers: ISplitMatcher[]
|
|
196
191
|
}
|
|
197
|
-
partitions
|
|
198
|
-
label
|
|
199
|
-
conditionType
|
|
200
|
-
}
|
|
201
|
-
|
|
202
|
-
export interface IRBSegment {
|
|
203
|
-
name: string,
|
|
204
|
-
changeNumber: number,
|
|
205
|
-
status: 'ACTIVE' | 'ARCHIVED',
|
|
206
|
-
conditions: ISplitCondition[],
|
|
207
|
-
excluded: {
|
|
208
|
-
keys: string[],
|
|
209
|
-
segments: string[]
|
|
210
|
-
}
|
|
192
|
+
partitions: ISplitPartition[]
|
|
193
|
+
label: string
|
|
194
|
+
conditionType: 'ROLLOUT' | 'WHITELIST'
|
|
211
195
|
}
|
|
212
196
|
|
|
213
197
|
export interface ISplit {
|
|
214
198
|
name: string,
|
|
215
199
|
changeNumber: number,
|
|
216
|
-
status: 'ACTIVE' | 'ARCHIVED',
|
|
217
|
-
conditions: ISplitCondition[],
|
|
218
200
|
killed: boolean,
|
|
219
201
|
defaultTreatment: string,
|
|
220
202
|
trafficTypeName: string,
|
|
203
|
+
conditions: ISplitCondition[],
|
|
204
|
+
status: 'ACTIVE' | 'ARCHIVED',
|
|
221
205
|
seed: number,
|
|
222
206
|
trafficAllocation?: number,
|
|
223
207
|
trafficAllocationSeed?: number
|
|
@@ -233,16 +217,8 @@ export type ISplitPartial = Pick<ISplit, 'conditions' | 'configurations' | 'traf
|
|
|
233
217
|
|
|
234
218
|
/** Interface of the parsed JSON response of `/splitChanges` */
|
|
235
219
|
export interface ISplitChangesResponse {
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
s?: number,
|
|
239
|
-
d: ISplit[]
|
|
240
|
-
},
|
|
241
|
-
rbs?: {
|
|
242
|
-
t: number,
|
|
243
|
-
s?: number,
|
|
244
|
-
d: IRBSegment[]
|
|
245
|
-
}
|
|
220
|
+
till: number,
|
|
221
|
+
splits: ISplit[]
|
|
246
222
|
}
|
|
247
223
|
|
|
248
224
|
/** Interface of the parsed JSON response of `/segmentChanges/{segmentName}` */
|
package/src/evaluator/Engine.ts
CHANGED
|
@@ -2,11 +2,10 @@ import { findIndex } from '../../utils/lang';
|
|
|
2
2
|
import { ILogger } from '../../logger/types';
|
|
3
3
|
import { thenable } from '../../utils/promise/thenable';
|
|
4
4
|
import { MaybeThenable } from '../../dtos/types';
|
|
5
|
-
import {
|
|
5
|
+
import { IMatcher } from '../types';
|
|
6
6
|
import { ENGINE_COMBINER_AND } from '../../logger/constants';
|
|
7
|
-
import SplitIO from '../../../types/splitio';
|
|
8
7
|
|
|
9
|
-
export function andCombinerContext(log: ILogger, matchers:
|
|
8
|
+
export function andCombinerContext(log: ILogger, matchers: IMatcher[]) {
|
|
10
9
|
|
|
11
10
|
function andResults(results: boolean[]): boolean {
|
|
12
11
|
// Array.prototype.every is supported by target environments
|
|
@@ -16,8 +15,8 @@ export function andCombinerContext(log: ILogger, matchers: Array<(key: SplitIO.S
|
|
|
16
15
|
return hasMatchedAll;
|
|
17
16
|
}
|
|
18
17
|
|
|
19
|
-
return function andCombiner(
|
|
20
|
-
const matcherResults = matchers.map(matcher => matcher(
|
|
18
|
+
return function andCombiner(...params: any): MaybeThenable<boolean> {
|
|
19
|
+
const matcherResults = matchers.map(matcher => matcher(...params));
|
|
21
20
|
|
|
22
21
|
// If any matching result is a thenable we should use Promise.all
|
|
23
22
|
if (findIndex(matcherResults, thenable) !== -1) {
|