@newrelic/browser-agent 0.1.230 → 0.1.231

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 (221) hide show
  1. package/README.md +25 -1
  2. package/dist/cjs/common/browser-version/ios-version.js +4 -3
  3. package/dist/cjs/common/config/state/configurable.js +1 -1
  4. package/dist/cjs/common/config/state/info.js +1 -1
  5. package/dist/cjs/common/config/state/init.js +1 -1
  6. package/dist/cjs/common/config/state/loader-config.js +1 -1
  7. package/dist/cjs/common/config/state/runtime.js +5 -5
  8. package/dist/cjs/common/constants/env.cdn.js +29 -0
  9. package/dist/cjs/common/constants/env.js +32 -0
  10. package/dist/cjs/common/constants/env.npm.js +30 -0
  11. package/dist/cjs/common/event-emitter/contextual-ee.test.js +282 -0
  12. package/dist/cjs/common/event-emitter/handle.test.js +58 -0
  13. package/dist/cjs/common/event-emitter/register-handler.test.js +55 -0
  14. package/dist/cjs/common/harvest/harvest.js +2 -2
  15. package/dist/cjs/common/ids/id.js +14 -6
  16. package/dist/cjs/common/ids/id.test.js +85 -0
  17. package/dist/cjs/common/ids/unique-id.js +75 -51
  18. package/dist/cjs/common/ids/unique-id.test.js +49 -0
  19. package/dist/cjs/common/timing/nav-timing.js +51 -30
  20. package/dist/cjs/common/timing/nav-timing.test.js +192 -0
  21. package/dist/cjs/common/url/clean-url.test.js +9 -0
  22. package/dist/cjs/common/url/encode.test.js +74 -0
  23. package/dist/cjs/common/url/location.js +4 -0
  24. package/dist/cjs/common/url/location.test.js +13 -0
  25. package/dist/cjs/common/url/parse-url.test.js +111 -0
  26. package/dist/cjs/common/url/protocol.js +2 -12
  27. package/dist/cjs/common/url/protocol.test.js +16 -0
  28. package/dist/cjs/common/util/console.js +1 -1
  29. package/dist/cjs/common/util/map-own.test.js +3 -3
  30. package/dist/cjs/common/util/obfuscate.js +1 -1
  31. package/dist/cjs/common/window/page-visibility.js +2 -1
  32. package/dist/cjs/common/wrap/index.js +0 -7
  33. package/dist/cjs/common/wrap/wrap-events.js +6 -9
  34. package/dist/cjs/common/wrap/wrap-fetch.js +6 -6
  35. package/dist/cjs/common/wrap/wrap-history.js +7 -6
  36. package/dist/cjs/common/wrap/wrap-jsonp.js +7 -6
  37. package/dist/cjs/common/wrap/wrap-mutation.js +7 -6
  38. package/dist/cjs/common/wrap/wrap-promise.js +7 -6
  39. package/dist/cjs/common/wrap/wrap-promise.test.js +119 -0
  40. package/dist/cjs/common/wrap/wrap-raf.js +6 -6
  41. package/dist/cjs/common/wrap/wrap-timer.js +6 -6
  42. package/dist/cjs/common/wrap/wrap-xhr.js +5 -6
  43. package/dist/cjs/features/ajax/aggregate/index.js +1 -1
  44. package/dist/cjs/features/jserrors/aggregate/compute-stack-trace.test.js +5 -5
  45. package/dist/cjs/features/jserrors/aggregate/format-stack-trace.test.js +1 -1
  46. package/dist/cjs/features/jserrors/aggregate/index.js +3 -3
  47. package/dist/cjs/features/jserrors/instrument/index.js +2 -2
  48. package/dist/cjs/features/metrics/aggregate/index.js +6 -7
  49. package/dist/cjs/features/metrics/instrument/index.js +0 -25
  50. package/dist/cjs/features/metrics/instrument/workers-helper.js +5 -5
  51. package/dist/cjs/features/page_action/aggregate/index.js +1 -1
  52. package/dist/cjs/features/page_view_event/aggregate/index.js +17 -6
  53. package/dist/cjs/features/page_view_timing/aggregate/index.js +36 -26
  54. package/dist/cjs/features/session_trace/aggregate/index.js +16 -13
  55. package/dist/cjs/features/utils/instrument-base.js +6 -2
  56. package/dist/cjs/features/utils/lazy-loader.js +1 -1
  57. package/dist/cjs/loaders/agent.js +1 -1
  58. package/dist/cjs/loaders/api/api.js +8 -5
  59. package/dist/cjs/loaders/features/enabled-features.js +1 -1
  60. package/dist/cjs/loaders/micro-agent.js +2 -1
  61. package/dist/esm/common/browser-version/ios-version.js +4 -3
  62. package/dist/esm/common/config/state/configurable.js +1 -1
  63. package/dist/esm/common/config/state/info.js +1 -1
  64. package/dist/esm/common/config/state/init.js +1 -1
  65. package/dist/esm/common/config/state/loader-config.js +1 -1
  66. package/dist/esm/common/config/state/runtime.js +2 -2
  67. package/dist/esm/common/constants/env.cdn.js +20 -0
  68. package/dist/esm/common/constants/env.js +23 -0
  69. package/dist/esm/common/constants/env.npm.js +21 -0
  70. package/dist/esm/common/event-emitter/contextual-ee.test.js +278 -0
  71. package/dist/esm/common/event-emitter/handle.test.js +54 -0
  72. package/dist/esm/common/event-emitter/register-handler.test.js +51 -0
  73. package/dist/esm/common/harvest/harvest.js +1 -1
  74. package/dist/esm/common/ids/id.js +16 -6
  75. package/dist/esm/common/ids/id.test.js +81 -0
  76. package/dist/esm/common/ids/unique-id.js +75 -51
  77. package/dist/esm/common/ids/unique-id.test.js +44 -0
  78. package/dist/esm/common/timing/nav-timing.js +51 -29
  79. package/dist/esm/common/timing/nav-timing.test.js +190 -0
  80. package/dist/esm/common/url/clean-url.test.js +7 -0
  81. package/dist/esm/common/url/encode.test.js +70 -0
  82. package/dist/esm/common/url/location.js +4 -0
  83. package/dist/esm/common/url/location.test.js +11 -0
  84. package/dist/esm/common/url/parse-url.test.js +107 -0
  85. package/dist/esm/common/url/protocol.js +3 -12
  86. package/dist/esm/common/url/protocol.test.js +14 -0
  87. package/dist/esm/common/util/console.js +1 -1
  88. package/dist/esm/common/util/map-own.test.js +3 -3
  89. package/dist/esm/common/util/obfuscate.js +2 -2
  90. package/dist/esm/common/window/page-visibility.js +2 -1
  91. package/dist/esm/common/wrap/index.js +1 -2
  92. package/dist/esm/common/wrap/wrap-events.js +6 -9
  93. package/dist/esm/common/wrap/wrap-fetch.js +6 -6
  94. package/dist/esm/common/wrap/wrap-history.js +7 -6
  95. package/dist/esm/common/wrap/wrap-jsonp.js +7 -6
  96. package/dist/esm/common/wrap/wrap-mutation.js +7 -6
  97. package/dist/esm/common/wrap/wrap-promise.js +7 -6
  98. package/dist/esm/common/wrap/wrap-promise.test.js +115 -0
  99. package/dist/esm/common/wrap/wrap-raf.js +6 -6
  100. package/dist/esm/common/wrap/wrap-timer.js +6 -6
  101. package/dist/esm/common/wrap/wrap-xhr.js +5 -6
  102. package/dist/esm/features/ajax/aggregate/index.js +1 -1
  103. package/dist/esm/features/jserrors/aggregate/compute-stack-trace.test.js +5 -5
  104. package/dist/esm/features/jserrors/aggregate/format-stack-trace.test.js +1 -1
  105. package/dist/esm/features/jserrors/aggregate/index.js +3 -3
  106. package/dist/esm/features/jserrors/instrument/index.js +2 -2
  107. package/dist/esm/features/metrics/aggregate/index.js +7 -8
  108. package/dist/esm/features/metrics/instrument/index.js +0 -25
  109. package/dist/esm/features/metrics/instrument/workers-helper.js +5 -5
  110. package/dist/esm/features/page_action/aggregate/index.js +1 -1
  111. package/dist/esm/features/page_view_event/aggregate/index.js +17 -6
  112. package/dist/esm/features/page_view_timing/aggregate/index.js +36 -26
  113. package/dist/esm/features/session_trace/aggregate/index.js +16 -13
  114. package/dist/esm/features/utils/instrument-base.js +1 -1
  115. package/dist/esm/features/utils/lazy-loader.js +1 -1
  116. package/dist/esm/loaders/agent.js +1 -1
  117. package/dist/esm/loaders/api/api.js +4 -4
  118. package/dist/esm/loaders/features/enabled-features.js +1 -1
  119. package/dist/types/common/config/state/runtime.d.ts.map +1 -1
  120. package/dist/types/common/constants/env.cdn.d.ts +18 -0
  121. package/dist/types/common/constants/env.cdn.d.ts.map +1 -0
  122. package/dist/types/common/constants/env.d.ts +13 -0
  123. package/dist/types/common/constants/env.d.ts.map +1 -0
  124. package/dist/types/common/constants/env.npm.d.ts +19 -0
  125. package/dist/types/common/constants/env.npm.d.ts.map +1 -0
  126. package/dist/types/common/ids/id.d.ts +11 -1
  127. package/dist/types/common/ids/id.d.ts.map +1 -1
  128. package/dist/types/common/ids/unique-id.d.ts +24 -1
  129. package/dist/types/common/ids/unique-id.d.ts.map +1 -1
  130. package/dist/types/common/timing/nav-timing.d.ts +1 -2
  131. package/dist/types/common/timing/nav-timing.d.ts.map +1 -1
  132. package/dist/types/common/unload/eol.d.ts.map +1 -1
  133. package/dist/types/common/url/location.d.ts +4 -0
  134. package/dist/types/common/url/location.d.ts.map +1 -1
  135. package/dist/types/common/url/parse-url.d.ts.map +1 -1
  136. package/dist/types/common/url/protocol.d.ts +1 -6
  137. package/dist/types/common/url/protocol.d.ts.map +1 -1
  138. package/dist/types/common/util/global-scope.d.ts.map +1 -1
  139. package/dist/types/common/wrap/index.d.ts +1 -2
  140. package/dist/types/common/wrap/index.d.ts.map +1 -1
  141. package/dist/types/common/wrap/wrap-fetch.d.ts.map +1 -1
  142. package/dist/types/common/wrap/wrap-history.d.ts.map +1 -1
  143. package/dist/types/common/wrap/wrap-jsonp.d.ts.map +1 -1
  144. package/dist/types/common/wrap/wrap-mutation.d.ts.map +1 -1
  145. package/dist/types/common/wrap/wrap-promise.d.ts.map +1 -1
  146. package/dist/types/common/wrap/wrap-raf.d.ts.map +1 -1
  147. package/dist/types/common/wrap/wrap-timer.d.ts.map +1 -1
  148. package/dist/types/common/wrap/wrap-xhr.d.ts.map +1 -1
  149. package/dist/types/features/jserrors/instrument/index.d.ts.map +1 -1
  150. package/dist/types/features/metrics/aggregate/index.d.ts.map +1 -1
  151. package/dist/types/features/metrics/instrument/index.d.ts +0 -1
  152. package/dist/types/features/metrics/instrument/index.d.ts.map +1 -1
  153. package/dist/types/features/metrics/instrument/workers-helper.d.ts.map +1 -1
  154. package/dist/types/features/page_view_event/aggregate/index.d.ts.map +1 -1
  155. package/dist/types/features/page_view_timing/aggregate/index.d.ts +1 -2
  156. package/dist/types/features/page_view_timing/aggregate/index.d.ts.map +1 -1
  157. package/dist/types/features/session_trace/aggregate/index.d.ts +1 -1
  158. package/dist/types/features/session_trace/aggregate/index.d.ts.map +1 -1
  159. package/dist/types/features/session_trace/instrument/index.d.ts.map +1 -1
  160. package/dist/types/features/spa/instrument/index.d.ts.map +1 -1
  161. package/package.json +9 -30
  162. package/src/common/browser-version/ios-version.js +4 -3
  163. package/src/common/config/state/runtime.js +26 -24
  164. package/src/common/constants/env.cdn.js +20 -0
  165. package/src/common/constants/env.js +23 -0
  166. package/src/common/constants/env.npm.js +21 -0
  167. package/src/common/event-emitter/contextual-ee.test.js +310 -0
  168. package/src/common/event-emitter/handle.test.js +56 -0
  169. package/src/common/event-emitter/register-handler.test.js +61 -0
  170. package/src/common/harvest/harvest.js +2 -2
  171. package/src/common/ids/id.js +15 -6
  172. package/src/common/ids/id.test.js +92 -0
  173. package/src/common/ids/unique-id.js +77 -54
  174. package/src/common/ids/unique-id.test.js +58 -0
  175. package/src/common/timing/nav-timing.js +50 -30
  176. package/src/common/timing/nav-timing.test.js +161 -0
  177. package/src/common/unload/eol.js +1 -2
  178. package/src/common/url/clean-url.test.js +25 -0
  179. package/src/common/url/encode.test.js +80 -0
  180. package/src/common/url/location.js +4 -0
  181. package/src/common/url/location.test.js +15 -0
  182. package/src/common/url/parse-url.js +1 -2
  183. package/src/common/url/parse-url.test.js +110 -0
  184. package/src/common/url/protocol.js +3 -13
  185. package/src/common/url/protocol.test.js +18 -0
  186. package/src/common/util/global-scope.js +1 -2
  187. package/src/common/util/obfuscate.js +2 -2
  188. package/src/common/window/page-visibility.js +1 -1
  189. package/src/common/wrap/index.js +1 -2
  190. package/src/common/wrap/wrap-events.js +5 -5
  191. package/src/common/wrap/wrap-fetch.js +4 -3
  192. package/src/common/wrap/wrap-history.js +6 -3
  193. package/src/common/wrap/wrap-jsonp.js +5 -3
  194. package/src/common/wrap/wrap-mutation.js +6 -3
  195. package/src/common/wrap/wrap-promise.js +7 -6
  196. package/src/common/wrap/wrap-promise.test.js +140 -0
  197. package/src/common/wrap/wrap-raf.js +5 -3
  198. package/src/common/wrap/wrap-timer.js +5 -3
  199. package/src/common/wrap/wrap-xhr.js +4 -3
  200. package/src/features/ajax/instrument/index.js +1 -1
  201. package/src/features/jserrors/instrument/index.js +4 -2
  202. package/src/features/metrics/aggregate/index.js +3 -4
  203. package/src/features/metrics/instrument/index.js +0 -30
  204. package/src/features/metrics/instrument/workers-helper.js +9 -6
  205. package/src/features/page_view_event/aggregate/index.js +15 -6
  206. package/src/features/page_view_timing/aggregate/index.js +36 -25
  207. package/src/features/page_view_timing/long-tasks.js +10 -10
  208. package/src/features/session_trace/aggregate/index.js +15 -12
  209. package/src/features/session_trace/instrument/index.js +3 -2
  210. package/src/features/spa/instrument/index.js +4 -2
  211. package/src/loaders/api/api.js +1 -1
  212. package/dist/cjs/common/constants/environment-variables.js +0 -20
  213. package/dist/cjs/common/wrap/wrap-console.js +0 -54
  214. package/dist/esm/common/constants/environment-variables.js +0 -11
  215. package/dist/esm/common/wrap/wrap-console.js +0 -46
  216. package/dist/types/common/constants/environment-variables.d.ts +0 -4
  217. package/dist/types/common/constants/environment-variables.d.ts.map +0 -1
  218. package/dist/types/common/wrap/wrap-console.d.ts +0 -16
  219. package/dist/types/common/wrap/wrap-console.d.ts.map +0 -1
  220. package/src/common/constants/environment-variables.js +0 -11
  221. package/src/common/wrap/wrap-console.js +0 -47
