@splitsoftware/splitio-commons 1.13.2-rc.2 → 1.13.2-rc.3

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 (66) hide show
  1. package/CHANGES.txt +0 -1
  2. package/cjs/evaluator/matchers/gte.js +3 -3
  3. package/cjs/evaluator/matchers/index.js +11 -1
  4. package/cjs/evaluator/matchers/matcherTypes.js +6 -1
  5. package/cjs/evaluator/matchers/semver_between.js +14 -0
  6. package/cjs/evaluator/matchers/semver_eq.js +12 -0
  7. package/cjs/evaluator/matchers/semver_gte.js +12 -0
  8. package/cjs/evaluator/matchers/semver_inlist.js +14 -0
  9. package/cjs/evaluator/matchers/semver_lte.js +12 -0
  10. package/cjs/evaluator/matchersTransform/index.js +20 -13
  11. package/cjs/services/splitApi.js +5 -5
  12. package/cjs/storages/KeyBuilder.js +2 -3
  13. package/cjs/utils/Semver.js +100 -0
  14. package/cjs/utils/constants/index.js +1 -2
  15. package/cjs/utils/settingsValidation/logger/pluggableLogger.js +1 -1
  16. package/esm/evaluator/matchers/gte.js +3 -3
  17. package/esm/evaluator/matchers/index.js +11 -1
  18. package/esm/evaluator/matchers/matcherTypes.js +6 -1
  19. package/esm/evaluator/matchers/semver_between.js +10 -0
  20. package/esm/evaluator/matchers/semver_eq.js +8 -0
  21. package/esm/evaluator/matchers/semver_gte.js +8 -0
  22. package/esm/evaluator/matchers/semver_inlist.js +10 -0
  23. package/esm/evaluator/matchers/semver_lte.js +8 -0
  24. package/esm/evaluator/matchersTransform/index.js +20 -13
  25. package/esm/services/splitApi.js +6 -6
  26. package/esm/storages/KeyBuilder.js +2 -3
  27. package/esm/utils/Semver.js +97 -0
  28. package/esm/utils/constants/index.js +0 -1
  29. package/esm/utils/settingsValidation/logger/pluggableLogger.js +1 -1
  30. package/package.json +1 -2
  31. package/src/dtos/types.ts +34 -1
  32. package/src/evaluator/matchers/between.ts +1 -1
  33. package/src/evaluator/matchers/boolean.ts +1 -1
  34. package/src/evaluator/matchers/cont_all.ts +1 -1
  35. package/src/evaluator/matchers/cont_any.ts +1 -1
  36. package/src/evaluator/matchers/cont_str.ts +1 -1
  37. package/src/evaluator/matchers/eq.ts +1 -1
  38. package/src/evaluator/matchers/eq_set.ts +1 -1
  39. package/src/evaluator/matchers/ew.ts +1 -1
  40. package/src/evaluator/matchers/gte.ts +4 -4
  41. package/src/evaluator/matchers/index.ts +28 -18
  42. package/src/evaluator/matchers/lte.ts +1 -1
  43. package/src/evaluator/matchers/matcherTypes.ts +6 -1
  44. package/src/evaluator/matchers/part_of.ts +1 -1
  45. package/src/evaluator/matchers/semver_between.ts +18 -0
  46. package/src/evaluator/matchers/semver_eq.ts +13 -0
  47. package/src/evaluator/matchers/semver_gte.ts +13 -0
  48. package/src/evaluator/matchers/semver_inlist.ts +15 -0
  49. package/src/evaluator/matchers/semver_lte.ts +13 -0
  50. package/src/evaluator/matchers/string.ts +1 -1
  51. package/src/evaluator/matchers/sw.ts +1 -1
  52. package/src/evaluator/matchers/whitelist.ts +1 -1
  53. package/src/evaluator/matchersTransform/index.ts +32 -21
  54. package/src/evaluator/types.ts +2 -2
  55. package/src/services/splitApi.ts +6 -5
  56. package/src/storages/KeyBuilder.ts +2 -3
  57. package/src/utils/Semver.ts +107 -0
  58. package/src/utils/constants/index.ts +0 -2
  59. package/src/utils/settingsValidation/logger/pluggableLogger.ts +1 -1
  60. package/types/dtos/types.d.ts +26 -1
  61. package/types/evaluator/matchers/semver_between.d.ts +2 -2
  62. package/types/evaluator/matchers/semver_inlist.d.ts +3 -0
  63. package/types/evaluator/types.d.ts +2 -2
  64. package/types/storages/KeyBuilder.d.ts +1 -1
  65. package/types/utils/constants/index.d.ts +0 -1
  66. package/types/utils/semVer.d.ts +8 -15
package/CHANGES.txt CHANGED
@@ -1,6 +1,5 @@
1
1
  1.14.0 (April XX, 2024)
2
2
  - Updated impression label to 'unsupported matcher type' when the matcher type is not supported by the SDK.
3
- - Updated Split API client to include the flags spec version query parameter for the `splitChanges` endpoint.
4
3
 
5
4
  1.13.1 (January 10, 2024)
6
5
  - Updated client `destroy` method to release SDK key immediately and avoid unexpected warning logs when a factory is created with the same SDK key after the previous one was destroyed.
@@ -4,9 +4,9 @@ exports.greaterThanEqualMatcherContext = void 0;
4
4
  var constants_1 = require("../../logger/constants");
5
5
  function greaterThanEqualMatcherContext(log, ruleAttr) {
6
6
  return function greaterThanEqualMatcher(runtimeAttr) {
7
- var isGreaterEqualThan = runtimeAttr >= ruleAttr;
8
- log.debug(constants_1.ENGINE_MATCHER_GREATER, [runtimeAttr, ruleAttr, isGreaterEqualThan]);
9
- return isGreaterEqualThan;
7
+ var isGreaterThanEqual = runtimeAttr >= ruleAttr;
8
+ log.debug(constants_1.ENGINE_MATCHER_GREATER, [runtimeAttr, ruleAttr, isGreaterThanEqual]);
9
+ return isGreaterThanEqual;
10
10
  };
11
11
  }
