@splitsoftware/splitio-commons 1.13.2-rc.1 → 1.13.2-rc.10

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 (164) hide show
  1. package/CHANGES.txt +4 -3
  2. package/cjs/evaluator/Engine.js +5 -6
  3. package/cjs/evaluator/combiners/ifelseif.js +2 -3
  4. package/cjs/evaluator/condition/index.js +2 -3
  5. package/cjs/evaluator/index.js +4 -5
  6. package/cjs/evaluator/matchers/all.js +1 -3
  7. package/cjs/evaluator/matchers/between.js +1 -3
  8. package/cjs/evaluator/matchers/boolean.js +1 -3
  9. package/cjs/evaluator/matchers/cont_all.js +1 -3
  10. package/cjs/evaluator/matchers/cont_any.js +1 -3
  11. package/cjs/evaluator/matchers/cont_str.js +1 -3
  12. package/cjs/evaluator/matchers/dependency.js +1 -1
  13. package/cjs/evaluator/matchers/eq.js +1 -3
  14. package/cjs/evaluator/matchers/eq_set.js +1 -3
  15. package/cjs/evaluator/matchers/ew.js +1 -3
  16. package/cjs/evaluator/matchers/gte.js +3 -5
  17. package/cjs/evaluator/matchers/index.js +12 -2
  18. package/cjs/evaluator/matchers/lte.js +3 -5
  19. package/cjs/evaluator/matchers/matcherTypes.js +6 -1
  20. package/cjs/evaluator/matchers/part_of.js +1 -3
  21. package/cjs/evaluator/matchers/segment.js +1 -6
  22. package/cjs/evaluator/matchers/semver_between.js +14 -0
  23. package/cjs/evaluator/matchers/semver_eq.js +13 -0
  24. package/cjs/evaluator/matchers/semver_gte.js +13 -0
  25. package/cjs/evaluator/matchers/semver_inlist.js +17 -0
  26. package/cjs/evaluator/matchers/semver_lte.js +13 -0
  27. package/cjs/evaluator/matchers/string.js +3 -12
  28. package/cjs/evaluator/matchers/sw.js +1 -3
  29. package/cjs/evaluator/matchers/whitelist.js +3 -4
  30. package/cjs/evaluator/matchersTransform/index.js +21 -16
  31. package/cjs/evaluator/matchersTransform/whitelist.js +2 -3
  32. package/cjs/evaluator/parser/index.js +23 -7
  33. package/cjs/logger/constants.js +5 -20
  34. package/cjs/logger/index.js +8 -1
  35. package/cjs/logger/messages/debug.js +5 -21
  36. package/cjs/logger/messages/error.js +2 -1
  37. package/cjs/logger/messages/warn.js +1 -1
  38. package/cjs/services/splitApi.js +5 -5
  39. package/cjs/storages/inLocalStorage/SplitsCacheInLocal.js +1 -1
  40. package/cjs/storages/pluggable/index.js +1 -1
  41. package/cjs/utils/Semver.js +103 -0
  42. package/cjs/utils/labels/index.js +1 -1
  43. package/cjs/utils/settingsValidation/logger/pluggableLogger.js +1 -1
  44. package/esm/evaluator/Engine.js +5 -5
  45. package/esm/evaluator/combiners/ifelseif.js +2 -2
  46. package/esm/evaluator/condition/index.js +2 -2
  47. package/esm/evaluator/index.js +4 -4
  48. package/esm/evaluator/matchers/all.js +1 -3
  49. package/esm/evaluator/matchers/between.js +1 -3
  50. package/esm/evaluator/matchers/boolean.js +1 -3
  51. package/esm/evaluator/matchers/cont_all.js +1 -3
  52. package/esm/evaluator/matchers/cont_any.js +1 -3
  53. package/esm/evaluator/matchers/cont_str.js +1 -3
  54. package/esm/evaluator/matchers/dependency.js +1 -1
  55. package/esm/evaluator/matchers/eq.js +1 -3
  56. package/esm/evaluator/matchers/eq_set.js +1 -3
  57. package/esm/evaluator/matchers/ew.js +1 -3
  58. package/esm/evaluator/matchers/gte.js +3 -5
  59. package/esm/evaluator/matchers/index.js +12 -2
  60. package/esm/evaluator/matchers/lte.js +3 -5
  61. package/esm/evaluator/matchers/matcherTypes.js +6 -1
  62. package/esm/evaluator/matchers/part_of.js +1 -3
  63. package/esm/evaluator/matchers/segment.js +1 -6
  64. package/esm/evaluator/matchers/semver_between.js +10 -0
  65. package/esm/evaluator/matchers/semver_eq.js +9 -0
  66. package/esm/evaluator/matchers/semver_gte.js +9 -0
  67. package/esm/evaluator/matchers/semver_inlist.js +13 -0
  68. package/esm/evaluator/matchers/semver_lte.js +9 -0
  69. package/esm/evaluator/matchers/string.js +3 -12
  70. package/esm/evaluator/matchers/sw.js +1 -3
  71. package/esm/evaluator/matchers/whitelist.js +4 -5
  72. package/esm/evaluator/matchersTransform/index.js +21 -16
  73. package/esm/evaluator/matchersTransform/whitelist.js +2 -3
  74. package/esm/evaluator/parser/index.js +23 -7
  75. package/esm/logger/constants.js +2 -17
  76. package/esm/logger/index.js +9 -2
  77. package/esm/logger/messages/debug.js +5 -21
  78. package/esm/logger/messages/error.js +2 -1
  79. package/esm/logger/messages/warn.js +1 -1
  80. package/esm/services/splitApi.js +5 -5
  81. package/esm/storages/inLocalStorage/SplitsCacheInLocal.js +1 -1
  82. package/esm/storages/pluggable/index.js +1 -1
  83. package/esm/utils/Semver.js +100 -0
  84. package/esm/utils/labels/index.js +1 -1
  85. package/esm/utils/settingsValidation/logger/pluggableLogger.js +1 -1
  86. package/package.json +1 -2
  87. package/src/dtos/types.ts +34 -1
  88. package/src/evaluator/Engine.ts +5 -6
  89. package/src/evaluator/combiners/ifelseif.ts +2 -2
  90. package/src/evaluator/condition/index.ts +2 -2
  91. package/src/evaluator/index.ts +4 -4
  92. package/src/evaluator/matchers/all.ts +1 -5
  93. package/src/evaluator/matchers/between.ts +3 -7
  94. package/src/evaluator/matchers/boolean.ts +2 -6
  95. package/src/evaluator/matchers/cont_all.ts +1 -5
  96. package/src/evaluator/matchers/cont_any.ts +1 -5
  97. package/src/evaluator/matchers/cont_str.ts +2 -6
  98. package/src/evaluator/matchers/dependency.ts +1 -1
  99. package/src/evaluator/matchers/eq.ts +2 -6
  100. package/src/evaluator/matchers/eq_set.ts +1 -5
  101. package/src/evaluator/matchers/ew.ts +2 -6
  102. package/src/evaluator/matchers/gte.ts +3 -7
  103. package/src/evaluator/matchers/index.ts +29 -19
  104. package/src/evaluator/matchers/lte.ts +3 -7
  105. package/src/evaluator/matchers/matcherTypes.ts +6 -1
  106. package/src/evaluator/matchers/part_of.ts +1 -5
  107. package/src/evaluator/matchers/segment.ts +1 -8
  108. package/src/evaluator/matchers/semver_between.ts +15 -0
  109. package/src/evaluator/matchers/semver_eq.ts +13 -0
  110. package/src/evaluator/matchers/semver_gte.ts +13 -0
  111. package/src/evaluator/matchers/semver_inlist.ts +17 -0
  112. package/src/evaluator/matchers/semver_lte.ts +13 -0
  113. package/src/evaluator/matchers/string.ts +3 -16
  114. package/src/evaluator/matchers/sw.ts +2 -6
  115. package/src/evaluator/matchers/whitelist.ts +5 -7
  116. package/src/evaluator/matchersTransform/index.ts +31 -23
  117. package/src/evaluator/matchersTransform/whitelist.ts +4 -5
  118. package/src/evaluator/parser/index.ts +24 -8
  119. package/src/evaluator/types.ts +3 -3
  120. package/src/logger/constants.ts +2 -17
  121. package/src/logger/index.ts +8 -2
  122. package/src/logger/messages/debug.ts +5 -21
  123. package/src/logger/messages/error.ts +2 -1
  124. package/src/logger/messages/warn.ts +1 -1
  125. package/src/services/splitApi.ts +4 -5
  126. package/src/storages/inLocalStorage/SplitsCacheInLocal.ts +1 -1
  127. package/src/storages/pluggable/index.ts +1 -1
  128. package/src/utils/Semver.ts +111 -0
  129. package/src/utils/labels/index.ts +1 -1
  130. package/src/utils/settingsValidation/logger/pluggableLogger.ts +1 -1
  131. package/types/dtos/types.d.ts +26 -1
  132. package/types/evaluator/matchers/all.d.ts +1 -2
  133. package/types/evaluator/matchers/between.d.ts +1 -2
  134. package/types/evaluator/matchers/boolean.d.ts +1 -2
  135. package/types/evaluator/matchers/cont_all.d.ts +1 -2
  136. package/types/evaluator/matchers/cont_any.d.ts +1 -2
  137. package/types/evaluator/matchers/cont_str.d.ts +1 -2
  138. package/types/evaluator/matchers/dependency.d.ts +1 -1
  139. package/types/evaluator/matchers/eq.d.ts +1 -2
  140. package/types/evaluator/matchers/eq_set.d.ts +1 -2
  141. package/types/evaluator/matchers/ew.d.ts +1 -2
  142. package/types/evaluator/matchers/gte.d.ts +1 -2
  143. package/types/evaluator/matchers/lte.d.ts +1 -2
  144. package/types/evaluator/matchers/part_of.d.ts +1 -2
  145. package/types/evaluator/matchers/segment.d.ts +1 -2
  146. package/types/evaluator/matchers/sember_inlist.d.ts +3 -0
  147. package/types/evaluator/matchers/semver_between.d.ts +2 -0
  148. package/types/evaluator/matchers/semver_eq.d.ts +1 -0
  149. package/types/evaluator/matchers/semver_gte.d.ts +1 -0
  150. package/types/evaluator/matchers/semver_inlist.d.ts +1 -0
  151. package/types/evaluator/matchers/semver_lte.d.ts +1 -0
  152. package/types/evaluator/matchers/string.d.ts +1 -2
  153. package/types/evaluator/matchers/sw.d.ts +1 -2
  154. package/types/evaluator/matchers/whitelist.d.ts +1 -3
  155. package/types/evaluator/matchersTransform/set.d.ts +2 -2
  156. package/types/evaluator/matchersTransform/string.d.ts +7 -0
  157. package/types/evaluator/matchersTransform/whitelist.d.ts +3 -3
  158. package/types/evaluator/types.d.ts +3 -3
  159. package/types/logger/constants.d.ts +2 -17
  160. package/types/utils/labels/index.d.ts +1 -1
  161. package/types/utils/semVer.d.ts +15 -0
  162. package/cjs/evaluator/matchersTransform/set.js +0 -10
  163. package/esm/evaluator/matchersTransform/set.js +0 -6
  164. package/src/evaluator/matchersTransform/set.ts +0 -8
