@splitsoftware/splitio-commons 2.0.1 → 2.0.3-rc.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 (69) hide show
  1. package/CHANGES.txt +8 -0
  2. package/cjs/evaluator/index.js +2 -0
  3. package/cjs/listeners/browser.js +4 -6
  4. package/cjs/readiness/readinessManager.js +16 -18
  5. package/cjs/sdkClient/client.js +13 -13
  6. package/cjs/sdkClient/sdkClient.js +1 -1
  7. package/cjs/sdkFactory/index.js +10 -14
  8. package/cjs/sdkManager/index.js +2 -1
  9. package/cjs/services/decorateHeaders.js +6 -1
  10. package/cjs/services/splitHttpClient.js +1 -1
  11. package/cjs/storages/inLocalStorage/index.js +6 -20
  12. package/cjs/storages/inMemory/InMemoryStorage.js +4 -12
  13. package/cjs/storages/inMemory/InMemoryStorageCS.js +6 -19
  14. package/cjs/storages/inRedis/index.js +9 -13
  15. package/cjs/storages/pluggable/index.js +15 -19
  16. package/cjs/sync/submitters/impressionCountsSubmitter.js +2 -4
  17. package/cjs/sync/submitters/submitterManager.js +3 -6
  18. package/cjs/trackers/impressionsTracker.js +17 -18
  19. package/cjs/trackers/strategy/strategyDebug.js +4 -11
  20. package/cjs/trackers/strategy/strategyNone.js +11 -16
  21. package/cjs/trackers/strategy/strategyOptimized.js +11 -21
  22. package/esm/evaluator/index.js +2 -0
  23. package/esm/listeners/browser.js +1 -3
  24. package/esm/readiness/readinessManager.js +16 -18
  25. package/esm/sdkClient/client.js +13 -13
  26. package/esm/sdkClient/sdkClient.js +1 -1
  27. package/esm/sdkFactory/index.js +11 -15
  28. package/esm/sdkManager/index.js +2 -1
  29. package/esm/services/decorateHeaders.js +4 -0
  30. package/esm/services/splitHttpClient.js +2 -2
  31. package/esm/storages/inLocalStorage/index.js +7 -21
  32. package/esm/storages/inMemory/InMemoryStorage.js +5 -13
  33. package/esm/storages/inMemory/InMemoryStorageCS.js +7 -20
  34. package/esm/storages/inRedis/index.js +10 -14
  35. package/esm/storages/pluggable/index.js +16 -20
  36. package/esm/sync/submitters/impressionCountsSubmitter.js +2 -4
  37. package/esm/sync/submitters/submitterManager.js +3 -6
  38. package/esm/trackers/impressionsTracker.js +17 -18
  39. package/esm/trackers/strategy/strategyDebug.js +4 -11
  40. package/esm/trackers/strategy/strategyNone.js +11 -16
  41. package/esm/trackers/strategy/strategyOptimized.js +11 -21
  42. package/package.json +1 -1
  43. package/src/dtos/types.ts +2 -1
  44. package/src/evaluator/index.ts +2 -0
  45. package/src/evaluator/types.ts +1 -1
  46. package/src/listeners/browser.ts +1 -3
  47. package/src/readiness/readinessManager.ts +15 -16
  48. package/src/sdkClient/client.ts +11 -11
  49. package/src/sdkClient/sdkClient.ts +1 -1
  50. package/src/sdkFactory/index.ts +12 -16
  51. package/src/sdkFactory/types.ts +1 -1
  52. package/src/sdkManager/index.ts +2 -1
  53. package/src/services/decorateHeaders.ts +5 -0
  54. package/src/services/splitHttpClient.ts +2 -2
  55. package/src/storages/inLocalStorage/index.ts +7 -20
  56. package/src/storages/inMemory/InMemoryStorage.ts +5 -13
  57. package/src/storages/inMemory/InMemoryStorageCS.ts +7 -20
  58. package/src/storages/inRedis/index.ts +10 -10
  59. package/src/storages/pluggable/index.ts +16 -20
  60. package/src/storages/types.ts +2 -2
  61. package/src/sync/submitters/impressionCountsSubmitter.ts +2 -4
  62. package/src/sync/submitters/submitterManager.ts +3 -4
  63. package/src/sync/submitters/uniqueKeysSubmitter.ts +2 -3
  64. package/src/trackers/impressionsTracker.ts +17 -18
  65. package/src/trackers/strategy/strategyDebug.ts +4 -11
  66. package/src/trackers/strategy/strategyNone.ts +11 -17
  67. package/src/trackers/strategy/strategyOptimized.ts +10 -20
  68. package/src/trackers/types.ts +2 -8
  69. package/types/splitio.d.ts +4 -0
package/CHANGES.txt CHANGED
@@ -1,3 +1,11 @@
1
+ 2.1.0 (January XX, 2025)
2
+ - Added `trackImpressions` property to SDK Manager's `SplitView` type.
3
+ - Updated implementation of the impressions tracker and strategies to support feature flags with impressions tracking disabled.
4
+
5
+ 2.0.2 (December 3, 2024)
6
+ - Updated the factory `init` and `destroy` methods to support re-initialization after destruction. This update ensures compatibility of the React SDK with React Strict Mode, where the factory's `init` and `destroy` effects are executed an extra time to validate proper resource cleanup.
7
+ - Bugfixing - Sanitize the `SplitSDKMachineName` header value to avoid exceptions on HTTP/S requests when it contains non ISO-8859-1 characters (Related to issue https://github.com/splitio/javascript-client/issues/847).
8
+
1
9
  2.0.1 (November 25, 2024)
2
10
  - Bugfixing - Fixed an issue with the SDK_UPDATE event on server-side, where it was not being emitted if there was an empty segment and the SDK received a feature flag update notification.
3
11
 
@@ -104,12 +104,14 @@ function getEvaluation(log, splitJSON, key, attributes, storage) {
104
104
  return evaluation.then(function (result) {
105
105
  result.changeNumber = split_1.getChangeNumber();
106
106
  result.config = splitJSON.configurations && splitJSON.configurations[result.treatment] || null;
107
+ result.track = splitJSON.trackImpressions;
107
108
  return result;
108
109
  });
109
110
  }
110
111
  else {
111
112
  evaluation.changeNumber = split_1.getChangeNumber(); // Always sync and optional
112
113
  evaluation.config = splitJSON.configurations && splitJSON.configurations[evaluation.treatment] || null;
114
+ evaluation.track = splitJSON.trackImpressions;
113
115
  }
114
116
  }
