@splitsoftware/splitio-commons 2.2.1-rc.4 → 2.3.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.
Files changed (147) hide show
  1. package/CHANGES.txt +26 -26
  2. package/README.md +0 -1
  3. package/cjs/evaluator/combiners/and.js +6 -2
  4. package/cjs/evaluator/combiners/ifelseif.js +6 -6
  5. package/cjs/evaluator/condition/index.js +5 -6
  6. package/cjs/evaluator/index.js +7 -7
  7. package/cjs/evaluator/matchers/index.js +1 -3
  8. package/cjs/evaluator/matchers/matcherTypes.js +0 -1
  9. package/cjs/evaluator/matchersTransform/index.js +0 -4
  10. package/cjs/evaluator/parser/index.js +2 -2
  11. package/cjs/evaluator/value/sanitize.js +0 -1
  12. package/cjs/logger/constants.js +3 -4
  13. package/cjs/logger/messages/debug.js +2 -3
  14. package/cjs/logger/messages/warn.js +1 -1
  15. package/cjs/services/splitApi.js +4 -3
  16. package/cjs/services/splitHttpClient.js +4 -1
  17. package/cjs/storages/AbstractSplitsCacheSync.js +2 -5
  18. package/cjs/storages/KeyBuilder.js +0 -9
  19. package/cjs/storages/KeyBuilderCS.js +0 -3
  20. package/cjs/storages/KeyBuilderSS.js +0 -3
  21. package/cjs/storages/inLocalStorage/SplitsCacheInLocal.js +14 -9
  22. package/cjs/storages/inLocalStorage/index.js +1 -5
  23. package/cjs/storages/inLocalStorage/validateCache.js +1 -2
  24. package/cjs/storages/inMemory/InMemoryStorage.js +0 -3
  25. package/cjs/storages/inMemory/InMemoryStorageCS.js +0 -4
  26. package/cjs/storages/inMemory/SplitsCacheInMemory.js +0 -1
  27. package/cjs/storages/inRedis/index.js +0 -2
  28. package/cjs/storages/pluggable/index.js +0 -2
  29. package/cjs/sync/polling/fetchers/splitChangesFetcher.js +4 -54
  30. package/cjs/sync/polling/pollingManagerCS.js +7 -7
  31. package/cjs/sync/polling/syncTasks/splitsSyncTask.js +1 -1
  32. package/cjs/sync/polling/updaters/mySegmentsUpdater.js +2 -2
  33. package/cjs/sync/polling/updaters/segmentChangesUpdater.js +1 -1
  34. package/cjs/sync/polling/updaters/splitChangesUpdater.js +33 -59
  35. package/cjs/sync/streaming/SSEHandler/index.js +0 -1
  36. package/cjs/sync/streaming/UpdateWorkers/SplitsUpdateWorker.js +77 -106
  37. package/cjs/sync/streaming/constants.js +1 -2
  38. package/cjs/sync/streaming/pushManager.js +16 -3
  39. package/cjs/sync/syncManagerOnline.js +2 -2
  40. package/cjs/utils/constants/index.js +2 -6
  41. package/esm/evaluator/combiners/and.js +6 -2
  42. package/esm/evaluator/combiners/ifelseif.js +7 -7
  43. package/esm/evaluator/condition/index.js +5 -6
  44. package/esm/evaluator/index.js +7 -7
  45. package/esm/evaluator/matchers/index.js +1 -3
  46. package/esm/evaluator/matchers/matcherTypes.js +0 -1
  47. package/esm/evaluator/matchersTransform/index.js +0 -4
  48. package/esm/evaluator/parser/index.js +2 -2
  49. package/esm/evaluator/value/sanitize.js +0 -1
  50. package/esm/logger/constants.js +0 -1
  51. package/esm/logger/messages/debug.js +2 -3
  52. package/esm/logger/messages/warn.js +1 -1
  53. package/esm/services/splitApi.js +4 -3
  54. package/esm/services/splitHttpClient.js +4 -1
  55. package/esm/storages/AbstractSplitsCacheSync.js +2 -5
  56. package/esm/storages/KeyBuilder.js +0 -9
  57. package/esm/storages/KeyBuilderCS.js +0 -3
  58. package/esm/storages/KeyBuilderSS.js +0 -3
  59. package/esm/storages/inLocalStorage/SplitsCacheInLocal.js +14 -9
  60. package/esm/storages/inLocalStorage/index.js +1 -5
  61. package/esm/storages/inLocalStorage/validateCache.js +1 -2
  62. package/esm/storages/inMemory/InMemoryStorage.js +0 -3
  63. package/esm/storages/inMemory/InMemoryStorageCS.js +0 -4
  64. package/esm/storages/inMemory/SplitsCacheInMemory.js +0 -1
  65. package/esm/storages/inRedis/index.js +0 -2
  66. package/esm/storages/pluggable/index.js +0 -2
  67. package/esm/sync/polling/fetchers/splitChangesFetcher.js +4 -54
  68. package/esm/sync/polling/pollingManagerCS.js +7 -7
  69. package/esm/sync/polling/syncTasks/splitsSyncTask.js +1 -1
  70. package/esm/sync/polling/updaters/mySegmentsUpdater.js +2 -2
  71. package/esm/sync/polling/updaters/segmentChangesUpdater.js +1 -1
  72. package/esm/sync/polling/updaters/splitChangesUpdater.js +33 -59
  73. package/esm/sync/streaming/SSEHandler/index.js +1 -2
  74. package/esm/sync/streaming/UpdateWorkers/SplitsUpdateWorker.js +73 -102
  75. package/esm/sync/streaming/constants.js +0 -1
  76. package/esm/sync/streaming/pushManager.js +19 -6
  77. package/esm/sync/syncManagerOnline.js +2 -2
  78. package/esm/utils/constants/index.js +1 -5
  79. package/package.json +1 -1
  80. package/src/dtos/types.ts +8 -37
  81. package/src/evaluator/Engine.ts +1 -1
  82. package/src/evaluator/combiners/and.ts +4 -5
  83. package/src/evaluator/combiners/ifelseif.ts +9 -7
  84. package/src/evaluator/condition/engineUtils.ts +1 -1
  85. package/src/evaluator/condition/index.ts +12 -12
  86. package/src/evaluator/index.ts +7 -7
  87. package/src/evaluator/matchers/index.ts +1 -3
  88. package/src/evaluator/matchers/matcherTypes.ts +0 -1
  89. package/src/evaluator/matchersTransform/index.ts +0 -3
  90. package/src/evaluator/parser/index.ts +3 -3
  91. package/src/evaluator/types.ts +2 -2
  92. package/src/evaluator/value/index.ts +2 -2
  93. package/src/evaluator/value/sanitize.ts +4 -5
  94. package/src/logger/constants.ts +0 -1
  95. package/src/logger/messages/debug.ts +2 -3
  96. package/src/logger/messages/warn.ts +1 -1
  97. package/src/sdkFactory/types.ts +1 -1
  98. package/src/sdkManager/index.ts +1 -1
  99. package/src/services/splitApi.ts +4 -3
  100. package/src/services/splitHttpClient.ts +4 -1
  101. package/src/services/types.ts +1 -1
  102. package/src/storages/AbstractSplitsCacheSync.ts +3 -6
  103. package/src/storages/KeyBuilder.ts +0 -12
  104. package/src/storages/KeyBuilderCS.ts +0 -4
  105. package/src/storages/KeyBuilderSS.ts +0 -4
  106. package/src/storages/inLocalStorage/SplitsCacheInLocal.ts +14 -10
  107. package/src/storages/inLocalStorage/index.ts +1 -5
  108. package/src/storages/inLocalStorage/validateCache.ts +1 -3
  109. package/src/storages/inMemory/InMemoryStorage.ts +0 -3
  110. package/src/storages/inMemory/InMemoryStorageCS.ts +0 -4
  111. package/src/storages/inMemory/SplitsCacheInMemory.ts +0 -1
  112. package/src/storages/inRedis/index.ts +0 -2
  113. package/src/storages/pluggable/index.ts +0 -2
  114. package/src/storages/types.ts +1 -33
  115. package/src/sync/polling/fetchers/splitChangesFetcher.ts +4 -65
  116. package/src/sync/polling/fetchers/types.ts +0 -1
  117. package/src/sync/polling/pollingManagerCS.ts +7 -7
  118. package/src/sync/polling/syncTasks/splitsSyncTask.ts +1 -1
  119. package/src/sync/polling/types.ts +2 -2
  120. package/src/sync/polling/updaters/mySegmentsUpdater.ts +2 -2
  121. package/src/sync/polling/updaters/segmentChangesUpdater.ts +1 -1
  122. package/src/sync/polling/updaters/splitChangesUpdater.ts +43 -70
  123. package/src/sync/streaming/SSEHandler/index.ts +1 -2
  124. package/src/sync/streaming/SSEHandler/types.ts +2 -2
  125. package/src/sync/streaming/UpdateWorkers/SplitsUpdateWorker.ts +68 -98
  126. package/src/sync/streaming/constants.ts +0 -1
  127. package/src/sync/streaming/parseUtils.ts +2 -2
  128. package/src/sync/streaming/pushManager.ts +18 -6
  129. package/src/sync/streaming/types.ts +2 -3
  130. package/src/sync/syncManagerOnline.ts +2 -2
  131. package/src/utils/constants/index.ts +1 -6
  132. package/src/utils/lang/index.ts +1 -1
  133. package/cjs/evaluator/matchers/rbsegment.js +0 -54
  134. package/cjs/storages/inLocalStorage/RBSegmentsCacheInLocal.js +0 -117
  135. package/cjs/storages/inMemory/RBSegmentsCacheInMemory.js +0 -61
  136. package/cjs/storages/inRedis/RBSegmentsCacheInRedis.js +0 -64
  137. package/cjs/storages/pluggable/RBSegmentsCachePluggable.js +0 -64
  138. package/esm/evaluator/matchers/rbsegment.js +0 -50
  139. package/esm/storages/inLocalStorage/RBSegmentsCacheInLocal.js +0 -114
  140. package/esm/storages/inMemory/RBSegmentsCacheInMemory.js +0 -58
  141. package/esm/storages/inRedis/RBSegmentsCacheInRedis.js +0 -61
  142. package/esm/storages/pluggable/RBSegmentsCachePluggable.js +0 -61
  143. package/src/evaluator/matchers/rbsegment.ts +0 -71
  144. package/src/storages/inLocalStorage/RBSegmentsCacheInLocal.ts +0 -136
  145. package/src/storages/inMemory/RBSegmentsCacheInMemory.ts +0 -68
  146. package/src/storages/inRedis/RBSegmentsCacheInRedis.ts +0 -79
  147. package/src/storages/pluggable/RBSegmentsCachePluggable.ts +0 -76
