flagsmith-nodejs 2.0.0-beta.1 → 2.0.0-beta.4

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 (85) hide show
  1. package/.tool-versions +1 -0
  2. package/example/package-lock.json +1070 -2
  3. package/example/package.json +1 -0
  4. package/example/server/api/index.js +14 -7
  5. package/example/server/index.js +3 -0
  6. package/flagsmith-engine/environments/models.ts +8 -8
  7. package/flagsmith-engine/features/models.ts +7 -7
  8. package/flagsmith-engine/identities/models.ts +8 -8
  9. package/flagsmith-engine/identities/util.ts +1 -1
  10. package/flagsmith-engine/index.ts +20 -19
  11. package/flagsmith-engine/organisations/models.ts +6 -6
  12. package/flagsmith-engine/projects/models.ts +2 -3
  13. package/flagsmith-engine/segments/evaluators.ts +2 -2
  14. package/flagsmith-engine/utils/errors.ts +1 -0
  15. package/flagsmith-engine/utils/hashing/index.ts +9 -14
  16. package/index.ts +4 -1
  17. package/package.json +11 -8
  18. package/sdk/analytics.ts +0 -28
  19. package/sdk/index.ts +73 -26
  20. package/sdk/models.ts +3 -1
  21. package/sdk/polling_manager.ts +4 -4
  22. package/sdk/utils.ts +1 -1
  23. package/tests/sdk/data/environment.json +39 -2
  24. package/tests/sdk/flagsmith.test.ts +21 -2
  25. package/tests/sdk/polling.test.ts +1 -1
  26. package/tests/sdk/utils.ts +1 -1
  27. package/.idea/flagsmith-nodejs-client.iml +0 -12
  28. package/.idea/modules.xml +0 -8
  29. package/.idea/vcs.xml +0 -6
  30. package/build/flagsmith-engine/environments/integrations/models.d.ts +0 -4
  31. package/build/flagsmith-engine/environments/integrations/models.js +0 -8
  32. package/build/flagsmith-engine/environments/models.d.ts +0 -25
  33. package/build/flagsmith-engine/environments/models.js +0 -40
  34. package/build/flagsmith-engine/environments/util.d.ts +0 -3
  35. package/build/flagsmith-engine/environments/util.js +0 -19
  36. package/build/flagsmith-engine/features/constants.d.ts +0 -4
  37. package/build/flagsmith-engine/features/constants.js +0 -7
  38. package/build/flagsmith-engine/features/models.d.ts +0 -32
  39. package/build/flagsmith-engine/features/models.js +0 -85
  40. package/build/flagsmith-engine/features/util.d.ts +0 -3
  41. package/build/flagsmith-engine/features/util.js +0 -20
  42. package/build/flagsmith-engine/identities/models.d.ts +0 -15
  43. package/build/flagsmith-engine/identities/models.js +0 -47
  44. package/build/flagsmith-engine/identities/traits/models.d.ts +0 -5
  45. package/build/flagsmith-engine/identities/traits/models.js +0 -12
  46. package/build/flagsmith-engine/identities/util.d.ts +0 -4
  47. package/build/flagsmith-engine/identities/util.js +0 -22
  48. package/build/flagsmith-engine/index.d.ts +0 -8
  49. package/build/flagsmith-engine/index.js +0 -60
  50. package/build/flagsmith-engine/organisations/models.d.ts +0 -9
  51. package/build/flagsmith-engine/organisations/models.js +0 -21
  52. package/build/flagsmith-engine/organisations/util.d.ts +0 -2
  53. package/build/flagsmith-engine/organisations/util.js +0 -8
  54. package/build/flagsmith-engine/projects/models.d.ts +0 -10
  55. package/build/flagsmith-engine/projects/models.js +0 -18
  56. package/build/flagsmith-engine/projects/util.d.ts +0 -2
  57. package/build/flagsmith-engine/projects/util.js +0 -15
  58. package/build/flagsmith-engine/segments/constants.d.ts +0 -26
  59. package/build/flagsmith-engine/segments/constants.js +0 -31
  60. package/build/flagsmith-engine/segments/evaluators.d.ts +0 -6
  61. package/build/flagsmith-engine/segments/evaluators.js +0 -29
  62. package/build/flagsmith-engine/segments/models.d.ts +0 -31
  63. package/build/flagsmith-engine/segments/models.js +0 -83
  64. package/build/flagsmith-engine/segments/util.d.ts +0 -4
  65. package/build/flagsmith-engine/segments/util.js +0 -23
  66. package/build/flagsmith-engine/utils/collections.d.ts +0 -4
  67. package/build/flagsmith-engine/utils/collections.js +0 -16
  68. package/build/flagsmith-engine/utils/hashing/index.d.ts +0 -1
  69. package/build/flagsmith-engine/utils/hashing/index.js +0 -56
  70. package/build/flagsmith-engine/utils/index.d.ts +0 -1
  71. package/build/flagsmith-engine/utils/index.js +0 -14
  72. package/build/index.d.ts +0 -2
  73. package/build/index.js +0 -7
  74. package/build/sdk/analytics.d.ts +0 -28
  75. package/build/sdk/analytics.js +0 -81
  76. package/build/sdk/errors.d.ts +0 -4
  77. package/build/sdk/errors.js +0 -9
  78. package/build/sdk/index.d.ts +0 -99
  79. package/build/sdk/index.js +0 -221
  80. package/build/sdk/models.d.ts +0 -55
  81. package/build/sdk/models.js +0 -102
  82. package/build/sdk/polling_manager.d.ts +0 -9
  83. package/build/sdk/polling_manager.js +0 -31
  84. package/build/sdk/utils.d.ts +0 -12
  85. package/build/sdk/utils.js +0 -45
