@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
@@ -0,0 +1,310 @@
1
+ import { faker } from '@faker-js/faker'
2
+
3
+ let mockNREUM
4
+ let runtime
5
+
6
+ beforeEach(() => {
7
+ mockNREUM = {}
8
+ runtime = {}
9
+
10
+ jest.doMock('../window/nreum', () => ({
11
+ __esModule: true,
12
+ gosNREUM: jest.fn(() => mockNREUM)
13
+ }))
14
+
15
+ jest.doMock('../config/config', () => ({
16
+ __esModule: true,
17
+ getRuntime: jest.fn(() => runtime)
18
+ }))
19
+ })
20
+
21
+ afterEach(() => {
22
+ jest.resetModules()
23
+ jest.resetAllMocks()
24
+ })
25
+
26
+ describe('global event-emitter', () => {
27
+ test('it returns the event-emitter defined on window.NREUM', async () => {
28
+ const mockEE = {}
29
+ mockNREUM.ee = mockEE
30
+
31
+ const { ee } = await import('./contextual-ee')
32
+
33
+ expect(ee).toEqual(mockEE)
34
+ })
35
+
36
+ test('it sets the global event-emitter on window.NREUM', async () => {
37
+ const { ee } = await import('./contextual-ee')
38
+
39
+ expect(ee).toEqual(mockNREUM.ee)
40
+ })
41
+ })
42
+
43
+ describe('scoping event-emitter', () => {
44
+ test('it creates a new child event-emitter', async () => {
45
+ const { ee } = await import('./contextual-ee')
46
+
47
+ const childName = faker.datatype.uuid()
48
+ const result = ee.get(childName)
49
+
50
+ expect(result).not.toEqual(mockNREUM.ee)
51
+ expect(result).toEqual(ee.get(childName)) // Should always return the same event-emitter
52
+ expect(result.debugId).toEqual(childName)
53
+ })
54
+
55
+ test('it creates a child event-emitter with an isolated backlog', async () => {
56
+ const childName = faker.random.alphaNumeric(16)
57
+
58
+ jest.doMock('../config/config', () => ({
59
+ __esModule: true,
60
+ getRuntime: jest.fn(id => {
61
+ if (id === childName) {
62
+ return { isolatedBacklog: true }
63
+ }
64
+
65
+ return runtime
66
+ })
67
+ }))
68
+
69
+ const { ee } = await import('./contextual-ee')
70
+ const result = ee.get(childName)
71
+
72
+ expect(ee.backlog).not.toBe(result.backlog)
73
+ })
74
+ })
75
+
76
+ describe('event-emitter context', () => {
77
+ test('it returns a new context', async () => {
78
+ const { ee } = await import('./contextual-ee')
79
+
80
+ const result = ee.context()
81
+
82
+ expect(result).toEqual({})
83
+ })
84
+
85
+ test('it returns the same context', async () => {
86
+ const { ee } = await import('./contextual-ee')
87
+
88
+ const result = ee.context()
89
+
90
+ expect(result).toEqual(ee.context(result))
91
+ })
92
+
93
+ test('it adds the context to the provided object', async () => {
94
+ const { ee } = await import('./contextual-ee')
95
+
96
+ const obj = {}
97
+ const result = ee.context(obj)
98
+
99
+ expect(result).toEqual(obj['nr@context'])
100
+ })
101
+ })
102
+
103
+ describe('event-emitter buffer', () => {
104
+ it('it should create a new buffer for the given group', async () => {
105
+ const { ee } = await import('./contextual-ee')
106
+ const eventType = faker.datatype.uuid()
107
+ const group = faker.datatype.uuid()
108
+
109
+ ee.buffer([eventType], group)
110
+
111
+ expect(ee.backlog).toEqual(expect.objectContaining({
112
+ [group]: []
113
+ }))
114
+ expect(ee.isBuffering(eventType)).toEqual(true)
115
+ })
116
+
117
+ it('it should default group to "feature"', async () => {
118
+ const { ee } = await import('./contextual-ee')
119
+ const eventType = faker.datatype.uuid()
120
+
121
+ ee.buffer([eventType])
122
+
123
+ expect(ee.backlog).toEqual(expect.objectContaining({
124
+ feature: []
125
+ }))
126
+ expect(ee.isBuffering(eventType)).toEqual(true)
127
+ })
128
+
129
+ it('it should not create buffer if event-emitter is aborted', async () => {
130
+ const { ee } = await import('./contextual-ee')
131
+ const eventType = faker.datatype.uuid()
132
+ const group = faker.datatype.uuid()
133
+
134
+ ee.backlog = {
135
+ api: ['foo', 'bar', 'baz']
136
+ }
137
+ ee.abort()
138
+ ee.buffer([eventType], group)
139
+
140
+ expect(ee.backlog).toEqual({})
141
+ expect(ee.isBuffering(eventType)).toEqual(false)
142
+ })
143
+ })
144
+
145
+ describe('event-emitter abort', () => {
146
+ test('it aborts if there is an API backlog', async () => {
147
+ const { ee } = await import('./contextual-ee')
148
+
149
+ ee.backlog = {
150
+ api: ['foo', 'bar', 'baz']
151
+ }
152
+ ee.abort()
153
+
154
+ expect(ee.aborted).toEqual(true)
155
+ expect(ee.backlog).toEqual({})
156
+ })
157
+
158
+ test('it aborts if there is a feature backlog', async () => {
159
+ const { ee } = await import('./contextual-ee')
160
+
161
+ ee.backlog = {
162
+ feature: ['foo', 'bar', 'baz']
163
+ }
164
+ ee.abort()
165
+
166
+ expect(ee.aborted).toEqual(true)
167
+ expect(ee.backlog).toEqual({})
168
+ })
169
+ })
170
+
171
+ describe('event-emitter emit', () => {
172
+ it('should execute the listener', async () => {
173
+ const { ee } = await import('./contextual-ee')
174
+ const mockListener = jest.fn()
175
+ const eventType = faker.datatype.uuid()
176
+ const eventArgs = ['a', 'b', 'c']
177
+
178
+ ee.on(eventType, mockListener)
179
+ ee.emit(eventType, eventArgs)
180
+
181
+ expect(mockListener).toHaveBeenCalledWith(eventArgs[0], eventArgs[1], eventArgs[2])
182
+ })
183
+
184
+ it('should not execute the listener after removal', async () => {
185
+ const { ee } = await import('./contextual-ee')
186
+ const mockListener = jest.fn()
187
+ const eventType = faker.datatype.uuid()
188
+ const eventArgs = ['a', 'b', 'c']
189
+
190
+ ee.on(eventType, mockListener)
191
+ ee.emit(eventType, eventArgs)
192
+ ee.removeEventListener(eventType, mockListener)
193
+ ee.emit(eventType, eventArgs)
194
+
195
+ expect(mockListener).toHaveBeenCalledTimes(1)
196
+ })
197
+
198
+ it('should return early if global event-emitter is aborted', async () => {
199
+ const { ee } = await import('./contextual-ee')
200
+ const mockListener = jest.fn()
201
+ const eventType = faker.datatype.uuid()
202
+ const eventArgs = ['a', 'b', 'c']
203
+
204
+ ee.backlog = {
205
+ api: ['foo', 'bar', 'baz']
206
+ }
207
+ ee.abort()
208
+ ee.on(eventType, mockListener)
209
+ ee.emit(eventType, eventArgs)
210
+
211
+ expect(mockListener).toHaveBeenCalledTimes(0)
212
+ })
213
+
214
+ it('should still emit if global event-emitter is aborted but force flag is true', async () => {
215
+ const { ee } = await import('./contextual-ee')
216
+ const scopeEE = ee.get(faker.datatype.uuid())
217
+ const mockScopeListener = jest.fn()
218
+ const eventType = faker.datatype.uuid()
219
+ const eventArgs = ['a', 'b', 'c']
220
+
221
+ ee.backlog = {
222
+ api: ['foo', 'bar', 'baz']
223
+ }
224
+ ee.abort()
225
+ scopeEE.on(eventType, mockScopeListener)
226
+ scopeEE.emit(eventType, eventArgs, {}, true)
227
+
228
+ expect(mockScopeListener).toHaveBeenCalledTimes(1)
229
+ })
230
+
231
+ it('should bubble the event if bubble flag is true', async () => {
232
+ const { ee } = await import('./contextual-ee')
233
+ const scopeEE = ee.get(faker.datatype.uuid())
234
+ const mockListener = jest.fn()
235
+ const mockScopeListener = jest.fn()
236
+ const eventType = faker.datatype.uuid()
237
+ const eventArgs = ['a', 'b', 'c']
238
+
239
+ ee.on(eventType, mockListener)
240
+ scopeEE.on(eventType, mockScopeListener)
241
+ scopeEE.emit(eventType, eventArgs, {}, false, true)
242
+
243
+ expect(mockScopeListener).toHaveBeenCalledTimes(1)
244
+ expect(mockListener).toHaveBeenCalledTimes(1)
245
+ })
246
+
247
+ it('should not bubble the event if bubble flag is false', async () => {
248
+ const { ee } = await import('./contextual-ee')
249
+ const scopeEE = ee.get(faker.datatype.uuid())
250
+ const mockListener = jest.fn()
251
+ const mockScopeListener = jest.fn()
252
+ const eventType = faker.datatype.uuid()
253
+ const eventArgs = ['a', 'b', 'c']
254
+
255
+ ee.on(eventType, mockListener)
256
+ scopeEE.on(eventType, mockScopeListener)
257
+ scopeEE.emit(eventType, eventArgs, {}, false, false)
258
+
259
+ expect(mockScopeListener).toHaveBeenCalledTimes(1)
260
+ expect(mockListener).not.toHaveBeenCalled()
261
+ })
262
+
263
+ it('should buffer the event on the scoped event-emitter', async () => {
264
+ const { ee } = await import('./contextual-ee')
265
+ const scopeEE = ee.get(faker.datatype.uuid())
266
+ const mockListener = jest.fn()
267
+ const mockScopeListener = jest.fn()
268
+ const eventType = faker.datatype.uuid()
269
+ const eventArgs = ['a', 'b', 'c']
270
+
271
+ ee.on(eventType, mockListener)
272
+ ee.buffer([eventType])
273
+ scopeEE.on(eventType, mockScopeListener)
274
+ scopeEE.buffer([eventType])
275
+ scopeEE.emit(eventType, eventArgs, {}, false, false)
276
+
277
+ expect(mockScopeListener).toHaveBeenCalledTimes(1)
278
+ expect(mockListener).not.toHaveBeenCalled()
279
+ expect(scopeEE.backlog['feature']).toEqual(expect.arrayContaining([
280
+ expect.arrayContaining([
281
+ scopeEE,
282
+ eventType,
283
+ eventArgs,
284
+ {}
285
+ ])
286
+ ]))
287
+ expect(ee.backlog['feature']).toEqual(scopeEE.backlog['feature'])
288
+ })
289
+ })
290
+
291
+ describe('getOrSetContext', () => {
292
+ test('it returns a new context', async () => {
293
+ const { getOrSetContext } = await import('./contextual-ee')
294
+
295
+ const obj = {}
296
+ const result = getOrSetContext(obj)
297
+
298
+ expect(result).toEqual({})
299
+ expect(result).toEqual(obj['nr@context'])
300
+ })
301
+
302
+ test('it returns the same context', async () => {
303
+ const { getOrSetContext } = await import('./contextual-ee')
304
+
305
+ const obj = {}
306
+ const result = getOrSetContext(obj)
307
+
308
+ expect(getOrSetContext(obj)).toEqual(result)
309
+ })
310
+ })
@@ -0,0 +1,56 @@
1
+ import { faker } from '@faker-js/faker'
2
+
3
+ jest.mock('./contextual-ee', () => ({
4
+ __esModule: true,
5
+ ee: {
6
+ buffer: jest.fn(),
7
+ emit: jest.fn(),
8
+ get: jest.fn(() => ({
9
+ buffer: jest.fn(),
10
+ emit: jest.fn()
11
+ }))
12
+ }
13
+ }))
14
+
15
+ afterEach(() => {
16
+ jest.resetModules()
17
+ jest.resetAllMocks()
18
+ })
19
+
20
+ test('it should create and use a default event-emitter', async () => {
21
+ const { ee } = await import('./contextual-ee')
22
+ const { handle } = await import('./handle')
23
+
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
+
30
+ handle(eventType, eventArgs, eventContext, eventGroup)
31
+
32
+ expect(handleEE.buffer).toHaveBeenCalledWith([eventType], eventGroup)
33
+ expect(handleEE.emit).toHaveBeenCalledWith(eventType, eventArgs, eventContext)
34
+ })
35
+
36
+ test('it should use the provided scoped event-emitter', async () => {
37
+ const { ee } = await import('./contextual-ee')
38
+ const { handle } = await import('./handle')
39
+ const scopedEE = {
40
+ buffer: jest.fn(),
41
+ emit: jest.fn()
42
+ }
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
+
50
+ handle(eventType, eventArgs, eventContext, eventGroup, scopedEE)
51
+
52
+ expect(handleEE.buffer).not.toHaveBeenCalled()
53
+ expect(handleEE.emit).not.toHaveBeenCalled()
54
+ expect(scopedEE.buffer).toHaveBeenCalledWith([eventType], eventGroup)
55
+ expect(scopedEE.emit).toHaveBeenCalledWith(eventType, eventArgs, eventContext)
56
+ })
@@ -0,0 +1,61 @@
1
+ import { faker } from '@faker-js/faker'
2
+
3
+ jest.mock('./handle', () => ({
4
+ __esModule: true,
5
+ handleEE: {}
6
+ }))
7
+
8
+ afterEach(() => {
9
+ jest.resetModules()
10
+ jest.resetAllMocks()
11
+ })
12
+
13
+ test('should default group to "feature"', async () => {
14
+ const { handleEE } = await import('./handle')
15
+ const { registerHandler } = await import('./register-handler')
16
+
17
+ const eventType = faker.datatype.uuid()
18
+ const eventHandler = jest.fn()
19
+
20
+ registerHandler(eventType, eventHandler)
21
+
22
+ expect(registerHandler.handlers['feature']).toEqual(expect.objectContaining({
23
+ [eventType]: [expect.arrayContaining([
24
+ handleEE, eventHandler
25
+ ])]
26
+ }))
27
+ })
28
+
29
+ test('should use the provided group', async () => {
30
+ const { handleEE } = await import('./handle')
31
+ const { registerHandler } = await import('./register-handler')
32
+
33
+ const eventType = faker.datatype.uuid()
34
+ const eventGroup = faker.datatype.uuid()
35
+ const eventHandler = jest.fn()
36
+
37
+ registerHandler(eventType, eventHandler, eventGroup)
38
+
39
+ expect(registerHandler.handlers[eventGroup]).toEqual(expect.objectContaining({
40
+ [eventType]: [expect.arrayContaining([
41
+ handleEE, eventHandler
42
+ ])]
43
+ }))
44
+ })
45
+
46
+ test('should use the provided event-emitter', async () => {
47
+ const { registerHandler } = await import('./register-handler')
48
+
49
+ const scopedEE = {}
50
+ const eventType = faker.datatype.uuid()
51
+ const eventGroup = faker.datatype.uuid()
52
+ const eventHandler = jest.fn()
53
+
54
+ registerHandler(eventType, eventHandler, eventGroup, scopedEE)
55
+
56
+ expect(registerHandler.handlers[eventGroup]).toEqual(expect.objectContaining({
57
+ [eventType]: [expect.arrayContaining([
58
+ scopedEE, eventHandler
59
+ ])]
60
+ }))
61
+ })
@@ -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'
19
19
  import { isBrowserScope, isWorkerScope } from '../util/global-scope'