@@ -1,64 +1,14 @@
1
- import { FLAG_SPEC_VERSION } from '../../../utils/constants';
2
- import { base } from '../../../utils/settingsValidation';
3
- import { LOG_PREFIX_SYNC_SPLITS } from '../../../logger/constants';
4
- var PROXY_CHECK_INTERVAL_MILLIS_CS = 60 * 60 * 1000; // 1 hour in Client Side
5
- var PROXY_CHECK_INTERVAL_MILLIS_SS = 24 * PROXY_CHECK_INTERVAL_MILLIS_CS; // 24 hours in Server Side
6
- function sdkEndpointOverriden(settings) {
7
- return settings.urls.sdk !== base.urls.sdk;
8
- }
9
1
  /**
10
2
  * Factory of SplitChanges fetcher.
11
3
  * SplitChanges fetcher is a wrapper around `splitChanges` API service that parses the response and handle errors.
12
4
  */
13
- // @TODO breaking: drop support for Split Proxy below v5.10.0 and simplify the implementation
14
- export function splitChangesFetcherFactory(fetchSplitChanges, settings, storage) {
15
- var log = settings.log;
16
- var PROXY_CHECK_INTERVAL_MILLIS = settings.core.key !== undefined ? PROXY_CHECK_INTERVAL_MILLIS_CS : PROXY_CHECK_INTERVAL_MILLIS_SS;
17
- var lastProxyCheckTimestamp;
18
- return function splitChangesFetcher(since, noCache, till, rbSince,
5
+ export function splitChangesFetcherFactory(fetchSplitChanges) {
6
+ return function splitChangesFetcher(since, noCache, till,
19
7
  // Optional decorator for `fetchSplitChanges` promise, such as timeout or time tracker
20
8
  decorator) {
21
- // Recheck proxy
22
- if (lastProxyCheckTimestamp && (Date.now() - lastProxyCheckTimestamp) > PROXY_CHECK_INTERVAL_MILLIS) {
23
- settings.sync.flagSpecVersion = FLAG_SPEC_VERSION;
24
- }
25
- var splitsPromise = fetchSplitChanges(since, noCache, till, settings.sync.flagSpecVersion === FLAG_SPEC_VERSION ? rbSince : undefined)
26
- // Handle proxy error with spec 1.3
27
- .catch(function (err) {
28
- if (err.statusCode === 400 && sdkEndpointOverriden(settings) && settings.sync.flagSpecVersion === FLAG_SPEC_VERSION) {
29
- log.error(LOG_PREFIX_SYNC_SPLITS + 'Proxy error detected. If you are using Split Proxy, please upgrade to latest version');
30
- lastProxyCheckTimestamp = Date.now();
31
- settings.sync.flagSpecVersion = '1.2'; // fallback to 1.2 spec
32
- return fetchSplitChanges(since, noCache, till); // retry request without rbSince
33
- }
34
- throw err;
35
- });
9
+ var splitsPromise = fetchSplitChanges(since, noCache, till);
36
10
  if (decorator)
37
11
  splitsPromise = decorator(splitsPromise);
38
- return splitsPromise
39
- .then(function (resp) { return resp.json(); })
40
- .then(function (data) {
41
- // Using flag spec version 1.2
42
- if (data.splits) {
43
- return {
44
- ff: {
45
- d: data.splits,
46
- s: data.since,
47
- t: data.till
48
- }
49
- };
50
- }
51
- // Proxy recovery
52
- if (lastProxyCheckTimestamp) {
53
- log.info(LOG_PREFIX_SYNC_SPLITS + 'Proxy error recovered');
54
- lastProxyCheckTimestamp = undefined;
55
- return splitChangesFetcher(-1, undefined, undefined, -1)
56
- .then(function (splitChangesResponse) {
57
- return Promise.all([storage.splits.clear(), storage.rbSegments.clear()])
58
- .then(function () { return splitChangesResponse; });
59
- });
60
- }
61
- return data;
62
- });
12
+ return splitsPromise.then(function (resp) { return resp.json(); });
63
13
  };
64
14
  }
@@ -31,10 +31,10 @@ export function pollingManagerCSFactory(params) {
31
31
  readiness.splits.on(SDK_SPLITS_ARRIVED, function () {
32
32
  if (!splitsSyncTask.isRunning())
33
33
  return; // noop if not doing polling
34
- var usingSegments = storage.splits.usesSegments() || storage.rbSegments.usesSegments();
35
- if (usingSegments !== mySegmentsSyncTask.isRunning()) {
36
- log.info(POLLING_SMART_PAUSING, [usingSegments ? 'ON' : 'OFF']);
37
- if (usingSegments) {
34
+ var splitsHaveSegments = storage.splits.usesSegments();
35
+ if (splitsHaveSegments !== mySegmentsSyncTask.isRunning()) {
36
+ log.info(POLLING_SMART_PAUSING, [splitsHaveSegments ? 'ON' : 'OFF']);
37
+ if (splitsHaveSegments) {
38
38
  startMySegmentsSyncTasks();
39
39
  }
40
40
  else {
@@ -46,10 +46,10 @@ export function pollingManagerCSFactory(params) {
46
46
  var mySegmentsSyncTask = mySegmentsSyncTaskFactory(splitApi.fetchMemberships, storage, readiness, settings, matchingKey);
47
47
  // smart ready
48
48
  function smartReady() {
49
- if (!readiness.isReady() && !storage.splits.usesSegments() && !storage.rbSegments.usesSegments())
49
+ if (!readiness.isReady() && !storage.splits.usesSegments())
50
50
  readiness.segments.emit(SDK_SEGMENTS_ARRIVED);
51
51
  }
52
- if (!storage.splits.usesSegments() && !storage.rbSegments.usesSegments())
52
+ if (!storage.splits.usesSegments())
53
53
  setTimeout(smartReady, 0);
54
54
  else
55
55
  readiness.splits.once(SDK_SPLITS_ARRIVED, smartReady);
@@ -63,7 +63,7 @@ export function pollingManagerCSFactory(params) {
63
63
  start: function () {
64
64
  log.info(POLLING_START);
65
65
  splitsSyncTask.start();
66
- if (storage.splits.usesSegments() || storage.rbSegments.usesSegments())
66
+ if (storage.splits.usesSegments())
67
67
  startMySegmentsSyncTasks();
68
68
  },
69
69
  // Stop periodic fetching (polling)
@@ -5,5 +5,5 @@ import { splitChangesUpdaterFactory } from '../updaters/splitChangesUpdater';
5
5
  * Creates a sync task that periodically executes a `splitChangesUpdater` task
6
6
  */
7
7
  export function splitsSyncTaskFactory(fetchSplitChanges, storage, readiness, settings, isClientSide) {
8
- return syncTaskFactory(settings.log, splitChangesUpdaterFactory(settings.log, splitChangesFetcherFactory(fetchSplitChanges, settings, storage), storage, settings.sync.__splitFiltersValidation, readiness.splits, settings.startup.requestTimeoutBeforeReady, settings.startup.retriesOnFailureBeforeReady, isClientSide), settings.scheduler.featuresRefreshRate, 'splitChangesUpdater');
8
+ return syncTaskFactory(settings.log, splitChangesUpdaterFactory(settings.log, splitChangesFetcherFactory(fetchSplitChanges), storage, settings.sync.__splitFiltersValidation, readiness.splits, settings.startup.requestTimeoutBeforeReady, settings.startup.retriesOnFailureBeforeReady, isClientSide), settings.scheduler.featuresRefreshRate, 'splitChangesUpdater');
9
9
  }
@@ -9,7 +9,7 @@ import { MEMBERSHIPS_LS_UPDATE } from '../../streaming/constants';
9
9
  * - uses `segmentsEventEmitter` to emit events related to segments data updates
10
10
  */
11
11
  export function mySegmentsUpdaterFactory(log, mySegmentsFetcher, storage, segmentsEventEmitter, requestTimeoutBeforeReady, retriesOnFailureBeforeReady, matchingKey) {
12
- var splits = storage.splits, rbSegments = storage.rbSegments, segments = storage.segments, largeSegments = storage.largeSegments;
12
+ var splits = storage.splits, segments = storage.segments, largeSegments = storage.largeSegments;
13
13
  var readyOnAlreadyExistentState = true;
14
14
  var startingUp = true;
15
15
  /** timeout and telemetry decorator for `splitChangesFetcher` promise */
@@ -31,7 +31,7 @@ export function mySegmentsUpdaterFactory(log, mySegmentsFetcher, storage, segmen
31
31
  shouldNotifyUpdate = largeSegments.resetSegments(segmentsData.ls || {}) || shouldNotifyUpdate;
32
32
  }
33
33
  // Notify update if required
34
- if ((splits.usesSegments() || rbSegments.usesSegments()) && (shouldNotifyUpdate || readyOnAlreadyExistentState)) {
34
+ if (splits.usesSegments() && (shouldNotifyUpdate || readyOnAlreadyExistentState)) {
35
35
  readyOnAlreadyExistentState = false;
36
36
  segmentsEventEmitter.emit(SDK_SEGMENTS_ARRIVED);
37
37
  }
@@ -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 or RB_SEGMENT_UPDATE notifications.
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 } from '../../../readiness/constants';
3
- import { SYNC_SPLITS_FETCH, SYNC_SPLITS_UPDATE, SYNC_RBS_UPDATE, SYNC_SPLITS_FETCH_FAILS, SYNC_SPLITS_FETCH_RETRY } from '../../../logger/constants';
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 { IN_RULE_BASED_SEGMENT, IN_SEGMENT, RULE_BASED_SEGMENT, STANDARD_SEGMENT } from '../../../utils/constants';
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
@@ -16,25 +15,16 @@ function checkAllSegmentsExist(segments) {
16
15
  });
17
16
  }
18
17
  /**
19
- * Collect segments from a raw FF or RBS definition.
18
+ * Collect segments from a raw split definition.
20
19
  * Exported for testing purposes.
21
20
  */
22
- export function parseSegments(ruleEntity, matcherType) {
23
- if (matcherType === void 0) { matcherType = IN_SEGMENT; }
24
- var _a = ruleEntity, _b = _a.conditions, conditions = _b === void 0 ? [] : _b, excluded = _a.excluded;
21
+ export function parseSegments(_a) {
22
+ var conditions = _a.conditions;
25
23
  var segments = new Set();
26
- if (excluded && excluded.segments) {
27
- excluded.segments.forEach(function (_a) {
28
- var type = _a.type, name = _a.name;
29
- if ((type === STANDARD_SEGMENT && matcherType === IN_SEGMENT) || (type === RULE_BASED_SEGMENT && matcherType === IN_RULE_BASED_SEGMENT)) {
30
- segments.add(name);
31
- }
32
- });
33
- }
34
24
  for (var i = 0; i < conditions.length; i++) {
35
25
  var matchers = conditions[i].matcherGroup.matchers;
36
26
  matchers.forEach(function (matcher) {
37
- if (matcher.matcherType === matcherType)
27
+ if (matcher.matcherType === IN_SEGMENT)
38
28
  segments.add(matcher.userDefinedSegmentMatcherData.segmentName);
39
29
  });
40
30
  }
@@ -64,19 +54,22 @@ function matchFilters(featureFlag, filters) {
64
54
  * i.e., an object with added splits, removed splits and used segments.
65
55
  * Exported for testing purposes.
66
56
  */
67
- export function computeMutation(rules, segments, filters) {
68
- return rules.reduce(function (accum, ruleEntity) {
69
- if (ruleEntity.status === 'ACTIVE' && (!filters || matchFilters(ruleEntity, filters))) {
70
- accum.added.push(ruleEntity);
71
- parseSegments(ruleEntity).forEach(function (segmentName) {
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) {
72
63
  segments.add(segmentName);
73
64
  });
74
65
  }
75
66
  else {
76
- accum.removed.push(ruleEntity);
67
+ accum.removed.push(split);
77
68
  }
78
69
  return accum;
79
- }, { added: [], removed: [] });
70
+ }, { added: [], removed: [], segments: [] });
71
+ computed.segments = setToArray(segments);
72
+ return computed;
80
73
  }
81
74
  /**
82
75
  * factory of SplitChanges updater, a task that:
@@ -95,7 +88,7 @@ export function computeMutation(rules, segments, filters) {
95
88
  export function splitChangesUpdaterFactory(log, splitChangesFetcher, storage, splitFiltersValidation, splitsEventEmitter, requestTimeoutBeforeReady, retriesOnFailureBeforeReady, isClientSide) {
96
89
  if (requestTimeoutBeforeReady === void 0) { requestTimeoutBeforeReady = 0; }
97
90
  if (retriesOnFailureBeforeReady === void 0) { retriesOnFailureBeforeReady = 0; }
98
- var splits = storage.splits, rbSegments = storage.rbSegments, segments = storage.segments;
91
+ var splits = storage.splits, segments = storage.segments;
99
92
  var startingUp = true;
100
93
  /** timeout decorator for `splitChangesFetcher` promise */
101
94
  function _promiseDecorator(promise) {
@@ -110,48 +103,29 @@ export function splitChangesUpdaterFactory(log, splitChangesFetcher, storage, sp
110
103
  * @param noCache - true to revalidate data to fetch
111
104
  * @param till - query param to bypass CDN requests
112
105
  */
113
- return function splitChangesUpdater(noCache, till, instantUpdate) {
106
+ return function splitChangesUpdater(noCache, till, splitUpdateNotification) {
114
107
  /**
115
108
  * @param since - current changeNumber at splitsCache
116
109
  * @param retry - current number of retry attempts
117
110
  */
118
- function _splitChangesUpdater(sinces, retry) {
111
+ function _splitChangesUpdater(since, retry) {
119
112
  if (retry === void 0) { retry = 0; }
120
- var since = sinces[0], rbSince = sinces[1];
121
- log.debug(SYNC_SPLITS_FETCH, sinces);
122
- return Promise.resolve(instantUpdate ?
123
- instantUpdate.type === SPLIT_UPDATE ?
124
- // IFFU edge case: a change to a flag that adds an IN_RULE_BASED_SEGMENT matcher that is not present yet
125
- Promise.resolve(rbSegments.contains(parseSegments(instantUpdate.payload, IN_RULE_BASED_SEGMENT))).then(function (contains) {
126
- return contains ?
127
- { ff: { d: [instantUpdate.payload], t: instantUpdate.changeNumber } } :
128
- splitChangesFetcher(since, noCache, till, rbSince, _promiseDecorator);
129
- }) :
130
- { rbs: { d: [instantUpdate.payload], t: instantUpdate.changeNumber } } :
131
- splitChangesFetcher(since, noCache, till, rbSince, _promiseDecorator))
113
+ log.debug(SYNC_SPLITS_FETCH, [since]);
114
+ return Promise.resolve(splitUpdateNotification ?
115
+ { splits: [splitUpdateNotification.payload], till: splitUpdateNotification.changeNumber } :
116
+ splitChangesFetcher(since, noCache, till, _promiseDecorator))
132
117
  .then(function (splitChanges) {
133
118
  startingUp = false;
134
- var usedSegments = new Set();
135
- var ffUpdate = false;
136
- if (splitChanges.ff) {
137
- var _a = computeMutation(splitChanges.ff.d, usedSegments, splitFiltersValidation), added = _a.added, removed = _a.removed;
138
- log.debug(SYNC_SPLITS_UPDATE, [added.length, removed.length]);
139
- ffUpdate = splits.update(added, removed, splitChanges.ff.t);
140
- }
141
- var rbsUpdate = false;
142
- if (splitChanges.rbs) {
143
- var _b = computeMutation(splitChanges.rbs.d, usedSegments), added = _b.added, removed = _b.removed;
144
- log.debug(SYNC_RBS_UPDATE, [added.length, removed.length]);
145
- rbsUpdate = rbSegments.update(added, removed, splitChanges.rbs.t);
146
- }
147
- return Promise.all([ffUpdate, rbsUpdate,
148
- // @TODO if at least 1 segment fetch fails due to 404 and other segments are updated in the storage, SDK_UPDATE is not emitted
149
- 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
+ return Promise.all([
122
+ splits.update(mutation.added, mutation.removed, splitChanges.till),
123
+ segments.registerSegments(mutation.segments)
150
124
  ]).then(function (_a) {
151
- var ffChanged = _a[0], rbsChanged = _a[1];
125
+ var isThereUpdate = _a[0];
152
126
  if (splitsEventEmitter) {
153
127
  // To emit SDK_SPLITS_ARRIVED for server-side SDK, we must check that all registered segments have been fetched
154
- return Promise.resolve(!splitsEventEmitter.splitsArrived || ((ffChanged || rbsChanged) && (isClientSide || checkAllSegmentsExist(segments))))
128
+ return Promise.resolve(!splitsEventEmitter.splitsArrived || (since !== splitChanges.till && isThereUpdate && (isClientSide || checkAllSegmentsExist(segments))))
155
129
  .catch(function () { return false; } /** noop. just to handle a possible `checkAllSegmentsExist` rejection, before emitting SDK event */)
156
130
  .then(function (emitSplitsArrivedEvent) {
157
131
  // emit SDK events
@@ -168,7 +142,7 @@ export function splitChangesUpdaterFactory(log, splitChangesFetcher, storage, sp
168
142
  if (startingUp && retriesOnFailureBeforeReady > retry) {
169
143
  retry += 1;
170
144
  log.info(SYNC_SPLITS_FETCH_RETRY, [retry, error]);
171
- return _splitChangesUpdater(sinces, retry);
145
+ return _splitChangesUpdater(since, retry);
172
146
  }
173
147
  else {
174
148
  startingUp = false;
@@ -176,7 +150,7 @@ export function splitChangesUpdaterFactory(log, splitChangesFetcher, storage, sp
176
150
  return false;
177
151
  });
178
152
  }
179
- // `getChangeNumber` never rejects or throws error
180
- return Promise.all([splits.getChangeNumber(), rbSegments.getChangeNumber()]).then(_splitChangesUpdater);
153
+ var sincePromise = Promise.resolve(splits.getChangeNumber()); // `getChangeNumber` never rejects or throws error
154
+ return sincePromise.then(_splitChangesUpdater);
181
155
  };
182
156
  }
@@ -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, RB_SEGMENT_UPDATE } from '../constants';
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, storage, splitsSyncTask, splitsEventEmitter, telemetryTracker, segmentsSyncTask) {
12
- var ff = SplitsUpdateWorker(storage.splits);
13
- var rbs = SplitsUpdateWorker(storage.rbSegments);
14
- function SplitsUpdateWorker(cache) {
15
- var maxChangeNumber = -1;
16
- var handleNewEvent = false;
17
- var isHandlingEvent;
18
- var cdnBypass;
19
- var instantUpdate;
20
- var backoff = new Backoff(__handleSplitUpdateCall, FETCH_BACKOFF_BASE, FETCH_BACKOFF_MAX_WAIT);
21
- function __handleSplitUpdateCall() {
22
- isHandlingEvent = true;
23
- if (maxChangeNumber > cache.getChangeNumber()) {
24
- handleNewEvent = false;
25
- // fetch splits revalidating data if cached
26
- splitsSyncTask.execute(true, cdnBypass ? maxChangeNumber : undefined, instantUpdate).then(function () {
27
- if (!isHandlingEvent)
28
- return; // halt if `stop` has been called
29
- if (handleNewEvent) {
30
- __handleSplitUpdateCall();
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
- if (instantUpdate)
34
- telemetryTracker.trackUpdatesFromSSE(SPLITS);
35
- // fetch new registered segments for server-side API. Not retrying on error
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
- if (backoff.timeoutID || !isHandlingEvent)
83
- __handleSplitUpdateCall();
84
- backoff.reset();
85
- },
86
- stop: function () {
87
- isHandlingEvent = false;
88
- backoff.reset();
89
- },
90
- isSync: function () {
91
- return maxChangeNumber <= cache.getChangeNumber();
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: function (parsedData) {
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 (storage.splits.killLocally(splitName, defaultTreatment, changeNumber)) {
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
- ff.put({ changeNumber: changeNumber });
96
+ put({ changeNumber: changeNumber });
126
97
  },
127
98
  stop: function () {
128
- ff.stop();
129
- rbs.stop();
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, RB_SEGMENT_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 } 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, splitsUpdateWorker.put);
183
- pushEmitter.on(RB_SEGMENT_UPDATE, splitsUpdateWorker.put);
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: {
@@ -127,7 +127,7 @@ export function syncManagerOnlineFactory(pollingManagerFactory, pushManagerFacto
127
127
  if (pushManager) {
128
128
  if (pollingManager.isRunning()) {
129
129
  // if doing polling, we must start the periodic fetch of data
130
- if (storage.splits.usesSegments() || storage.rbSegments.usesSegments())
130
+ if (storage.splits.usesSegments())
131
131
  mySegmentsSyncTask.start();
132
132
  }
133
133
  else {
@@ -137,7 +137,7 @@ export function syncManagerOnlineFactory(pollingManagerFactory, pushManagerFacto
137
137
  }
138
138
  }
139
139
  else {
140
- if (storage.splits.usesSegments() || storage.rbSegments.usesSegments())
140
+ if (storage.splits.usesSegments())
141
141
  mySegmentsSyncTask.start();
142
142
  }
143
143
  }
@@ -86,11 +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.3';
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';
94
- export var STANDARD_SEGMENT = 'standard';
95
- export var LARGE_SEGMENT = 'large';
96
- export var RULE_BASED_SEGMENT = 'rule-based';
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@splitsoftware/splitio-commons",
3
- "version": "2.2.1-rc.4",
3
+ "version": "2.3.0",
4
4
  "description": "Split JavaScript SDK common components",
5
5
  "main": "cjs/index.js",
6
6
  "module": "esm/index.js",