12
12
  exports.greaterThanEqualMatcherContext = greaterThanEqualMatcherContext;
@@ -18,6 +18,11 @@ var cont_str_1 = require("./cont_str");
18
18
  var dependency_1 = require("./dependency");
19
19
  var boolean_1 = require("./boolean");
20
20
  var string_1 = require("./string");
21
+ var semver_eq_1 = require("./semver_eq");
22
+ var semver_gte_1 = require("./semver_gte");
23
+ var semver_lte_1 = require("./semver_lte");
24
+ var semver_between_1 = require("./semver_between");
25
+ var semver_inlist_1 = require("./semver_inlist");
21
26
  var matchers = [
22
27
  undefined,
23
28
  all_1.allMatcherContext,
@@ -36,7 +41,12 @@ var matchers = [
36
41
  cont_str_1.containsStringMatcherContext,
37
42
  dependency_1.dependencyMatcherContext,
38
43
  boolean_1.booleanMatcherContext,
39
- string_1.stringMatcherContext // MATCHES_STRING: 17
44
+ string_1.stringMatcherContext,
45
+ semver_eq_1.equalToSemverMatcherContext,
46
+ semver_gte_1.greaterThanEqualToSemverMatcherContext,
47
+ semver_lte_1.lessThanEqualToSemverMatcherContext,
48
+ semver_between_1.betweenSemverMatcherContext,
49
+ semver_inlist_1.inListSemverMatcherContext, // IN_LIST_SEMVER: 22
40
50
  ];
41
51
  /**
42
52
  * Matcher factory.
@@ -19,7 +19,12 @@ exports.matcherTypes = {
19
19
  CONTAINS_STRING: 14,
20
20
  IN_SPLIT_TREATMENT: 15,
21
21
  EQUAL_TO_BOOLEAN: 16,
22
- MATCHES_STRING: 17
22
+ MATCHES_STRING: 17,
23
+ EQUAL_TO_SEMVER: 18,
24
+ GREATER_THAN_OR_EQUAL_TO_SEMVER: 19,
25
+ LESS_THAN_OR_EQUAL_TO_SEMVER: 20,
26
+ BETWEEN_SEMVER: 21,
27
+ IN_LIST_SEMVER: 22,
23
28
  };
24
29
  exports.matcherDataTypes = {
25
30
  BOOLEAN: 'BOOLEAN',
@@ -0,0 +1,14 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.betweenSemverMatcherContext = void 0;
4
+ var Semver_1 = require("../../utils/Semver");
5
+ function betweenSemverMatcherContext(log, ruleAttr) {
6
+ var startSemver = new Semver_1.Semver(ruleAttr.start);
7
+ var endSemver = new Semver_1.Semver(ruleAttr.end);
8
+ return function betweenSemverMatcher(key) {
9
+ var runtimeSemver = new Semver_1.Semver(key);
10
+ var isBetween = startSemver.compare(runtimeSemver) <= 0 && endSemver.compare(runtimeSemver) >= 0;
11
+ return isBetween;
12
+ };
13
+ }
14
+ exports.betweenSemverMatcherContext = betweenSemverMatcherContext;
@@ -0,0 +1,12 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.equalToSemverMatcherContext = void 0;
4
+ var Semver_1 = require("../../utils/Semver");
5
+ function equalToSemverMatcherContext(log, ruleAttr) {
6
+ var ruleSemver = new Semver_1.Semver(ruleAttr);
7
+ return function equalToSemverMatcher(runtimeAttr) {
8
+ var isEqual = ruleSemver.version === new Semver_1.Semver(runtimeAttr).version;
9
+ return isEqual;
10
+ };
11
+ }
12
+ exports.equalToSemverMatcherContext = equalToSemverMatcherContext;
@@ -0,0 +1,12 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.greaterThanEqualToSemverMatcherContext = void 0;
4
+ var Semver_1 = require("../../utils/Semver");
5
+ function greaterThanEqualToSemverMatcherContext(log, ruleAttr) {
6
+ var ruleSemver = new Semver_1.Semver(ruleAttr);
7
+ return function greaterThanEqualToSemverMatcher(runtimeAttr) {
8
+ var isGreaterThanEqual = new Semver_1.Semver(runtimeAttr).compare(ruleSemver) >= 0;
9
+ return isGreaterThanEqual;
10
+ };
11
+ }
12
+ exports.greaterThanEqualToSemverMatcherContext = greaterThanEqualToSemverMatcherContext;
@@ -0,0 +1,14 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.inListSemverMatcherContext = void 0;
4
+ var sets_1 = require("../../utils/lang/sets");
5
+ var Semver_1 = require("../../utils/Semver");
6
+ function inListSemverMatcherContext(log, ruleAttr) {
7
+ var listOfSemvers = new sets_1._Set((0, sets_1.setToArray)(ruleAttr).map(function (version) { return new Semver_1.Semver(version).version; }));
8
+ return function inListSemverMatcher(runtimeAttr) {
9
+ var runtimeSemver = new Semver_1.Semver(runtimeAttr).version;
10
+ var isInList = listOfSemvers.has(runtimeSemver);
11
+ return isInList;
12
+ };
13
+ }
14
+ exports.inListSemverMatcherContext = inListSemverMatcherContext;
@@ -0,0 +1,12 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.lessThanEqualToSemverMatcherContext = void 0;
4
+ var Semver_1 = require("../../utils/Semver");
5
+ function lessThanEqualToSemverMatcherContext(log, ruleAttr) {
6
+ var ruleSemver = new Semver_1.Semver(ruleAttr);
7
+ return function lessThanEqualToSemverMatcher(runtimeAttr) {
8
+ var isLessThenEqual = new Semver_1.Semver(runtimeAttr).compare(ruleSemver) <= 0;
9
+ return isLessThenEqual;
10
+ };
11
+ }
12
+ exports.lessThanEqualToSemverMatcherContext = lessThanEqualToSemverMatcherContext;
@@ -13,37 +13,38 @@ var convertions_1 = require("../convertions");
13
13
  */
14
14
  function matchersTransform(matchers) {
15
15
  var parsedMatchers = matchers.map(function (matcher) {
16
- var matcherType = matcher.matcherType /* string */, negate = matcher.negate /* boolean */, keySelector = matcher.keySelector /* keySelectorObject */, segmentObject = matcher.userDefinedSegmentMatcherData /* segmentObject */, whitelistObject = matcher.whitelistMatcherData /* whiteListObject, provided by 'WHITELIST', set and string matchers */, unaryNumericObject = matcher.unaryNumericMatcherData /* unaryNumericObject */, betweenObject = matcher.betweenMatcherData /* betweenObject */, dependencyObject = matcher.dependencyMatcherData /* dependencyObject */, booleanMatcherData = matcher.booleanMatcherData, stringMatcherData = matcher.stringMatcherData;
16
+ var matcherType = matcher.matcherType, negate = matcher.negate, keySelector = matcher.keySelector, userDefinedSegmentMatcherData = matcher.userDefinedSegmentMatcherData, whitelistMatcherData = matcher.whitelistMatcherData, /* whitelistObject, provided by 'WHITELIST', 'IN_LIST_SEMVER', set and string matchers */ unaryNumericMatcherData = matcher.unaryNumericMatcherData, betweenMatcherData = matcher.betweenMatcherData, dependencyMatcherData = matcher.dependencyMatcherData, booleanMatcherData = matcher.booleanMatcherData, stringMatcherData = matcher.stringMatcherData, betweenStringMatcherData = matcher.betweenStringMatcherData;
17
17
  var attribute = keySelector && keySelector.attribute;
18
18
  var type = (0, matcherTypes_1.matcherTypesMapper)(matcherType);
19
19
  // As default input data type we use string (even for ALL_KEYS)
20
20
  var dataType = matcherTypes_1.matcherDataTypes.STRING;
21
21
  var value = undefined;
22
22
  if (type === matcherTypes_1.matcherTypes.IN_SEGMENT) {
23
- value = (0, segment_1.segmentTransform)(segmentObject);
23
+ value = (0, segment_1.segmentTransform)(userDefinedSegmentMatcherData);
24
24
  }
25
- else if (type === matcherTypes_1.matcherTypes.WHITELIST) {
26
- value = (0, whitelist_1.whitelistTransform)(whitelistObject);
25
+ else if (type === matcherTypes_1.matcherTypes.WHITELIST ||
26
+ type === matcherTypes_1.matcherTypes.IN_LIST_SEMVER) {
27
+ value = (0, whitelist_1.whitelistTransform)(whitelistMatcherData);
27
28
  }
28
29
  else if (type === matcherTypes_1.matcherTypes.EQUAL_TO) {
29
- value = (0, unaryNumeric_1.numericTransform)(unaryNumericObject);
30
+ value = (0, unaryNumeric_1.numericTransform)(unaryNumericMatcherData);
30
31
  dataType = matcherTypes_1.matcherDataTypes.NUMBER;
31
- if (unaryNumericObject.dataType === 'DATETIME') {
32
+ if (unaryNumericMatcherData.dataType === 'DATETIME') {
32
33
  value = (0, convertions_1.zeroSinceHH)(value);
33
34
  dataType = matcherTypes_1.matcherDataTypes.DATETIME;
34
35
  }
35
36
  }
36
37
  else if (type === matcherTypes_1.matcherTypes.GREATER_THAN_OR_EQUAL_TO ||
37
38
  type === matcherTypes_1.matcherTypes.LESS_THAN_OR_EQUAL_TO) {
38
- value = (0, unaryNumeric_1.numericTransform)(unaryNumericObject);
39
+ value = (0, unaryNumeric_1.numericTransform)(unaryNumericMatcherData);
39
40
  dataType = matcherTypes_1.matcherDataTypes.NUMBER;
40
- if (unaryNumericObject.dataType === 'DATETIME') {
41
+ if (unaryNumericMatcherData.dataType === 'DATETIME') {
41
42
  value = (0, convertions_1.zeroSinceSS)(value);
42
43
  dataType = matcherTypes_1.matcherDataTypes.DATETIME;
43
44
  }
44
45
  }
45
46
  else if (type === matcherTypes_1.matcherTypes.BETWEEN) {
46
- value = betweenObject;
47
+ value = betweenMatcherData;
47
48
  dataType = matcherTypes_1.matcherDataTypes.NUMBER;
48
49
  if (value.dataType === 'DATETIME') {
49
50
  value.start = (0, convertions_1.zeroSinceSS)(value.start);
@@ -51,27 +52,33 @@ function matchersTransform(matchers) {
51
52
  dataType = matcherTypes_1.matcherDataTypes.DATETIME;
52
53
  }
53
54
  }
55
+ else if (type === matcherTypes_1.matcherTypes.BETWEEN_SEMVER) {
56
+ value = betweenStringMatcherData;
57
+ }
54
58
  else if (type === matcherTypes_1.matcherTypes.EQUAL_TO_SET ||
55
59
  type === matcherTypes_1.matcherTypes.CONTAINS_ANY_OF_SET ||
56
60
  type === matcherTypes_1.matcherTypes.CONTAINS_ALL_OF_SET ||
57
61
  type === matcherTypes_1.matcherTypes.PART_OF_SET) {
58
- value = (0, set_1.setTransform)(whitelistObject);
62
+ value = (0, set_1.setTransform)(whitelistMatcherData);
59
63
  dataType = matcherTypes_1.matcherDataTypes.SET;
60
64
  }
61
65
  else if (type === matcherTypes_1.matcherTypes.STARTS_WITH ||
62
66
  type === matcherTypes_1.matcherTypes.ENDS_WITH ||
63
67
  type === matcherTypes_1.matcherTypes.CONTAINS_STRING) {
64
- value = (0, set_1.setTransform)(whitelistObject);
68
+ value = (0, set_1.setTransform)(whitelistMatcherData);
65
69
  }
66
70
  else if (type === matcherTypes_1.matcherTypes.IN_SPLIT_TREATMENT) {
67
- value = dependencyObject;
71
+ value = dependencyMatcherData;
68
72
  dataType = matcherTypes_1.matcherDataTypes.NOT_SPECIFIED;
69
73
  }
70
74
  else if (type === matcherTypes_1.matcherTypes.EQUAL_TO_BOOLEAN) {
71
75
  dataType = matcherTypes_1.matcherDataTypes.BOOLEAN;
72
76
  value = booleanMatcherData;
73
77
  }
74
- else if (type === matcherTypes_1.matcherTypes.MATCHES_STRING) {
78
+ else if (type === matcherTypes_1.matcherTypes.MATCHES_STRING ||
79
+ type === matcherTypes_1.matcherTypes.EQUAL_TO_SEMVER ||
80
+ type === matcherTypes_1.matcherTypes.GREATER_THAN_OR_EQUAL_TO_SEMVER ||
81
+ type === matcherTypes_1.matcherTypes.LESS_THAN_OR_EQUAL_TO_SEMVER) {
75
82
  value = stringMatcherData;
76
83
  }
77
84
  return {
@@ -32,16 +32,16 @@ function splitApiFactory(settings, platform, telemetryTracker) {
32
32
  return splitHttpClient(url).then(function () { return true; }).catch(function () { return false; });
33
33
  },
34
34
  fetchAuth: function (userMatchingKeys) {
35
- var url = urls.auth + "/v2/auth?s=" + constants_1.FLAGS_SPEC;
36
- if (userMatchingKeys) { // `userMatchingKeys` is undefined in server-side
35
+ var url = urls.auth + "/v2/auth";
36
+ if (userMatchingKeys) { // accounting the possibility that `userMatchingKeys` is undefined (server-side API)
37
37
  var queryParams = userMatchingKeys.map(userKeyToQueryParam).join('&');
38
- if (queryParams)
39
- url += '&' + queryParams;
38
+ if (queryParams) // accounting the possibility that `userKeys` and thus `queryParams` are empty
39
+ url += '?' + queryParams;
40
40
  }
41
41
  return splitHttpClient(url, undefined, telemetryTracker.trackHttp(constants_1.TOKEN));
42
42
  },
43
43
  fetchSplitChanges: function (since, noCache, till) {
44
- var url = urls.sdk + "/splitChanges?s=" + constants_1.FLAGS_SPEC + "&since=" + since + (till ? '&till=' + till : '') + (filterQueryString || '');
44
+ var url = urls.sdk + "/splitChanges?since=" + since + (till ? '&till=' + till : '') + (filterQueryString || '');
45
45
  return splitHttpClient(url, noCache ? noCacheHeaderOptions : undefined, telemetryTracker.trackHttp(constants_1.SPLITS))
46
46
  .catch(function (err) {
47
47
  if (err.statusCode === 414)
@@ -1,7 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.getStorageHash = exports.KeyBuilder = exports.validatePrefix = void 0;
4
- var constants_1 = require("../utils/constants");
5
4
  var lang_1 = require("../utils/lang");
6
5
  var murmur3_1 = require("../utils/murmur3/murmur3");
7
6
  var everythingAtTheEnd = /[^.]+$/;
@@ -67,10 +66,10 @@ var KeyBuilder = /** @class */ (function () {
67
66
  }());
68
67
  exports.KeyBuilder = KeyBuilder;
69
68
  /**
70
- * Generates a murmur32 hash based on the authorization key, the feature flags filter query, and version of SplitChanges API.
69
+ * Generates a murmur32 hash based on the authorization key and the feature flags filter query.
71
70
  * The hash is in hexadecimal format (8 characters max, 32 bits).
72
71
  */
73
72
  function getStorageHash(settings) {
74
- return (0, murmur3_1.hash)(settings.core.authorizationKey + "::" + settings.sync.__splitFiltersValidation.queryString + "::" + constants_1.FLAGS_SPEC).toString(16);
73
+ return (0, murmur3_1.hash)(settings.core.authorizationKey + "::" + settings.sync.__splitFiltersValidation.queryString).toString(16);
75
74
  }
76
75
  exports.getStorageHash = getStorageHash;
@@ -0,0 +1,100 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Semver = void 0;
4
+ var NUMERIC_IDENTIFIER_REGEX = /^[0-9]+$/;
5
+ var METADATA_DELIMITER = '+';
6
+ var PRERELEASE_DELIMITER = '-';
7
+ var VALUE_DELIMITER = '.';
8
+ /**
9
+ * Compares two strings. If both strings are numeric identifiers, they are compared numerically. Otherwise, they are compared lexicographically.
10
+ * This could be implemented using `a.localeCompare(b, undefined, { numeric: true })` but locale options are not broadly supported.
11
+ */
12
+ function compareStrings(a, b) {
13
+ if (NUMERIC_IDENTIFIER_REGEX.test(a) && NUMERIC_IDENTIFIER_REGEX.test(b)) {
14
+ var result = a.length - b.length;
15
+ if (result !== 0) {
16
+ return result;
17
+ }
18
+ }
19
+ return a < b ? -1 : a > b ? 1 : 0;
20
+ }
21
+ // Sanitizes a numeric identifier by removing leading zeros
22
+ function sanitizeNumericIdentifier(value) {
23
+ return value.replace(/^0+(?=\d)/, '');
24
+ }
25
+ function throwError(version) {
26
+ throw new Error('Unable to convert to Semver, incorrect format: ' + version);
27
+ }
28
+ var Semver = /** @class */ (function () {
29
+ function Semver(version) {
30
+ // Separate metadata if exists
31
+ var index = version.indexOf(METADATA_DELIMITER);
32
+ var _a = index === -1 ? [version] : [version.slice(0, index), version.slice(index + 1)], vWithoutMetadata = _a[0], metadata = _a[1];
33
+ if (metadata === '')
34
+ throwError(version);
35
+ // Set pre-release versions if exists
36
+ index = vWithoutMetadata.indexOf(PRERELEASE_DELIMITER);
37
+ if (index === -1) {
38
+ this._isStable = true;
39
+ this._preRelease = [];
40
+ }
41
+ else {
42
+ this._isStable = false;
43
+ this._preRelease = vWithoutMetadata.slice(index + 1).split(VALUE_DELIMITER).map(function (value) {
44
+ if (!value)
45
+ throwError(version);
46
+ return NUMERIC_IDENTIFIER_REGEX.test(value) ?
47
+ sanitizeNumericIdentifier(value) :
48
+ value;
49
+ });
50
+ vWithoutMetadata = vWithoutMetadata.slice(0, index);
51
+ }
52
+ // Set major, minor, and patch versions
53
+ var vParts = vWithoutMetadata.split(VALUE_DELIMITER).map(function (value) {
54
+ if (!value || !NUMERIC_IDENTIFIER_REGEX.test(value))
55
+ throwError(version);
56
+ return sanitizeNumericIdentifier(value);
57
+ });
58
+ if (vParts.length !== 3)
59
+ throwError(version);
60
+ this._major = vParts[0];
61
+ this._minor = vParts[1];
62
+ this._patch = vParts[2];
63
+ // Set version string
64
+ this.version = vParts.join(VALUE_DELIMITER);
65
+ if (this._preRelease.length)
66
+ this.version += PRERELEASE_DELIMITER + this._preRelease.join(VALUE_DELIMITER);
67
+ if (metadata)
68
+ this.version += METADATA_DELIMITER + metadata;
69
+ }
70
+ /**
71
+ * Precedence comparision between 2 Semver objects.
72
+ *
73
+ * @return `0` if `this === toCompare`, `-1` if `this < toCompare`, and `1` if `this > toCompare`
74
+ */
75
+ Semver.prototype.compare = function (toCompare) {
76
+ if (this.version === toCompare.version)
77
+ return 0;
78
+ var result = compareStrings(this._major, toCompare._major);
79
+ if (result !== 0)
80
+ return result;
81
+ result = compareStrings(this._minor, toCompare._minor);
82
+ if (result !== 0)
83
+ return result;
84
+ result = compareStrings(this._patch, toCompare._patch);
85
+ if (result !== 0)
86
+ return result;
87
+ if (!this._isStable && toCompare._isStable)
88
+ return -1;
89
+ if (this._isStable && !toCompare._isStable)
90
+ return 1;
91
+ for (var i = 0, length_1 = Math.min(this._preRelease.length, toCompare._preRelease.length); i < length_1; i++) {
92
+ var result_1 = compareStrings(this._preRelease[i], toCompare._preRelease[i]);
93
+ if (result_1 !== 0)
94
+ return result_1;
95
+ }
96
+ return this._preRelease.length - toCompare._preRelease.length;
97
+ };
98
+ return Semver;
99
+ }());
100
+ exports.Semver = Semver;
@@ -1,7 +1,7 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.MY_SEGMENT = exports.SEGMENT = exports.TOKEN = exports.TELEMETRY = exports.EVENTS = exports.IMPRESSIONS_COUNT = exports.IMPRESSIONS = exports.SPLITS = exports.NONE_ENUM = exports.DEBUG_ENUM = exports.OPTIMIZED_ENUM = exports.CONSUMER_PARTIAL_ENUM = exports.CONSUMER_ENUM = exports.STANDALONE_ENUM = exports.DEDUPED = exports.DROPPED = exports.QUEUED = exports.NAMES_FN_LABEL = exports.SPLITS_FN_LABEL = exports.SPLIT_FN_LABEL = exports.TRACK_FN_LABEL = exports.GET_TREATMENTS_WITH_CONFIG_BY_FLAG_SETS = exports.GET_TREATMENTS_WITH_CONFIG_BY_FLAG_SET = exports.GET_TREATMENTS_BY_FLAG_SETS = exports.GET_TREATMENTS_BY_FLAG_SET = exports.GET_TREATMENTS_WITH_CONFIG = exports.GET_TREATMENT_WITH_CONFIG = exports.GET_TREATMENTS = exports.GET_TREATMENT = exports.CONSENT_UNKNOWN = exports.CONSENT_DECLINED = exports.CONSENT_GRANTED = exports.STORAGE_PLUGGABLE = exports.STORAGE_REDIS = exports.STORAGE_LOCALSTORAGE = exports.STORAGE_MEMORY = exports.CONSUMER_PARTIAL_MODE = exports.CONSUMER_MODE = exports.PRODUCER_MODE = exports.STANDALONE_MODE = exports.LOCALHOST_MODE = exports.NONE = exports.OPTIMIZED = exports.DEBUG = exports.SPLIT_EVENT = exports.SPLIT_IMPRESSION = exports.NA = exports.UNKNOWN = exports.CONTROL_WITH_CONFIG = exports.CONTROL = void 0;
4
- exports.FLAGS_SPEC = exports.PAUSED = exports.ENABLED = exports.DISABLED = exports.NON_REQUESTED = exports.REQUESTED = exports.POLLING = exports.STREAMING = exports.AUTH_REJECTION = exports.SYNC_MODE_UPDATE = exports.ABLY_ERROR = exports.TOKEN_REFRESH = exports.SSE_CONNECTION_ERROR = exports.STREAMING_STATUS = exports.OCCUPANCY_SEC = exports.OCCUPANCY_PRI = exports.CONNECTION_ESTABLISHED = exports.TRACK = exports.TREATMENTS_WITH_CONFIG_BY_FLAGSETS = exports.TREATMENTS_WITH_CONFIG_BY_FLAGSET = exports.TREATMENTS_BY_FLAGSETS = exports.TREATMENTS_BY_FLAGSET = exports.TREATMENTS_WITH_CONFIG = exports.TREATMENT_WITH_CONFIG = exports.TREATMENTS = exports.TREATMENT = void 0;
4
+ exports.PAUSED = exports.ENABLED = exports.DISABLED = exports.NON_REQUESTED = exports.REQUESTED = exports.POLLING = exports.STREAMING = exports.AUTH_REJECTION = exports.SYNC_MODE_UPDATE = exports.ABLY_ERROR = exports.TOKEN_REFRESH = exports.SSE_CONNECTION_ERROR = exports.STREAMING_STATUS = exports.OCCUPANCY_SEC = exports.OCCUPANCY_PRI = exports.CONNECTION_ESTABLISHED = exports.TRACK = exports.TREATMENTS_WITH_CONFIG_BY_FLAGSETS = exports.TREATMENTS_WITH_CONFIG_BY_FLAGSET = exports.TREATMENTS_BY_FLAGSETS = exports.TREATMENTS_BY_FLAGSET = exports.TREATMENTS_WITH_CONFIG = exports.TREATMENT_WITH_CONFIG = exports.TREATMENTS = exports.TREATMENT = void 0;
5
5
  // Special treatments
6
6
  exports.CONTROL = 'control';
7
7
  exports.CONTROL_WITH_CONFIG = {
@@ -90,4 +90,3 @@ exports.NON_REQUESTED = 1;
90
90
  exports.DISABLED = 0;
91
91
  exports.ENABLED = 1;
92
92
  exports.PAUSED = 2;
93
- exports.FLAGS_SPEC = '1.1';
@@ -4,7 +4,7 @@ exports.validateLogger = void 0;
4
4
  var logger_1 = require("../../../logger");
5
5
  var commons_1 = require("./commons");
6
6
  function isLogger(log) {
7
- return log && typeof log.debug === 'function' && typeof log.info === 'function' && typeof log.warn === 'function' && typeof log.error === 'function' && typeof log.setLogLevel === 'function';
7
+ return log !== null && typeof log === 'object' && typeof log.debug === 'function' && typeof log.info === 'function' && typeof log.warn === 'function' && typeof log.error === 'function' && typeof log.setLogLevel === 'function';
8
8
  }
9
9
  // By default it starts disabled.
10
10
  var initialLogLevel = logger_1.LogLevels.NONE;
@@ -1,8 +1,8 @@
1
1
  import { ENGINE_MATCHER_GREATER } from '../../logger/constants';
2
2
  export function greaterThanEqualMatcherContext(log, ruleAttr) {
3
3
  return function greaterThanEqualMatcher(runtimeAttr) {
4
- var isGreaterEqualThan = runtimeAttr >= ruleAttr;
5
- log.debug(ENGINE_MATCHER_GREATER, [runtimeAttr, ruleAttr, isGreaterEqualThan]);
6
- return isGreaterEqualThan;
4
+ var isGreaterThanEqual = runtimeAttr >= ruleAttr;
5
+ log.debug(ENGINE_MATCHER_GREATER, [runtimeAttr, ruleAttr, isGreaterThanEqual]);
6
+ return isGreaterThanEqual;
7
7
  };
8
8
  }
@@ -15,6 +15,11 @@ import { containsStringMatcherContext } from './cont_str';
15
15
  import { dependencyMatcherContext } from './dependency';
16
16
  import { booleanMatcherContext } from './boolean';
17
17
  import { stringMatcherContext } from './string';
18
+ import { equalToSemverMatcherContext } from './semver_eq';
19
+ import { greaterThanEqualToSemverMatcherContext } from './semver_gte';
20
+ import { lessThanEqualToSemverMatcherContext } from './semver_lte';
21
+ import { betweenSemverMatcherContext } from './semver_between';
22
+ import { inListSemverMatcherContext } from './semver_inlist';
18
23
  var matchers = [
19
24
  undefined,
20
25
  allMatcherContext,
@@ -33,7 +38,12 @@ var matchers = [
33
38
  containsStringMatcherContext,
34
39
  dependencyMatcherContext,
35
40
  booleanMatcherContext,
36
- stringMatcherContext // MATCHES_STRING: 17
41
+ stringMatcherContext,
42
+ equalToSemverMatcherContext,
43
+ greaterThanEqualToSemverMatcherContext,
44
+ lessThanEqualToSemverMatcherContext,
45
+ betweenSemverMatcherContext,
46
+ inListSemverMatcherContext, // IN_LIST_SEMVER: 22
37
47
  ];
38
48
  /**
39
49
  * Matcher factory.
@@ -16,7 +16,12 @@ export var matcherTypes = {
16
16
  CONTAINS_STRING: 14,
17
17
  IN_SPLIT_TREATMENT: 15,
18
18
  EQUAL_TO_BOOLEAN: 16,
19
- MATCHES_STRING: 17
19
+ MATCHES_STRING: 17,
20
+ EQUAL_TO_SEMVER: 18,
21
+ GREATER_THAN_OR_EQUAL_TO_SEMVER: 19,
22
+ LESS_THAN_OR_EQUAL_TO_SEMVER: 20,
23
+ BETWEEN_SEMVER: 21,
24
+ IN_LIST_SEMVER: 22,
20
25
  };
21
26
  export var matcherDataTypes = {
22
27
  BOOLEAN: 'BOOLEAN',
@@ -0,0 +1,10 @@
1
+ import { Semver } from '../../utils/Semver';
2
+ export function betweenSemverMatcherContext(log, ruleAttr) {
3
+ var startSemver = new Semver(ruleAttr.start);
4
+ var endSemver = new Semver(ruleAttr.end);
5
+ return function betweenSemverMatcher(key) {
6
+ var runtimeSemver = new Semver(key);
7
+ var isBetween = startSemver.compare(runtimeSemver) <= 0 && endSemver.compare(runtimeSemver) >= 0;
8
+ return isBetween;
9
+ };
10
+ }
@@ -0,0 +1,8 @@
1
+ import { Semver } from '../../utils/Semver';
2
+ export function equalToSemverMatcherContext(log, ruleAttr) {
3
+ var ruleSemver = new Semver(ruleAttr);
4
+ return function equalToSemverMatcher(runtimeAttr) {
5
+ var isEqual = ruleSemver.version === new Semver(runtimeAttr).version;
6
+ return isEqual;
7
+ };
8
+ }
@@ -0,0 +1,8 @@
1
+ import { Semver } from '../../utils/Semver';
2
+ export function greaterThanEqualToSemverMatcherContext(log, ruleAttr) {
3
+ var ruleSemver = new Semver(ruleAttr);
4
+ return function greaterThanEqualToSemverMatcher(runtimeAttr) {
5
+ var isGreaterThanEqual = new Semver(runtimeAttr).compare(ruleSemver) >= 0;
6
+ return isGreaterThanEqual;
7
+ };
8
+ }
@@ -0,0 +1,10 @@
1
+ import { setToArray, _Set } from '../../utils/lang/sets';
2
+ import { Semver } from '../../utils/Semver';
3
+ export function inListSemverMatcherContext(log, ruleAttr) {
4
+ var listOfSemvers = new _Set(setToArray(ruleAttr).map(function (version) { return new Semver(version).version; }));
5
+ return function inListSemverMatcher(runtimeAttr) {
6
+ var runtimeSemver = new Semver(runtimeAttr).version;
7
+ var isInList = listOfSemvers.has(runtimeSemver);
8
+ return isInList;
9
+ };
10
+ }
@@ -0,0 +1,8 @@
1
+ import { Semver } from '../../utils/Semver';
2
+ export function lessThanEqualToSemverMatcherContext(log, ruleAttr) {
3
+ var ruleSemver = new Semver(ruleAttr);
4
+ return function lessThanEqualToSemverMatcher(runtimeAttr) {
5
+ var isLessThenEqual = new Semver(runtimeAttr).compare(ruleSemver) <= 0;
6
+ return isLessThenEqual;
7
+ };
8
+ }
@@ -10,37 +10,38 @@ import { zeroSinceHH, zeroSinceSS } from '../convertions';
10
10
  */
11
11
  export function matchersTransform(matchers) {
12
12
  var parsedMatchers = matchers.map(function (matcher) {
13
- var matcherType = matcher.matcherType /* string */, negate = matcher.negate /* boolean */, keySelector = matcher.keySelector /* keySelectorObject */, segmentObject = matcher.userDefinedSegmentMatcherData /* segmentObject */, whitelistObject = matcher.whitelistMatcherData /* whiteListObject, provided by 'WHITELIST', set and string matchers */, unaryNumericObject = matcher.unaryNumericMatcherData /* unaryNumericObject */, betweenObject = matcher.betweenMatcherData /* betweenObject */, dependencyObject = matcher.dependencyMatcherData /* dependencyObject */, booleanMatcherData = matcher.booleanMatcherData, stringMatcherData = matcher.stringMatcherData;
13
+ var matcherType = matcher.matcherType, negate = matcher.negate, keySelector = matcher.keySelector, userDefinedSegmentMatcherData = matcher.userDefinedSegmentMatcherData, whitelistMatcherData = matcher.whitelistMatcherData, /* whitelistObject, provided by 'WHITELIST', 'IN_LIST_SEMVER', set and string matchers */ unaryNumericMatcherData = matcher.unaryNumericMatcherData, betweenMatcherData = matcher.betweenMatcherData, dependencyMatcherData = matcher.dependencyMatcherData, booleanMatcherData = matcher.booleanMatcherData, stringMatcherData = matcher.stringMatcherData, betweenStringMatcherData = matcher.betweenStringMatcherData;
14
14
  var attribute = keySelector && keySelector.attribute;
15
15
  var type = matcherTypesMapper(matcherType);
16
16
  // As default input data type we use string (even for ALL_KEYS)
17
17
  var dataType = matcherDataTypes.STRING;
18
18
  var value = undefined;
19
19
  if (type === matcherTypes.IN_SEGMENT) {
20
- value = segmentTransform(segmentObject);
20
+ value = segmentTransform(userDefinedSegmentMatcherData);
21
21
  }
22
- else if (type === matcherTypes.WHITELIST) {
23
- value = whitelistTransform(whitelistObject);
22
+ else if (type === matcherTypes.WHITELIST ||
23
+ type === matcherTypes.IN_LIST_SEMVER) {
24
+ value = whitelistTransform(whitelistMatcherData);
24
25
  }
25
26
  else if (type === matcherTypes.EQUAL_TO) {
26
- value = numericTransform(unaryNumericObject);
27
+ value = numericTransform(unaryNumericMatcherData);
27
28
  dataType = matcherDataTypes.NUMBER;
28
- if (unaryNumericObject.dataType === 'DATETIME') {
29
+ if (unaryNumericMatcherData.dataType === 'DATETIME') {
29
30
  value = zeroSinceHH(value);
30
31
  dataType = matcherDataTypes.DATETIME;
31
32
  }
32
33
  }
33
34
  else if (type === matcherTypes.GREATER_THAN_OR_EQUAL_TO ||
34
35
  type === matcherTypes.LESS_THAN_OR_EQUAL_TO) {
35
- value = numericTransform(unaryNumericObject);
36
+ value = numericTransform(unaryNumericMatcherData);
36
37
  dataType = matcherDataTypes.NUMBER;
37
- if (unaryNumericObject.dataType === 'DATETIME') {
38
+ if (unaryNumericMatcherData.dataType === 'DATETIME') {
38
39
  value = zeroSinceSS(value);
39
40
  dataType = matcherDataTypes.DATETIME;
40
41
  }
41
42
  }
42
43
  else if (type === matcherTypes.BETWEEN) {
43
- value = betweenObject;
44
+ value = betweenMatcherData;
44
45
  dataType = matcherDataTypes.NUMBER;
45
46
  if (value.dataType === 'DATETIME') {
46
47
  value.start = zeroSinceSS(value.start);
@@ -48,27 +49,33 @@ export function matchersTransform(matchers) {
48
49
  dataType = matcherDataTypes.DATETIME;
49
50
  }
50
51
  }
52
+ else if (type === matcherTypes.BETWEEN_SEMVER) {
53
+ value = betweenStringMatcherData;
54
+ }
51
55
  else if (type === matcherTypes.EQUAL_TO_SET ||
52
56
  type === matcherTypes.CONTAINS_ANY_OF_SET ||
53
57
  type === matcherTypes.CONTAINS_ALL_OF_SET ||
54
58
  type === matcherTypes.PART_OF_SET) {
55
- value = setTransform(whitelistObject);
59
+ value = setTransform(whitelistMatcherData);
56
60
  dataType = matcherDataTypes.SET;
57
61
  }
58
62
  else if (type === matcherTypes.STARTS_WITH ||
59
63
  type === matcherTypes.ENDS_WITH ||
60
64
  type === matcherTypes.CONTAINS_STRING) {
61
- value = setTransform(whitelistObject);
65
+ value = setTransform(whitelistMatcherData);
62
66
  }
63
67
  else if (type === matcherTypes.IN_SPLIT_TREATMENT) {
64
- value = dependencyObject;
68
+ value = dependencyMatcherData;
65
69
  dataType = matcherDataTypes.NOT_SPECIFIED;
66
70
  }
67
71
  else if (type === matcherTypes.EQUAL_TO_BOOLEAN) {
68
72
  dataType = matcherDataTypes.BOOLEAN;
69
73
  value = booleanMatcherData;
70
74
  }
71
- else if (type === matcherTypes.MATCHES_STRING) {
75
+ else if (type === matcherTypes.MATCHES_STRING ||
76
+ type === matcherTypes.EQUAL_TO_SEMVER ||
77
+ type === matcherTypes.GREATER_THAN_OR_EQUAL_TO_SEMVER ||
78
+ type === matcherTypes.LESS_THAN_OR_EQUAL_TO_SEMVER) {
72
79
  value = stringMatcherData;
73
80
  }
74
81
  return {
@@ -1,6 +1,6 @@
1
1
  import { splitHttpClientFactory } from './splitHttpClient';
2
2
  import { objectAssign } from '../utils/lang/objectAssign';
3
- import { SPLITS, IMPRESSIONS, IMPRESSIONS_COUNT, EVENTS, TELEMETRY, TOKEN, SEGMENT, MY_SEGMENT, FLAGS_SPEC } from '../utils/constants';
3
+ import { SPLITS, IMPRESSIONS, IMPRESSIONS_COUNT, EVENTS, TELEMETRY, TOKEN, SEGMENT, MY_SEGMENT } from '../utils/constants';
4
4
  import { ERROR_TOO_MANY_SETS } from '../logger/constants';
5
5
  var noCacheHeaderOptions = { headers: { 'Cache-Control': 'no-cache' } };
6
6
  function userKeyToQueryParam(userKey) {
@@ -29,16 +29,16 @@ export function splitApiFactory(settings, platform, telemetryTracker) {
29
29
  return splitHttpClient(url).then(function () { return true; }).catch(function () { return false; });
30
30
  },
31
31
  fetchAuth: function (userMatchingKeys) {
32
- var url = urls.auth + "/v2/auth?s=" + FLAGS_SPEC;
33
- if (userMatchingKeys) { // `userMatchingKeys` is undefined in server-side
32
+ var url = urls.auth + "/v2/auth";
33
+ if (userMatchingKeys) { // accounting the possibility that `userMatchingKeys` is undefined (server-side API)
34
34
  var queryParams = userMatchingKeys.map(userKeyToQueryParam).join('&');
35
- if (queryParams)
36
- url += '&' + queryParams;
35
+ if (queryParams) // accounting the possibility that `userKeys` and thus `queryParams` are empty
36
+ url += '?' + queryParams;
37
37
  }
38
38
  return splitHttpClient(url, undefined, telemetryTracker.trackHttp(TOKEN));
39
39
  },
40
40
  fetchSplitChanges: function (since, noCache, till) {
41
- var url = urls.sdk + "/splitChanges?s=" + FLAGS_SPEC + "&since=" + since + (till ? '&till=' + till : '') + (filterQueryString || '');
41
+ var url = urls.sdk + "/splitChanges?since=" + since + (till ? '&till=' + till : '') + (filterQueryString || '');
42
42
  return splitHttpClient(url, noCache ? noCacheHeaderOptions : undefined, telemetryTracker.trackHttp(SPLITS))
43
43
  .catch(function (err) {
44
44
  if (err.statusCode === 414)