@@ -11,7 +11,7 @@ const model = {
11
11
  const _cache = {};
12
12
  export function getLoaderConfig(id) {
13
13
  if (!id) throw new Error('All loader-config objects require an agent identifier!');
14
- if (!_cache[id]) throw new Error(`LoaderConfig for ${id} was never set`);
14
+ if (!_cache[id]) throw new Error("LoaderConfig for ".concat(id, " was never set"));
15
15
  return _cache[id];
16
16
  }
17
17
  export function setLoaderConfig(id, obj) {
@@ -4,7 +4,7 @@ import { gosNREUMInitializedAgents } from '../../window/nreum';
4
4
  import { getCurrentSessionIdOrMakeNew } from '../../window/session-storage';
5
5
  import { getConfigurationValue } from '../config';
6
6
  import { globalScope } from '../../util/global-scope';
7
- import { BUILD_ENV, DIST_METHOD, VERSION } from '../../constants/environment-variables';
7
+ import { BUILD_ENV, DIST_METHOD, VERSION } from "../../constants/env.npm";
8
8
  const model = agentId => {
9
9
  return {
10
10
  buildEnv: BUILD_ENV,
@@ -32,7 +32,7 @@ const model = agentId => {
32
32
  const _cache = {};
33
33
  export function getRuntime(id) {
34
34
  if (!id) throw new Error('All runtime objects require an agent identifier!');
35
- if (!_cache[id]) throw new Error(`Runtime for ${id} was never set`);
35
+ if (!_cache[id]) throw new Error("Runtime for ".concat(id, " was never set"));
36
36
  return _cache[id];
37
37
  }
38
38
  export function setRuntime(id, obj) {
@@ -0,0 +1,20 @@
1
+ /**
2
+ * @file This file exposes CDN build environment variables. These variables will
3
+ * be overridden with babel.
4
+ */
5
+
6
+ /**
7
+ * Exposes the version of the agent
8
+ */
9
+ export const VERSION = "1.231.0";
10
+
11
+ /**
12
+ * Exposes the build type of the agent
13
+ * Valid values are LOCAL, PROD, DEV
14
+ */
15
+ export const BUILD_ENV = undefined;
16
+
17
+ /**
18
+ * Exposes the distribution method of the agent
19
+ */
20
+ export const DIST_METHOD = 'CDN';
@@ -0,0 +1,23 @@
1
+ /**
2
+ * @file This file exposes build time environment variables that will be included in the
3
+ * build output of the agent. This file specifically contains the normal environment variables
4
+ * for the NPM agent build and will be overridden by webpack/babel during the build based on the
5
+ * type of build being performed.
6
+ */
7
+
8
+ import pkgJSON from '../../../package.json';
9
+
10
+ /**
11
+ * Exposes the version of the agent
12
+ */
13
+ export const VERSION = pkgJSON.version;
14
+
15
+ /**
16
+ * Exposes the build type of the agent
17
+ */
18
+ export const BUILD_ENV = 'NPM';
19
+
20
+ /**
21
+ * Exposes the distribution method of the agent
22
+ */
23
+ export const DIST_METHOD = 'NPM';
@@ -0,0 +1,21 @@
1
+ /**
2
+ * @file This file exposes NPM build environment variables. These variables will
3
+ * be overridden with babel.
4
+ */
5
+
6
+ /**
7
+ * Exposes the version of the agent
8
+ */
9
+ export const VERSION = "1.231.0";
10
+
11
+ /**
12
+ * Exposes the build type of the agent
13
+ * Valid values are LOCAL, PROD, DEV
14
+ */
15
+ export const BUILD_ENV = 'NPM';
16
+
17
+ /**
18
+ * Exposes the distribution method of the agent
19
+ * Valid valuse are CDN, NPM
20
+ */
21
+ export const DIST_METHOD = 'NPM';
@@ -0,0 +1,278 @@
1
+ import { faker } from '@faker-js/faker';
2
+ let mockNREUM;
3
+ let runtime;
4
+ beforeEach(() => {
5
+ mockNREUM = {};
6
+ runtime = {};
7
+ jest.doMock('../window/nreum', () => ({
8
+ __esModule: true,
9
+ gosNREUM: jest.fn(() => mockNREUM)
10
+ }));
11
+ jest.doMock('../config/config', () => ({
12
+ __esModule: true,
13
+ getRuntime: jest.fn(() => runtime)
14
+ }));
15
+ });
16
+ afterEach(() => {
17
+ jest.resetModules();
18
+ jest.resetAllMocks();
19
+ });
20
+ describe('global event-emitter', () => {
21
+ test('it returns the event-emitter defined on window.NREUM', async () => {
22
+ const mockEE = {};
23
+ mockNREUM.ee = mockEE;
24
+ const {
25
+ ee
26
+ } = await import('./contextual-ee');
27
+ expect(ee).toEqual(mockEE);
28
+ });
29
+ test('it sets the global event-emitter on window.NREUM', async () => {
30
+ const {
31
+ ee
32
+ } = await import('./contextual-ee');
33
+ expect(ee).toEqual(mockNREUM.ee);
34
+ });
35
+ });
36
+ describe('scoping event-emitter', () => {
37
+ test('it creates a new child event-emitter', async () => {
38
+ const {
39
+ ee
40
+ } = await import('./contextual-ee');
41
+ const childName = faker.datatype.uuid();
42
+ const result = ee.get(childName);
43
+ expect(result).not.toEqual(mockNREUM.ee);
44
+ expect(result).toEqual(ee.get(childName)); // Should always return the same event-emitter
45
+ expect(result.debugId).toEqual(childName);
46
+ });
47
+ test('it creates a child event-emitter with an isolated backlog', async () => {
48
+ const childName = faker.random.alphaNumeric(16);
49
+ jest.doMock('../config/config', () => ({
50
+ __esModule: true,
51
+ getRuntime: jest.fn(id => {
52
+ if (id === childName) {
53
+ return {
54
+ isolatedBacklog: true
55
+ };
56
+ }
57
+ return runtime;
58
+ })
59
+ }));
60
+ const {
61
+ ee
62
+ } = await import('./contextual-ee');
63
+ const result = ee.get(childName);
64
+ expect(ee.backlog).not.toBe(result.backlog);
65
+ });
66
+ });
67
+ describe('event-emitter context', () => {
68
+ test('it returns a new context', async () => {
69
+ const {
70
+ ee
71
+ } = await import('./contextual-ee');
72
+ const result = ee.context();
73
+ expect(result).toEqual({});
74
+ });
75
+ test('it returns the same context', async () => {
76
+ const {
77
+ ee
78
+ } = await import('./contextual-ee');
79
+ const result = ee.context();
80
+ expect(result).toEqual(ee.context(result));
81
+ });
82
+ test('it adds the context to the provided object', async () => {
83
+ const {
84
+ ee
85
+ } = await import('./contextual-ee');
86
+ const obj = {};
87
+ const result = ee.context(obj);
88
+ expect(result).toEqual(obj['nr@context']);
89
+ });
90
+ });
91
+ describe('event-emitter buffer', () => {
92
+ it('it should create a new buffer for the given group', async () => {
93
+ const {
94
+ ee
95
+ } = await import('./contextual-ee');
96
+ const eventType = faker.datatype.uuid();
97
+ const group = faker.datatype.uuid();
98
+ ee.buffer([eventType], group);
99
+ expect(ee.backlog).toEqual(expect.objectContaining({
100
+ [group]: []
101
+ }));
102
+ expect(ee.isBuffering(eventType)).toEqual(true);
103
+ });
104
+ it('it should default group to "feature"', async () => {
105
+ const {
106
+ ee
107
+ } = await import('./contextual-ee');
108
+ const eventType = faker.datatype.uuid();
109
+ ee.buffer([eventType]);
110
+ expect(ee.backlog).toEqual(expect.objectContaining({
111
+ feature: []
112
+ }));
113
+ expect(ee.isBuffering(eventType)).toEqual(true);
114
+ });
115
+ it('it should not create buffer if event-emitter is aborted', async () => {
116
+ const {
117
+ ee
118
+ } = await import('./contextual-ee');
119
+ const eventType = faker.datatype.uuid();
120
+ const group = faker.datatype.uuid();
121
+ ee.backlog = {
122
+ api: ['foo', 'bar', 'baz']
123
+ };
124
+ ee.abort();
125
+ ee.buffer([eventType], group);
126
+ expect(ee.backlog).toEqual({});
127
+ expect(ee.isBuffering(eventType)).toEqual(false);
128
+ });
129
+ });
130
+ describe('event-emitter abort', () => {
131
+ test('it aborts if there is an API backlog', async () => {
132
+ const {
133
+ ee
134
+ } = await import('./contextual-ee');
135
+ ee.backlog = {
136
+ api: ['foo', 'bar', 'baz']
137
+ };
138
+ ee.abort();
139
+ expect(ee.aborted).toEqual(true);
140
+ expect(ee.backlog).toEqual({});
141
+ });
142
+ test('it aborts if there is a feature backlog', async () => {
143
+ const {
144
+ ee
145
+ } = await import('./contextual-ee');
146
+ ee.backlog = {
147
+ feature: ['foo', 'bar', 'baz']
148
+ };
149
+ ee.abort();
150
+ expect(ee.aborted).toEqual(true);
151
+ expect(ee.backlog).toEqual({});
152
+ });
153
+ });
154
+ describe('event-emitter emit', () => {
155
+ it('should execute the listener', async () => {
156
+ const {
157
+ ee
158
+ } = await import('./contextual-ee');
159
+ const mockListener = jest.fn();
160
+ const eventType = faker.datatype.uuid();
161
+ const eventArgs = ['a', 'b', 'c'];
162
+ ee.on(eventType, mockListener);
163
+ ee.emit(eventType, eventArgs);
164
+ expect(mockListener).toHaveBeenCalledWith(eventArgs[0], eventArgs[1], eventArgs[2]);
165
+ });
166
+ it('should not execute the listener after removal', async () => {
167
+ const {
168
+ ee
169
+ } = await import('./contextual-ee');
170
+ const mockListener = jest.fn();
171
+ const eventType = faker.datatype.uuid();
172
+ const eventArgs = ['a', 'b', 'c'];
173
+ ee.on(eventType, mockListener);
174
+ ee.emit(eventType, eventArgs);
175
+ ee.removeEventListener(eventType, mockListener);
176
+ ee.emit(eventType, eventArgs);
177
+ expect(mockListener).toHaveBeenCalledTimes(1);
178
+ });
179
+ it('should return early if global event-emitter is aborted', async () => {
180
+ const {
181
+ ee
182
+ } = await import('./contextual-ee');
183
+ const mockListener = jest.fn();
184
+ const eventType = faker.datatype.uuid();
185
+ const eventArgs = ['a', 'b', 'c'];
186
+ ee.backlog = {
187
+ api: ['foo', 'bar', 'baz']
188
+ };
189
+ ee.abort();
190
+ ee.on(eventType, mockListener);
191
+ ee.emit(eventType, eventArgs);
192
+ expect(mockListener).toHaveBeenCalledTimes(0);
193
+ });
194
+ it('should still emit if global event-emitter is aborted but force flag is true', async () => {
195
+ const {
196
+ ee
197
+ } = await import('./contextual-ee');
198
+ const scopeEE = ee.get(faker.datatype.uuid());
199
+ const mockScopeListener = jest.fn();
200
+ const eventType = faker.datatype.uuid();
201
+ const eventArgs = ['a', 'b', 'c'];
202
+ ee.backlog = {
203
+ api: ['foo', 'bar', 'baz']
204
+ };
205
+ ee.abort();
206
+ scopeEE.on(eventType, mockScopeListener);
207
+ scopeEE.emit(eventType, eventArgs, {}, true);
208
+ expect(mockScopeListener).toHaveBeenCalledTimes(1);
209
+ });
210
+ it('should bubble the event if bubble flag is true', async () => {
211
+ const {
212
+ ee
213
+ } = await import('./contextual-ee');
214
+ const scopeEE = ee.get(faker.datatype.uuid());
215
+ const mockListener = jest.fn();
216
+ const mockScopeListener = jest.fn();
217
+ const eventType = faker.datatype.uuid();
218
+ const eventArgs = ['a', 'b', 'c'];
219
+ ee.on(eventType, mockListener);
220
+ scopeEE.on(eventType, mockScopeListener);
221
+ scopeEE.emit(eventType, eventArgs, {}, false, true);
222
+ expect(mockScopeListener).toHaveBeenCalledTimes(1);
223
+ expect(mockListener).toHaveBeenCalledTimes(1);
224
+ });
225
+ it('should not bubble the event if bubble flag is false', async () => {
226
+ const {
227
+ ee
228
+ } = await import('./contextual-ee');
229
+ const scopeEE = ee.get(faker.datatype.uuid());
230
+ const mockListener = jest.fn();
231
+ const mockScopeListener = jest.fn();
232
+ const eventType = faker.datatype.uuid();
233
+ const eventArgs = ['a', 'b', 'c'];
234
+ ee.on(eventType, mockListener);
235
+ scopeEE.on(eventType, mockScopeListener);
236
+ scopeEE.emit(eventType, eventArgs, {}, false, false);
237
+ expect(mockScopeListener).toHaveBeenCalledTimes(1);
238
+ expect(mockListener).not.toHaveBeenCalled();
239
+ });
240
+ it('should buffer the event on the scoped event-emitter', async () => {
241
+ const {
242
+ ee
243
+ } = await import('./contextual-ee');
244
+ const scopeEE = ee.get(faker.datatype.uuid());
245
+ const mockListener = jest.fn();
246
+ const mockScopeListener = jest.fn();
247
+ const eventType = faker.datatype.uuid();
248
+ const eventArgs = ['a', 'b', 'c'];
249
+ ee.on(eventType, mockListener);
250
+ ee.buffer([eventType]);
251
+ scopeEE.on(eventType, mockScopeListener);
252
+ scopeEE.buffer([eventType]);
253
+ scopeEE.emit(eventType, eventArgs, {}, false, false);
254
+ expect(mockScopeListener).toHaveBeenCalledTimes(1);
255
+ expect(mockListener).not.toHaveBeenCalled();
256
+ expect(scopeEE.backlog['feature']).toEqual(expect.arrayContaining([expect.arrayContaining([scopeEE, eventType, eventArgs, {}])]));
257
+ expect(ee.backlog['feature']).toEqual(scopeEE.backlog['feature']);
258
+ });
259
+ });
260
+ describe('getOrSetContext', () => {
261
+ test('it returns a new context', async () => {
262
+ const {
263
+ getOrSetContext
264
+ } = await import('./contextual-ee');
265
+ const obj = {};
266
+ const result = getOrSetContext(obj);
267
+ expect(result).toEqual({});
268
+ expect(result).toEqual(obj['nr@context']);
269
+ });
270
+ test('it returns the same context', async () => {
271
+ const {
272
+ getOrSetContext
273
+ } = await import('./contextual-ee');
274
+ const obj = {};
275
+ const result = getOrSetContext(obj);
276
+ expect(getOrSetContext(obj)).toEqual(result);
277
+ });
278
+ });
@@ -0,0 +1,54 @@
1
+ import { faker } from '@faker-js/faker';
2
+ jest.mock('./contextual-ee', () => ({
3
+ __esModule: true,
4
+ ee: {
5
+ buffer: jest.fn(),
6
+ emit: jest.fn(),
7
+ get: jest.fn(() => ({
8
+ buffer: jest.fn(),
9
+ emit: jest.fn()
10
+ }))
11
+ }
12
+ }));
13
+ afterEach(() => {
14
+ jest.resetModules();
15
+ jest.resetAllMocks();
16
+ });
17
+ test('it should create and use a default event-emitter', async () => {
18
+ const {
19
+ ee
20
+ } = await import('./contextual-ee');
21
+ const {
22
+ handle
23
+ } = await import('./handle');
24
+ const handleEE = jest.mocked(ee.get).mock.results[0].value;
25
+ const eventType = faker.datatype.uuid();
26
+ const eventArgs = ['a', 'b', 'c'];
27
+ const eventContext = {};
28
+ const eventGroup = faker.datatype.uuid();
29
+ handle(eventType, eventArgs, eventContext, eventGroup);
30
+ expect(handleEE.buffer).toHaveBeenCalledWith([eventType], eventGroup);
31
+ expect(handleEE.emit).toHaveBeenCalledWith(eventType, eventArgs, eventContext);
32
+ });
33
+ test('it should use the provided scoped event-emitter', async () => {
34
+ const {
35
+ ee
36
+ } = await import('./contextual-ee');
37
+ const {
38
+ handle
39
+ } = await import('./handle');
40
+ const scopedEE = {
41
+ buffer: jest.fn(),
42
+ emit: jest.fn()
43
+ };
44
+ const handleEE = jest.mocked(ee.get).mock.results[0].value;
45
+ const eventType = faker.datatype.uuid();
46
+ const eventArgs = ['a', 'b', 'c'];
47
+ const eventContext = {};
48
+ const eventGroup = faker.datatype.uuid();
49
+ handle(eventType, eventArgs, eventContext, eventGroup, scopedEE);
50
+ expect(handleEE.buffer).not.toHaveBeenCalled();
51
+ expect(handleEE.emit).not.toHaveBeenCalled();
52
+ expect(scopedEE.buffer).toHaveBeenCalledWith([eventType], eventGroup);
53
+ expect(scopedEE.emit).toHaveBeenCalledWith(eventType, eventArgs, eventContext);
54
+ });
@@ -0,0 +1,51 @@
1
+ import { faker } from '@faker-js/faker';
2
+ jest.mock('./handle', () => ({
3
+ __esModule: true,
4
+ handleEE: {}
5
+ }));
6
+ afterEach(() => {
7
+ jest.resetModules();
8
+ jest.resetAllMocks();
9
+ });
10
+ test('should default group to "feature"', async () => {
11
+ const {
12
+ handleEE
13
+ } = await import('./handle');
14
+ const {
15
+ registerHandler
16
+ } = await import('./register-handler');
17
+ const eventType = faker.datatype.uuid();
18
+ const eventHandler = jest.fn();
19
+ registerHandler(eventType, eventHandler);
20
+ expect(registerHandler.handlers['feature']).toEqual(expect.objectContaining({
21
+ [eventType]: [expect.arrayContaining([handleEE, eventHandler])]
22
+ }));
23
+ });
24
+ test('should use the provided group', async () => {
25
+ const {
26
+ handleEE
27
+ } = await import('./handle');
28
+ const {
29
+ registerHandler
30
+ } = await import('./register-handler');
31
+ const eventType = faker.datatype.uuid();
32
+ const eventGroup = faker.datatype.uuid();
33
+ const eventHandler = jest.fn();
34
+ registerHandler(eventType, eventHandler, eventGroup);
35
+ expect(registerHandler.handlers[eventGroup]).toEqual(expect.objectContaining({
36
+ [eventType]: [expect.arrayContaining([handleEE, eventHandler])]
37
+ }));
38
+ });
39
+ test('should use the provided event-emitter', async () => {
40
+ const {
41
+ registerHandler
42
+ } = await import('./register-handler');
43
+ const scopedEE = {};
44
+ const eventType = faker.datatype.uuid();
45
+ const eventGroup = faker.datatype.uuid();
46
+ const eventHandler = jest.fn();
47
+ registerHandler(eventType, eventHandler, eventGroup, scopedEE);
48
+ expect(registerHandler.handlers[eventGroup]).toEqual(expect.objectContaining({
49
+ [eventType]: [expect.arrayContaining([scopedEE, eventHandler])]
50
+ }));
51
+ });
@@ -15,7 +15,7 @@ import { eventListenerOpts } from '../event-listener/event-listener-opts';
15
15
  import { Obfuscator } from '../util/obfuscate';
16
16
  import { applyFnToProps } from '../util/traverse';
17
17
  import { SharedContext } from '../context/shared-context';
18
- import { VERSION } from '../constants/environment-variables';
18
+ import { VERSION } from "../constants/env.npm";
19
19
  import { isBrowserScope, isWorkerScope } from '../util/global-scope';
20
20
  const haveSendBeacon = !!navigator.sendBeacon; // only the web window obj has sendBeacon at this time, so 'false' for other envs
21
21
 
@@ -3,16 +3,26 @@
3
3
  * SPDX-License-Identifier: Apache-2.0
4
4
  */
5
5
 
6
- // Start assigning ids at 1 so 0 can always be used for WindowOrWorkerGlobalScope, without
7
- // actually setting it (which would create a global variable).
8
6
  import { getOrSet } from '../util/get-or-set';
9
7
  import { globalScope } from '../util/global-scope';
10
- var index = 1;
11
- var prop = 'nr@id';
12
8
 
13
- // Always returns id of obj, may tag obj with an id in the process.
9
+ // Start assigning ids at 1 so 0 can always be used for Window or WorkerGlobalScope, without
10
+ // actually setting it (which would create a global variable).
11
+ let index = 1;
12
+ const prop = 'nr@id';
13
+
14
+ /**
15
+ * Tags a specified object with an identifier if it does not already
16
+ * have one. If the object is the global scope, zero will be returned
17
+ * and the object will not be modified. If the object already contains
18
+ * an identifier, it will be returned without modification. If the passed
19
+ * value is not an object, function, or array, -1 will be returned without
20
+ * modifying the passed value.
21
+ * @param {object|function|array} obj Object to be tagged with an identifier
22
+ * @returns {number} Identifier of the given object
23
+ */
14
24
  export function id(obj) {
15
- var type = typeof obj;
25
+ const type = typeof obj;
16
26
  // We can only tag objects, functions, and arrays with ids.
17
27
  // For all primitive values we instead return -1.
18
28
  if (!obj || !(type === 'object' || type === 'function')) return -1;
@@ -0,0 +1,81 @@
1
+ let idFn;
2
+ beforeEach(async () => {
3
+ idFn = (await import('./id')).id;
4
+ });
5
+ afterEach(() => {
6
+ jest.resetModules();
7
+ jest.clearAllMocks();
8
+ });
9
+ test.each([{
10
+ input: undefined,
11
+ expected: -1,
12
+ title: 'id of undefined is -1'
13
+ }, {
14
+ input: null,
15
+ expected: -1,
16
+ title: 'id of null is -1'
17
+ }, {
18
+ input: 2,
19
+ expected: -1,
20
+ title: 'id of number is -1'
21
+ }, {
22
+ input: 'foo',
23
+ expected: -1,
24
+ title: 'id of string is -1'
25
+ }])('$title', _ref => {
26
+ let {
27
+ input,
28
+ expected
29
+ } = _ref;
30
+ const result = idFn(input);
31
+ expect(typeof result).toEqual('number');
32
+ expect(result).toEqual(expected);
33
+ });
34
+ test('id values increment sequentially', () => {
35
+ const inputA = {};
36
+ const inputB = {};
37
+ const resultA = idFn(inputA);
38
+ const resultB = idFn(inputB);
39
+ expect(resultA - resultB).toEqual(-1);
40
+ });
41
+ test('id is correctly assigned to function type', () => {
42
+ const input = jest.fn();
43
+ const result = idFn(input);
44
+ expect(result).toEqual(1);
45
+ expect(input['nr@id']).toEqual(1);
46
+ });
47
+ test('id is correctly assigned to object type', () => {
48
+ const input = {};
49
+ const result = idFn(input);
50
+ expect(result).toEqual(1);
51
+ expect(input['nr@id']).toEqual(1);
52
+ });
53
+ test('id is the same when called twice on the same input', () => {
54
+ const input = {};
55
+ const result1 = idFn(input);
56
+ const result2 = idFn(input);
57
+ expect(result1).toEqual(1);
58
+ expect(result2).toEqual(1);
59
+ expect(input['nr@id']).toEqual(1);
60
+ });
61
+ test('id is zero on global scope', async () => {
62
+ const result = idFn(global);
63
+ expect(result).toEqual(0);
64
+ });
65
+ test('id on prototype is correctly inherited', () => {
66
+ const fn = jest.fn();
67
+ const a = {};
68
+ idFn(a);
69
+ fn.prototype = a;
70
+ const b = new fn();
71
+ expect(b['nr@id']).toEqual(a['nr@id']);
72
+ });
73
+ test('id on prototype is different from instance', () => {
74
+ const fn = jest.fn();
75
+ const a = {};
76
+ idFn(a);
77
+ fn.prototype = a;
78
+ const b = new fn();
79
+ const result = idFn(b);
80
+ expect(b['nr@id']).toEqual(result);
81
+ });