115
117
  return evaluation;
@@ -3,9 +3,8 @@ Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.BrowserSignalListener = void 0;
4
4
  var impressionsSubmitter_1 = require("../sync/submitters/impressionsSubmitter");
5
5
  var impressionCountsSubmitter_1 = require("../sync/submitters/impressionCountsSubmitter");
6
- var constants_1 = require("../utils/constants");
7
6
  var objectAssign_1 = require("../utils/lang/objectAssign");
8
- var constants_2 = require("../logger/constants");
7
+ var constants_1 = require("../logger/constants");
9
8
  var consent_1 = require("../consent");
10
9
  var VISIBILITYCHANGE_EVENT = 'visibilitychange';
11
10
  var PAGEHIDE_EVENT = 'pagehide';
@@ -28,7 +27,7 @@ var BrowserSignalListener = /** @class */ (function () {
28
27
  * Called when SplitFactory is initialized, it adds event listeners to close streaming and flush impressions and events.
29
28
  */
30
29
  BrowserSignalListener.prototype.start = function () {
31
- this.settings.log.debug(constants_2.CLEANUP_REGISTERING, [EVENT_NAME]);
30
+ this.settings.log.debug(constants_1.CLEANUP_REGISTERING, [EVENT_NAME]);
32
31
  if (typeof document !== 'undefined' && document.addEventListener) {
33
32
  // Flush data whenever the page is hidden or unloaded.
34
33
  document.addEventListener(VISIBILITYCHANGE_EVENT, this.flushDataIfHidden);
@@ -44,7 +43,7 @@ var BrowserSignalListener = /** @class */ (function () {
44
43
  * Called when client is destroyed, it removes event listeners.
45
44
  */
46
45
  BrowserSignalListener.prototype.stop = function () {
47
- this.settings.log.debug(constants_2.CLEANUP_DEREGISTERING, [EVENT_NAME]);
46
+ this.settings.log.debug(constants_1.CLEANUP_DEREGISTERING, [EVENT_NAME]);
48
47
  if (typeof document !== 'undefined' && document.removeEventListener) {
49
48
  document.removeEventListener(VISIBILITYCHANGE_EVENT, this.flushDataIfHidden);
50
49
  }
@@ -63,10 +62,9 @@ var BrowserSignalListener = /** @class */ (function () {
63
62
  var _a = this.settings.urls, events = _a.events, telemetry = _a.telemetry;
64
63
  // Flush impressions & events data if there is user consent
65
64
  if ((0, consent_1.isConsentGranted)(this.settings)) {
66
- var sim = this.settings.sync.impressionsMode;
67
65
  var extraMetadata = {
68
66
  // sim stands for Sync/Split Impressions Mode
69
- sim: sim === constants_1.OPTIMIZED ? constants_1.OPTIMIZED : sim === constants_1.DEBUG ? constants_1.DEBUG : constants_1.NONE
67
+ sim: this.settings.sync.impressionsMode
70
68
  };
71
69
  this._flushData(events + '/testImpressions/beacon', this.storage.impressions, this.serviceApi.postTestImpressionsBulk, this.fromImpressionsCollector, extraMetadata);
72
70
  this._flushData(events + '/events/beacon', this.storage.events, this.serviceApi.postEventsBulk);
@@ -28,7 +28,7 @@ function segmentsEventEmitterFactory(EventEmitter) {
28
28
  /**
29
29
  * Factory of readiness manager, which handles the ready / update event propagation.
30
30
  */
31
- function readinessManagerFactory(EventEmitter, settings, splits) {
31
+ function readinessManagerFactory(EventEmitter, settings, splits, isShared) {
32
32
  if (splits === void 0) { splits = splitsEventEmitterFactory(EventEmitter); }
33
33
  var readyTimeout = settings.startup.readyTimeout;
34
34
  var segments = segmentsEventEmitterFactory(EventEmitter);
@@ -54,22 +54,24 @@ function readinessManagerFactory(EventEmitter, settings, splits) {
54
54
  syncLastUpdate();
55
55
  gate.emit(constants_1.SDK_READY_TIMED_OUT, 'Split SDK emitted SDK_READY_TIMED_OUT event.');
56
56
  }
57
- var readyTimeoutId;
58
- if (readyTimeout > 0) {
59
- if (splits.hasInit)
60
- readyTimeoutId = setTimeout(timeout, readyTimeout);
61
- else
62
- splits.initCallbacks.push(function () { readyTimeoutId = setTimeout(timeout, readyTimeout); });
63
- }
64
57
  // emit SDK_READY and SDK_UPDATE
65
58
  var isReady = false;
66
59
  splits.on(constants_1.SDK_SPLITS_ARRIVED, checkIsReadyOrUpdate);
67
60
  segments.on(constants_1.SDK_SEGMENTS_ARRIVED, checkIsReadyOrUpdate);
68
61
  var isDestroyed = false;
62
+ var readyTimeoutId;
63
+ function __init() {
64
+ isDestroyed = false;
65
+ if (readyTimeout > 0 && !isReady)
66
+ readyTimeoutId = setTimeout(timeout, readyTimeout);
67
+ }
68
+ splits.initCallbacks.push(__init);
69
+ if (splits.hasInit)
70
+ __init();
69
71
  function checkIsReadyFromCache() {
70
72
  isReadyFromCache = true;
71
73
  // Don't emit SDK_READY_FROM_CACHE if SDK_READY has been emitted
72
- if (!isReady) {
74
+ if (!isReady && !isDestroyed) {
73
75
  try {
74
76
  syncLastUpdate();
75
77
  gate.emit(constants_1.SDK_READY_FROM_CACHE);
@@ -81,6 +83,8 @@ function readinessManagerFactory(EventEmitter, settings, splits) {
81
83
  }
82
84
  }
83
85
  function checkIsReadyOrUpdate(diff) {
86
+ if (isDestroyed)
87
+ return;
84
88
  if (isReady) {
85
89
  try {
86
90
  syncLastUpdate();
@@ -106,14 +110,12 @@ function readinessManagerFactory(EventEmitter, settings, splits) {
106
110
  }
107
111
  }
108
112
  }
109
- var refCount = 1;
110
113
  return {
111
114
  splits: splits,
112
115
  segments: segments,
113
116
  gate: gate,
114
117
  shared: function () {
115
- refCount++;
116
- return readinessManagerFactory(EventEmitter, settings, splits);
118
+ return readinessManagerFactory(EventEmitter, settings, splits, true);
117
119
  },
118
120
  // @TODO review/remove next methods when non-recoverable errors are reworked
119
121
  // Called on consumer mode, when storage fails to connect
@@ -130,13 +132,9 @@ function readinessManagerFactory(EventEmitter, settings, splits) {
130
132
  destroy: function () {
131
133
  isDestroyed = true;
132
134
  syncLastUpdate();
133
- segments.removeAllListeners();
134
- gate.removeAllListeners();
135
135
  clearTimeout(readyTimeoutId);
136
- if (refCount > 0)
137
- refCount--;
138
- if (refCount === 0)
139
- splits.removeAllListeners();
136
+ if (!isShared)
137
+ splits.hasInit = false;
140
138
  },
141
139
  isReady: function () { return isReady; },
142
140
  isReadyFromCache: function () { return isReadyFromCache; },
@@ -33,7 +33,7 @@ function clientFactory(params) {
33
33
  var queue = [];
34
34
  var treatment = processEvaluation(evaluationResult, featureFlagName, key, attributes, withConfig, methodName, queue);
35
35
  impressionsTracker.track(queue, attributes);
36
- stopTelemetryTracker(queue[0] && queue[0].label);
36
+ stopTelemetryTracker(queue[0] && queue[0][0].label);
37
37
  return treatment;
38
38
  };
39
39
  var evaluation = readinessManager.isReady() || readinessManager.isReadyFromCache() ?
@@ -57,7 +57,7 @@ function clientFactory(params) {
57
57
  treatments[featureFlagName] = processEvaluation(evaluationResults[featureFlagName], featureFlagName, key, attributes, withConfig, methodName, queue);
58
58
  });
59
59
  impressionsTracker.track(queue, attributes);
60
- stopTelemetryTracker(queue[0] && queue[0].label);
60
+ stopTelemetryTracker(queue[0] && queue[0][0].label);
61
61
  return treatments;
62
62
  };
63
63
  var evaluations = readinessManager.isReady() || readinessManager.isReadyFromCache() ?
@@ -83,7 +83,7 @@ function clientFactory(params) {
83
83
  treatments[featureFlagName] = processEvaluation(evaluations[featureFlagName], featureFlagName, key, attributes, withConfig, methodName, queue);
84
84
  });
85
85
  impressionsTracker.track(queue, attributes);
86
- stopTelemetryTracker(queue[0] && queue[0].label);
86
+ stopTelemetryTracker(queue[0] && queue[0][0].label);
87
87
  return treatments;
88
88
  };
89
89
  var evaluations = readinessManager.isReady() || readinessManager.isReadyFromCache() ?
@@ -106,19 +106,19 @@ function clientFactory(params) {
106
106
  function processEvaluation(evaluation, featureFlagName, key, attributes, withConfig, invokingMethodName, queue) {
107
107
  var matchingKey = (0, key_1.getMatching)(key);
108
108
  var bucketingKey = (0, key_1.getBucketing)(key);
109
- var treatment = evaluation.treatment, label = evaluation.label, changeNumber = evaluation.changeNumber, _a = evaluation.config, config = _a === void 0 ? null : _a;
109
+ var treatment = evaluation.treatment, label = evaluation.label, changeNumber = evaluation.changeNumber, _a = evaluation.config, config = _a === void 0 ? null : _a, track = evaluation.track;
110
110
  log.info(constants_2.IMPRESSION, [featureFlagName, matchingKey, treatment, label]);
111
111
  if ((0, splitExistence_1.validateSplitExistence)(log, readinessManager, featureFlagName, label, invokingMethodName)) {
112
112
  log.info(constants_2.IMPRESSION_QUEUEING);
113
- queue.push({
114
- feature: featureFlagName,
115
- keyName: matchingKey,
116
- treatment: treatment,
117
- time: Date.now(),
118
- bucketingKey: bucketingKey,
119
- label: label,
120
- changeNumber: changeNumber
121
- });
113
+ queue.push([{
114
+ feature: featureFlagName,
115
+ keyName: matchingKey,
116
+ treatment: treatment,
117
+ time: Date.now(),
118
+ bucketingKey: bucketingKey,
119
+ label: label,
120
+ changeNumber: changeNumber,
121
+ }, track]);
122
122
  }
123
123
  if (withConfig) {
124
124
  return {
@@ -50,7 +50,7 @@ function sdkClientFactory(params, isSharedClient) {
50
50
  (0, apiKey_1.releaseApiKey)(settings.core.authorizationKey);
51
51
  telemetryTracker.sessionLength();
52
52
  signalListener && signalListener.stop();
53
- uniqueKeysTracker && uniqueKeysTracker.stop();
53
+ uniqueKeysTracker.stop();
54
54
  }
55
55
  // Stop background jobs
56
56
  syncManager && syncManager.stop();
@@ -51,19 +51,14 @@ function sdkFactory(params) {
51
51
  var telemetryTracker = (0, telemetryTracker_1.telemetryTrackerFactory)(storage.telemetry, platform.now);
52
52
  var integrationsManager = integrationsManagerFactory && integrationsManagerFactory({ settings: settings, storage: storage, telemetryTracker: telemetryTracker });
53
53
  var observer = impressionsObserverFactory();
54
- var uniqueKeysTracker = impressionsMode === constants_3.NONE ? (0, uniqueKeysTracker_1.uniqueKeysTrackerFactory)(log, storage.uniqueKeys, filterAdapterFactory && filterAdapterFactory()) : undefined;
55
- var strategy;
56
- switch (impressionsMode) {
57
- case constants_3.OPTIMIZED:
58
- strategy = (0, strategyOptimized_1.strategyOptimizedFactory)(observer, storage.impressionCounts);
59
- break;
60
- case constants_3.NONE:
61
- strategy = (0, strategyNone_1.strategyNoneFactory)(storage.impressionCounts, uniqueKeysTracker);
62
- break;
63
- default:
64
- strategy = (0, strategyDebug_1.strategyDebugFactory)(observer);
65
- }
66
- var impressionsTracker = (0, impressionsTracker_1.impressionsTrackerFactory)(settings, storage.impressions, strategy, whenInit, integrationsManager, storage.telemetry);
54
+ var uniqueKeysTracker = (0, uniqueKeysTracker_1.uniqueKeysTrackerFactory)(log, storage.uniqueKeys, filterAdapterFactory && filterAdapterFactory());
55
+ var noneStrategy = (0, strategyNone_1.strategyNoneFactory)(storage.impressionCounts, uniqueKeysTracker);
56
+ var strategy = impressionsMode === constants_3.OPTIMIZED ?
57
+ (0, strategyOptimized_1.strategyOptimizedFactory)(observer, storage.impressionCounts) :
58
+ impressionsMode === constants_3.DEBUG ?
59
+ (0, strategyDebug_1.strategyDebugFactory)(observer) :
60
+ noneStrategy;
61
+ var impressionsTracker = (0, impressionsTracker_1.impressionsTrackerFactory)(settings, storage.impressions, noneStrategy, strategy, whenInit, integrationsManager, storage.telemetry);
67
62
  var eventTracker = (0, eventTracker_1.eventTrackerFactory)(settings, storage.events, whenInit, integrationsManager, storage.telemetry);
68
63
  // splitApi is used by SyncManager and Browser signal listener
69
64
  var splitApi = splitApiFactory && splitApiFactory(settings, platform, telemetryTracker);
@@ -82,7 +77,7 @@ function sdkFactory(params) {
82
77
  // We will just log and allow for the SDK to end up throwing an SDK_TIMEOUT event for devs to handle.
83
78
  (0, apiKey_1.validateAndTrackApiKey)(log, settings.core.authorizationKey);
84
79
  readiness.init();
85
- uniqueKeysTracker && uniqueKeysTracker.start();
80
+ uniqueKeysTracker.start();
86
81
  syncManager && syncManager.start();
87
82
  signalListener && signalListener.start();
88
83
  initCallbacks.forEach(function (cb) { return cb(); });
@@ -102,6 +97,7 @@ function sdkFactory(params) {
102
97
  Logger: (0, sdkLogger_1.createLoggerAPI)(log),
103
98
  settings: settings,
104
99
  destroy: function () {
100
+ hasInit = false;
105
101
  return Promise.all(Object.keys(clients).map(function (key) { return clients[key].destroy(); })).then(function () { });
106
102
  }
107
103
  }, extraProps && extraProps(ctx), lazyInit ? { init: init } : init());
@@ -28,7 +28,8 @@ function objectToView(splitObject) {
28
28
  treatments: collectTreatments(splitObject),
29
29
  configs: splitObject.configurations || {},
30
30
  sets: splitObject.sets || [],
31
- defaultTreatment: splitObject.defaultTreatment
31
+ defaultTreatment: splitObject.defaultTreatment,
32
+ trackImpressions: splitObject.trackImpressions !== false
32
33
  };
33
34
  }
34
35
  function objectsToViews(splitObjects) {
@@ -1,6 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.decorateHeaders = void 0;
3
+ exports.removeNonISO88591 = exports.decorateHeaders = void 0;
4
4
  var objectAssign_1 = require("../utils/lang/objectAssign");
5
5
  var FORBIDDEN_HEADERS = new Set([
6
6
  'splitsdkclientkey',
@@ -33,3 +33,8 @@ function decorateHeaders(settings, headers) {
33
33
  return headers;
34
34
  }
35
35
  exports.decorateHeaders = decorateHeaders;
36
+ function removeNonISO88591(input) {
37
+ // eslint-disable-next-line no-control-regex
38
+ return input.replace(/[^\x00-\xFF]/g, '');
39
+ }
40
+ exports.removeNonISO88591 = removeNonISO88591;
@@ -28,7 +28,7 @@ function splitHttpClientFactory(settings, _a) {
28
28
  if (ip)
29
29
  commonHeaders['SplitSDKMachineIP'] = ip;
30
30
  if (hostname)
31
- commonHeaders['SplitSDKMachineName'] = hostname;
31
+ commonHeaders['SplitSDKMachineName'] = (0, decorateHeaders_1.removeNonISO88591)(hostname);
32
32
  return function httpClient(url, reqOpts, latencyTracker, logErrorsAsInfo) {
33
33
  if (reqOpts === void 0) { reqOpts = {}; }
34
34
  if (latencyTracker === void 0) { latencyTracker = function () { }; }
@@ -9,8 +9,6 @@ var KeyBuilderCS_1 = require("../KeyBuilderCS");
9
9
  var isLocalStorageAvailable_1 = require("../../utils/env/isLocalStorageAvailable");
10
10
  var SplitsCacheInLocal_1 = require("./SplitsCacheInLocal");
11
11
  var MySegmentsCacheInLocal_1 = require("./MySegmentsCacheInLocal");
12
- var MySegmentsCacheInMemory_1 = require("../inMemory/MySegmentsCacheInMemory");
13
- var SplitsCacheInMemory_1 = require("../inMemory/SplitsCacheInMemory");
14
12
  var browser_1 = require("../../utils/constants/browser");
15
13
  var InMemoryStorageCS_1 = require("../inMemory/InMemoryStorageCS");
16
14
  var constants_1 = require("./constants");
@@ -30,7 +28,7 @@ function InLocalStorage(options) {
30
28
  params.settings.log.warn(constants_1.LOG_PREFIX + 'LocalStorage API is unavailable. Falling back to default MEMORY storage');
31
29
  return (0, InMemoryStorageCS_1.InMemoryStorageCSFactory)(params);
32
30
  }
33
- var settings = params.settings, _a = params.settings, log = _a.log, _b = _a.scheduler, impressionsQueueSize = _b.impressionsQueueSize, eventsQueueSize = _b.eventsQueueSize, _c = _a.sync, impressionsMode = _c.impressionsMode, __splitFiltersValidation = _c.__splitFiltersValidation;
31
+ var settings = params.settings, _a = params.settings, log = _a.log, _b = _a.scheduler, impressionsQueueSize = _b.impressionsQueueSize, eventsQueueSize = _b.eventsQueueSize;
34
32
  var matchingKey = (0, key_1.getMatching)(settings.core.key);
35
33
  var keys = new KeyBuilderCS_1.KeyBuilderCS(prefix, matchingKey);
36
34
  var expirationTimestamp = Date.now() - browser_1.DEFAULT_CACHE_EXPIRATION_IN_MILLIS;
@@ -42,20 +40,11 @@ function InLocalStorage(options) {
42
40
  segments: segments,
43
41
  largeSegments: largeSegments,
44
42
  impressions: new ImpressionsCacheInMemory_1.ImpressionsCacheInMemory(impressionsQueueSize),
45
- impressionCounts: impressionsMode !== constants_2.DEBUG ? new ImpressionCountsCacheInMemory_1.ImpressionCountsCacheInMemory() : undefined,
43
+ impressionCounts: new ImpressionCountsCacheInMemory_1.ImpressionCountsCacheInMemory(),
46
44
  events: new EventsCacheInMemory_1.EventsCacheInMemory(eventsQueueSize),
47
45
  telemetry: (0, TelemetryCacheInMemory_1.shouldRecordTelemetry)(params) ? new TelemetryCacheInMemory_1.TelemetryCacheInMemory(splits, segments) : undefined,
48
- uniqueKeys: impressionsMode === constants_2.NONE ? new UniqueKeysCacheInMemoryCS_1.UniqueKeysCacheInMemoryCS() : undefined,
49
- destroy: function () {
50
- var _a;
51
- this.splits = new SplitsCacheInMemory_1.SplitsCacheInMemory(__splitFiltersValidation);
52
- this.segments = new MySegmentsCacheInMemory_1.MySegmentsCacheInMemory();
53
- this.largeSegments = new MySegmentsCacheInMemory_1.MySegmentsCacheInMemory();
54
- this.impressions.clear();
55
- this.impressionCounts && this.impressionCounts.clear();
56
- this.events.clear();
57
- (_a = this.uniqueKeys) === null || _a === void 0 ? void 0 : _a.clear();
58
- },
46
+ uniqueKeys: new UniqueKeysCacheInMemoryCS_1.UniqueKeysCacheInMemoryCS(),
47
+ destroy: function () { },
59
48
  // When using shared instantiation with MEMORY we reuse everything but segments (they are customer per key).
60
49
  shared: function (matchingKey) {
61
50
  return {
@@ -66,11 +55,8 @@ function InLocalStorage(options) {
66
55
  impressionCounts: this.impressionCounts,
67
56
  events: this.events,
68
57
  telemetry: this.telemetry,
69
- destroy: function () {
70
- this.splits = new SplitsCacheInMemory_1.SplitsCacheInMemory(__splitFiltersValidation);
71
- this.segments = new MySegmentsCacheInMemory_1.MySegmentsCacheInMemory();
72
- this.largeSegments = new MySegmentsCacheInMemory_1.MySegmentsCacheInMemory();
73
- }
58
+ uniqueKeys: this.uniqueKeys,
59
+ destroy: function () { }
74
60
  };
75
61
  },
76
62
  };
@@ -15,26 +15,18 @@ var UniqueKeysCacheInMemory_1 = require("./UniqueKeysCacheInMemory");
15
15
  * @param params - parameters required by EventsCacheSync
16
16
  */
17
17
  function InMemoryStorageFactory(params) {
18
- var _a = params.settings, _b = _a.scheduler, impressionsQueueSize = _b.impressionsQueueSize, eventsQueueSize = _b.eventsQueueSize, _c = _a.sync, impressionsMode = _c.impressionsMode, __splitFiltersValidation = _c.__splitFiltersValidation;
18
+ var _a = params.settings, _b = _a.scheduler, impressionsQueueSize = _b.impressionsQueueSize, eventsQueueSize = _b.eventsQueueSize, __splitFiltersValidation = _a.sync.__splitFiltersValidation;
19
19
  var splits = new SplitsCacheInMemory_1.SplitsCacheInMemory(__splitFiltersValidation);
20
20
  var segments = new SegmentsCacheInMemory_1.SegmentsCacheInMemory();
21
21
  var storage = {
22
22
  splits: splits,
23
23
  segments: segments,
24
24
  impressions: new ImpressionsCacheInMemory_1.ImpressionsCacheInMemory(impressionsQueueSize),
25
- impressionCounts: impressionsMode !== constants_1.DEBUG ? new ImpressionCountsCacheInMemory_1.ImpressionCountsCacheInMemory() : undefined,
25
+ impressionCounts: new ImpressionCountsCacheInMemory_1.ImpressionCountsCacheInMemory(),
26
26
  events: new EventsCacheInMemory_1.EventsCacheInMemory(eventsQueueSize),
27
27
  telemetry: (0, TelemetryCacheInMemory_1.shouldRecordTelemetry)(params) ? new TelemetryCacheInMemory_1.TelemetryCacheInMemory(splits, segments) : undefined,
28
- uniqueKeys: impressionsMode === constants_1.NONE ? new UniqueKeysCacheInMemory_1.UniqueKeysCacheInMemory() : undefined,
29
- // When using MEMORY we should clean all the caches to leave them empty
30
- destroy: function () {
31
- this.splits.clear();
32
- this.segments.clear();
33
- this.impressions.clear();
34
- this.impressionCounts && this.impressionCounts.clear();
35
- this.events.clear();
36
- this.uniqueKeys && this.uniqueKeys.clear();
37
- }
28
+ uniqueKeys: new UniqueKeysCacheInMemory_1.UniqueKeysCacheInMemory(),
29
+ destroy: function () { }
38
30
  };
39
31
  // @TODO revisit storage logic in localhost mode
40
32
  // No tracking data in localhost mode to avoid memory leaks
@@ -15,7 +15,7 @@ var UniqueKeysCacheInMemoryCS_1 = require("./UniqueKeysCacheInMemoryCS");
15
15
  * @param params - parameters required by EventsCacheSync
16
16
  */
17
17
  function InMemoryStorageCSFactory(params) {
18
- var _a = params.settings, _b = _a.scheduler, impressionsQueueSize = _b.impressionsQueueSize, eventsQueueSize = _b.eventsQueueSize, _c = _a.sync, impressionsMode = _c.impressionsMode, __splitFiltersValidation = _c.__splitFiltersValidation;
18
+ var _a = params.settings, _b = _a.scheduler, impressionsQueueSize = _b.impressionsQueueSize, eventsQueueSize = _b.eventsQueueSize, __splitFiltersValidation = _a.sync.__splitFiltersValidation;
19
19
  var splits = new SplitsCacheInMemory_1.SplitsCacheInMemory(__splitFiltersValidation);
20
20
  var segments = new MySegmentsCacheInMemory_1.MySegmentsCacheInMemory();
21
21
  var largeSegments = new MySegmentsCacheInMemory_1.MySegmentsCacheInMemory();
@@ -24,20 +24,11 @@ function InMemoryStorageCSFactory(params) {
24
24
  segments: segments,
25
25
  largeSegments: largeSegments,
26
26
  impressions: new ImpressionsCacheInMemory_1.ImpressionsCacheInMemory(impressionsQueueSize),
27
- impressionCounts: impressionsMode !== constants_1.DEBUG ? new ImpressionCountsCacheInMemory_1.ImpressionCountsCacheInMemory() : undefined,
27
+ impressionCounts: new ImpressionCountsCacheInMemory_1.ImpressionCountsCacheInMemory(),
28
28
  events: new EventsCacheInMemory_1.EventsCacheInMemory(eventsQueueSize),
29
29
  telemetry: (0, TelemetryCacheInMemory_1.shouldRecordTelemetry)(params) ? new TelemetryCacheInMemory_1.TelemetryCacheInMemory(splits, segments) : undefined,
30
- uniqueKeys: impressionsMode === constants_1.NONE ? new UniqueKeysCacheInMemoryCS_1.UniqueKeysCacheInMemoryCS() : undefined,
31
- // When using MEMORY we should clean all the caches to leave them empty
32
- destroy: function () {
33
- this.splits.clear();
34
- this.segments.clear();
35
- this.largeSegments.clear();
36
- this.impressions.clear();
37
- this.impressionCounts && this.impressionCounts.clear();
38
- this.events.clear();
39
- this.uniqueKeys && this.uniqueKeys.clear();
40
- },
30
+ uniqueKeys: new UniqueKeysCacheInMemoryCS_1.UniqueKeysCacheInMemoryCS(),
31
+ destroy: function () { },
41
32
  // When using shared instantiation with MEMORY we reuse everything but segments (they are unique per key)
42
33
  shared: function () {
43
34
  return {
@@ -48,12 +39,8 @@ function InMemoryStorageCSFactory(params) {
48
39
  impressionCounts: this.impressionCounts,
49
40
  events: this.events,
50
41
  telemetry: this.telemetry,
51
- // Set a new splits cache to clean it for the client without affecting other clients
52
- destroy: function () {
53
- this.splits = new SplitsCacheInMemory_1.SplitsCacheInMemory(__splitFiltersValidation);
54
- this.segments.clear();
55
- this.largeSegments.clear();
56
- }
42
+ uniqueKeys: this.uniqueKeys,
43
+ destroy: function () { }
57
44
  };
58
45
  },
59
46
  };
@@ -23,20 +23,18 @@ function InRedisStorage(options) {
23
23
  var RD = require('./RedisAdapter').RedisAdapter;
24
24
  var prefix = (0, KeyBuilder_1.validatePrefix)(options.prefix);
25
25
  function InRedisStorageFactory(params) {
26
- var onReadyCb = params.onReadyCb, settings = params.settings, _a = params.settings, log = _a.log, impressionsMode = _a.sync.impressionsMode;
26
+ var onReadyCb = params.onReadyCb, settings = params.settings, log = params.settings.log;
27
27
  var metadata = (0, utils_1.metadataBuilder)(settings);
28
28
  var keys = new KeyBuilderSS_1.KeyBuilderSS(prefix, metadata);
29
29
  var redisClient = new RD(log, options.options || {});
30
30
  var telemetry = new TelemetryCacheInRedis_1.TelemetryCacheInRedis(log, keys, redisClient);
31
- var impressionCountsCache = impressionsMode !== constants_1.DEBUG ? new ImpressionCountsCacheInRedis_1.ImpressionCountsCacheInRedis(log, keys.buildImpressionsCountKey(), redisClient) : undefined;
32
- var uniqueKeysCache = impressionsMode === constants_1.NONE ? new UniqueKeysCacheInRedis_1.UniqueKeysCacheInRedis(log, keys.buildUniqueKeysKey(), redisClient) : undefined;
31
+ var impressionCountsCache = new ImpressionCountsCacheInRedis_1.ImpressionCountsCacheInRedis(log, keys.buildImpressionsCountKey(), redisClient);
32
+ var uniqueKeysCache = new UniqueKeysCacheInRedis_1.UniqueKeysCacheInRedis(log, keys.buildUniqueKeysKey(), redisClient);
33
33
  // subscription to Redis connect event in order to emit SDK_READY event on consumer mode
34
34
  redisClient.on('connect', function () {
35
35
  onReadyCb();
36
- if (impressionCountsCache)
37
- impressionCountsCache.start();
38
- if (uniqueKeysCache)
39
- uniqueKeysCache.start();
36
+ impressionCountsCache.start();
37
+ uniqueKeysCache.start();
40
38
  // Synchronize config
41
39
  telemetry.recordConfig();
42
40
  });
@@ -51,12 +49,10 @@ function InRedisStorage(options) {
51
49
  // When using REDIS we should:
52
50
  // 1- Disconnect from the storage
53
51
  destroy: function () {
54
- var promises = [];
55
- if (impressionCountsCache)
56
- promises.push(impressionCountsCache.stop());
57
- if (uniqueKeysCache)
58
- promises.push(uniqueKeysCache.stop());
59
- return Promise.all(promises).then(function () { redisClient.disconnect(); });
52
+ return Promise.all([
53
+ impressionCountsCache.stop(),
54
+ uniqueKeysCache.stop()
55
+ ]).then(function () { redisClient.disconnect(); });
60
56
  // @TODO check that caches works as expected when redisClient is disconnected
61
57
  }
62
58
  };
@@ -54,30 +54,26 @@ function PluggableStorage(options) {
54
54
  validatePluggableStorageOptions(options);
55
55
  var prefix = (0, KeyBuilder_1.validatePrefix)(options.prefix);
56
56
  function PluggableStorageFactory(params) {
57
- var onReadyCb = params.onReadyCb, settings = params.settings, _a = params.settings, log = _a.log, mode = _a.mode, impressionsMode = _a.sync.impressionsMode, _b = _a.scheduler, impressionsQueueSize = _b.impressionsQueueSize, eventsQueueSize = _b.eventsQueueSize;
57
+ var onReadyCb = params.onReadyCb, settings = params.settings, _a = params.settings, log = _a.log, mode = _a.mode, _b = _a.scheduler, impressionsQueueSize = _b.impressionsQueueSize, eventsQueueSize = _b.eventsQueueSize;
58
58
  var metadata = (0, utils_1.metadataBuilder)(settings);
59
59
  var keys = new KeyBuilderSS_1.KeyBuilderSS(prefix, metadata);
60
60
  var wrapper = (0, wrapperAdapter_1.wrapperAdapter)(log, options.wrapper);
61
- var isSyncronizer = mode === undefined; // If mode is not defined, the synchronizer is running
61
+ var isSynchronizer = mode === undefined; // If mode is not defined, the synchronizer is running
62
62
  var isPartialConsumer = mode === constants_1.CONSUMER_PARTIAL_MODE;
63
- var telemetry = (0, TelemetryCacheInMemory_1.shouldRecordTelemetry)(params) || isSyncronizer ?
63
+ var telemetry = (0, TelemetryCacheInMemory_1.shouldRecordTelemetry)(params) || isSynchronizer ?
64
64
  isPartialConsumer ?
65
65
  new TelemetryCacheInMemory_1.TelemetryCacheInMemory() :
66
66
  new TelemetryCachePluggable_1.TelemetryCachePluggable(log, keys, wrapper) :
67
67
  undefined;
68
- var impressionCountsCache = impressionsMode !== constants_1.DEBUG || isSyncronizer ?
69
- isPartialConsumer ?
70
- new ImpressionCountsCacheInMemory_1.ImpressionCountsCacheInMemory() :
71
- new ImpressionCountsCachePluggable_1.ImpressionCountsCachePluggable(log, keys.buildImpressionsCountKey(), wrapper) :
72
- undefined;
73
- var uniqueKeysCache = impressionsMode === constants_1.NONE || isSyncronizer ?
74
- isPartialConsumer ?
75
- settings.core.key === undefined ? new UniqueKeysCacheInMemory_1.UniqueKeysCacheInMemory() : new UniqueKeysCacheInMemoryCS_1.UniqueKeysCacheInMemoryCS() :
76
- new UniqueKeysCachePluggable_1.UniqueKeysCachePluggable(log, keys.buildUniqueKeysKey(), wrapper) :
77
- undefined;
68
+ var impressionCountsCache = isPartialConsumer ?
69
+ new ImpressionCountsCacheInMemory_1.ImpressionCountsCacheInMemory() :
70
+ new ImpressionCountsCachePluggable_1.ImpressionCountsCachePluggable(log, keys.buildImpressionsCountKey(), wrapper);
71
+ var uniqueKeysCache = isPartialConsumer ?
72
+ settings.core.key === undefined ? new UniqueKeysCacheInMemory_1.UniqueKeysCacheInMemory() : new UniqueKeysCacheInMemoryCS_1.UniqueKeysCacheInMemoryCS() :
73
+ new UniqueKeysCachePluggable_1.UniqueKeysCachePluggable(log, keys.buildUniqueKeysKey(), wrapper);
78
74
  // Connects to wrapper and emits SDK_READY event on main client
79
75
  var connectPromise = wrapper.connect().then(function () {
80
- if (isSyncronizer) {
76
+ if (isSynchronizer) {
81
77
  // In standalone or producer mode, clear storage if SDK key or feature flag filter has changed
82
78
  return wrapper.get(keys.buildHashKey()).then(function (hash) {
83
79
  var currentHash = (0, KeyBuilder_1.getStorageHash)(settings);
@@ -93,9 +89,9 @@ function PluggableStorage(options) {
93
89
  }
94
90
  else {
95
91
  // Start periodic flush of async storages if not running synchronizer (producer mode)
96
- if (impressionCountsCache && impressionCountsCache.start)
92
+ if (impressionCountsCache.start)
97
93
  impressionCountsCache.start();
98
- if (uniqueKeysCache && uniqueKeysCache.start)
94
+ if (uniqueKeysCache.start)
99
95
  uniqueKeysCache.start();
100
96
  if (telemetry && telemetry.recordConfig)
101
97
  telemetry.recordConfig();
@@ -116,9 +112,9 @@ function PluggableStorage(options) {
116
112
  uniqueKeys: uniqueKeysCache,
117
113
  // Stop periodic flush and disconnect the underlying storage
118
114
  destroy: function () {
119
- return Promise.all(isSyncronizer ? [] : [
120
- impressionCountsCache && impressionCountsCache.stop && impressionCountsCache.stop(),
121
- uniqueKeysCache && uniqueKeysCache.stop && uniqueKeysCache.stop(),
115
+ return Promise.all(isSynchronizer ? [] : [
116
+ impressionCountsCache.stop && impressionCountsCache.stop(),
117
+ uniqueKeysCache.stop && uniqueKeysCache.stop(),
122
118
  ]).then(function () { return wrapper.disconnect(); });
123
119
  },
124
120
  // emits SDK_READY event on shared clients and returns a reference to the storage
@@ -30,9 +30,7 @@ var IMPRESSIONS_COUNT_RATE = 1800000; // 30 minutes
30
30
  */
31
31
  function impressionCountsSubmitterFactory(params) {
32
32
  var log = params.settings.log, postTestImpressionsCount = params.splitApi.postTestImpressionsCount, impressionCounts = params.storage.impressionCounts;
33
- if (impressionCounts) {
34
- // retry impressions counts only once.
35
- return (0, submitter_1.submitterFactory)(log, postTestImpressionsCount, impressionCounts, IMPRESSIONS_COUNT_RATE, 'impression counts', fromImpressionCountsCollector, 1);
36
- }
33
+ // retry impressions counts only once.
34
+ return (0, submitter_1.submitterFactory)(log, postTestImpressionsCount, impressionCounts, IMPRESSIONS_COUNT_RATE, 'impression counts', fromImpressionCountsCollector, 1);
37
35
  }
38
36
  exports.impressionCountsSubmitterFactory = impressionCountsSubmitterFactory;
@@ -9,14 +9,11 @@ var uniqueKeysSubmitter_1 = require("./uniqueKeysSubmitter");
9
9
  function submitterManagerFactory(params) {
10
10
  var submitters = [
11
11
  (0, impressionsSubmitter_1.impressionsSubmitterFactory)(params),
12
- (0, eventsSubmitter_1.eventsSubmitterFactory)(params)
12
+ (0, eventsSubmitter_1.eventsSubmitterFactory)(params),
13
+ (0, impressionCountsSubmitter_1.impressionCountsSubmitterFactory)(params),
14
+ (0, uniqueKeysSubmitter_1.uniqueKeysSubmitterFactory)(params)
13
15
  ];
14
- var impressionCountsSubmitter = (0, impressionCountsSubmitter_1.impressionCountsSubmitterFactory)(params);
15
- if (impressionCountsSubmitter)
16
- submitters.push(impressionCountsSubmitter);
17
16
  var telemetrySubmitter = (0, telemetrySubmitter_1.telemetrySubmitterFactory)(params);
18
- if (params.storage.uniqueKeys)
19
- submitters.push((0, uniqueKeysSubmitter_1.uniqueKeysSubmitterFactory)(params));
20
17
  return {
21
18
  // `onlyTelemetry` true if SDK is created with userConsent not GRANTED
22
19
  start: function (onlyTelemetry) {