@@ -0,0 +1,13 @@
1
+ import { Semver } from '../../utils/Semver';
2
+
3
+ export function equalToSemverMatcherContext(ruleAttr: string) {
4
+ const ruleSemver = new Semver(ruleAttr);
5
+
6
+ return function equalToSemverMatcher(runtimeAttr: string): boolean {
7
+ const runtimeSemver = new Semver(runtimeAttr);
8
+
9
+ const isEqual = ruleSemver.version === runtimeSemver.version;
10
+
11
+ return isEqual;
12
+ };
13
+ }
@@ -0,0 +1,13 @@
1
+ import { Semver } from '../../utils/Semver';
2
+
3
+ export function greaterThanEqualToSemverMatcherContext(ruleAttr: string) {
4
+ const ruleSemver = new Semver(ruleAttr);
5
+
6
+ return function greaterThanEqualToSemverMatcher(runtimeAttr: string): boolean {
7
+ const runtimeSemver = new Semver(runtimeAttr);
8
+
9
+ const isGreaterThanEqual = runtimeSemver.compare(ruleSemver) >= 0;
10
+
11
+ return isGreaterThanEqual;
12
+ };
13
+ }
@@ -0,0 +1,17 @@
1
+ import { _Set } from '../../utils/lang/sets';
2
+ import { Semver } from '../../utils/Semver';
3
+
4
+ export function inListSemverMatcherContext(ruleAttr: string[]) {
5
+ // @TODO ruleAttr validation should be done at the `parser` or `matchersTransform` level to reuse for all matchers
6
+ if (!ruleAttr || ruleAttr.length === 0) throw new Error('whitelistMatcherData is required for IN_LIST_SEMVER matcher type');
7
+
8
+ const listOfSemvers = new _Set(ruleAttr.map((version) => new Semver(version).version));
9
+
10
+ return function inListSemverMatcher(runtimeAttr: string): boolean {
11
+ const runtimeSemver = new Semver(runtimeAttr).version;
12
+
13
+ const isInList = listOfSemvers.has(runtimeSemver);
14
+
15
+ return isInList;
16
+ };
17
+ }
@@ -0,0 +1,13 @@
1
+ import { Semver } from '../../utils/Semver';
2
+
3
+ export function lessThanEqualToSemverMatcherContext(ruleAttr: string) {
4
+ const ruleSemver = new Semver(ruleAttr);
5
+
6
+ return function lessThanEqualToSemverMatcher(runtimeAttr: string): boolean {
7
+ const runtimeSemver = new Semver(runtimeAttr);
8
+
9
+ const isLessThanEqual = runtimeSemver.compare(ruleSemver) <= 0;
10
+
11
+ return isLessThanEqual;
12
+ };
13
+ }
@@ -1,21 +1,8 @@
1
- import { ENGINE_MATCHER_STRING_INVALID, ENGINE_MATCHER_STRING } from '../../logger/constants';
2
- import { ILogger } from '../../logger/types';
1
+ export function stringMatcherContext(ruleAttr: string) {
2
+ const regex = new RegExp(ruleAttr);
3
3
 
4
- export function stringMatcherContext(log: ILogger, ruleAttr: string) /*: Function */ {
5
4
  return function stringMatcher(runtimeAttr: string): boolean {
6
- let re;
7
-
8
- try {
9
- re = new RegExp(ruleAttr);
10
- } catch (e) {
11
- log.debug(ENGINE_MATCHER_STRING_INVALID, [ruleAttr]);
12
-
13
- return false;
14
- }
15
-
16
- let regexMatches = re.test(runtimeAttr);
17
-
18
- log.debug(ENGINE_MATCHER_STRING, [runtimeAttr, ruleAttr, regexMatches ? 'yes' : 'no']);
5
+ const regexMatches = regex.test(runtimeAttr);
19
6
 
20
7
  return regexMatches;
21
8
  };
@@ -1,12 +1,8 @@
1
- import { ENGINE_MATCHER_STARTS_WITH } from '../../logger/constants';
2
- import { ILogger } from '../../logger/types';
3
1
  import { startsWith } from '../../utils/lang';
4
2
 
5
- export function startsWithMatcherContext(log: ILogger, ruleAttr: string[]) /*: Function */ {
3
+ export function startsWithMatcherContext(ruleAttr: string[]) {
6
4
  return function startsWithMatcher(runtimeAttr: string): boolean {
7
- let matches = ruleAttr.some(e => startsWith(runtimeAttr, e));
8
-
9
- log.debug(ENGINE_MATCHER_STARTS_WITH, [runtimeAttr, ruleAttr, matches]);
5
+ const matches = ruleAttr.some(e => startsWith(runtimeAttr, e));
10
6
 
11
7
  return matches;
12
8
  };
@@ -1,12 +1,10 @@
1
- import { setToArray, ISet } from '../../utils/lang/sets';
2
- import { ILogger } from '../../logger/types';
3
- import { ENGINE_MATCHER_WHITELIST } from '../../logger/constants';
1
+ import { _Set } from '../../utils/lang/sets';
4
2
 
5
- export function whitelistMatcherContext(log: ILogger, ruleAttr: ISet<string>) /*: Function */ {
6
- return function whitelistMatcher(runtimeAttr: string): boolean {
7
- let isInWhitelist = ruleAttr.has(runtimeAttr);
3
+ export function whitelistMatcherContext(ruleAttr: string[]) {
4
+ const whitelistSet = new _Set(ruleAttr);
8
5
 
9
- log.debug(ENGINE_MATCHER_WHITELIST, [runtimeAttr, setToArray(ruleAttr).join(','), isInWhitelist]);
6
+ return function whitelistMatcher(runtimeAttr: string): boolean {
7
+ const isInWhitelist = whitelistSet.has(runtimeAttr);
10
8
 
11
9
  return isInWhitelist;
12
10
  };
@@ -2,10 +2,9 @@ import { findIndex } from '../../utils/lang';
2
2
  import { matcherTypes, matcherTypesMapper, matcherDataTypes } from '../matchers/matcherTypes';
3
3
  import { segmentTransform } from './segment';
4
4
  import { whitelistTransform } from './whitelist';
5
- import { setTransform } from './set';
6
5
  import { numericTransform } from './unaryNumeric';
7
6
  import { zeroSinceHH, zeroSinceSS } from '../convertions';
8
- import { IBetweenMatcherData, IInSegmentMatcherData, ISplitMatcher, IUnaryNumericMatcherData, IWhitelistMatcherData } from '../../dtos/types';
7
+ import { IBetweenMatcherData, IInSegmentMatcherData, ISplitMatcher, IUnaryNumericMatcherData } from '../../dtos/types';
9
8
  import { IMatcherDto } from '../types';
10
9
 
11
10
  /**
@@ -15,16 +14,17 @@ export function matchersTransform(matchers: ISplitMatcher[]): IMatcherDto[] {
15
14
 
16
15
  let parsedMatchers = matchers.map(matcher => {
17
16
  let {
18
- matcherType /* string */,
19
- negate /* boolean */,
20
- keySelector /* keySelectorObject */,
21
- userDefinedSegmentMatcherData: segmentObject /* segmentObject */,
22
- whitelistMatcherData: whitelistObject /* whiteListObject, provided by 'WHITELIST', set and string matchers */,
23
- unaryNumericMatcherData: unaryNumericObject /* unaryNumericObject */,
24
- betweenMatcherData: betweenObject /* betweenObject */,
25
- dependencyMatcherData: dependencyObject /* dependencyObject */,
17
+ matcherType,
18
+ negate,
19
+ keySelector,
20
+ userDefinedSegmentMatcherData,
21
+ whitelistMatcherData, /* whitelistObject, provided by 'WHITELIST', 'IN_LIST_SEMVER', set and string matchers */
22
+ unaryNumericMatcherData,
23
+ betweenMatcherData,
24
+ dependencyMatcherData,
26
25
  booleanMatcherData,
27
- stringMatcherData
26
+ stringMatcherData,
27
+ betweenStringMatcherData
28
28
  } = matcher;
29
29
 
30
30
  let attribute = keySelector && keySelector.attribute;
@@ -34,28 +34,26 @@ export function matchersTransform(matchers: ISplitMatcher[]): IMatcherDto[] {
34
34
  let value = undefined;
35
35
 
36
36
  if (type === matcherTypes.IN_SEGMENT) {
37
- value = segmentTransform(segmentObject as IInSegmentMatcherData);
38
- } else if (type === matcherTypes.WHITELIST) {
39
- value = whitelistTransform(whitelistObject as IWhitelistMatcherData);
37
+ value = segmentTransform(userDefinedSegmentMatcherData as IInSegmentMatcherData);
40
38
  } else if (type === matcherTypes.EQUAL_TO) {
41
- value = numericTransform(unaryNumericObject as IUnaryNumericMatcherData);
39
+ value = numericTransform(unaryNumericMatcherData as IUnaryNumericMatcherData);
42
40
  dataType = matcherDataTypes.NUMBER;
43
41
 
44
- if ((unaryNumericObject as IUnaryNumericMatcherData).dataType === 'DATETIME') {
42
+ if ((unaryNumericMatcherData as IUnaryNumericMatcherData).dataType === 'DATETIME') {
45
43
  value = zeroSinceHH(value);
46
44
  dataType = matcherDataTypes.DATETIME;
47
45
  }
48
46
  } else if (type === matcherTypes.GREATER_THAN_OR_EQUAL_TO ||
49
47
  type === matcherTypes.LESS_THAN_OR_EQUAL_TO) {
50
- value = numericTransform(unaryNumericObject as IUnaryNumericMatcherData);
48
+ value = numericTransform(unaryNumericMatcherData as IUnaryNumericMatcherData);
51
49
  dataType = matcherDataTypes.NUMBER;
52
50
 
53
- if ((unaryNumericObject as IUnaryNumericMatcherData).dataType === 'DATETIME') {
51
+ if ((unaryNumericMatcherData as IUnaryNumericMatcherData).dataType === 'DATETIME') {
54
52
  value = zeroSinceSS(value);
55
53
  dataType = matcherDataTypes.DATETIME;
56
54
  }
57
55
  } else if (type === matcherTypes.BETWEEN) {
58
- value = betweenObject as IBetweenMatcherData;
56
+ value = betweenMatcherData as IBetweenMatcherData;
59
57
  dataType = matcherDataTypes.NUMBER;
60
58
 
61
59
  if (value.dataType === 'DATETIME') {
@@ -63,27 +61,36 @@ export function matchersTransform(matchers: ISplitMatcher[]): IMatcherDto[] {
63
61
  value.end = zeroSinceSS(value.end);
64
62
  dataType = matcherDataTypes.DATETIME;
65
63
  }
64
+ } else if (type === matcherTypes.BETWEEN_SEMVER) {
65
+ value = betweenStringMatcherData;
66
66
  } else if (
67
67
  type === matcherTypes.EQUAL_TO_SET ||
68
68
  type === matcherTypes.CONTAINS_ANY_OF_SET ||
69
69
  type === matcherTypes.CONTAINS_ALL_OF_SET ||
70
70
  type === matcherTypes.PART_OF_SET
71
71
  ) {
72
- value = setTransform(whitelistObject as IWhitelistMatcherData);
72
+ value = whitelistTransform(whitelistMatcherData);
73
73
  dataType = matcherDataTypes.SET;
74
74
  } else if (
75
+ type === matcherTypes.WHITELIST ||
76
+ type === matcherTypes.IN_LIST_SEMVER ||
75
77
  type === matcherTypes.STARTS_WITH ||
76
78
  type === matcherTypes.ENDS_WITH ||
77
79
  type === matcherTypes.CONTAINS_STRING
78
80
  ) {
79
- value = setTransform(whitelistObject as IWhitelistMatcherData);
81
+ value = whitelistTransform(whitelistMatcherData);
80
82
  } else if (type === matcherTypes.IN_SPLIT_TREATMENT) {
81
- value = dependencyObject;
83
+ value = dependencyMatcherData;
82
84
  dataType = matcherDataTypes.NOT_SPECIFIED;
83
85
  } else if (type === matcherTypes.EQUAL_TO_BOOLEAN) {
84
86
  dataType = matcherDataTypes.BOOLEAN;
85
87
  value = booleanMatcherData;
86
- } else if (type === matcherTypes.MATCHES_STRING) {
88
+ } else if (
89
+ type === matcherTypes.MATCHES_STRING ||
90
+ type === matcherTypes.EQUAL_TO_SEMVER ||
91
+ type === matcherTypes.GREATER_THAN_OR_EQUAL_TO_SEMVER ||
92
+ type === matcherTypes.LESS_THAN_OR_EQUAL_TO_SEMVER
93
+ ) {
87
94
  value = stringMatcherData;
88
95
  }
89
96
 
@@ -91,6 +98,7 @@ export function matchersTransform(matchers: ISplitMatcher[]): IMatcherDto[] {
91
98
  attribute, // attribute over we should do the matching, undefined means 'use the key'
92
99
  negate, // should we negate the result?
93
100
  type, // which kind of matcher we should evaluate
101
+ name: matcherType,// name of the matcher for logging purposes
94
102
  value, // metadata used for the matching
95
103
  dataType // runtime input data type
96
104
  };
@@ -1,9 +1,8 @@
1
- import { IWhitelistMatcherData } from '../../dtos/types';
2
- import { _Set } from '../../utils/lang/sets';
1
+ import { ISplitMatcher } from '../../dtos/types';
3
2
 
4
3
  /**
5
- * Extract whitelist as a set. Used by 'WHITELIST' matcher.
4
+ * Extract whitelist array.
6
5
  */
7
- export function whitelistTransform(whitelistObject: IWhitelistMatcherData) {
8
- return new _Set(whitelistObject.whitelist);
6
+ export function whitelistTransform(whitelistObject: ISplitMatcher['whitelistMatcherData']) {
7
+ return whitelistObject && whitelistObject.whitelist;
9
8
  }
@@ -7,10 +7,11 @@ import { ifElseIfCombinerContext } from '../combiners/ifelseif';
7
7
  import { andCombinerContext } from '../combiners/and';
8
8
  import { thenable } from '../../utils/promise/thenable';
9
9
  import { IEvaluator, IMatcherDto, ISplitEvaluator } from '../types';
10
- import { ISplitCondition } from '../../dtos/types';
10
+ import { ISplitCondition, MaybeThenable } from '../../dtos/types';
11
11
  import { IStorageAsync, IStorageSync } from '../../storages/types';
12
12
  import { SplitIO } from '../../types';
13
13
  import { ILogger } from '../../logger/types';
14
+ import { ENGINE_MATCHER_ERROR, ENGINE_MATCHER_RESULT } from '../../logger/constants';
14
15
 
15
16
  export function parser(log: ILogger, conditions: ISplitCondition[], storage: IStorageSync | IStorageAsync): IEvaluator {
16
17
  let predicates = [];
@@ -28,19 +29,34 @@ export function parser(log: ILogger, conditions: ISplitCondition[], storage: ISt
28
29
 
29
30
  // create a set of pure functions from the matcher's dto
30
31
  const expressions = matchers.map((matcherDto: IMatcherDto) => {
31
- const matcher = matcherFactory(log, matcherDto, storage);
32
+ let matcher: ReturnType<typeof matcherFactory>;
33
+ try {
34
+ matcher = matcherFactory(log, matcherDto, storage);
35
+ } catch (error) {
36
+ log.error(ENGINE_MATCHER_ERROR, [matcherDto.name, error]);
37
+ }
32
38
 
33
39
  // Evaluator function.
34
40
  return (key: string, attributes: SplitIO.Attributes | undefined, splitEvaluator: ISplitEvaluator) => {
35
41
  const value = sanitizeValue(log, key, matcherDto, attributes);
36
- const result = value !== undefined && matcher ? matcher(value, splitEvaluator) : false;
42
+ let result: MaybeThenable<boolean> = false;
37
43
 
38
- if (thenable(result)) {
39
- // @ts-ignore
40
- return result.then(res => Boolean(res ^ matcherDto.negate));
44
+ if (value !== undefined && matcher) {
45
+ try {
46
+ result = matcher(value, splitEvaluator);
47
+ } catch (error) {
48
+ log.error(ENGINE_MATCHER_ERROR, [matcherDto.name, error]);
49
+ }
41
50
  }
42
- // @ts-ignore
43
- return Boolean(result ^ matcherDto.negate);
51
+
52
+ function handleResult(result: boolean) {
53
+ log.debug(ENGINE_MATCHER_RESULT, [matcherDto.name, result, matcherDto.value, value]); // @ts-ignore
54
+ return Boolean(result ^ matcherDto.negate);
55
+ }
56
+
57
+ return thenable(result) ?
58
+ result.then(handleResult) :
59
+ handleResult(result);
44
60
  };
45
61
  });
46
62
 
@@ -1,6 +1,5 @@
1
- import { IBetweenMatcherData, IDependencyMatcherData, MaybeThenable } from '../dtos/types';
1
+ import { IBetweenMatcherData, IBetweenStringMatcherData, IDependencyMatcherData, MaybeThenable } from '../dtos/types';
2
2
  import { IStorageAsync, IStorageSync } from '../storages/types';
3
- import { ISet } from '../utils/lang/sets';
4
3
  import { SplitIO } from '../types';
5
4
  import { ILogger } from '../logger/types';
6
5
 
@@ -11,7 +10,8 @@ export interface IDependencyMatcherValue {
11
10
 
12
11
  export interface IMatcherDto {
13
12
  type: number
14
- value?: string | number | boolean | string[] | IDependencyMatcherData | ISet<string> | IBetweenMatcherData | null
13
+ name: string
14
+ value?: string | number | boolean | string[] | IDependencyMatcherData | IBetweenMatcherData | IBetweenStringMatcherData | null
15
15
 
16
16
  attribute: string | null
17
17
  negate: boolean
@@ -9,25 +9,8 @@ export const ENGINE_COMBINER_AND = 0;
9
9
  export const ENGINE_COMBINER_IFELSEIF = 1;
10
10
  export const ENGINE_COMBINER_IFELSEIF_NO_TREATMENT = 2;
11
11
  export const ENGINE_BUCKET = 3;
12
- export const ENGINE_MATCHER_ALL = 4;
13
- export const ENGINE_MATCHER_BETWEEN = 5;
14
- export const ENGINE_MATCHER_BOOLEAN = 6;
15
- export const ENGINE_MATCHER_CONTAINS_ALL = 7;
16
- export const ENGINE_MATCHER_CONTAINS_ANY = 8;
17
- export const ENGINE_MATCHER_CONTAINS_STRING = 9;
18
12
  export const ENGINE_MATCHER_DEPENDENCY = 10;
19
13
  export const ENGINE_MATCHER_DEPENDENCY_PRE = 11;
20
- export const ENGINE_MATCHER_EQUAL = 12;
21
- export const ENGINE_MATCHER_EQUAL_TO_SET = 13;
22
- export const ENGINE_MATCHER_ENDS_WITH = 14;
23
- export const ENGINE_MATCHER_GREATER = 15;
24
- export const ENGINE_MATCHER_LESS = 16;
25
- export const ENGINE_MATCHER_PART_OF = 17;
26
- export const ENGINE_MATCHER_SEGMENT = 18;
27
- export const ENGINE_MATCHER_STRING = 19;
28
- export const ENGINE_MATCHER_STRING_INVALID = 20;
29
- export const ENGINE_MATCHER_STARTS_WITH = 21;
30
- export const ENGINE_MATCHER_WHITELIST = 22;
31
14
  export const ENGINE_VALUE = 23;
32
15
  export const ENGINE_SANITIZE = 24;
33
16
  export const CLEANUP_REGISTERING = 25;
@@ -45,6 +28,7 @@ export const SYNC_TASK_START = 36;
45
28
  export const SYNC_TASK_EXECUTE = 37;
46
29
  export const SYNC_TASK_STOP = 38;
47
30
  export const SETTINGS_SPLITS_FILTER = 39;
31
+ export const ENGINE_MATCHER_RESULT = 40;
48
32
 
49
33
  export const CLIENT_READY_FROM_CACHE = 100;
50
34
  export const CLIENT_READY = 101;
@@ -131,6 +115,7 @@ export const ERROR_NOT_BOOLEAN = 325;
131
115
  export const ERROR_MIN_CONFIG_PARAM = 326;
132
116
  export const ERROR_TOO_MANY_SETS = 327;
133
117
  export const ERROR_SETS_FILTER_EXCLUSIVE = 328;
118
+ export const ENGINE_MATCHER_ERROR = 329;
134
119
 
135
120
  // Log prefixes (a.k.a. tags or categories)
136
121
  export const LOG_PREFIX_SETTINGS = 'settings';
@@ -1,6 +1,6 @@
1
1
  import { objectAssign } from '../utils/lang/objectAssign';
2
2
  import { ILoggerOptions, ILogger } from './types';
3
- import { find } from '../utils/lang';
3
+ import { find, isObject } from '../utils/lang';
4
4
  import { LogLevel } from '../types';
5
5
  import { IMap, _Map } from '../utils/lang/maps';
6
6
 
@@ -28,7 +28,13 @@ export function isLogLevelString(str: string): str is LogLevel {
28
28
  export function _sprintf(format: string = '', args: any[] = []): string {
29
29
  let i = 0;
30
30
  return format.replace(/%s/g, function () {
31
- return args[i++];
31
+ let arg = args[i++];
32
+ if (isObject(arg) || Array.isArray(arg)) {
33
+ try {
34
+ arg = JSON.stringify(arg);
35
+ } catch (e) { /* empty */ }
36
+ }
37
+ return arg;
32
38
  });
33
39
  }
34
40
 
@@ -7,27 +7,11 @@ export const codesDebug: [number, string][] = codesInfo.concat([
7
7
  [c.ENGINE_COMBINER_IFELSEIF, c.LOG_PREFIX_ENGINE_COMBINER + 'Treatment found: %s'],
8
8
  [c.ENGINE_COMBINER_IFELSEIF_NO_TREATMENT, c.LOG_PREFIX_ENGINE_COMBINER + 'All predicates evaluated, no treatment found.'],
9
9
  [c.ENGINE_BUCKET, c.LOG_PREFIX_ENGINE + ': using algo "murmur" bucket %s for key %s using seed %s - treatment %s'],
10
- [c.ENGINE_MATCHER_ALL, c.LOG_PREFIX_ENGINE_MATCHER + '[allMatcher] is always true'],
11
- [c.ENGINE_MATCHER_BETWEEN, c.LOG_PREFIX_ENGINE_MATCHER + '[betweenMatcher] is %s between %s and %s? %s'],
12
- [c.ENGINE_MATCHER_BOOLEAN, c.LOG_PREFIX_ENGINE_MATCHER + '[booleanMatcher] %s === %s'],
13
- [c.ENGINE_MATCHER_CONTAINS_ALL, c.LOG_PREFIX_ENGINE_MATCHER + '[containsAllMatcher] %s contains all elements of %s? %s'],
14
- [c.ENGINE_MATCHER_CONTAINS_ANY, c.LOG_PREFIX_ENGINE_MATCHER + '[containsAnyMatcher] %s contains at least an element of %s? %s'],
15
- [c.ENGINE_MATCHER_CONTAINS_STRING, c.LOG_PREFIX_ENGINE_MATCHER + '[containsStringMatcher] %s contains %s? %s'],
16
- [c.ENGINE_MATCHER_DEPENDENCY, c.LOG_PREFIX_ENGINE_MATCHER + '[dependencyMatcher] parent feature flag "%s" evaluated to "%s" with label "%s". %s evaluated treatment is part of [%s] ? %s.'],
17
- [c.ENGINE_MATCHER_DEPENDENCY_PRE, c.LOG_PREFIX_ENGINE_MATCHER + '[dependencyMatcher] will evaluate parent feature flag: "%s" with key: %s %s'],
18
- [c.ENGINE_MATCHER_EQUAL, c.LOG_PREFIX_ENGINE_MATCHER + '[equalToMatcher] is %s equal to %s? %s'],
19
- [c.ENGINE_MATCHER_EQUAL_TO_SET, c.LOG_PREFIX_ENGINE_MATCHER + '[equalToSetMatcher] is %s equal to set %s? %s'],
20
- [c.ENGINE_MATCHER_ENDS_WITH, c.LOG_PREFIX_ENGINE_MATCHER + '[endsWithMatcher] %s ends with %s? %s'],
21
- [c.ENGINE_MATCHER_GREATER, c.LOG_PREFIX_ENGINE_MATCHER + '[greaterThanEqualMatcher] is %s greater than %s? %s'],
22
- [c.ENGINE_MATCHER_LESS, c.LOG_PREFIX_ENGINE_MATCHER + '[lessThanEqualMatcher] is %s less than %s? %s'],
23
- [c.ENGINE_MATCHER_PART_OF, c.LOG_PREFIX_ENGINE_MATCHER + '[partOfMatcher] %s is part of %s? %s'],
24
- [c.ENGINE_MATCHER_SEGMENT, c.LOG_PREFIX_ENGINE_MATCHER + '[segmentMatcher] evaluated %s / %s => %s'],
25
- [c.ENGINE_MATCHER_STRING, c.LOG_PREFIX_ENGINE_MATCHER + '[stringMatcher] does %s matches with %s? %s'],
26
- [c.ENGINE_MATCHER_STRING_INVALID, c.LOG_PREFIX_ENGINE_MATCHER + '[stringMatcher] %s is an invalid regex'],
27
- [c.ENGINE_MATCHER_STARTS_WITH, c.LOG_PREFIX_ENGINE_MATCHER + '[startsWithMatcher] %s starts with %s? %s'],
28
- [c.ENGINE_MATCHER_WHITELIST, c.LOG_PREFIX_ENGINE_MATCHER + '[whitelistMatcher] evaluated %s in [%s] => %s'],
29
- [c.ENGINE_VALUE, c.LOG_PREFIX_ENGINE_VALUE + 'Extracted attribute [%s], [%s] will be used for matching.'],
30
- [c.ENGINE_SANITIZE, c.LOG_PREFIX_ENGINE + ':sanitize: Attempted to sanitize [%s] which should be of type [%s]. Sanitized and processed value => [%s]'],
10
+ [c.ENGINE_MATCHER_DEPENDENCY, c.LOG_PREFIX_ENGINE_MATCHER + '[IN_SPLIT_TREATMENT] parent feature flag "%s" evaluated to "%s" with label "%s". %s evaluated treatment is part of %s ? %s.'],
11
+ [c.ENGINE_MATCHER_DEPENDENCY_PRE, c.LOG_PREFIX_ENGINE_MATCHER + '[IN_SPLIT_TREATMENT] will evaluate parent feature flag: "%s" with key: %s %s'],
12
+ [c.ENGINE_VALUE, c.LOG_PREFIX_ENGINE_VALUE + 'Extracted attribute `%s`. %s will be used for matching.'],
13
+ [c.ENGINE_SANITIZE, c.LOG_PREFIX_ENGINE + ':sanitize: Attempted to sanitize %s which should be of type %s. Sanitized and processed value => %s'],
14
+ [c.ENGINE_MATCHER_RESULT, c.LOG_PREFIX_ENGINE_MATCHER + '[%s] Result: %s. Rule value: %s. Evaluation value: %s'],
31
15
  // SDK
32
16
  [c.CLEANUP_REGISTERING, c.LOG_PREFIX_CLEANUP + 'Registering cleanup handler %s'],
33
17
  [c.CLEANUP_DEREGISTERING, c.LOG_PREFIX_CLEANUP + 'Deregistering cleanup handler %s'],
@@ -2,7 +2,8 @@ import * as c from '../constants';
2
2
 
3
3
  export const codesError: [number, string][] = [
4
4
  // evaluator
5
- [c.ERROR_ENGINE_COMBINER_IFELSEIF, c.LOG_PREFIX_ENGINE_COMBINER + 'Invalid feature flag, no valid rules or unsupported matcher type found'],
5
+ [c.ERROR_ENGINE_COMBINER_IFELSEIF, c.LOG_PREFIX_ENGINE_COMBINER + 'Invalid feature flag, no valid rules or unsupported targeting rule type found'],
6
+ [c.ENGINE_MATCHER_ERROR, c.LOG_PREFIX_ENGINE_MATCHER + '[%s] %s'],
6
7
  // SDK
7
8
  [c.ERROR_LOGLEVEL_INVALID, 'logger: Invalid Log Level - No changes to the logs will be applied.'],
8
9
  [c.ERROR_CLIENT_CANNOT_GET_READY, 'The SDK will not get ready. Reason: %s'],
@@ -4,7 +4,7 @@ import { codesError } from './error';
4
4
  export const codesWarn: [number, string][] = codesError.concat([
5
5
  // evaluator
6
6
  [c.ENGINE_VALUE_INVALID, c.LOG_PREFIX_ENGINE_VALUE + 'Value %s doesn\'t match with expected type.'],
7
- [c.ENGINE_VALUE_NO_ATTRIBUTES, c.LOG_PREFIX_ENGINE_VALUE + 'Defined attribute [%s], no attributes received.'],
7
+ [c.ENGINE_VALUE_NO_ATTRIBUTES, c.LOG_PREFIX_ENGINE_VALUE + 'Defined attribute `%s`. No attributes received.'],
8
8
  // synchronizer
9
9
  [c.SYNC_MYSEGMENTS_FETCH_RETRY, c.LOG_PREFIX_SYNC_MYSEGMENTS + 'Retrying download of segments #%s. Reason: %s'],
10
10
  [c.SYNC_SPLITS_FETCH_FAILS, c.LOG_PREFIX_SYNC_SPLITS + 'Error while doing fetch of feature flags. %s'],
@@ -44,17 +44,16 @@ export function splitApiFactory(
44
44
  },
45
45
 
46
46
  fetchAuth(userMatchingKeys?: string[]) {
47
- let url = `${urls.auth}/v2/auth`;
48
- if (userMatchingKeys) { // accounting the possibility that `userMatchingKeys` is undefined (server-side API)
47
+ let url = `${urls.auth}/v2/auth?s=${FLAGS_SPEC}`;
48
+ if (userMatchingKeys) { // `userMatchingKeys` is undefined in server-side
49
49
  const queryParams = userMatchingKeys.map(userKeyToQueryParam).join('&');
50
- if (queryParams) // accounting the possibility that `userKeys` and thus `queryParams` are empty
51
- url += '?' + queryParams;
50
+ if (queryParams) url += '&' + queryParams;
52
51
  }
53
52
  return splitHttpClient(url, undefined, telemetryTracker.trackHttp(TOKEN));
54
53
  },
55
54
 
56
55
  fetchSplitChanges(since: number, noCache?: boolean, till?: number) {
57
- const url = `${urls.sdk}/splitChanges?s=${FLAGS_SPEC}&since=${since}${till ? '&till=' + till : ''}${filterQueryString || ''}`;
56
+ const url = `${urls.sdk}/splitChanges?s=${FLAGS_SPEC}&since=${since}${filterQueryString || ''}${till ? '&till=' + till : ''}`;
58
57
  return splitHttpClient(url, noCache ? noCacheHeaderOptions : undefined, telemetryTracker.trackHttp(SPLITS))
59
58
  .catch((err) => {
60
59
  if (err.statusCode === 414) settings.log.error(ERROR_TOO_MANY_SETS);
@@ -146,7 +146,7 @@ export class SplitsCacheInLocal extends AbstractSplitsCacheSync {
146
146
 
147
147
  // when using a new split query, we must update it at the store
148
148
  if (this.updateNewFilter) {
149
- this.log.info(LOG_PREFIX + 'SDK key or feature flag filter criteria was modified. Updating cache');
149
+ this.log.info(LOG_PREFIX + 'SDK key, flags filter criteria or flags spec version was modified. Updating cache');
150
150
  const storageHashKey = this.keys.buildHashKey();
151
151
  try {
152
152
  localStorage.setItem(storageHashKey, this.storageHash);
@@ -96,7 +96,7 @@ export function PluggableStorage(options: PluggableStorageOptions): IStorageAsyn
96
96
  return wrapper.get(keys.buildHashKey()).then((hash) => {
97
97
  const currentHash = getStorageHash(settings);
98
98
  if (hash !== currentHash) {
99
- log.info(LOG_PREFIX + 'Storage HASH has changed (SDK key or feature flag filter criteria was modified). Clearing cache');
99
+ log.info(LOG_PREFIX + 'Storage HASH has changed (SDK key, flags filter criteria or flags spec version was modified). Clearing cache');
100
100
  return wrapper.getKeysByPrefix(`${keys.prefix}.`).then(storageKeys => {
101
101
  return Promise.all(storageKeys.map(storageKey => wrapper.del(storageKey)));
102
102
  }).then(() => wrapper.set(keys.buildHashKey(), currentHash));
@@ -0,0 +1,111 @@
1
+ import { isString } from '../utils/lang';
2
+
3
+ const NUMERIC_IDENTIFIER_REGEX = /^[0-9]+$/;
4
+
5
+ const METADATA_DELIMITER = '+';
6
+ const PRERELEASE_DELIMITER = '-';
7
+ const VALUE_DELIMITER = '.';
8
+
9
+ /**
10
+ * Compares two strings. If both strings are numeric identifiers, they are compared numerically. Otherwise, they are compared lexicographically.
11
+ * This could be implemented using `a.localeCompare(b, undefined, { numeric: true })` but locale options are not broadly supported.
12
+ */
13
+ function compareStrings(a: string, b: string): number {
14
+ if (NUMERIC_IDENTIFIER_REGEX.test(a) && NUMERIC_IDENTIFIER_REGEX.test(b)) {
15
+ const result = a.length - b.length;
16
+ if (result !== 0) {
17
+ return result;
18
+ }
19
+ }
20
+ return a < b ? -1 : a > b ? 1 : 0;
21
+ }
22
+
23
+ // Sanitizes a numeric identifier by removing leading zeros
24
+ function sanitizeNumericIdentifier(value: string): string {
25
+ return value.replace(/^0+(?=\d)/, '');
26
+ }
27
+
28
+ function throwError(version: string) {
29
+ throw new Error('Unable to convert to Semver, incorrect format: ' + version);
30
+ }
31
+
32
+ export class Semver {
33
+
34
+ private readonly _major: string;
35
+ private readonly _minor: string;
36
+ private readonly _patch: string;
37
+ private readonly _preRelease: string[];
38
+ private readonly _isStable: boolean;
39
+
40
+ // Version string for 'equal' and 'in list' comparisons
41
+ public readonly version: string;
42
+
43
+ public constructor(version: string) {
44
+ if (!isString(version)) throwError(version);
45
+
46
+ // Separate metadata if exists
47
+ let index = version.indexOf(METADATA_DELIMITER);
48
+ let [vWithoutMetadata, metadata] = index === -1 ? [version] : [version.slice(0, index), version.slice(index + 1)];
49
+ if (metadata === '') throwError(version);
50
+
51
+ // Set pre-release versions if exists
52
+ index = vWithoutMetadata.indexOf(PRERELEASE_DELIMITER);
53
+ if (index === -1) {
54
+ this._isStable = true;
55
+ this._preRelease = [];
56
+ } else {
57
+ this._isStable = false;
58
+ this._preRelease = vWithoutMetadata.slice(index + 1).split(VALUE_DELIMITER).map((value) => {
59
+ if (!value) throwError(version);
60
+ return NUMERIC_IDENTIFIER_REGEX.test(value) ?
61
+ sanitizeNumericIdentifier(value) :
62
+ value;
63
+ });
64
+ vWithoutMetadata = vWithoutMetadata.slice(0, index);
65
+ }
66
+
67
+ // Set major, minor, and patch versions
68
+ const vParts = vWithoutMetadata.split(VALUE_DELIMITER).map((value) => {
69
+ if (!value || !NUMERIC_IDENTIFIER_REGEX.test(value)) throwError(version);
70
+ return sanitizeNumericIdentifier(value);
71
+ });
72
+
73
+ if (vParts.length !== 3) throwError(version);
74
+ this._major = vParts[0];
75
+ this._minor = vParts[1];
76
+ this._patch = vParts[2];
77
+
78
+ // Set version string
79
+ this.version = vParts.join(VALUE_DELIMITER);
80
+ if (this._preRelease.length) this.version += PRERELEASE_DELIMITER + this._preRelease.join(VALUE_DELIMITER);
81
+ if (metadata) this.version += METADATA_DELIMITER + metadata;
82
+ }
83
+
84
+ /**
85
+ * Precedence comparision between 2 Semver objects.
86
+ *
87
+ * @return `0` if `this === toCompare`, `-1` if `this < toCompare`, and `1` if `this > toCompare`
88
+ */
89
+ public compare(toCompare: Semver): number {
90
+ if (this.version === toCompare.version) return 0;
91
+
92
+ let result = compareStrings(this._major, toCompare._major);
93
+ if (result !== 0) return result;
94
+
95
+ result = compareStrings(this._minor, toCompare._minor);
96
+ if (result !== 0) return result;
97
+
98
+ result = compareStrings(this._patch, toCompare._patch);
99
+ if (result !== 0) return result;
100
+
101
+ if (!this._isStable && toCompare._isStable) return -1;
102
+ if (this._isStable && !toCompare._isStable) return 1;
103
+
104
+ for (let i = 0, length = Math.min(this._preRelease.length, toCompare._preRelease.length); i < length; i++) {
105
+ const result = compareStrings(this._preRelease[i], toCompare._preRelease[i]);
106
+ if (result !== 0) return result;
107
+ }
108
+
109
+ return this._preRelease.length - toCompare._preRelease.length;
110
+ }
111
+ }
@@ -5,4 +5,4 @@ export const SDK_NOT_READY = 'not ready';
5
5
  export const EXCEPTION = 'exception';
6
6
  export const SPLIT_ARCHIVED = 'archived';
7
7
  export const NOT_IN_SPLIT = 'not in split';
8
- export const UNSUPPORTED_MATCHER_TYPE = 'unsupported matcher type';
8
+ export const UNSUPPORTED_MATCHER_TYPE = 'targeting rule type unsupported by sdk';
@@ -4,7 +4,7 @@ import { LogLevel } from '../../../types';
4
4
  import { getLogLevel } from './commons';
5
5
 
6
6
  function isLogger(log: any): log is ILogger {
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
 
10
10
  // By default it starts disabled.