Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

prepare 4.3.0 release #73

Merged
merged 210 commits into from
Oct 17, 2022
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
210 commits
Select commit Hold shift + click to select a range
ad35601
initial move of code from js-client-sdk-private
eli-darkly Oct 30, 2019
abd39cd
changelog note
eli-darkly Oct 30, 2019
d64ad13
rm obsolete comment
eli-darkly Oct 30, 2019
05fa2ea
add npm audit helper
eli-darkly Oct 30, 2019
722b6f4
update babel, jest, rollup
eli-darkly Oct 30, 2019
83156f4
fix rollup config
eli-darkly Oct 30, 2019
c062853
fix ES build, dependency cleanup
eli-darkly Oct 30, 2019
5c0e2ac
add Releaser metadata
eli-darkly Oct 30, 2019
2a16af0
Update babel config to work in `test` without `useBuiltIns`
zmdavis Oct 30, 2019
e333469
Merge pull request #1 from launchdarkly/eb/update-dev-tools
eli-darkly Oct 31, 2019
f09a83a
Merge pull request #3 from launchdarkly/zdavis/babel-config
eli-darkly Oct 31, 2019
55c3852
copyedits
eli-darkly Oct 31, 2019
d361672
fix misnamed directory
eli-darkly Nov 1, 2019
f4eb20a
Merge pull request #2 from launchdarkly/eb/ch48393/initial
eli-darkly Nov 4, 2019
f072529
use spread operator instead of Object.assign
eli-darkly Nov 4, 2019
7c1299e
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Nov 4, 2019
957f477
Merge pull request #4 from launchdarkly/eb/ch54985/object-assign
eli-darkly Nov 4, 2019
a57c404
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Nov 4, 2019
1539659
add issue templates
eli-darkly Nov 4, 2019
013c273
add babel-eslint
eli-darkly Nov 4, 2019
db12eca
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Nov 4, 2019
087773b
merge from public after release
LaunchDarklyCI Nov 4, 2019
5bac01d
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
bwoskow-ld Nov 7, 2019
e145270
add event capacity config property
eli-darkly Dec 3, 2019
34ed8a5
re-add deprecation warning on samplingInterval
eli-darkly Dec 3, 2019
2ef59bc
Merge pull request #5 from launchdarkly/eb/ch57883/event-capacity
eli-darkly Dec 3, 2019
33d0311
Merge pull request #6 from launchdarkly/eb/ch38635/deprecation-comment
eli-darkly Dec 3, 2019
3c3e22a
better config validation
eli-darkly Dec 3, 2019
5cfb5a0
remove rollup-plugins-node-resolve
eli-darkly Dec 4, 2019
3c0794a
use newer Rollup node-resolve plugin
eli-darkly Dec 4, 2019
6385bb7
rm rollup-plugin-includepaths (unused)
eli-darkly Dec 4, 2019
c1b6f09
npm audit fix (handlebars dependency from jest)
eli-darkly Dec 4, 2019
11e46e1
comment
eli-darkly Dec 4, 2019
7cc15ba
copyedit
eli-darkly Dec 4, 2019
c44052b
Merge pull request #7 from launchdarkly/eb/ch57985/config-validation
eli-darkly Dec 4, 2019
9dbb6b2
Merge pull request #8 from launchdarkly/eb/ch57982/rollup-builtins
eli-darkly Dec 4, 2019
1cf982b
use new test helpers + misc test cleanup
eli-darkly Dec 5, 2019
9077edf
clean up stream testing logic
eli-darkly Dec 6, 2019
3f77514
fix hash parameter
eli-darkly Dec 6, 2019
d1fe985
Merge pull request #9 from launchdarkly/eb/ch57131/test-cleanup
eli-darkly Dec 7, 2019
d0de600
linter
eli-darkly Dec 7, 2019
b0168db
Merge pull request #10 from launchdarkly/eb/ch57131/stream-tests
eli-darkly Dec 7, 2019
b489fb2
clearer way to model the config option defaults/types
eli-darkly Dec 9, 2019
a32383c
test improvements
eli-darkly Dec 9, 2019
cdc1e96
change internal param name
eli-darkly Dec 9, 2019
2fcfa79
comment
eli-darkly Dec 9, 2019
88aa416
fix default logger logic
eli-darkly Dec 9, 2019
ceb9e3d
simpler way to enforce minimum values
eli-darkly Dec 10, 2019
58ee382
Merge pull request #12 from launchdarkly/eb/ch57985/config-validation…
eli-darkly Dec 10, 2019
50bc6aa
implement diagnostic events in common JS package (#11)
eli-darkly Dec 11, 2019
53a8590
add support for function type in config options
eli-darkly Dec 12, 2019
0a782ca
add support for function type in config options (#13)
eli-darkly Dec 12, 2019
04130f7
add wrapper metadata options and fix custom header logic
eli-darkly Dec 12, 2019
9491e16
lint
eli-darkly Dec 12, 2019
c9e7d7e
Merge branch 'eb/ch57741/wrapper-desc' into eb/ch57741/diagnostic-wra…
eli-darkly Dec 12, 2019
2610fc7
lint
eli-darkly Dec 12, 2019
fd7e2bb
remove image-loading logic from common code, replace it with an abstr…
eli-darkly Dec 12, 2019
0f28ba9
Merge branch 'eb/ch57741/wrapper-desc' into eb/ch59034/no-image-loading
eli-darkly Dec 12, 2019
f40f19b
Merge branch 'eb/ch59034/no-image-loading' into eb/ch57741/diagnostic…
eli-darkly Dec 12, 2019
5bdb140
Merge branch 'master' into diagnostic-events
eli-darkly Dec 12, 2019
fedb366
Merge branch 'eb/ch57741/diagnostic-wrapper' into diagnostic-events
eli-darkly Dec 12, 2019
385cbcd
add validation for options.streaming
eli-darkly Dec 12, 2019
2affe2f
Merge branch 'eb/ch57741/wrapper-desc' into eb/ch59034/no-image-loading
eli-darkly Dec 12, 2019
9e653e8
typo
eli-darkly Dec 12, 2019
c88839c
rm unused params
eli-darkly Dec 12, 2019
6f50448
merge from public after release
LaunchDarklyCI Dec 13, 2019
b4e0bfd
typo in comment
eli-darkly Dec 13, 2019
c5494b1
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Dec 13, 2019
6ef084a
Merge pull request #14 from launchdarkly/eb/ch57741/wrapper-desc
eli-darkly Dec 13, 2019
f3815bc
Merge pull request #15 from launchdarkly/eb/ch59034/no-image-loading
eli-darkly Dec 13, 2019
32d767f
misc fixes to merged code from external PR
eli-darkly Dec 13, 2019
7d4f67d
merge from public after release
LaunchDarklyCI Dec 13, 2019
3ba6e27
add event payload ID header
eli-darkly Jan 14, 2020
d9499df
Merge pull request #16 from launchdarkly/eb/ch61092/payload-id
eli-darkly Jan 15, 2020
a670c7f
merge from public after release
LaunchDarklyCI Jan 15, 2020
547b309
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Jan 30, 2020
15b5072
npm audit fix
eli-darkly Jan 30, 2020
6294b02
change exact dependencies to best-compatible
eli-darkly Jan 30, 2020
f190577
Merge pull request #17 from launchdarkly/eb/ch63559/relax-deps
eli-darkly Feb 1, 2020
7068968
merge from public after release
LaunchDarklyCI Feb 1, 2020
8c42690
standardize linting
eli-darkly Feb 11, 2020
f838fb3
disallow "window" and "document"
eli-darkly Feb 11, 2020
5064c3b
Merge branch 'master' into diagnostic-events
eli-darkly Feb 11, 2020
ade0261
Merge pull request #18 from launchdarkly/eb/ch65303/linting
eli-darkly Feb 11, 2020
8db5bc2
Merge branch 'diagnostic-events'
eli-darkly Feb 11, 2020
b4affad
improve diag event tests + debug logging
eli-darkly Feb 12, 2020
a01e65f
misc cleanup
eli-darkly Feb 12, 2020
7862b29
Merge pull request #19 from launchdarkly/eb/ch57741/diag-tests-fixes
eli-darkly Feb 12, 2020
4140b85
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Feb 12, 2020
20c655d
fix updating secure mode hash with identify()
eli-darkly Feb 13, 2020
8bf67c4
Merge pull request #20 from launchdarkly/eb/ch65728/identify-hash
eli-darkly Feb 13, 2020
ad44049
merge from public after release
LaunchDarklyCI Feb 13, 2020
232a404
don't omit streamInits.failed when it's false
eli-darkly Feb 14, 2020
bafa8da
Merge pull request #21 from launchdarkly/eb/ch65817/diag-explicit-false
eli-darkly Feb 14, 2020
ab182f0
merge from public after release
LaunchDarklyCI Feb 14, 2020
f954124
clean up init state logic, prevent unhandled rejections
eli-darkly Mar 5, 2020
78db776
lint
eli-darkly Mar 5, 2020
3136564
Merge pull request #22 from launchdarkly/eb/ch68119/init-state-error
eli-darkly Mar 6, 2020
bd3f238
merge from public after release
LaunchDarklyCI Mar 6, 2020
54099c8
less strict matching of json content-type header
eli-darkly Mar 18, 2020
e4c792c
Merge pull request #23 from launchdarkly/eb/ch69815/content-type
eli-darkly Mar 18, 2020
c610ffd
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Mar 18, 2020
5a323d8
merge from public after release
LaunchDarklyCI Mar 18, 2020
c974e2c
remove unsafe usage of hasOwnProperty
eli-darkly Mar 18, 2020
f5177c7
Merge pull request #24 from launchdarkly/eb/ch69855/has-own-property
eli-darkly Mar 18, 2020
93ee649
merge from public after release
LaunchDarklyCI Mar 18, 2020
e7ee64a
console logger must tolerate console object not always existing
eli-darkly Mar 30, 2020
c2402de
Merge pull request #25 from launchdarkly/eb/ch71507/no-console
eli-darkly Mar 31, 2020
68f359b
merge from public after release
LaunchDarklyCI Mar 31, 2020
a84b943
fix double initialization of diagnostics manager
eli-darkly May 1, 2020
14134c9
Merge pull request #26 from launchdarkly/eb/ch75557/extra-diagnostics
eli-darkly May 1, 2020
bdc9335
merge from public after release
LaunchDarklyCI May 1, 2020
5aac8a2
fix TypeScript declaration for track() and add more TS compilation te…
eli-darkly May 13, 2020
ca937d3
merge from public after release
LaunchDarklyCI May 13, 2020
fc3a857
remove startsWith usage (#28)
bwoskow-ld Jul 10, 2020
88125a7
merge from public after release
LaunchDarklyCI Jul 10, 2020
b572761
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Aug 5, 2020
1359aaf
prevent poll caused by a stream ping from overwriting later poll for …
eli-darkly Sep 12, 2020
b18f56f
merge from public after release
LaunchDarklyCI Sep 14, 2020
a6037a5
upgrade jest dependency and transitive yargs-parser dependency (#30)
bwoskow-ld Oct 15, 2020
bcb1573
Add null to LDEvaluationDetail.reason type (#31)
msiadak Oct 15, 2020
924213f
Merge branch 'contrib' of github.com:launchdarkly/js-sdk-common
bwoskow-ld Oct 15, 2020
88e42fb
Revert "Add null to LDEvaluationDetail.reason type (#31)"
bwoskow-ld Oct 16, 2020
6c26507
Revert "Add null to LDEvaluationDetail.reason type (#31)"
bwoskow-ld Oct 16, 2020
5125396
Merge branch 'contrib' of github.com:launchdarkly/js-sdk-common
bwoskow-ld Oct 16, 2020
b0f9c53
nullable evaluation reason (#32)
bwoskow-ld Nov 17, 2020
fdd97ee
merge from public after release
LaunchDarklyCI Nov 17, 2020
48bee4d
adding alias event functionality (#33)
bwoskow-ld Jan 20, 2021
7845b9a
set stream read timeout
eli-darkly Jan 26, 2021
f68c5b5
Merge pull request #35 from launchdarkly/eb/ch73764/stream-read-timeout
eli-darkly Jan 26, 2021
b36bb51
merge from public after release
LaunchDarklyCI Jan 26, 2021
efc11b0
Merge branch 'alias' of github.com:launchdarkly/js-sdk-common-private
bwoskow-ld Jan 27, 2021
b53732c
merge from public after release
LaunchDarklyCI Jan 27, 2021
a6ad9c8
Add prepare script (#34)
bwoskow-ld Jan 27, 2021
3911145
add a missing typescript verification (#36)
bwoskow-ld Jan 28, 2021
63c2ddd
Removed the guides link
bwoskow-ld Feb 3, 2021
fea10ae
Merge branch 'master' of github.com:launchdarkly/js-sdk-common into c…
bwoskow-ld Feb 9, 2021
4a5d31c
Correct doc link (#36)
sinchang Feb 9, 2021
72e9801
Merge branch 'contrib' of github.com:launchdarkly/js-sdk-common
bwoskow-ld Feb 9, 2021
0f83533
Fix typo in LDClient.on jsdoc (#37)
Doesntmeananything Mar 18, 2021
b493884
Merge branch 'contrib' of github.com:launchdarkly/js-sdk-common
bwoskow-ld Mar 18, 2021
94400df
add inlineUsersInEvents option in TypeScript (#37)
eli-darkly Apr 1, 2021
8d396dd
merge from public after release
LaunchDarklyCI Apr 1, 2021
4dfbc45
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
bwoskow-ld May 11, 2021
bb60a02
Filter private attributes on debug event users. Send variation for de…
gwhelanLD Jun 7, 2021
aef52df
Merge pull request #38 from launchdarkly/gw/ch110704/debug-private-attrs
gwhelanLD Jun 7, 2021
d7fc275
Merge remote-tracking branch 'public/master'
gwhelanLD Jun 8, 2021
717322a
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Aug 13, 2021
3b2ff6c
update uuid package version (#39)
eli-darkly Aug 23, 2021
c7b2645
merge from public after release
Aug 23, 2021
e05d244
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
bwoskow-ld Sep 21, 2021
8689021
use Releaser v2 config + newer CI image
eli-darkly Sep 21, 2021
75380ab
Merge pull request #40 from launchdarkly/eb/ch118712/releaser-v2
eli-darkly Sep 21, 2021
fe2bd87
First half, add the type, create the new options, add the new util me…
louis-launchdarkly Sep 24, 2021
d3cb835
Second half, call the tranform util method before calling any HTTP re…
louis-launchdarkly Sep 24, 2021
80f3098
Update the transform to work on a copy of headers instead of mutating it
louis-launchdarkly Sep 29, 2021
decda5f
add comments about removing custom event warning logic in the future
eli-darkly Oct 13, 2021
c0ab7c6
Merge pull request #42 from launchdarkly/eb/sc-124547/deprecate-goals…
eli-darkly Oct 13, 2021
f3b4a9a
revert updating of UUID dependency (#43)
eli-darkly Oct 15, 2021
98f9a86
Merge branch 'master' into lc/sc108033/add-lazily-evaluated-header
eli-darkly Oct 15, 2021
b508c75
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Oct 15, 2021
37af758
Merge branch 'master' into lc/sc108033/add-lazily-evaluated-header
eli-darkly Oct 15, 2021
b4294d8
Merge pull request #41 from launchdarkly/lc/sc108033/add-lazily-evalu…
louis-launchdarkly Oct 15, 2021
97c39ea
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Oct 15, 2021
455d381
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Oct 15, 2021
0444a31
merge from public after release
Oct 15, 2021
6edc3f6
better error handling for local storage operations (#44)
eli-darkly Jan 10, 2022
6fb8de6
add basic logger similar to server-side Node SDK (#45)
eli-darkly Jan 11, 2022
20bfe46
fix exports and add validation of custom logger (#46)
eli-darkly Jan 14, 2022
1f2076e
remove typedoc.js file that interferes with Releaser's docs build
eli-darkly Jan 14, 2022
52bc2af
update typescript version
eli-darkly Jan 14, 2022
00c0649
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Jan 15, 2022
9474648
add maintenance branch
eli-darkly Jan 15, 2022
c26d831
remove deprecated things (#48)
eli-darkly Jan 15, 2022
cf9ec23
fix EvaluationDetail.reason to be nullable so we can get rid of NonNu…
eli-darkly Jan 15, 2022
ca4468b
re-bump uuid package (#50)
eli-darkly Jan 15, 2022
0ef060e
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Jan 15, 2022
edf5c7a
use regular User-Agent header name unless overridden by js-client-sdk…
eli-darkly Jan 19, 2022
89c7a41
switch to publishing js-sdk-common as a regular Node module (#51)
eli-darkly Jan 20, 2022
59a9edc
fix CI
eli-darkly Jan 21, 2022
4e499a3
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Jan 21, 2022
02895b1
remove `version` constant which can't be exported from js-sdk-common …
eli-darkly Jan 26, 2022
90098a2
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Jan 26, 2022
7da82bb
Merge branch 'master' of github.com:launchdarkly/js-sdk-common
eli-darkly Jan 26, 2022
7b665ca
merge from public after release
Jan 26, 2022
d1b2875
catch errors in JSON parsing of stream data (#54)
eli-darkly Feb 17, 2022
df351a6
merge from public after release
Feb 17, 2022
7618644
backport sc-142333 fix
eli-darkly Feb 17, 2022
c2e5f5d
prepare 3.5.1 release (#63)
LaunchDarklyReleaseBot Feb 17, 2022
a736e46
Releasing version 3.5.1
Feb 17, 2022
b71603e
rm obsolete file to fix merge
eli-darkly Feb 17, 2022
a945a0c
Merge branch '3.x' of github.com:launchdarkly/js-sdk-common into 3.x
eli-darkly Feb 17, 2022
b6d90c7
Releasing version 3.5.1
Feb 17, 2022
ac38b52
merge from public after release
Feb 17, 2022
8173212
make URL path concatenation work right whether base URL has a trailin…
eli-darkly Apr 18, 2022
ea13ce5
Implement support for application tags. (#55)
kinyoklion Apr 20, 2022
3398942
Fix typing of LDOptionsBase. (#63)
kinyoklion Apr 21, 2022
c810e6f
Implement application tags for 3.x. (#62)
kinyoklion Apr 21, 2022
d3680b4
Merge branch '3.x'
eli-darkly Apr 21, 2022
088198c
lint
eli-darkly Apr 21, 2022
6f1feeb
merge from public after release
Apr 21, 2022
b58db4d
Add a line to refer to sendEventsOnlyForVariation
louis-launchdarkly Apr 22, 2022
3f01976
Merge pull request #65 from launchdarkly/lc/sc-151042/enhance-allFlag…
louis-launchdarkly Apr 22, 2022
784be57
don't include deleted flags in allFlags (#66)
eli-darkly Apr 29, 2022
075f8e7
Merge branch 'main' of github.com:launchdarkly/js-sdk-common
eli-darkly May 7, 2022
7915ede
Enforce 64 character limit for tag value. (#68)
kinyoklion May 16, 2022
0df97d6
merge from public after release
Jun 7, 2022
06d1457
Remove the last seen cache, deprecate allowFrequentDuplicateEvents. (…
kinyoklion Sep 28, 2022
26dd615
merge from public after release
Oct 3, 2022
c8685e8
Inspector proposal V2. (#71)
kinyoklion Oct 12, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
120 changes: 120 additions & 0 deletions src/InspectorManager.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,120 @@
const { messages } = require('.');
const SafeInspector = require('./SafeInspector');
const { onNextTick } = require('./utils');

/**
* The types of supported inspectors.
*/
const InspectorTypes = {
flagUsed: 'flag-used',
flagDetailsChanged: 'flag-details-changed',
flagDetailChanged: 'flag-detail-changed',
clientIdentityChanged: 'client-identity-changed',
};

Object.freeze(InspectorTypes);

/**
* Manages dispatching of inspection data to registered inspectors.
*/
function InspectorManager(inspectors, logger) {
const manager = {};

/**
* Collection of inspectors keyed by type.
* @type {{[type: string]: object[]}}
*/
const inspectorsByType = {
[InspectorTypes.flagUsed]: [],
[InspectorTypes.flagDetailsChanged]: [],
[InspectorTypes.flagDetailChanged]: [],
[InspectorTypes.clientIdentityChanged]: [],
};

const safeInspectors = inspectors?.map(inspector => SafeInspector(inspector, logger));

safeInspectors.forEach(safeInspector => {
// Only add inspectors of supported types.
if (Object.prototype.hasOwnProperty.call(inspectorsByType, safeInspector.type)) {
inspectorsByType[safeInspector.type].push(safeInspector);
} else {
logger.warn(messages.invalidInspector(safeInspector.type, safeInspector.name));
}
});

/**
* Check if there is an inspector of a specific type registered.
*
* @param {string} type The type of the inspector to check.
* @returns True if there are any inspectors of that type registered.
*/
manager.hasListeners = type => inspectorsByType[type]?.length;

/**
* Notify registered inspectors of a flag being used.
*
* The notification itself will be dispatched asynchronously.
*
* @param {string} flagKey The key for the flag.
* @param {Object} detail The LDEvaluationDetail for the flag.
* @param {Object} user The LDUser for the flag.
*/
manager.onFlagUsed = (flagKey, detail, user) => {
if (inspectorsByType[InspectorTypes.flagUsed].length) {
onNextTick(() => {
inspectorsByType[InspectorTypes.flagUsed].forEach(inspector => inspector.method(flagKey, detail, user));
});
}
};

/**
* Notify registered inspectors that the flags have been replaced.
*
* The notification itself will be dispatched asynchronously.
*
* @param {Record<string, Object>} flags The current flags as a Record<string, LDEvaluationDetail>.
*/
manager.onFlags = flags => {
if (inspectorsByType[InspectorTypes.flagDetailsChanged].length) {
onNextTick(() => {
inspectorsByType[InspectorTypes.flagDetailsChanged].forEach(inspector => inspector.method(flags));
});
}
};

/**
* Notify registered inspectors that a flag value has changed.
*
* The notification itself will be dispatched asynchronously.
*
* @param {string} flagKey The key for the flag that changed.
* @param {Object} flag An `LDEvaluationDetail` for the flag.
*/
manager.onFlagChanged = (flagKey, flag) => {
if (inspectorsByType[InspectorTypes.flagDetailChanged].length) {
onNextTick(() => {
console.log('what?');
inspectorsByType[InspectorTypes.flagDetailChanged].forEach(inspector => inspector.method(flagKey, flag));
});
}
};

/**
* Notify the registered inspectors that the user identity has changed.
*
* The notification itself will be dispatched asynchronously.
*
* @param {Object} user The `LDUser` which is now identified.
*/
manager.onIdentityChanged = user => {
if (inspectorsByType[InspectorTypes.clientIdentityChanged].length) {
onNextTick(() => {
inspectorsByType[InspectorTypes.clientIdentityChanged].forEach(inspector => inspector.method(user));
});
}
};

return manager;
}

module.exports = { InspectorTypes, InspectorManager };
34 changes: 34 additions & 0 deletions src/SafeInspector.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
const { messages } = require('.');

/**
* Wrap an inspector ensuring that calling its methods are safe.
* @param {object} inspector Inspector to wrap.
*/
function SafeInspector(inspector, logger) {
let errorLogged = false;
const wrapper = {
type: inspector.type,
name: inspector.name,
};

wrapper.method = (...args) => {
try {
inspector.method(...args);
} catch {
// If something goes wrong in an inspector we want to log that something
// went wrong. We don't want to flood the logs, so we only log something
// the first time that something goes wrong.
// We do not include the exception in the log, because we do not know what
// kind of data it may contain.
if (!errorLogged) {
errorLogged = true;
logger.warn(messages.inspectorMethodError(wrapper.type, wrapper.name));
}
// Prevent errors.
}
};

return wrapper;
}

module.exports = SafeInspector;
186 changes: 186 additions & 0 deletions src/__tests__/InspectorManager-test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,186 @@
const { AsyncQueue } = require('launchdarkly-js-test-helpers');
const { InspectorTypes, InspectorManager } = require('../InspectorManager');
const stubPlatform = require('./stubPlatform');

describe('given an inspector manager with no registered inspectors', () => {
const platform = stubPlatform.defaults();
const manager = InspectorManager([], platform.testing.logger);

it('does not cause errors', () => {
manager.onIdentityChanged({ key: 'key' });
manager.onFlagUsed(
'flag-key',
{
value: null,
},
{ key: 'key' }
);
manager.onFlags({});
manager.onFlagChanged('flag-key', { value: null });
});

it('does not report any registered listeners', () => {
expect(manager.hasListeners(InspectorTypes.clientIdentityChanged)).toBeFalsy();
expect(manager.hasListeners(InspectorTypes.flagDetailChanged)).toBeFalsy();
expect(manager.hasListeners(InspectorTypes.flagDetailsChanged)).toBeFalsy();
expect(manager.hasListeners(InspectorTypes.flagUsed)).toBeFalsy();
expect(manager.hasListeners('potato')).toBeFalsy();
});
});

describe('given an inspector with callbacks of every type', () => {
/**
* @type {AsyncQueue}
*/
const eventQueue = new AsyncQueue();
const platform = stubPlatform.defaults();
const manager = InspectorManager(
[
{
type: 'flag-used',
name: 'my-flag-used-inspector',
method: (flagKey, flagDetail, user) => {
eventQueue.add({ type: 'flag-used', flagKey, flagDetail, user });
},
},
// 'flag-used registered twice.
{
type: 'flag-used',
name: 'my-other-flag-used-inspector',
method: (flagKey, flagDetail, user) => {
eventQueue.add({ type: 'flag-used', flagKey, flagDetail, user });
},
},
{
type: 'flag-details-changed',
name: 'my-flag-details-inspector',
method: details => {
eventQueue.add({
type: 'flag-details-changed',
details,
});
},
},
{
type: 'flag-detail-changed',
name: 'my-flag-detail-inspector',
method: (flagKey, flagDetail) => {
eventQueue.add({
type: 'flag-detail-changed',
flagKey,
flagDetail,
});
},
},
{
type: 'client-identity-changed',
name: 'my-identity-inspector',
method: user => {
eventQueue.add({
type: 'client-identity-changed',
user,
});
},
},
// Invalid inspector shouldn't have an effect.
{
type: 'potato',
name: 'my-potato-inspector',
method: () => {},
},
],
platform.testing.logger
);

afterEach(() => {
expect(eventQueue.length()).toEqual(0);
});

afterAll(() => {
eventQueue.close();
});

it('logged that there was a bad inspector', () => {
expect(platform.testing.logger.output.warn).toEqual([
'an inspector: "my-potato-inspector" of an invalid type (potato) was configured',
]);
});

it('reports any registered listeners', () => {
expect(manager.hasListeners(InspectorTypes.clientIdentityChanged)).toBeTruthy();
expect(manager.hasListeners(InspectorTypes.flagDetailChanged)).toBeTruthy();
expect(manager.hasListeners(InspectorTypes.flagDetailsChanged)).toBeTruthy();
expect(manager.hasListeners(InspectorTypes.flagUsed)).toBeTruthy();
expect(manager.hasListeners('potato')).toBeFalsy();
});

it('executes `onFlagUsed` handlers', async () => {
manager.onFlagUsed(
'flag-key',
{
value: 'test',
variationIndex: 1,
reason: {
kind: 'OFF',
},
},
{ key: 'test-key' }
);

const expectedEvent = {
type: 'flag-used',
flagKey: 'flag-key',
flagDetail: {
value: 'test',
variationIndex: 1,
reason: {
kind: 'OFF',
},
},
user: { key: 'test-key' },
};
const event1 = await eventQueue.take();
expect(event1).toMatchObject(expectedEvent);

// There are two handlers, so there should be another event.
const event2 = await eventQueue.take();
expect(event2).toMatchObject(expectedEvent);
});

it('executes `onFlags` handler', async () => {
manager.onFlags({
example: { value: 'a-value' },
});

const event = await eventQueue.take();
expect(event).toMatchObject({
type: 'flag-details-changed',
details: {
example: { value: 'a-value' },
},
});
});

it('executes `onFlagChanged` handler', async () => {
manager.onFlagChanged('the-flag', { value: 'a-value' });

const event = await eventQueue.take();
expect(event).toMatchObject({
type: 'flag-detail-changed',
flagKey: 'the-flag',
flagDetail: {
value: 'a-value',
},
});
});

it('executes `onIdentityChanged` handler', async () => {
manager.onIdentityChanged({ key: 'the-key' });

const event = await eventQueue.take();
expect(event).toMatchObject({
type: 'client-identity-changed',
user: { key: 'the-key' },
});
});
});
Loading