@@ -15,6 +15,7 @@
15
15
  "npm": "3.10.x"
16
16
  },
17
17
  "dependencies": {
18
+ "flagsmith-nodejs": "^2.0.0-beta.1",
18
19
  "node-cache": "^5.1.2",
19
20
  "ssg-node-express": "4.16.4",
20
21
  "ssg-util": "0.0.3"
@@ -1,14 +1,16 @@
1
1
  const Router = require('express').Router;
2
- const { Flagsmith } = require('../../../build/sdk');
3
- const environmentKey = ""
4
- if(!environmentKey) {
5
- throw new Error("Please generate a Server Side SDK Key in environment settings to run the example")
2
+ const Flagsmith = require('../../../build');
3
+ const environmentKey = '';
4
+ if (!environmentKey) {
5
+ throw new Error(
6
+ 'Please generate a Server Side SDK Key in environment settings to run the example'
7
+ );
6
8
  }
7
9
  const flagsmith = new Flagsmith({
8
10
  environmentKey,
9
11
  enableLocalEvaluation: true,
10
- defaultFlagHandler: (str)=> {
11
- return {enabled:false, isDefault:true, value:null}
12
+ defaultFlagHandler: str => {
13
+ return { enabled: false, isDefault: true, value: null };
12
14
  }
13
15
  });
14
16
 
@@ -21,11 +23,16 @@ module.exports = () => {
21
23
  });
22
24
 
23
25
  api.get('/:user', async (req, res) => {
24
- const flags = await flagsmith.getIdentityFlags(req.params.user, {checkout_v2: 1});
26
+ const flags = await flagsmith.getIdentityFlags(req.params.user, { checkout_v2: 1 });
25
27
  const fontSize = flags.getFeatureValue('font_size');
26
28
  const checkoutV2 = flags.isFeatureEnabled('checkout_v2');
27
29
  res.json({ fontSize, checkoutV2 });
28
30
  });
29
31
 
32
+ api.get('/:user/segments', async (req, res) => {
33
+ const segments = await flagsmith.getIdentitySegments(req.params.user, { checkout_v2: 1 });
34
+ res.json(segments.map(v => v.name));
35
+ });
36
+
30
37
  return api;
31
38
  };
@@ -22,5 +22,8 @@ console.log('To get an example response for getFlags');
22
22
  console.log();
23
23
  console.log('Go to http://localhost:' + PORT + '/api/flagsmith_sample_user');
24
24
  console.log('To get an example feature state for a user');
25
+ console.log();
26
+ console.log('Go to http://localhost:' + PORT + '/api/flagsmith_sample_user/segments');
27
+ console.log('To get the segments which the user belongs to');
25
28
 
26
29
  module.exports = app;
@@ -14,17 +14,17 @@ export class EnvironmentAPIKeyModel {
14
14
  constructor(
15
15
  id: number,
16
16
  key: string,
17
- created_at: number,
17
+ createdAt: number,
18
18
  name: string,
19
- client_api_key: string,
20
- expires_at?: number
19
+ clientApiKey: string,
20
+ expiresAt?: number
21
21
  ) {
22
22
  this.id = id;
23
23
  this.key = key;
24
- this.createdAt = created_at;
24
+ this.createdAt = createdAt;
25
25
  this.name = name;
26
- this.clientApiKey = client_api_key;
27
- this.expiresAt = expires_at;
26
+ this.clientApiKey = clientApiKey;
27
+ this.expiresAt = expiresAt;
28
28
  }
29
29
 
30
30
  isValid() {
@@ -42,9 +42,9 @@ export class EnvironmentModel {
42
42
  mixpanel_config?: IntegrationModel;
43
43
  heap_config?: IntegrationModel;
44
44
 
45
- constructor(id: number, api_key: string, project: ProjectModel) {
45
+ constructor(id: number, apiKey: string, project: ProjectModel) {
46
46
  this.id = id;
47
- this.apiKey = api_key;
47
+ this.apiKey = apiKey;
48
48
  this.project = project;
49
49
  }
50
50
  }
@@ -51,7 +51,7 @@ export class FeatureStateModel {
51
51
  enabled: boolean;
52
52
  djangoID: number;
53
53
  featurestateUUID: string = uuidv4();
54
- _value: any;
54
+ private value: any;
55
55
  multivariateFeatureStateValues: MultivariateFeatureStateValueModel[] = [];
56
56
 
57
57
  constructor(
@@ -59,24 +59,24 @@ export class FeatureStateModel {
59
59
  enabled: boolean,
60
60
  djangoID: number,
61
61
  value?: any,
62
- featurestate_uuid: string = uuidv4()
62
+ featurestateUuid: string = uuidv4()
63
63
  ) {
64
64
  this.feature = feature;
65
65
  this.enabled = enabled;
66
66
  this.djangoID = djangoID;
67
- this._value = value;
68
- this.featurestateUUID = featurestate_uuid;
67
+ this.value = value;
68
+ this.featurestateUUID = featurestateUuid;
69
69
  }
70
70
 
71
71
  setValue(value: any) {
72
- this._value = value;
72
+ this.value = value;
73
73
  }
74
74
 
75
75
  getValue(identityId?: number | string) {
76
76
  if (!!identityId && this.multivariateFeatureStateValues.length > 0) {
77
77
  return this.getMultivariateValue(identityId);
78
78
  }
79
- return this._value;
79
+ return this.value;
80
80
  }
81
81
 
82
82
  get_feature_state_value() {
@@ -100,6 +100,6 @@ export class FeatureStateModel {
100
100
  }
101
101
  startPercentage = limit;
102
102
  }
103
- return this._value;
103
+ return this.value;
104
104
  }
105
105
  }
@@ -15,17 +15,17 @@ export class IdentityModel {
15
15
 
16
16
  constructor(
17
17
  created_date: string,
18
- identity_traits: TraitModel[],
19
- identity_features: IdentityFeaturesList,
20
- environment_api_key: string,
18
+ identityTraits: TraitModel[],
19
+ identityFeatures: IdentityFeaturesList,
20
+ environmentApiKey: string,
21
21
  identifier: string,
22
- identity_uuid?: string
22
+ identityUuid?: string
23
23
  ) {
24
- this.identityUuid = identity_uuid || uuidv4();
24
+ this.identityUuid = identityUuid || uuidv4();
25
25
  this.createdDate = Date.parse(created_date) || Date.now();
26
- this.identityTraits = identity_traits;
27
- this.identityFeatures = new IdentityFeaturesList(...identity_features);
28
- this.environmentApiKey = environment_api_key;
26
+ this.identityTraits = identityTraits;
27
+ this.identityFeatures = new IdentityFeaturesList(...identityFeatures);
28
+ this.environmentApiKey = environmentApiKey;
29
29
  this.identifier = identifier;
30
30
  }
31
31
 
@@ -25,6 +25,6 @@ export function buildIdentityModel(identityJSON: any): IdentityModel {
25
25
  identityJSON.identity_uuid
26
26
  );
27
27
 
28
- model.djangoID = identityJSON['django_id'];
28
+ model.djangoID = identityJSON.django_id;
29
29
  return model;
30
30
  }
@@ -4,56 +4,57 @@ import { IdentityModel } from './identities/models';
4
4
  import { TraitModel } from './identities/traits/models';
5
5
  import { getIdentitySegments } from './segments/evaluators';
6
6
  import { SegmentModel } from './segments/models';
7
+ import { FeatureStateNotFound } from './utils/errors';
7
8
 
8
9
  function getIdentityFeatureStatesDict(
9
10
  environment: EnvironmentModel,
10
11
  identity: IdentityModel,
11
- override_traits?: TraitModel[]
12
+ overrideTraits?: TraitModel[]
12
13
  ) {
13
14
  // Get feature states from the environment
14
- const feature_states: { [key: number]: FeatureStateModel } = {};
15
+ const featureStates: { [key: number]: FeatureStateModel } = {};
15
16
  for (const fs of environment.featureStates) {
16
- feature_states[fs.feature.id] = fs;
17
+ featureStates[fs.feature.id] = fs;
17
18
  }
18
19
 
19
20
  // Override with any feature states defined by matching segments
20
- const identity_segments: SegmentModel[] = getIdentitySegments(
21
+ const identitySegments: SegmentModel[] = getIdentitySegments(
21
22
  environment,
22
23
  identity,
23
- override_traits
24
+ overrideTraits
24
25
  );
25
- for (const matching_segment of identity_segments) {
26
- for (const feature_state of matching_segment.featureStates) {
26
+ for (const matchingSegment of identitySegments) {
27
+ for (const featureState of matchingSegment.featureStates) {
27
28
  // note that feature states are stored on the segment in descending priority
28
29
  // order so we only care that the last one is added
29
30
  // TODO: can we optimise this?
30
- feature_states[feature_state.feature.id] = feature_state;
31
+ featureStates[featureState.feature.id] = featureState;
31
32
  }
32
33
  }
33
34
 
34
35
  // Override with any feature states defined directly the identity
35
36
  for (const fs of identity.identityFeatures || []) {
36
- if (feature_states[fs.feature.id]) {
37
- feature_states[fs.feature.id] = fs;
37
+ if (featureStates[fs.feature.id]) {
38
+ featureStates[fs.feature.id] = fs;
38
39
  }
39
40
  }
40
- return feature_states;
41
+ return featureStates;
41
42
  }
42
43
 
43
44
  export function getIdentityFeatureState(
44
45
  environment: EnvironmentModel,
45
46
  identity: IdentityModel,
46
- feature_name: string,
47
- override_traits?: TraitModel[]
47
+ featureName: string,
48
+ overrideTraits?: TraitModel[]
48
49
  ): FeatureStateModel {
49
- const featureStates = getIdentityFeatureStatesDict(environment, identity, override_traits);
50
+ const featureStates = getIdentityFeatureStatesDict(environment, identity, overrideTraits);
50
51
 
51
52
  const matchingFeature = Object.values(featureStates).filter(
52
- f => f.feature.name === feature_name
53
+ f => f.feature.name === featureName
53
54
  );
54
55
 
55
56
  if (matchingFeature.length === 0) {
56
- throw new Error('Feature State Not Found');
57
+ throw new FeatureStateNotFound('Feature State Not Found');
57
58
  }
58
59
 
59
60
  return matchingFeature[0];
@@ -64,14 +65,14 @@ export function getIdentityFeatureStates(
64
65
  identity: IdentityModel,
65
66
  overrideTraits?: TraitModel[]
66
67
  ): FeatureStateModel[] {
67
- const feature_states = Object.values(
68
+ const featureStates = Object.values(
68
69
  getIdentityFeatureStatesDict(environment, identity, overrideTraits)
69
70
  );
70
71
 
71
72
  if (environment.project.hideDisabledFlags) {
72
- return feature_states.filter(fs => !!fs.enabled);
73
+ return featureStates.filter(fs => !!fs.enabled);
73
74
  }
74
- return feature_states;
75
+ return featureStates;
75
76
  }
76
77
 
77
78
  export function getEnvironmentFeatureState(environment: EnvironmentModel, featureName: string) {
@@ -8,15 +8,15 @@ export class OrganisationModel {
8
8
  constructor(
9
9
  id: number,
10
10
  name: string,
11
- feature_analytics: boolean,
12
- stop_serving_flags: boolean,
13
- persist_trait_data: boolean
11
+ featureAnalytics: boolean,
12
+ stopServingFlags: boolean,
13
+ persistTraitData: boolean
14
14
  ) {
15
15
  this.id = id;
16
16
  this.name = name;
17
- this.featureAnalytics = feature_analytics;
18
- this.stopServingFlags = stop_serving_flags;
19
- this.persistTraitData = persist_trait_data;
17
+ this.featureAnalytics = featureAnalytics;
18
+ this.stopServingFlags = stopServingFlags;
19
+ this.persistTraitData = persistTraitData;
20
20
  }
21
21
 
22
22
  get unique_slug() {
@@ -6,18 +6,17 @@ export class ProjectModel {
6
6
  name: string;
7
7
  organisation: OrganisationModel;
8
8
  hideDisabledFlags: boolean;
9
- // FIXME
10
9
  segments: SegmentModel[] = [];
11
10
 
12
11
  constructor(
13
12
  id: number,
14
13
  name: string,
15
- hide_disabled_flags: boolean,
14
+ hideDisabledFlags: boolean,
16
15
  organization: OrganisationModel
17
16
  ) {
18
17
  this.id = id;
19
18
  this.name = name;
20
- this.hideDisabledFlags = hide_disabled_flags;
19
+ this.hideDisabledFlags = hideDisabledFlags;
21
20
  this.organisation = organization;
22
21
  }
23
22
  }
@@ -56,7 +56,7 @@ function traitsMatchSegmentRule(
56
56
  }
57
57
 
58
58
  function traitsMatchSegmentCondition(
59
- identity_traits: TraitModel[],
59
+ identityTraits: TraitModel[],
60
60
  condition: SegmentConditionModel,
61
61
  segmentId: number | string,
62
62
  identityId: number | string
@@ -65,7 +65,7 @@ function traitsMatchSegmentCondition(
65
65
  return getHashedPercentateForObjIds([segmentId, identityId]) <= parseFloat(condition.value);
66
66
  }
67
67
 
68
- const traits = identity_traits.filter(t => t.traitKey === condition.property_);
68
+ const traits = identityTraits.filter(t => t.traitKey === condition.property_);
69
69
  const trait = traits.length > 0 ? traits[0] : undefined;
70
70
 
71
71
  return trait ? condition.matchesTraitValue(trait.traitValue) : false;
@@ -0,0 +1 @@
1
+ export class FeatureStateNotFound extends Error {}
@@ -1,22 +1,10 @@
1
1
  import md5 from 'md5';
2
2
  import bigInt from 'big-integer';
3
3
 
4
- // def get_hashed_percentage_for_object_ids(
5
- // object_ids: typing.Iterable[typing.Any], iterations: int = 1
6
- // ) -> float:
7
- // """
8
- // Given a list of object ids, get a floating point number between 0 and 1 based on
9
- // the hash of those ids. This should give the same value every time for any
10
- // list of ids.
11
-
12
- // :param object_ids: list of object ids to calculate the has for
13
- // :param iterations: num times to include each id in the generated string to hash
14
- // :return: (float) number between 0 (inclusive) and 100 (exclusive)
15
- // """
16
-
17
4
  const makeRepeated = (arr: Array<any>, repeats: number) =>
18
5
  Array.from({ length: repeats }, () => arr).flat();
19
6
 
7
+ // https://stackoverflow.com/questions/12532871/how-to-convert-a-very-large-hex-number-to-decimal-in-javascript
20
8
  function h2d(s: any): string {
21
9
  function add(x: any, y: any) {
22
10
  var c = 0,
@@ -42,7 +30,14 @@ function h2d(s: any): string {
42
30
  });
43
31
  return dec;
44
32
  }
45
-
33
+ /**
34
+ * Given a list of object ids, get a floating point number between 0 and 1 based on
35
+ * the hash of those ids. This should give the same value every time for any list of ids.
36
+ *
37
+ * @param {Array<any>} objectIds list of object ids to calculate the has for
38
+ * @param {} iterations=1 num times to include each id in the generated string to hash
39
+ * @returns number number between 0 (inclusive) and 100 (exclusive)
40
+ */
46
41
  export function getHashedPercentateForObjIds(objectIds: Array<any>, iterations = 1): number {
47
42
  let to_hash = makeRepeated(objectIds, iterations).join(',');
48
43
  const hashedValue = md5(to_hash);
package/index.ts CHANGED
@@ -1,3 +1,6 @@
1
1
  import Flagsmith from './sdk';
2
2
 
3
- export default Flagsmith;
3
+ export { Flagsmith } from './sdk';
4
+ // export default Flagsmith;
5
+
6
+ module.exports = Flagsmith;
package/package.json CHANGED
@@ -1,24 +1,25 @@
1
1
  {
2
2
  "name": "flagsmith-nodejs",
3
- "version": "2.0.0-beta.1",
3
+ "version": "2.0.0-beta.4",
4
4
  "description": "Flagsmith lets you manage features flags and remote config across web, mobile and server side applications. Deliver true Continuous Integration. Get builds out faster. Control who has access to new features.",
5
5
  "main": "build/index.js",
6
6
  "repository": {
7
7
  "type": "git",
8
- "url": "https://github.com/SolidStateGroup/bullet-train-nodejs-client"
8
+ "url": "https://github.com/Flagsmith/flagsmith-nodejs-client"
9
9
  },
10
10
  "keywords": [
11
11
  "nodejs",
12
- "bullet train",
13
12
  "flagsmith",
14
- "feature flagger",
13
+ "feature flags",
14
+ "feature toggles",
15
+ "remote configuration",
15
16
  "continuous deployment"
16
17
  ],
17
18
  "bugs": {
18
- "url": "https://github.com/SolidStateGroup/bullet-train-nodejs-client/issues"
19
+ "url": "https://github.com/Flagsmith/flagsmith-nodejs-client/issues"
19
20
  },
20
- "homepage": "http://bullet-train.io",
21
- "author": "SSG",
21
+ "homepage": "http://flagsmith.com/",
22
+ "author": "Flagsmith",
22
23
  "contributors": [
23
24
  {
24
25
  "name": "Tom Stuart",
@@ -26,7 +27,7 @@
26
27
  },
27
28
  {
28
29
  "name": "Kyle Johnson",
29
- "email": "kyle@solidstategroup.com",
30
+ "email": "kyle.johnson@flagsmith.com",
30
31
  "url": "https://www.npmjs.com/~kyle-ssg"
31
32
  },
32
33
  {
@@ -42,6 +43,8 @@
42
43
  "scripts": {
43
44
  "lint": "prettier --write .",
44
45
  "test": "jest --coverage --coverageReporters='text'",
46
+ "test:watch": "jest --coverage --watch --coverageReporters='text'",
47
+ "test:debug": "node --inspect-brk node_modules/.bin/jest --coverage --watch --coverageReporters='text'",
45
48
  "build": "tsc",
46
49
  "prepare": "husky install"
47
50
  },
package/sdk/analytics.ts CHANGED
@@ -7,34 +7,6 @@ const ANALYTICS_TIMER = 10;
7
7
 
8
8
  const delay = (ms: number) => new Promise(resolve => setTimeout(() => resolve(undefined), ms));
9
9
 
10
- const retryFetch = (
11
- url: string,
12
- fetchOptions = {},
13
- retries = 3,
14
- retryDelay = 1000,
15
- timeout: number
16
- ) => {
17
- return new Promise((resolve, reject) => {
18
- // check for timeout
19
- if (timeout) setTimeout(() => reject('error: timeout'), timeout);
20
-
21
- const wrapper = (n: number) => {
22
- fetch(url, fetchOptions)
23
- .then(res => resolve(res))
24
- .catch(async err => {
25
- if (n > 0) {
26
- await delay(retryDelay);
27
- wrapper(--n);
28
- } else {
29
- reject(err);
30
- }
31
- });
32
- };
33
-
34
- wrapper(retries);
35
- });
36
- };
37
-
38
10
  export class AnalyticsProcessor {
39
11
  private analyticsEndpoint: string;
40
12
  private environmentKey: string;
package/sdk/index.ts CHANGED
@@ -10,6 +10,8 @@ import { FlagsmithAPIError, FlagsmithClientError } from './errors';
10
10
  import { DefaultFlag, Flags } from './models';
11
11
  import { EnvironmentDataPollingManager } from './polling_manager';
12
12
  import { generateIdentitiesData, retryFetch } from './utils';
13
+ import { SegmentModel } from '../flagsmith-engine/segments/models';
14
+ import { getIdentitySegments } from '../flagsmith-engine/segments/evaluators';
13
15
 
14
16
  const DEFAULT_API_URL = 'https://api.flagsmith.com/api/v1/';
15
17
 
@@ -20,7 +22,7 @@ export class Flagsmith {
20
22
  requestTimeoutSeconds?: number;
21
23
  enableLocalEvaluation?: boolean = false;
22
24
  environmentRefreshIntervalSeconds: number = 60;
23
- retries?: any;
25
+ retries?: number;
24
26
  enableAnalytics: boolean = false;
25
27
  defaultFlagHandler?: (featureName: string) => DefaultFlag;
26
28
 
@@ -29,7 +31,7 @@ export class Flagsmith {
29
31
  environmentUrl: string;
30
32
 
31
33
  environmentDataPollingManager?: EnvironmentDataPollingManager;
32
- environment?: EnvironmentModel;
34
+ environment!: EnvironmentModel;
33
35
  private analyticsProcessor?: AnalyticsProcessor;
34
36
  /**
35
37
  * A Flagsmith client.
@@ -68,7 +70,7 @@ export class Flagsmith {
68
70
  requestTimeoutSeconds?: number;
69
71
  enableLocalEvaluation?: boolean;
70
72
  environmentRefreshIntervalSeconds?: number;
71
- retries?: any;
73
+ retries?: number;
72
74
  enableAnalytics?: boolean;
73
75
  defaultFlagHandler?: (featureName: string) => DefaultFlag;
74
76
  }) {
@@ -88,11 +90,17 @@ export class Flagsmith {
88
90
  this.environmentUrl = `${this.apiUrl}environment-document/`;
89
91
 
90
92
  if (this.enableLocalEvaluation) {
93
+ if (!this.environmentKey.startsWith('ser.')) {
94
+ console.error(
95
+ 'In order to use local evaluation, please generate a server key in the environment settings page.'
96
+ );
97
+ }
91
98
  this.environmentDataPollingManager = new EnvironmentDataPollingManager(
92
99
  this,
93
100
  this.environmentRefreshIntervalSeconds
94
101
  );
95
102
  this.environmentDataPollingManager.start();
103
+ this.updateEnvironment();
96
104
  }
97
105
 
98
106
  this.analyticsProcessor = data.enableAnalytics
@@ -128,20 +136,67 @@ export class Flagsmith {
128
136
  */
129
137
  getIdentityFlags(identifier: string, traits?: { [key: string]: any }): Promise<Flags> {
130
138
  traits = traits || {};
131
- if (this.environment) {
139
+ if (this.enableLocalEvaluation) {
132
140
  return new Promise(resolve =>
133
- resolve(this.getIdentityFlagsFromDocument(identifier, traits || {}))
141
+ this.environmentPromise!.then(() => {
142
+ resolve(this.getIdentityFlagsFromDocument(identifier, traits || {}));
143
+ })
134
144
  );
135
145
  }
136
146
  return this.getIdentityFlagsFromApi(identifier, traits);
137
147
  }
138
148
 
149
+ /**
150
+ * Get the segments for the current environment for a given identity. Will also
151
+ upsert all traits to the Flagsmith API for future evaluations. Providing a
152
+ trait with a value of None will remove the trait from the identity if it exists.
153
+ *
154
+ * @param {string} identifier a unique identifier for the identity in the current
155
+ environment, e.g. email address, username, uuid
156
+ * @param {{[key:string]:any}} traits? a dictionary of traits to add / update on the identity in
157
+ Flagsmith, e.g. {"num_orders": 10}
158
+ * @returns Segments that the given identity belongs to.
159
+ */
160
+ getIdentitySegments(
161
+ identifier: string,
162
+ traits?: { [key: string]: any }
163
+ ): Promise<SegmentModel[]> {
164
+ traits = traits || {};
165
+ if (this.enableLocalEvaluation) {
166
+ return this.environmentPromise!.then(() => {
167
+ return new Promise(resolve => {
168
+ const identityModel = this.buildIdentityModel(
169
+ identifier,
170
+ Object.keys(traits || {}).map(key => ({
171
+ key,
172
+ value: traits?.[key]
173
+ }))
174
+ );
175
+
176
+ const segments = getIdentitySegments(this.environment, identityModel);
177
+ return resolve(segments);
178
+ });
179
+ });
180
+ }
181
+ console.error('This function is only permitted with local evaluation.');
182
+ return Promise.resolve([]);
183
+ }
184
+
139
185
  /**
140
186
  * Updates the environment state for local flag evaluation.
187
+ * Sets a local promise to prevent race conditions in getIdentityFlags / getIdentitySegments.
141
188
  * You only need to call this if you wish to bypass environmentRefreshIntervalSeconds.
142
189
  */
143
190
  async updateEnvironment() {
144
- this.environment = await this.getEnvironmentFromApi();
191
+ const request = this.getEnvironmentFromApi();
192
+ if (!this.environmentPromise) {
193
+ this.environmentPromise = request.then(res => {
194
+ this.environment = res;
195
+ });
196
+ await this.environmentPromise;
197
+ } else {
198
+ this.environment = await request;
199
+ }
145
200
  }
146
201
 
147
202
  private async getJSONResponse(
@@ -149,7 +204,7 @@ export class Flagsmith {
149
204
  method: string,
150
205
  body?: { [key: string]: any }
151
206
  ): Promise<any> {
152
- const headers: { [key: string]: any } = {};
207
+ const headers: { [key: string]: any } = { 'Content-Type': 'application/json' };
153
208
  if (this.environmentKey) {
154
209
  headers['X-Environment-Key'] = this.environmentKey as string;
155
210
  }
@@ -182,18 +237,17 @@ export class Flagsmith {
182
237
  return data.json();
183
238
  }
184
239
 
240
+ /**
241
+ * This promise ensures that the environment is retrieved before attempting to locally evaluate.
242
+ */
243
+ private environmentPromise: Promise<any> | undefined;
244
+
185
245
  private async getEnvironmentFromApi() {
186
246
  const environment_data = await this.getJSONResponse(this.environmentUrl, 'GET');
187
247
  return buildEnvironmentModel(environment_data);
188
248
  }
189
249
 
190
250
  private getEnvironmentFlagsFromDocument() {
191
- if (!this.environment) {
192
- throw new FlagsmithClientError(
193
- 'Unable to build identity model when no local environment present.'
194
- );
195
- }
196
-
197
251
  return Flags.fromFeatureStateModels({
198
252
  featureStates: getEnvironmentFeatureStates(this.environment),
199
253
  analyticsProcessor: this.analyticsProcessor,
@@ -202,18 +256,13 @@ export class Flagsmith {
202
256
  }
203
257
 
204
258
  private getIdentityFlagsFromDocument(identifier: string, traits: { [key: string]: any }) {
205
- if (!this.environment) {
206
- throw new FlagsmithClientError(
207
- 'Unable to build identity model when no local environment present.'
208
- );
209
- }
210
-
211
- const identityModel = this.buildIdentityModel(identifier, Object.keys(traits).map((key)=>(
212
- {
259
+ const identityModel = this.buildIdentityModel(
260
+ identifier,
261
+ Object.keys(traits).map(key => ({
213
262
  key,
214
263
  value: traits[key]
215
- }
216
- )));
264
+ }))
265
+ );
217
266
 
218
267
  const featureStates = getIdentityFeatureStates(this.environment, identityModel);
219
268
 
@@ -265,7 +314,7 @@ export class Flagsmith {
265
314
  }
266
315
  }
267
316
 
268
- private buildIdentityModel(identifier: string, traits: { key:string, value:any }[]) {
317
+ private buildIdentityModel(identifier: string, traits: { key: string; value: any }[]) {
269
318
  if (!this.environment) {
270
319
  throw new FlagsmithClientError(
271
320
  'Unable to build identity model when no local environment present.'
@@ -275,8 +324,6 @@ export class Flagsmith {
275
324
  const traitModels = traits.map(trait => new TraitModel(trait.key, trait.value));
276
325
  return new IdentityModel('0', traitModels, [], this.environment.apiKey, identifier);
277
326
  }
278
-
279
- stop() {}
280
327
  }
281
328
 
282
329
  export default Flagsmith;