20
20
 
21
21
  const haveSendBeacon = !!navigator.sendBeacon // only the web window obj has sendBeacon at this time, so 'false' for other envs
@@ -176,7 +176,7 @@ export class Harvest extends SharedContext {
176
176
  createPayload (type, options) {
177
177
  var makeBody = createAccumulator()
178
178
  var makeQueryString = createAccumulator()
179
- var listeners = (this._events[type] && this._events[type] || [])
179
+ var listeners = ((this._events[type] && this._events[type]) || [])
180
180
 
181
181
  for (var i = 0; i < listeners.length; i++) {
182
182
  var singlePayload = listeners[i](options)
@@ -3,17 +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
8
 
11
- var index = 1
12
- var prop = 'nr@id'
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
13
 
14
- // Always returns id of obj, may tag obj with an id in the process.
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
+ */
15
24
  export function id (obj) {
16
- var type = typeof obj
25
+ const type = typeof obj
17
26
  // We can only tag objects, functions, and arrays with ids.
18
27
  // For all primitive values we instead return -1.
19
28
  if (!obj || !(type === 'object' || type === 'function')) return -1
@@ -0,0 +1,92 @@
1
+ let idFn
2
+
3
+ beforeEach(async () => {
4
+ idFn = (await import('./id')).id
5
+ })
6
+
7
+ afterEach(() => {
8
+ jest.resetModules()
9
+ jest.clearAllMocks()
10
+ })
11
+
12
+ test.each([
13
+ { input: undefined, expected: -1, title: 'id of undefined is -1' },
14
+ { input: null, expected: -1, title: 'id of null is -1' },
15
+ { input: 2, expected: -1, title: 'id of number is -1' },
16
+ { input: 'foo', expected: -1, title: 'id of string is -1' }
17
+ ])('$title', ({ input, expected }) => {
18
+ const result = idFn(input)
19
+
20
+ expect(typeof result).toEqual('number')
21
+ expect(result).toEqual(expected)
22
+ })
23
+
24
+ test('id values increment sequentially', () => {
25
+ const inputA = {}
26
+ const inputB = {}
27
+
28
+ const resultA = idFn(inputA)
29
+ const resultB = idFn(inputB)
30
+
31
+ expect(resultA - resultB).toEqual(-1)
32
+ })
33
+
34
+ test('id is correctly assigned to function type', () => {
35
+ const input = jest.fn()
36
+
37
+ const result = idFn(input)
38
+
39
+ expect(result).toEqual(1)
40
+ expect(input['nr@id']).toEqual(1)
41
+ })
42
+
43
+ test('id is correctly assigned to object type', () => {
44
+ const input = {}
45
+
46
+ const result = idFn(input)
47
+
48
+ expect(result).toEqual(1)
49
+ expect(input['nr@id']).toEqual(1)
50
+ })
51
+
52
+ test('id is the same when called twice on the same input', () => {
53
+ const input = {}
54
+
55
+ const result1 = idFn(input)
56
+ const result2 = idFn(input)
57
+
58
+ expect(result1).toEqual(1)
59
+ expect(result2).toEqual(1)
60
+ expect(input['nr@id']).toEqual(1)
61
+ })
62
+
63
+ test('id is zero on global scope', async () => {
64
+ const result = idFn(global)
65
+
66
+ expect(result).toEqual(0)
67
+ })
68
+
69
+ test('id on prototype is correctly inherited', () => {
70
+ const fn = jest.fn()
71
+ const a = {}
72
+
73
+ idFn(a)
74
+ fn.prototype = a
75
+
76
+ const b = new fn()
77
+
78
+ expect(b['nr@id']).toEqual(a['nr@id'])
79
+ })
80
+
81
+ test('id on prototype is different from instance', () => {
82
+ const fn = jest.fn()
83
+ const a = {}
84
+
85
+ idFn(a)
86
+ fn.prototype = a
87
+
88
+ const b = new fn()
89
+ const result = idFn(b)
90
+
91
+ expect(b['nr@id']).toEqual(result)
92
+ })