@dynatrace/rum-javascript-sdk 1.331.18 → 1.333.2

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.
@@ -1,4 +1,5 @@
1
1
  import { expect, test as base } from "@playwright/test";
2
+ import { DEFAULT_IGNORED_EVENTS, DEFAULT_IGNORED_FIELDS, DEFAULT_REMOVED_FIELDS, filterIgnoredEvents, processEventProperties, readSnapshot, sortEventsByCharacteristics, writeSnapshot } from "./snapshot.js";
2
3
  import { fetchRumJavaScript } from "./install.js";
3
4
  import { uncompress } from "snappyjs";
4
5
  const DEFAULT_TIMEOUT = 5_000;
@@ -22,12 +23,31 @@ export const test = base.extend({
22
23
  const cache = new Map();
23
24
  await use(cache);
24
25
  }, { scope: "worker", auto: true, box: true }],
25
- dynatraceTesting: async ({ page, dynatraceConfig, rumJavaScriptCache }, use) => {
26
+ dynatraceTesting: async ({ page, dynatraceConfig, rumJavaScriptCache }, use, testInfo) => {
26
27
  validateDynatraceConfig(dynatraceConfig);
27
- // Check if page has already navigated and warn about fixture ordering
28
- const currentUrl = page.url();
29
- if (currentUrl !== "about:blank") {
30
- console.warn(`[Dynatrace Testing] Warning: Page has already navigated to "${currentUrl}".
28
+ warnIfPageAlreadyNavigated(page);
29
+ const { ignoreWarnings = [], dumpEventsOnFail = false } = dynatraceConfig;
30
+ const context = {
31
+ events: [],
32
+ beacons: [],
33
+ warnings: [],
34
+ dumpEventsOnFail
35
+ };
36
+ setupConsoleWarningListener(page, ignoreWarnings, context);
37
+ await setupRumJavaScriptAndListeners(page, dynatraceConfig, rumJavaScriptCache, context);
38
+ await use(createDynatraceTestingApi(context, testInfo));
39
+ }
40
+ });
41
+ /**
42
+ * Warns if the page has already navigated away from about:blank.
43
+ * This indicates the fixture may not inject RUM JavaScript correctly.
44
+ *
45
+ * @param page The Playwright page instance
46
+ */
47
+ function warnIfPageAlreadyNavigated(page) {
48
+ const currentUrl = page.url();
49
+ if (currentUrl !== "about:blank") {
50
+ console.warn(`[Dynatrace Testing] Warning: Page has already navigated to "${currentUrl}".
31
51
  The RUM JavaScript may not be injected. Ensure dynatraceTesting is destructured
32
52
  before any fixtures that navigate the page.
33
53
 
@@ -35,178 +55,235 @@ Example of correct fixture order:
35
55
  test("my test", async ({ dynatraceTesting, myCustomFixture, page }) => { ... });
36
56
 
37
57
  See testing.md for more information.`);
58
+ }
59
+ }
60
+ /**
61
+ * Sets up a console listener to capture Dynatrace-related warnings.
62
+ *
63
+ * @param page The Playwright page instance
64
+ * @param ignoreWarnings Array of regex patterns for warnings to ignore
65
+ * @param context The testing context to store warnings
66
+ */
67
+ function setupConsoleWarningListener(page, ignoreWarnings, context) {
68
+ const ignoreRegexes = ignoreWarnings.map(x => new RegExp(x, "i"));
69
+ page.on("console", (msg) => {
70
+ if (msg.text().includes("ynatrace") && !ignoreRegexes.some(regex => regex.test(msg.text()))) {
71
+ context.warnings.push(msg.text());
72
+ }
73
+ });
74
+ }
75
+ /**
76
+ * Sets up RUM JavaScript injection and beacon request listeners.
77
+ *
78
+ * @param page The Playwright page instance
79
+ * @param config The Dynatrace configuration
80
+ * @param cache The worker-scoped cache for RUM JavaScript
81
+ * @param context The testing context to store beacons and events
82
+ */
83
+ async function setupRumJavaScriptAndListeners(page, config, cache, context) {
84
+ const { endpointUrl, appId, token } = config;
85
+ const cacheKey = `${endpointUrl}:${appId}`;
86
+ let rum = cache.get(cacheKey);
87
+ if (!rum) {
88
+ rum = await fetchRumJavaScriptContent(endpointUrl, appId, token);
89
+ const beaconUri = extractBeaconUri(rum);
90
+ cache.set(cacheKey, rum);
91
+ cache.set(`${cacheKey}:beaconUri`, beaconUri);
92
+ }
93
+ else {
94
+ const beaconUri = cache.get(`${cacheKey}:beaconUri`);
95
+ if (!beaconUri) {
96
+ throw new Error("Failed to retrieve beaconUri from cache.");
38
97
  }
39
- const { endpointUrl, appId, token, ignoreWarnings = [], dumpEventsOnFail = false } = dynatraceConfig;
40
- const beacons = [];
41
- const events = [];
42
- const warnings = [];
43
- const ignoreRegexes = ignoreWarnings.map(x => new RegExp(x, "i"));
44
- page.on("console", (msg) => {
45
- if (msg.text().includes("ynatrace") && !ignoreRegexes.some(regex => regex.test(msg.text()))) {
46
- warnings.push(msg.text());
98
+ }
99
+ page.on("request", (request) => {
100
+ const url = request.url();
101
+ // Gen 3 beacons should always have ty=js
102
+ if (!url.includes("ty=js")) {
103
+ return;
104
+ }
105
+ const beaconBodyString = extractBeaconBody(request, url);
106
+ if (!beaconBodyString) {
107
+ return;
108
+ }
109
+ try {
110
+ const beaconBody = JSON.parse(beaconBodyString);
111
+ context.beacons.push({
112
+ url: url,
113
+ body: beaconBody
114
+ });
115
+ context.events.push(...beaconBody.data.events);
116
+ }
117
+ catch (error) {
118
+ console.error("[Dynatrace Testing] Failed to parse beacon body:", error);
119
+ }
120
+ });
121
+ await page.addInitScript(rum);
122
+ }
123
+ /**
124
+ * Creates the DynatraceTesting API object with all testing methods.
125
+ *
126
+ * @param context The testing context containing events, beacons, and warnings
127
+ * @param testInfo The Playwright test info for snapshot paths
128
+ * @returns The DynatraceTesting API object
129
+ */
130
+ function createDynatraceTestingApi(context, testInfo) {
131
+ const { events, beacons, warnings } = context;
132
+ return {
133
+ async waitForBeacons(options = {}) {
134
+ const { minCount = 0, timeout = DEFAULT_TIMEOUT } = options;
135
+ const start = Date.now();
136
+ while (beacons.length < minCount && (Date.now() - start) <= timeout) {
137
+ if (warnings.length > 0) {
138
+ throwWithEventDump(context, "Unexpected Dynatrace API warnings: " + warnings.join("\n"), "waitForBeacons - unexpected warnings");
139
+ }
140
+ await wait(100);
47
141
  }
48
- });
49
- await setupRumJavaScript();
50
- await use({
51
- async waitForBeacons(options = {}) {
52
- const { minCount = 0, timeout = DEFAULT_TIMEOUT } = options;
53
- const start = Date.now();
54
- while (beacons.length < minCount && (Date.now() - start) <= timeout) {
55
- if (warnings.length > 0) {
56
- throwWithEventDump("Unexpected Dynatrace API warnings: " + warnings.join("\n"), "waitForBeacons - unexpected warnings");
142
+ if (beacons.length < minCount) {
143
+ throwWithEventDump(context, `Found only ${beacons.length} Dynatrace beacons after timeout of ${timeout}ms, but expected at least ${minCount}.`, "waitForBeacons - timeout");
144
+ }
145
+ return beacons;
146
+ },
147
+ async expectToHaveSentEvent(expectedEvent, options = {}) {
148
+ const { timeout = DEFAULT_TIMEOUT } = options;
149
+ const start = Date.now();
150
+ let lastNonMatch = void 0;
151
+ let nextCheckIndex = 0;
152
+ while ((Date.now() - start) <= timeout) {
153
+ for (let i = nextCheckIndex; i < events.length; i++) {
154
+ const beaconEvent = events[i];
155
+ try {
156
+ expect(beaconEvent).toMatchObject(expectedEvent);
157
+ return;
57
158
  }
58
- await wait(100);
59
- }
60
- if (beacons.length < minCount) {
61
- throwWithEventDump(`Found only ${beacons.length} Dynatrace beacons after timeout of ${timeout}ms, but expected at least ${minCount}.`, "waitForBeacons - timeout");
62
- }
63
- return beacons;
64
- },
65
- async expectToHaveSentEvent(expectedEvent, options = {}) {
66
- const { timeout = DEFAULT_TIMEOUT } = options;
67
- const start = Date.now();
68
- let lastNonMatch = void 0;
69
- let nextCheckIndex = 0;
70
- while ((Date.now() - start) <= timeout) {
71
- for (let i = nextCheckIndex; i < events.length; i++) {
72
- const beaconEvent = events[i];
73
- try {
74
- expect(beaconEvent).toMatchObject(expectedEvent);
75
- return;
76
- }
77
- catch {
78
- lastNonMatch = beaconEvent;
79
- }
80
- nextCheckIndex = i + 1;
159
+ catch {
160
+ lastNonMatch = beaconEvent;
81
161
  }
82
- await wait(100);
162
+ nextCheckIndex = i + 1;
83
163
  }
84
- if (!lastNonMatch) {
85
- throwWithEventDump("Dynatrace didn't send any events.", "expectToHaveSentEvent - no events");
86
- }
87
- dumpEventsIfEnabled("expectToHaveSentEvent - no match");
88
- expect(lastNonMatch).toMatchObject(expectedEvent);
89
- },
90
- async expectToHaveSentEventTimes(expectedEvent, times, options = {}) {
91
- const { timeout = DEFAULT_TIMEOUT } = options;
92
- const start = Date.now();
93
- let lastNonMatch = void 0;
94
- let nextCheckIndex = 0;
95
- let foundTimes = 0;
96
- while ((Date.now() - start) <= timeout) {
97
- for (let i = nextCheckIndex; i < events.length; i++) {
98
- const beaconEvent = events[i];
99
- try {
100
- expect(beaconEvent).toMatchObject(expectedEvent);
101
- foundTimes++;
102
- }
103
- catch {
104
- lastNonMatch = beaconEvent;
105
- }
106
- nextCheckIndex = i + 1;
107
- }
108
- if (foundTimes === times) {
109
- return;
164
+ await wait(100);
165
+ }
166
+ if (!lastNonMatch) {
167
+ throwWithEventDump(context, "Dynatrace didn't send any events.", "expectToHaveSentEvent - no events");
168
+ }
169
+ dumpEventsIfEnabled(context, "expectToHaveSentEvent - no match");
170
+ expect(lastNonMatch).toMatchObject(expectedEvent);
171
+ },
172
+ async expectToHaveSentEventTimes(expectedEvent, times, options = {}) {
173
+ const { timeout = DEFAULT_TIMEOUT } = options;
174
+ const start = Date.now();
175
+ let lastNonMatch = void 0;
176
+ let nextCheckIndex = 0;
177
+ let foundTimes = 0;
178
+ while ((Date.now() - start) <= timeout) {
179
+ for (let i = nextCheckIndex; i < events.length; i++) {
180
+ const beaconEvent = events[i];
181
+ try {
182
+ expect(beaconEvent).toMatchObject(expectedEvent);
183
+ foundTimes++;
110
184
  }
111
- if (foundTimes > times) {
112
- throwWithEventDump(`Expected ${times} event occurrences, found ${foundTimes}.`, "expectToHaveSentEventTimes - too many matches");
185
+ catch {
186
+ lastNonMatch = beaconEvent;
113
187
  }
114
- await wait(100);
188
+ nextCheckIndex = i + 1;
115
189
  }
116
- if (!lastNonMatch) {
117
- throwWithEventDump("Dynatrace didn't send any events", "expectToHaveSentEventTimes - no events");
190
+ if (foundTimes === times) {
191
+ return;
118
192
  }
119
- if (foundTimes < times) {
120
- throwWithEventDump(`Didn't find the expected amount of ${times} matching events, found ${foundTimes}.`, "expectToHaveSentEventTimes - count mismatch");
193
+ if (foundTimes > times) {
194
+ throwWithEventDump(context, `Expected ${times} event occurrences, found ${foundTimes}.`, "expectToHaveSentEventTimes - too many matches");
121
195
  }
122
- dumpEventsIfEnabled("expectToHaveSentEventTimes - no match");
123
- expect(lastNonMatch).toMatchObject(expectedEvent);
124
- },
125
- clearEvents() {
126
- events.length = 0;
127
- beacons.length = 0;
196
+ await wait(100);
128
197
  }
129
- });
130
- /**
131
- * Dumps received events to console if dumpEventsOnFail is enabled.
132
- * Formats events for readability.
133
- *
134
- * @param context Additional context about why events are being dumped
135
- */
136
- function dumpEventsIfEnabled(context) {
137
- if (!dumpEventsOnFail) {
138
- return;
198
+ if (!lastNonMatch) {
199
+ throwWithEventDump(context, "Dynatrace didn't send any events", "expectToHaveSentEventTimes - no events");
139
200
  }
140
- console.log(`\n[Dynatrace Testing] Event Dump (${context})`);
141
- console.log(`Total events received: ${events.length}`);
142
- console.log(`Total beacons received: ${beacons.length}`);
143
- if (events.length === 0) {
144
- console.log("No events were received.");
201
+ if (foundTimes < times) {
202
+ throwWithEventDump(context, `Didn't find the expected amount of ${times} matching events, found ${foundTimes}.`, "expectToHaveSentEventTimes - count mismatch");
145
203
  }
146
- else {
147
- console.log("\nReceived events:");
148
- events.forEach((event, index) => {
149
- console.log(`\nEvent ${index + 1}:`, JSON.stringify(event, null, 2));
150
- });
204
+ dumpEventsIfEnabled(context, "expectToHaveSentEventTimes - no match");
205
+ expect(lastNonMatch).toMatchObject(expectedEvent);
206
+ },
207
+ clearEvents() {
208
+ events.length = 0;
209
+ beacons.length = 0;
210
+ },
211
+ toMatchEventSnapshot(options = {}) {
212
+ const { ignoreEvents = [...DEFAULT_IGNORED_EVENTS], ignoredFields = [...DEFAULT_IGNORED_FIELDS], removedFields = [...DEFAULT_REMOVED_FIELDS], name } = options;
213
+ const snapshotName = name ?? "events";
214
+ const snapshotPath = testInfo.snapshotPath(`${snapshotName}.events.snap`);
215
+ const updateSnapshots = testInfo.config.updateSnapshots === "all";
216
+ const filteredEvents = filterIgnoredEvents(events, ignoreEvents);
217
+ const processedEvents = processEventProperties(filteredEvents, ignoredFields, removedFields);
218
+ const snapshotResult = readSnapshot(snapshotPath);
219
+ if (!snapshotResult.exists || updateSnapshots) {
220
+ writeSnapshot(snapshotPath, processedEvents);
221
+ if (!snapshotResult.exists) {
222
+ console.log(`[Dynatrace Testing] Created new snapshot: ${snapshotPath}`);
223
+ }
224
+ else {
225
+ console.log(`[Dynatrace Testing] Updated snapshot: ${snapshotPath}`);
226
+ }
227
+ return Promise.resolve();
151
228
  }
152
- }
153
- /**
154
- * Throws an error with a custom message. If event dumping is enabled, logs the received Dynatrace events before
155
- * throwing the error.
156
- *
157
- * @param message The error message to be thrown
158
- * @param context Context about the failure for event dumping
159
- */
160
- function throwWithEventDump(message, context) {
161
- dumpEventsIfEnabled(context ?? "assertion failure");
162
- throw new Error(message);
163
- }
164
- async function setupRumJavaScript() {
165
- // Fetch RUM JavaScript once and cache it
166
- const cacheKey = `${endpointUrl}:${appId}`;
167
- let rum = rumJavaScriptCache.get(cacheKey);
168
- let beaconUri;
169
- if (!rum) {
170
- rum = await fetchRumJavaScriptContent(endpointUrl, appId, token);
171
- beaconUri = extractBeaconUri(rum);
172
- rumJavaScriptCache.set(cacheKey, rum);
173
- rumJavaScriptCache.set(`${cacheKey}:beaconUri`, beaconUri);
229
+ const snapshotData = snapshotResult.data;
230
+ if (!snapshotData) {
231
+ return Promise.reject(new Error("[Dynatrace Testing] Snapshot file is empty"));
174
232
  }
175
- else {
176
- beaconUri = rumJavaScriptCache.get(`${cacheKey}:beaconUri`);
177
- if (!beaconUri) {
178
- throw new Error("Failed to retrieve beaconUri from cache.");
179
- }
233
+ // Sort both arrays by characteristics for order-independent comparison
234
+ const sortedActual = sortEventsByCharacteristics(processedEvents);
235
+ const sortedExpected = sortEventsByCharacteristics(snapshotData);
236
+ try {
237
+ // Use Playwright's expect for comparison - provides excellent diff output
238
+ expect(sortedActual).toStrictEqual(sortedExpected);
239
+ return Promise.resolve();
240
+ }
241
+ catch (error) {
242
+ dumpEventsIfEnabled(context, "toMatchEventSnapshot - mismatch");
243
+ const errorMessage = `[Dynatrace Testing] Snapshot mismatch for "${snapshotName}".\n\n`
244
+ + `Snapshot path: ${snapshotPath}\n\n`
245
+ + `${error instanceof Error ? error.message : String(error)}\n\n`
246
+ + `To update the snapshot, run with --update-snapshots flag.`;
247
+ return Promise.reject(new Error(errorMessage));
180
248
  }
181
- // Listen to network requests to capture beacon data
182
- // This approach doesn't interfere with user-defined routes
183
- page.on("request", (request) => {
184
- const url = request.url();
185
- // Gen 3 beacons should always have ty=js
186
- if (!url.includes("ty=js")) {
187
- return;
188
- }
189
- const beaconBodyString = extractBeaconBody(request, url);
190
- if (!beaconBodyString) {
191
- return;
192
- }
193
- try {
194
- const beaconBody = JSON.parse(beaconBodyString);
195
- beacons.push({
196
- url: url,
197
- body: beaconBody
198
- });
199
- events.push(...beaconBody.data.events);
200
- }
201
- catch (error) {
202
- console.error("[Dynatrace Testing] Failed to parse beacon body:", error);
203
- }
204
- });
205
- // Use addInitScript to inject RUM script tag before any page scripts execute
206
- await page.addInitScript(rum);
207
249
  }
250
+ };
251
+ }
252
+ /**
253
+ * Dumps received events to console if dumpEventsOnFail is enabled.
254
+ *
255
+ * @param context The testing context containing events and beacons
256
+ * @param description Additional context about why events are being dumped
257
+ */
258
+ function dumpEventsIfEnabled(context, description) {
259
+ if (!context.dumpEventsOnFail) {
260
+ return;
208
261
  }
209
- });
262
+ console.log(`\n[Dynatrace Testing] Event Dump (${description})`);
263
+ console.log(`Total events received: ${context.events.length}`);
264
+ console.log(`Total beacons received: ${context.beacons.length}`);
265
+ if (context.events.length === 0) {
266
+ console.log("No events were received.");
267
+ }
268
+ else {
269
+ console.log("\nReceived events:");
270
+ context.events.forEach((event, index) => {
271
+ console.log(`\nEvent ${index + 1}:`, JSON.stringify(event, null, 2));
272
+ });
273
+ }
274
+ }
275
+ /**
276
+ * Throws an error with a custom message. If event dumping is enabled, logs the received Dynatrace events before
277
+ * throwing the error.
278
+ *
279
+ * @param context The testing context for event dumping
280
+ * @param message The error message to be thrown
281
+ * @param description Context about the failure for event dumping
282
+ */
283
+ function throwWithEventDump(context, message, description) {
284
+ dumpEventsIfEnabled(context, description ?? "assertion failure");
285
+ throw new Error(message);
286
+ }
210
287
  /**
211
288
  * Creates a helpful error for missing configuration fields.
212
289
  *
@@ -316,4 +393,4 @@ function extractBeaconBody(request, url) {
316
393
  }
317
394
  return request.postData();
318
395
  }
319
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.js","sourceRoot":"","sources":["../../source/testing/test.ts"],"names":[],"mappings":"AACA,OAAO,EACH,MAAM,EACN,IAAI,IAAI,IAAI,EACf,MAAM,kBAAkB,CAAC;AAC1B,OAAO,EAAE,kBAAkB,EAAE,MAAM,cAAc,CAAC;AAClD,OAAO,EAAE,UAAU,EAAE,MAAM,UAAU,CAAC;AAkEtC,MAAM,eAAe,GAAG,KAAK,CAAC;AAE9B,MAAM,OAAO,GAAG,IAAI,WAAW,EAAE,CAAC;AA+DlC;;GAEG;AACH,MAAM,kBAAkB,GAA2B;IAC/C,WAAW,EAAE,+EAA+E;IAC5F,KAAK,EAAE,iEAAiE;IACxE,KAAK,EAAE,kEAAkE;CAC5E,CAAC;AAEF,MAAM,CAAC,MAAM,IAAI,GAAG,IAAI,CAAC,MAAM,CAAyD;IACpF,eAAe,EAAE,CAAC;YACd,KAAK,EAAE,EAAE;YACT,WAAW,EAAE,EAAE;YACf,KAAK,EAAE,EAAE;SACZ,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;IAEpB,6HAA6H;IAC7H,kBAAkB,EAAE,CAAC,KAAK,EAAE,EAAG,EAAE,GAAG,EAAE,EAAE;YACpC,MAAM,KAAK,GAAG,IAAI,GAAG,EAAkB,CAAC;YACxC,MAAM,GAAG,CAAC,KAAK,CAAC,CAAC;QACrB,CAAC,EAAE,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,IAAI,EAAE,GAAG,EAAE,IAAI,EAAE,CAAC;IAE9C,gBAAgB,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,eAAe,EAAE,kBAAkB,EAAE,EAAE,GAAG,EAAE,EAAE;QAC3E,uBAAuB,CAAC,eAAe,CAAC,CAAC;QAEzC,sEAAsE;QACtE,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC9B,IAAI,UAAU,KAAK,aAAa,EAAE,CAAC;YAC/B,OAAO,CAAC,IAAI,CACR,+DAA+D,UAAU;;;;;;;qCAOpD,CACxB,CAAC;QACN,CAAC;QAED,MAAM,EAAE,WAAW,EAAE,KAAK,EAAE,KAAK,EAAE,cAAc,GAAG,EAAE,EAAE,gBAAgB,GAAG,KAAK,EAAE,GAAG,eAAe,CAAC;QACrG,MAAM,OAAO,GAAqD,EAAE,CAAC;QACrE,MAAM,MAAM,GAA8B,EAAE,CAAC;QAC7C,MAAM,QAAQ,GAAa,EAAE,CAAC;QAC9B,MAAM,aAAa,GAAG,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,MAAM,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC;QAElE,IAAI,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,GAAG,EAAE,EAAE;YACvB,IAAI,GAAG,CAAC,IAAI,EAAE,CAAC,QAAQ,CAAC,UAAU,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC;gBAC1F,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC;YAC9B,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,MAAM,kBAAkB,EAAE,CAAC;QAE3B,MAAM,GAAG,CAAC;YACN,KAAK,CAAC,cAAc,CAAC,UAAiC,EAAE;gBACpD,MAAM,EAAE,QAAQ,GAAG,CAAC,EAAE,OAAO,GAAG,eAAe,EAAE,GAAG,OAAO,CAAC;gBAC5D,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBAEzB,OAAO,OAAO,CAAC,MAAM,GAAG,QAAQ,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC,IAAI,OAAO,EAAE,CAAC;oBAClE,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;wBACtB,kBAAkB,CACd,qCAAqC,GAAG,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,EAC3D,sCAAsC,CACzC,CAAC;oBACN,CAAC;oBACD,MAAM,IAAI,CAAC,GAAG,CAAC,CAAC;gBACpB,CAAC;gBAED,IAAI,OAAO,CAAC,MAAM,GAAG,QAAQ,EAAE,CAAC;oBAC5B,kBAAkB,CACd,cAAc,OAAO,CAAC,MAAM,uCAAuC,OAAO,6BAA6B,QAAQ,GAAG,EAClH,0BAA0B,CAC7B,CAAC;gBACN,CAAC;gBAED,OAAO,OAAO,CAAC;YACnB,CAAC;YAED,KAAK,CAAC,qBAAqB,CACvB,aAAsC,EACtC,UAAyB,EAAE;gBAE3B,MAAM,EAAE,OAAO,GAAG,eAAe,EAAE,GAAG,OAAO,CAAC;gBAC9C,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBACzB,IAAI,YAAY,GAAwC,KAAK,CAAC,CAAC;gBAC/D,IAAI,cAAc,GAAG,CAAC,CAAC;gBAEvB,OAAO,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC,IAAI,OAAO,EAAE,CAAC;oBACrC,KAAK,IAAI,CAAC,GAAG,cAAc,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;wBAClD,MAAM,WAAW,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;wBAC9B,IAAI,CAAC;4BACD,MAAM,CAAC,WAAW,CAAC,CAAC,aAAa,CAAC,aAAa,CAAC,CAAC;4BACjD,OAAO;wBACX,CAAC;wBAAC,MAAM,CAAC;4BACL,YAAY,GAAG,WAAW,CAAC;wBAC/B,CAAC;wBACD,cAAc,GAAG,CAAC,GAAG,CAAC,CAAC;oBAC3B,CAAC;oBACD,MAAM,IAAI,CAAC,GAAG,CAAC,CAAC;gBACpB,CAAC;gBAED,IAAI,CAAC,YAAY,EAAE,CAAC;oBAChB,kBAAkB,CACd,mCAAmC,EACnC,mCAAmC,CACtC,CAAC;gBACN,CAAC;gBAED,mBAAmB,CAAC,kCAAkC,CAAC,CAAC;gBACxD,MAAM,CAAC,YAAY,CAAC,CAAC,aAAa,CAAC,aAAa,CAAC,CAAC;YACtD,CAAC;YAED,KAAK,CAAC,0BAA0B,CAC5B,aAAsC,EACtC,KAAa,EACb,UAAyB,EAAE;gBAE3B,MAAM,EAAE,OAAO,GAAG,eAAe,EAAE,GAAG,OAAO,CAAC;gBAC9C,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBACzB,IAAI,YAAY,GAAwC,KAAK,CAAC,CAAC;gBAC/D,IAAI,cAAc,GAAG,CAAC,CAAC;gBACvB,IAAI,UAAU,GAAG,CAAC,CAAC;gBAEnB,OAAO,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC,IAAI,OAAO,EAAE,CAAC;oBACrC,KAAK,IAAI,CAAC,GAAG,cAAc,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;wBAClD,MAAM,WAAW,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;wBAC9B,IAAI,CAAC;4BACD,MAAM,CAAC,WAAW,CAAC,CAAC,aAAa,CAAC,aAAa,CAAC,CAAC;4BACjD,UAAU,EAAE,CAAC;wBACjB,CAAC;wBAAC,MAAM,CAAC;4BACL,YAAY,GAAG,WAAW,CAAC;wBAC/B,CAAC;wBACD,cAAc,GAAG,CAAC,GAAG,CAAC,CAAC;oBAC3B,CAAC;oBACD,IAAI,UAAU,KAAK,KAAK,EAAE,CAAC;wBACvB,OAAO;oBACX,CAAC;oBACD,IAAI,UAAU,GAAG,KAAK,EAAE,CAAC;wBACrB,kBAAkB,CACd,YAAY,KAAK,6BAA6B,UAAU,GAAG,EAC3D,+CAA+C,CAClD,CAAC;oBACN,CAAC;oBACD,MAAM,IAAI,CAAC,GAAG,CAAC,CAAC;gBACpB,CAAC;gBAED,IAAI,CAAC,YAAY,EAAE,CAAC;oBAChB,kBAAkB,CACd,kCAAkC,EAClC,wCAAwC,CAC3C,CAAC;gBACN,CAAC;gBAED,IAAI,UAAU,GAAG,KAAK,EAAE,CAAC;oBACrB,kBAAkB,CACd,sCAAsC,KAAK,2BAA2B,UAAU,GAAG,EACnF,6CAA6C,CAChD,CAAC;gBACN,CAAC;gBAED,mBAAmB,CAAC,uCAAuC,CAAC,CAAC;gBAC7D,MAAM,CAAC,YAAY,CAAC,CAAC,aAAa,CAAC,aAAa,CAAC,CAAC;YACtD,CAAC;YAED,WAAW;gBACP,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC;gBAClB,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC;YACvB,CAAC;SACJ,CAAC,CAAC;QAEH;;;;;WAKG;QACH,SAAS,mBAAmB,CAAC,OAAe;YACxC,IAAI,CAAC,gBAAgB,EAAE,CAAC;gBACpB,OAAO;YACX,CAAC;YAED,OAAO,CAAC,GAAG,CAAC,qCAAqC,OAAO,GAAG,CAAC,CAAC;YAC7D,OAAO,CAAC,GAAG,CAAC,0BAA0B,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC;YACvD,OAAO,CAAC,GAAG,CAAC,2BAA2B,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;YAEzD,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACtB,OAAO,CAAC,GAAG,CAAC,0BAA0B,CAAC,CAAC;YAC5C,CAAC;iBAAM,CAAC;gBACJ,OAAO,CAAC,GAAG,CAAC,oBAAoB,CAAC,CAAC;gBAClC,MAAM,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,EAAE;oBAC5B,OAAO,CAAC,GAAG,CAAC,WAAW,KAAK,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;gBACzE,CAAC,CAAC,CAAC;YACP,CAAC;QACL,CAAC;QAED;;;;;;WAMG;QACH,SAAS,kBAAkB,CAAC,OAAe,EAAE,OAAgB;YACzD,mBAAmB,CAAC,OAAO,IAAI,mBAAmB,CAAC,CAAC;YACpD,MAAM,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC;QAC7B,CAAC;QAED,KAAK,UAAU,kBAAkB;YAC7B,yCAAyC;YACzC,MAAM,QAAQ,GAAG,GAAG,WAAW,IAAI,KAAK,EAAE,CAAC;YAC3C,IAAI,GAAG,GAAG,kBAAkB,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;YAC3C,IAAI,SAA6B,CAAC;YAClC,IAAI,CAAC,GAAG,EAAE,CAAC;gBACP,GAAG,GAAG,MAAM,yBAAyB,CAAC,WAAW,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC;gBACjE,SAAS,GAAG,gBAAgB,CAAC,GAAG,CAAC,CAAC;gBAElC,kBAAkB,CAAC,GAAG,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC;gBACtC,kBAAkB,CAAC,GAAG,CAAC,GAAG,QAAQ,YAAY,EAAE,SAAS,CAAC,CAAC;YAC/D,CAAC;iBAAM,CAAC;gBACJ,SAAS,GAAG,kBAAkB,CAAC,GAAG,CAAC,GAAG,QAAQ,YAAY,CAAC,CAAC;gBAC5D,IAAI,CAAC,SAAS,EAAE,CAAC;oBACb,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;gBAChE,CAAC;YACL,CAAC;YAED,oDAAoD;YACpD,2DAA2D;YAC3D,IAAI,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,OAAO,EAAE,EAAE;gBAC3B,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;gBAC1B,yCAAyC;gBACzC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;oBACzB,OAAO;gBACX,CAAC;gBACD,MAAM,gBAAgB,GAAG,iBAAiB,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;gBACzD,IAAI,CAAC,gBAAgB,EAAE,CAAC;oBACpB,OAAO;gBACX,CAAC;gBACD,IAAI,CAAC;oBACD,MAAM,UAAU,GAIZ,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC;oBACjC,OAAO,CAAC,IAAI,CAAC;wBACT,GAAG,EAAE,GAAG;wBACR,IAAI,EAAE,UAAU;qBACnB,CAAC,CAAC;oBACH,MAAM,CAAC,IAAI,CAAC,GAAG,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAC3C,CAAC;gBAAC,OAAO,KAAK,EAAE,CAAC;oBACb,OAAO,CAAC,KAAK,CAAC,kDAAkD,EAAE,KAAK,CAAC,CAAC;gBAC7E,CAAC;YACL,CAAC,CAAC,CAAC;YAEH,6EAA6E;YAC7E,MAAM,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;QAClC,CAAC;IACL,CAAC;CACJ,CAAC,CAAC;AAEH;;;;;GAKG;AACH,SAAS,wBAAwB,CAAC,aAAuB;IACrD,MAAM,cAAc,GAAG,aAAa;SAC/B,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,SAAS,KAAK,KAAK,kBAAkB,CAAC,KAAK,CAAC,EAAE,CAAC;SAC5D,IAAI,CAAC,IAAI,CAAC,CAAC;IAEhB,OAAO,IAAI,KAAK,CACZ,2DAA2D,aAAa,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,KAAK,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC;;EAEjI,cAAc;;;;;;;;;;;gDAWgC,CAC3C,CAAC;AACN,CAAC;AAED;;;;;GAKG;AACH,SAAS,uBAAuB,CAAC,MAAuB;IACpD,MAAM,aAAa,GAAa,EAAE,CAAC;IAEnC,IAAI,CAAC,MAAM,CAAC,WAAW,IAAI,MAAM,CAAC,WAAW,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC;QAC1D,aAAa,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;IACtC,CAAC;IAED,IAAI,CAAC,MAAM,CAAC,KAAK,IAAI,MAAM,CAAC,KAAK,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC;QAC9C,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAChC,CAAC;IAED,IAAI,CAAC,MAAM,CAAC,KAAK,IAAI,MAAM,CAAC,KAAK,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC;QAC9C,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAChC,CAAC;IAED,IAAI,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC3B,MAAM,wBAAwB,CAAC,aAAa,CAAC,CAAC;IAClD,CAAC;AACL,CAAC;AAED;;;;;;;GAOG;AACH,KAAK,UAAU,yBAAyB,CAAC,WAAmB,EAAE,KAAa,EAAE,KAAa;IACtF,MAAM,YAAY,GAAG,MAAM,kBAAkB,CAAC,WAAW,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC;IACzE,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,cAAc,CAAC,EAAE,CAAC;QACzC,MAAM,IAAI,KAAK,CAAC,4FAA4F,YAAY,EAAE,CAAC,CAAC;IAChI,CAAC;IAED,0CAA0C;IAC1C,MAAM,QAAQ,GAAG,YAAY,CAAC,KAAK,CAAC,eAAe,CAAC,CAAC;IACrD,IAAI,CAAC,QAAQ,EAAE,CAAC;QACZ,MAAM,IAAI,KAAK,CAAC,oDAAoD,CAAC,CAAC;IAC1E,CAAC;IACD,MAAM,SAAS,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC;IAE9B,sCAAsC;IACtC,MAAM,cAAc,GAAG,MAAM,KAAK,CAAC,SAAS,CAAC,CAAC;IAC9C,OAAO,cAAc,CAAC,IAAI,EAAE,CAAC;AACjC,CAAC;AAED;;;;;GAKG;AACH,SAAS,gBAAgB,CAAC,GAAW;IACjC,oDAAoD;IACpD,MAAM,cAAc,GAAG,GAAG,CAAC,KAAK,CAAC,uBAAuB,CAAC,CAAC;IAC1D,IAAI,CAAC,cAAc,EAAE,CAAC;QAClB,MAAM,IAAI,KAAK,CAAC,0DAA0D,CAAC,CAAC;IAChF,CAAC;IACD,OAAO,cAAc,CAAC,CAAC,CAAC,CAAC;AAC7B,CAAC;AAED;;;;;GAKG;AACH,SAAS,IAAI,CAAC,IAAY;IACtB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC3B,UAAU,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IAC9B,CAAC,CAAC,CAAC;AACP,CAAC;AAED;;;;;;GAMG;AACH,SAAS,iBAAiB,CAAC,OAAgB,EAAE,GAAW;IACpD,MAAM,MAAM,GAAG,OAAO,CAAC,cAAc,EAAE,CAAC;IACxC,IAAI,GAAG,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,MAAM,EAAE,CAAC;QACtC,MAAM,SAAS,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC;QACrC,OAAO,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IACrC,CAAC;IACD,OAAO,OAAO,CAAC,QAAQ,EAAE,CAAC;AAC9B,CAAC","sourcesContent":["import type { Request } from \"@playwright/test\";\nimport {\n    expect,\n    test as base\n} from \"@playwright/test\";\nimport { fetchRumJavaScript } from \"./install.js\";\nimport { uncompress } from \"snappyjs\";\n\nexport interface WaitForBeaconsOptions {\n    /**\n     * The minimum number of beacon requests to wait for\n     */\n    minCount?: number;\n    /**\n     * The maximum time to wait for beacon requests, in milliseconds - Default: 5_000\n     */\n    timeout?: number;\n}\n\nexport interface ExpectOptions {\n    /**\n     * The maximum time to wait for the event to be sent, in milliseconds - Default: 5_000\n     */\n    timeout?: number;\n}\n\nexport interface DynatraceTesting {\n    /**\n     * Waits for a specified number of beacon requests or until a timeout occurs.\n     *\n     * @param options Configuration options for waiting for beacons\n     * @returns       A promise that resolves with an array of beacon requests\n     */\n    waitForBeacons(options?: WaitForBeaconsOptions): Promise<BeaconRequest[]>;\n\n    /**\n     * Verifies that a specific event has been sent, optionally within a timeout period.\n     *\n     * @param event   The event to check, represented as a key-value pair object. This uses Playwrights expect(event).toMatchObject.\n     * @param options Configuration options for the verification\n     * @returns       A promise that resolves when the event is confirmed to have been sent\n     */\n    expectToHaveSentEvent(event: Record<string, unknown>, options?: ExpectOptions): Promise<void>;\n\n    /**\n     * Verifies that a specific event has been sent a specified number of times, optionally within a timeout period.\n     *\n     * @param event   The event to check, represented as a key-value pair object. This uses Playwrights expect(event).toMatchObject.\n     * @param times   The exact number of times the event is expected to have been sent\n     * @param options Configuration options for the verification\n     * @returns       A promise that resolves when the event is confirmed to have been sent the specified number of times\n     */\n    expectToHaveSentEventTimes(event: Record<string, unknown>, times: number, options?: ExpectOptions): Promise<void>;\n\n    /**\n     * Clears all events and beacons, allowing for subsequent expectations to ignore events that have been sent\n     * in the past. Example:\n     * ```\n     * sendFooEvent();\n     * await dynatraceTesting.expectToHaveSentEventTimes({ foo: \"bar\" }, 1);\n     * dynatraceTesting.clearEvents();\n     * await dynatraceTesting.expectToHaveSentEventTimes({ foo: \"bar\" }, 1);\n     * ```\n     */\n    clearEvents(): void;\n}\n\nexport interface BeaconRequest {\n    url: string;\n    body: Record<string, unknown>;\n}\n\nconst DEFAULT_TIMEOUT = 5_000;\n\nconst decoder = new TextDecoder();\n\nexport interface DynatraceTestingFixture {\n    /**\n     * The testing API providing expect methods.\n     */\n    dynatraceTesting: DynatraceTesting;\n\n    /**\n     * Configuration to enable Dynatrace testing with a given environment.\n     */\n    dynatraceConfig: DynatraceConfig;\n}\n\nexport interface DynatraceConfig {\n    /**\n     * The URL of the Dynatrace API endpoint to connect to, see\n     * [RUM manual insertion tags API](https://docs.dynatrace.com/docs/discover-dynatrace/references/dynatrace-api/environment-api/rum/rum-manual-insertion-tags)\n     * for details. Example: https://{your-environment-id}.live.dynatrace.com or\n     * https://{your-activegate-domain}/e/{your-environment-id}\n     */\n    endpointUrl: string;\n\n    /**\n     * The application ID to identify the correct RUM JavaScript to fetch. You can find\n     * this in the URL if you open the Experience Vitals app and select the frontend you\n     * want to test. Example: APPLICATION-ABCDEF0123456789\n     */\n    appId: string;\n\n    /**\n     * The authentication token for the installation. See\n     * [Tokens and authentication](https://docs.dynatrace.com/docs/discover-dynatrace/references/dynatrace-api/basics/dynatrace-api-authentication).\n     */\n    token: string;\n\n    /**\n     * Accepts an array of regular expressions to match against Dynatrace-related warnings. By default, these warnings\n     * trigger test fails. If you anticipate warnings and don't want your tests to fail, use this setting to ignore them\n     * by message. Example:\n     * [\"invalid property my_app_data\\.\\\\w+[0-9_]*\"]\n     */\n    ignoreWarnings?: string[];\n\n    /**\n     * Dump the array of Dynatrace events into the console in case an assertion fails.\n     */\n    dumpEventsOnFail?: boolean;\n}\n\n/**\n * Worker-scoped fixtures for caching RUM JavaScript across tests in the same worker.\n * This improves performance by avoiding redundant network requests.\n */\nexport interface DynatraceTestingWorkerFixture {\n    /**\n     * Cache for RUM JavaScript snippets, scoped to the worker.\n     *\n     * @internal\n     */\n    rumJavaScriptCache: Map<string, string>;\n}\n\n/**\n * Field descriptions for configuration error messages.\n */\nconst FIELD_DESCRIPTIONS: Record<string, string> = {\n    endpointUrl: \"The Dynatrace environment URL (e.g., \\\"https://abc12345.live.dynatrace.com\\\")\",\n    appId: \"The RUM application ID (e.g., \\\"APPLICATION-ABCDEF0123456789\\\")\",\n    token: \"The API token with \\\"Read RUM manual insertion tags\\\" permission\"\n};\n\nexport const test = base.extend<DynatraceTestingFixture, DynatraceTestingWorkerFixture>({\n    dynatraceConfig: [{\n        appId: \"\",\n        endpointUrl: \"\",\n        token: \"\"\n    }, { option: true }],\n\n    // eslint-disable-next-line no-empty-pattern -- Playwright requires object destructuring even when no dependencies are needed\n    rumJavaScriptCache: [async ({ }, use) => { // NOSONAR\n        const cache = new Map<string, string>();\n        await use(cache);\n    }, { scope: \"worker\", auto: true, box: true }],\n\n    dynatraceTesting: async ({ page, dynatraceConfig, rumJavaScriptCache }, use) => {\n        validateDynatraceConfig(dynatraceConfig);\n\n        // Check if page has already navigated and warn about fixture ordering\n        const currentUrl = page.url();\n        if (currentUrl !== \"about:blank\") {\n            console.warn(\n                `[Dynatrace Testing] Warning: Page has already navigated to \"${currentUrl}\".\nThe RUM JavaScript may not be injected. Ensure dynatraceTesting is destructured\nbefore any fixtures that navigate the page.\n\nExample of correct fixture order:\n    test(\"my test\", async ({ dynatraceTesting, myCustomFixture, page }) => { ... });\n\nSee testing.md for more information.`\n            );\n        }\n\n        const { endpointUrl, appId, token, ignoreWarnings = [], dumpEventsOnFail = false } = dynatraceConfig;\n        const beacons: { url: string, body: Record<string, unknown> }[] = [];\n        const events: Record<string, unknown>[] = [];\n        const warnings: string[] = [];\n        const ignoreRegexes = ignoreWarnings.map(x => new RegExp(x, \"i\"));\n\n        page.on(\"console\", (msg) => {\n            if (msg.text().includes(\"ynatrace\") && !ignoreRegexes.some(regex => regex.test(msg.text()))) {\n                warnings.push(msg.text());\n            }\n        });\n\n        await setupRumJavaScript();\n\n        await use({\n            async waitForBeacons(options: WaitForBeaconsOptions = {}): Promise<BeaconRequest[]> {\n                const { minCount = 0, timeout = DEFAULT_TIMEOUT } = options;\n                const start = Date.now();\n\n                while (beacons.length < minCount && (Date.now() - start) <= timeout) {\n                    if (warnings.length > 0) {\n                        throwWithEventDump(\n                            \"Unexpected Dynatrace API warnings: \" + warnings.join(\"\\n\"),\n                            \"waitForBeacons - unexpected warnings\"\n                        );\n                    }\n                    await wait(100);\n                }\n\n                if (beacons.length < minCount) {\n                    throwWithEventDump(\n                        `Found only ${beacons.length} Dynatrace beacons after timeout of ${timeout}ms, but expected at least ${minCount}.`,\n                        \"waitForBeacons - timeout\"\n                    );\n                }\n\n                return beacons;\n            },\n\n            async expectToHaveSentEvent(\n                expectedEvent: Record<string, unknown>,\n                options: ExpectOptions = {}\n            ): Promise<void> {\n                const { timeout = DEFAULT_TIMEOUT } = options;\n                const start = Date.now();\n                let lastNonMatch: Record<string, unknown> | undefined = void 0;\n                let nextCheckIndex = 0;\n\n                while ((Date.now() - start) <= timeout) {\n                    for (let i = nextCheckIndex; i < events.length; i++) {\n                        const beaconEvent = events[i];\n                        try {\n                            expect(beaconEvent).toMatchObject(expectedEvent);\n                            return;\n                        } catch {\n                            lastNonMatch = beaconEvent;\n                        }\n                        nextCheckIndex = i + 1;\n                    }\n                    await wait(100);\n                }\n\n                if (!lastNonMatch) {\n                    throwWithEventDump(\n                        \"Dynatrace didn't send any events.\",\n                        \"expectToHaveSentEvent - no events\"\n                    );\n                }\n\n                dumpEventsIfEnabled(\"expectToHaveSentEvent - no match\");\n                expect(lastNonMatch).toMatchObject(expectedEvent);\n            },\n\n            async expectToHaveSentEventTimes(\n                expectedEvent: Record<string, unknown>,\n                times: number,\n                options: ExpectOptions = {}\n            ): Promise<void> {\n                const { timeout = DEFAULT_TIMEOUT } = options;\n                const start = Date.now();\n                let lastNonMatch: Record<string, unknown> | undefined = void 0;\n                let nextCheckIndex = 0;\n                let foundTimes = 0;\n\n                while ((Date.now() - start) <= timeout) {\n                    for (let i = nextCheckIndex; i < events.length; i++) {\n                        const beaconEvent = events[i];\n                        try {\n                            expect(beaconEvent).toMatchObject(expectedEvent);\n                            foundTimes++;\n                        } catch {\n                            lastNonMatch = beaconEvent;\n                        }\n                        nextCheckIndex = i + 1;\n                    }\n                    if (foundTimes === times) {\n                        return;\n                    }\n                    if (foundTimes > times) {\n                        throwWithEventDump(\n                            `Expected ${times} event occurrences, found ${foundTimes}.`,\n                            \"expectToHaveSentEventTimes - too many matches\"\n                        );\n                    }\n                    await wait(100);\n                }\n\n                if (!lastNonMatch) {\n                    throwWithEventDump(\n                        \"Dynatrace didn't send any events\",\n                        \"expectToHaveSentEventTimes - no events\"\n                    );\n                }\n\n                if (foundTimes < times) {\n                    throwWithEventDump(\n                        `Didn't find the expected amount of ${times} matching events, found ${foundTimes}.`,\n                        \"expectToHaveSentEventTimes - count mismatch\"\n                    );\n                }\n\n                dumpEventsIfEnabled(\"expectToHaveSentEventTimes - no match\");\n                expect(lastNonMatch).toMatchObject(expectedEvent);\n            },\n\n            clearEvents(): void {\n                events.length = 0;\n                beacons.length = 0;\n            }\n        });\n\n        /**\n         * Dumps received events to console if dumpEventsOnFail is enabled.\n         * Formats events for readability.\n         *\n         * @param context Additional context about why events are being dumped\n         */\n        function dumpEventsIfEnabled(context: string): void {\n            if (!dumpEventsOnFail) {\n                return;\n            }\n\n            console.log(`\\n[Dynatrace Testing] Event Dump (${context})`);\n            console.log(`Total events received: ${events.length}`);\n            console.log(`Total beacons received: ${beacons.length}`);\n\n            if (events.length === 0) {\n                console.log(\"No events were received.\");\n            } else {\n                console.log(\"\\nReceived events:\");\n                events.forEach((event, index) => {\n                    console.log(`\\nEvent ${index + 1}:`, JSON.stringify(event, null, 2));\n                });\n            }\n        }\n\n        /**\n         * Throws an error with a custom message. If event dumping is enabled, logs the received Dynatrace events before\n         * throwing the error.\n         *\n         * @param message The error message to be thrown\n         * @param context Context about the failure for event dumping\n         */\n        function throwWithEventDump(message: string, context?: string): never {\n            dumpEventsIfEnabled(context ?? \"assertion failure\");\n            throw new Error(message);\n        }\n\n        async function setupRumJavaScript(): Promise<void> {\n            // Fetch RUM JavaScript once and cache it\n            const cacheKey = `${endpointUrl}:${appId}`;\n            let rum = rumJavaScriptCache.get(cacheKey);\n            let beaconUri: string | undefined;\n            if (!rum) {\n                rum = await fetchRumJavaScriptContent(endpointUrl, appId, token);\n                beaconUri = extractBeaconUri(rum);\n\n                rumJavaScriptCache.set(cacheKey, rum);\n                rumJavaScriptCache.set(`${cacheKey}:beaconUri`, beaconUri);\n            } else {\n                beaconUri = rumJavaScriptCache.get(`${cacheKey}:beaconUri`);\n                if (!beaconUri) {\n                    throw new Error(\"Failed to retrieve beaconUri from cache.\");\n                }\n            }\n\n            // Listen to network requests to capture beacon data\n            // This approach doesn't interfere with user-defined routes\n            page.on(\"request\", (request) => {\n                const url = request.url();\n                // Gen 3 beacons should always have ty=js\n                if (!url.includes(\"ty=js\")) {\n                    return;\n                }\n                const beaconBodyString = extractBeaconBody(request, url);\n                if (!beaconBodyString) {\n                    return;\n                }\n                try {\n                    const beaconBody: {\n                        data: {\n                            events: Record<string, unknown>[];\n                        };\n                    } = JSON.parse(beaconBodyString);\n                    beacons.push({\n                        url: url,\n                        body: beaconBody\n                    });\n                    events.push(...beaconBody.data.events);\n                } catch (error) {\n                    console.error(\"[Dynatrace Testing] Failed to parse beacon body:\", error);\n                }\n            });\n\n            // Use addInitScript to inject RUM script tag before any page scripts execute\n            await page.addInitScript(rum);\n        }\n    }\n});\n\n/**\n * Creates a helpful error for missing configuration fields.\n *\n * @param missingFields Array of field names that are missing or empty\n * @returns             Error with detailed setup instructions\n */\nfunction createMissingConfigError(missingFields: string[]): Error {\n    const missingDetails = missingFields\n        .map(field => `    - ${field}: ${FIELD_DESCRIPTIONS[field]}`)\n        .join(\"\\n\");\n\n    return new Error(\n        `[Dynatrace Testing] Missing required configuration field${missingFields.length > 1 ? \"s\" : \"\"}: ${missingFields.join(\", \")}\n\n${missingDetails}\n\nConfigure via test.use():\n    test.use({\n        dynatraceConfig: {\n            endpointUrl: process.env.DT_ENDPOINT_URL!,\n            appId: process.env.DT_APP_ID!,\n            token: process.env.DT_TOKEN!\n        }\n    });\n\nSee testing.md for detailed setup instructions.`\n    );\n}\n\n/**\n * Validates the Dynatrace configuration and throws a helpful error if invalid.\n *\n * @param config The configuration to validate\n * @throws Error with detailed setup instructions if configuration is invalid\n */\nfunction validateDynatraceConfig(config: DynatraceConfig): void {\n    const missingFields: string[] = [];\n\n    if (!config.endpointUrl || config.endpointUrl.trim() === \"\") {\n        missingFields.push(\"endpointUrl\");\n    }\n\n    if (!config.appId || config.appId.trim() === \"\") {\n        missingFields.push(\"appId\");\n    }\n\n    if (!config.token || config.token.trim() === \"\") {\n        missingFields.push(\"token\");\n    }\n\n    if (missingFields.length > 0) {\n        throw createMissingConfigError(missingFields);\n    }\n}\n\n/**\n * Fetches the RUM JavaScript content from the Dynatrace API endpoint.\n *\n * @param endpointUrl The URL of the Dynatrace API endpoint\n * @param appId       The application ID to identify the correct RUM JavaScript\n * @param token       The authentication token for the installation\n * @returns           A promise that resolves to the RUM JavaScript content\n */\nasync function fetchRumJavaScriptContent(endpointUrl: string, appId: string, token: string): Promise<string> {\n    const rumApiResult = await fetchRumJavaScript(endpointUrl, appId, token);\n    if (!rumApiResult.includes(\"_complete.js\")) {\n        throw new Error(`Dynatrace received unexpected RUM JavaScript when requesting the JavaScript Tag via API: ${rumApiResult}`);\n    }\n\n    // Extract the src URL from the script tag\n    const srcMatch = rumApiResult.match(/src=\"([^\"]+)\"/);\n    if (!srcMatch) {\n        throw new Error(\"Failed to extract src URL from RUM JavaScript tag.\");\n    }\n    const scriptUrl = srcMatch[1];\n\n    // Fetch the actual JavaScript content\n    const scriptResponse = await fetch(scriptUrl);\n    return scriptResponse.text();\n}\n\n/**\n * Extracts the beacon URI from the RUM JavaScript content.\n *\n * @param rum The RUM JavaScript content to extract the beacon URI from\n * @returns   The extracted beacon URI\n */\nfunction extractBeaconUri(rum: string): string {\n    // Extract the beaconUri from the JavaScript content\n    const beaconUriMatch = rum.match(/\"beaconUri\":\"([^\"]+)\"/);\n    if (!beaconUriMatch) {\n        throw new Error(\"Failed to extract beaconUri from RUM JavaScript content.\");\n    }\n    return beaconUriMatch[1];\n}\n\n/**\n * Waits for a given time and resolves upon timeout.\n *\n * @param time The time to wait in milliseconds\n * @returns    A resolved promise when the time ran out\n */\nfunction wait(time: number): Promise<void> {\n    return new Promise((resolve) => {\n        setTimeout(resolve, time);\n    });\n}\n\n/**\n * Extracts the body of a beacon request, optionally decompressing it if the URL indicates compressed data.\n *\n * @param request The request object containing the beacon data\n * @param url     The URL of the request to check for specific compression indicators\n * @returns       The extracted body of the beacon request, or null if no request body is present\n */\nfunction extractBeaconBody(request: Request, url: string): string | null {\n    const buffer = request.postDataBuffer();\n    if (url.includes(\"co=snappy\") && buffer) {\n        const uint8Data = uncompress(buffer);\n        return decoder.decode(uint8Data);\n    }\n    return request.postData();\n}\n"]}
396
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.js","sourceRoot":"","sources":["../../source/testing/test.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n    Page,\n    Request,\n    TestInfo\n} from \"@playwright/test\";\nimport type { SnapshotOptions } from \"./snapshot.js\";\nimport {\n    expect,\n    test as base\n} from \"@playwright/test\";\nimport {\n    DEFAULT_IGNORED_EVENTS,\n    DEFAULT_IGNORED_FIELDS,\n    DEFAULT_REMOVED_FIELDS,\n    filterIgnoredEvents,\n    processEventProperties,\n    readSnapshot,\n    sortEventsByCharacteristics,\n    writeSnapshot\n} from \"./snapshot.js\";\nimport { fetchRumJavaScript } from \"./install.js\";\nimport { uncompress } from \"snappyjs\";\n\nexport interface WaitForBeaconsOptions {\n    /**\n     * The minimum number of beacon requests to wait for\n     */\n    minCount?: number;\n    /**\n     * The maximum time to wait for beacon requests, in milliseconds - Default: 5_000\n     */\n    timeout?: number;\n}\n\nexport interface ExpectOptions {\n    /**\n     * The maximum time to wait for the event to be sent, in milliseconds - Default: 5_000\n     */\n    timeout?: number;\n}\n\nexport interface DynatraceTesting {\n    /**\n     * Waits for a specified number of beacon requests or until a timeout occurs.\n     *\n     * @param options Configuration options for waiting for beacons\n     * @returns       A promise that resolves with an array of beacon requests\n     */\n    waitForBeacons(options?: WaitForBeaconsOptions): Promise<BeaconRequest[]>;\n\n    /**\n     * Verifies that a specific event has been sent, optionally within a timeout period.\n     *\n     * @param event   The event to check, represented as a key-value pair object. This uses Playwrights expect(event).toMatchObject.\n     * @param options Configuration options for the verification\n     * @returns       A promise that resolves when the event is confirmed to have been sent\n     */\n    expectToHaveSentEvent(event: Record<string, unknown>, options?: ExpectOptions): Promise<void>;\n\n    /**\n     * Verifies that a specific event has been sent a specified number of times, optionally within a timeout period.\n     *\n     * @param event   The event to check, represented as a key-value pair object. This uses Playwrights expect(event).toMatchObject.\n     * @param times   The exact number of times the event is expected to have been sent\n     * @param options Configuration options for the verification\n     * @returns       A promise that resolves when the event is confirmed to have been sent the specified number of times\n     */\n    expectToHaveSentEventTimes(event: Record<string, unknown>, times: number, options?: ExpectOptions): Promise<void>;\n\n    /**\n     * Clears all events and beacons, allowing for subsequent expectations to ignore events that have been sent\n     * in the past. Example:\n     * ```\n     * sendFooEvent();\n     * await dynatraceTesting.expectToHaveSentEventTimes({ foo: \"bar\" }, 1);\n     * dynatraceTesting.clearEvents();\n     * await dynatraceTesting.expectToHaveSentEventTimes({ foo: \"bar\" }, 1);\n     * ```\n     */\n    clearEvents(): void;\n\n    /**\n     * Compares captured events against a stored snapshot file.\n     * On first run, creates the snapshot. On subsequent runs, compares and fails if different.\n     * Volatile fields (timestamps, IDs, etc.) are masked by default to prevent flaky tests.\n     *\n     * @param options Configuration for snapshot comparison\n     */\n    toMatchEventSnapshot(options?: SnapshotOptions): Promise<void>;\n}\n\nexport interface BeaconRequest {\n    url: string;\n    body: Record<string, unknown>;\n}\n\nconst DEFAULT_TIMEOUT = 5_000;\n\nconst decoder = new TextDecoder();\n\n/**\n * Internal context shared between testing API methods.\n */\ninterface TestingContext {\n    events: Record<string, unknown>[];\n    beacons: BeaconRequest[];\n    warnings: string[];\n    dumpEventsOnFail: boolean;\n}\n\nexport interface DynatraceTestingFixture {\n    /**\n     * The testing API providing expect methods.\n     */\n    dynatraceTesting: DynatraceTesting;\n\n    /**\n     * Configuration to enable Dynatrace testing with a given environment.\n     */\n    dynatraceConfig: DynatraceConfig;\n}\n\nexport interface DynatraceConfig {\n    /**\n     * The URL of the Dynatrace API endpoint to connect to, see\n     * [RUM manual insertion tags API](https://docs.dynatrace.com/docs/discover-dynatrace/references/dynatrace-api/environment-api/rum/rum-manual-insertion-tags)\n     * for details. Example: https://{your-environment-id}.live.dynatrace.com or\n     * https://{your-activegate-domain}/e/{your-environment-id}\n     */\n    endpointUrl: string;\n\n    /**\n     * The application ID to identify the correct RUM JavaScript to fetch. You can find\n     * this in the URL if you open the Experience Vitals app and select the frontend you\n     * want to test. Example: APPLICATION-ABCDEF0123456789\n     */\n    appId: string;\n\n    /**\n     * The authentication token for the installation. See\n     * [Tokens and authentication](https://docs.dynatrace.com/docs/discover-dynatrace/references/dynatrace-api/basics/dynatrace-api-authentication).\n     */\n    token: string;\n\n    /**\n     * Accepts an array of regular expressions to match against Dynatrace-related warnings. By default, these warnings\n     * trigger test fails. If you anticipate warnings and don't want your tests to fail, use this setting to ignore them\n     * by message. Example:\n     * [\"invalid property my_app_data\\.\\\\w+[0-9_]*\"]\n     */\n    ignoreWarnings?: string[];\n\n    /**\n     * Dump the array of Dynatrace events into the console in case an assertion fails.\n     */\n    dumpEventsOnFail?: boolean;\n}\n\n/**\n * Worker-scoped fixtures for caching RUM JavaScript across tests in the same worker.\n * This improves performance by avoiding redundant network requests.\n */\nexport interface DynatraceTestingWorkerFixture {\n    /**\n     * Cache for RUM JavaScript snippets, scoped to the worker.\n     *\n     * @internal\n     */\n    rumJavaScriptCache: Map<string, string>;\n}\n\n/**\n * Field descriptions for configuration error messages.\n */\nconst FIELD_DESCRIPTIONS: Record<string, string> = {\n    endpointUrl: \"The Dynatrace environment URL (e.g., \\\"https://abc12345.live.dynatrace.com\\\")\",\n    appId: \"The RUM application ID (e.g., \\\"APPLICATION-ABCDEF0123456789\\\")\",\n    token: \"The API token with \\\"Read RUM manual insertion tags\\\" permission\"\n};\n\nexport const test = base.extend<DynatraceTestingFixture, DynatraceTestingWorkerFixture>({\n    dynatraceConfig: [{\n        appId: \"\",\n        endpointUrl: \"\",\n        token: \"\"\n    }, { option: true }],\n\n    // eslint-disable-next-line no-empty-pattern -- Playwright requires object destructuring even when no dependencies are needed\n    rumJavaScriptCache: [async ({ }, use) => { // NOSONAR\n        const cache = new Map<string, string>();\n        await use(cache);\n    }, { scope: \"worker\", auto: true, box: true }],\n\n    dynatraceTesting: async (\n        { page, dynatraceConfig, rumJavaScriptCache },\n        use,\n        testInfo: TestInfo\n    ) => {\n        validateDynatraceConfig(dynatraceConfig);\n        warnIfPageAlreadyNavigated(page);\n\n        const { ignoreWarnings = [], dumpEventsOnFail = false } = dynatraceConfig;\n        const context: TestingContext = {\n            events: [],\n            beacons: [],\n            warnings: [],\n            dumpEventsOnFail\n        };\n\n        setupConsoleWarningListener(page, ignoreWarnings, context);\n        await setupRumJavaScriptAndListeners(page, dynatraceConfig, rumJavaScriptCache, context);\n        await use(createDynatraceTestingApi(context, testInfo));\n    }\n});\n\n/**\n * Warns if the page has already navigated away from about:blank.\n * This indicates the fixture may not inject RUM JavaScript correctly.\n *\n * @param page The Playwright page instance\n */\nfunction warnIfPageAlreadyNavigated(page: Page): void {\n    const currentUrl = page.url();\n    if (currentUrl !== \"about:blank\") {\n        console.warn(\n            `[Dynatrace Testing] Warning: Page has already navigated to \"${currentUrl}\".\nThe RUM JavaScript may not be injected. Ensure dynatraceTesting is destructured\nbefore any fixtures that navigate the page.\n\nExample of correct fixture order:\n    test(\"my test\", async ({ dynatraceTesting, myCustomFixture, page }) => { ... });\n\nSee testing.md for more information.`\n        );\n    }\n}\n\n/**\n * Sets up a console listener to capture Dynatrace-related warnings.\n *\n * @param page           The Playwright page instance\n * @param ignoreWarnings Array of regex patterns for warnings to ignore\n * @param context        The testing context to store warnings\n */\nfunction setupConsoleWarningListener(\n    page: Page,\n    ignoreWarnings: string[],\n    context: TestingContext\n): void {\n    const ignoreRegexes = ignoreWarnings.map(x => new RegExp(x, \"i\"));\n    page.on(\"console\", (msg) => {\n        if (msg.text().includes(\"ynatrace\") && !ignoreRegexes.some(regex => regex.test(msg.text()))) {\n            context.warnings.push(msg.text());\n        }\n    });\n}\n\n/**\n * Sets up RUM JavaScript injection and beacon request listeners.\n *\n * @param page    The Playwright page instance\n * @param config  The Dynatrace configuration\n * @param cache   The worker-scoped cache for RUM JavaScript\n * @param context The testing context to store beacons and events\n */\nasync function setupRumJavaScriptAndListeners(\n    page: Page,\n    config: DynatraceConfig,\n    cache: Map<string, string>,\n    context: TestingContext\n): Promise<void> {\n    const { endpointUrl, appId, token } = config;\n    const cacheKey = `${endpointUrl}:${appId}`;\n    let rum = cache.get(cacheKey);\n\n    if (!rum) {\n        rum = await fetchRumJavaScriptContent(endpointUrl, appId, token);\n        const beaconUri = extractBeaconUri(rum);\n        cache.set(cacheKey, rum);\n        cache.set(`${cacheKey}:beaconUri`, beaconUri);\n    } else {\n        const beaconUri = cache.get(`${cacheKey}:beaconUri`);\n        if (!beaconUri) {\n            throw new Error(\"Failed to retrieve beaconUri from cache.\");\n        }\n    }\n\n    page.on(\"request\", (request) => {\n        const url = request.url();\n        // Gen 3 beacons should always have ty=js\n        if (!url.includes(\"ty=js\")) {\n            return;\n        }\n        const beaconBodyString = extractBeaconBody(request, url);\n        if (!beaconBodyString) {\n            return;\n        }\n        try {\n            const beaconBody: {\n                data: {\n                    events: Record<string, unknown>[];\n                };\n            } = JSON.parse(beaconBodyString);\n            context.beacons.push({\n                url: url,\n                body: beaconBody\n            });\n            context.events.push(...beaconBody.data.events);\n        } catch (error) {\n            console.error(\"[Dynatrace Testing] Failed to parse beacon body:\", error);\n        }\n    });\n\n    await page.addInitScript(rum);\n}\n\n/**\n * Creates the DynatraceTesting API object with all testing methods.\n *\n * @param context  The testing context containing events, beacons, and warnings\n * @param testInfo The Playwright test info for snapshot paths\n * @returns        The DynatraceTesting API object\n */\nfunction createDynatraceTestingApi(context: TestingContext, testInfo: TestInfo): DynatraceTesting {\n    const { events, beacons, warnings } = context;\n\n    return {\n        async waitForBeacons(options: WaitForBeaconsOptions = {}): Promise<BeaconRequest[]> {\n            const { minCount = 0, timeout = DEFAULT_TIMEOUT } = options;\n            const start = Date.now();\n\n            while (beacons.length < minCount && (Date.now() - start) <= timeout) {\n                if (warnings.length > 0) {\n                    throwWithEventDump(\n                        context,\n                        \"Unexpected Dynatrace API warnings: \" + warnings.join(\"\\n\"),\n                        \"waitForBeacons - unexpected warnings\"\n                    );\n                }\n                await wait(100);\n            }\n\n            if (beacons.length < minCount) {\n                throwWithEventDump(\n                    context,\n                    `Found only ${beacons.length} Dynatrace beacons after timeout of ${timeout}ms, but expected at least ${minCount}.`,\n                    \"waitForBeacons - timeout\"\n                );\n            }\n\n            return beacons;\n        },\n\n        async expectToHaveSentEvent(\n            expectedEvent: Record<string, unknown>,\n            options: ExpectOptions = {}\n        ): Promise<void> {\n            const { timeout = DEFAULT_TIMEOUT } = options;\n            const start = Date.now();\n            let lastNonMatch: Record<string, unknown> | undefined = void 0;\n            let nextCheckIndex = 0;\n\n            while ((Date.now() - start) <= timeout) {\n                for (let i = nextCheckIndex; i < events.length; i++) {\n                    const beaconEvent = events[i];\n                    try {\n                        expect(beaconEvent).toMatchObject(expectedEvent);\n                        return;\n                    } catch {\n                        lastNonMatch = beaconEvent;\n                    }\n                    nextCheckIndex = i + 1;\n                }\n                await wait(100);\n            }\n\n            if (!lastNonMatch) {\n                throwWithEventDump(\n                    context,\n                    \"Dynatrace didn't send any events.\",\n                    \"expectToHaveSentEvent - no events\"\n                );\n            }\n\n            dumpEventsIfEnabled(context, \"expectToHaveSentEvent - no match\");\n            expect(lastNonMatch).toMatchObject(expectedEvent);\n        },\n\n        async expectToHaveSentEventTimes(\n            expectedEvent: Record<string, unknown>,\n            times: number,\n            options: ExpectOptions = {}\n        ): Promise<void> {\n            const { timeout = DEFAULT_TIMEOUT } = options;\n            const start = Date.now();\n            let lastNonMatch: Record<string, unknown> | undefined = void 0;\n            let nextCheckIndex = 0;\n            let foundTimes = 0;\n\n            while ((Date.now() - start) <= timeout) {\n                for (let i = nextCheckIndex; i < events.length; i++) {\n                    const beaconEvent = events[i];\n                    try {\n                        expect(beaconEvent).toMatchObject(expectedEvent);\n                        foundTimes++;\n                    } catch {\n                        lastNonMatch = beaconEvent;\n                    }\n                    nextCheckIndex = i + 1;\n                }\n                if (foundTimes === times) {\n                    return;\n                }\n                if (foundTimes > times) {\n                    throwWithEventDump(\n                        context,\n                        `Expected ${times} event occurrences, found ${foundTimes}.`,\n                        \"expectToHaveSentEventTimes - too many matches\"\n                    );\n                }\n                await wait(100);\n            }\n\n            if (!lastNonMatch) {\n                throwWithEventDump(\n                    context,\n                    \"Dynatrace didn't send any events\",\n                    \"expectToHaveSentEventTimes - no events\"\n                );\n            }\n\n            if (foundTimes < times) {\n                throwWithEventDump(\n                    context,\n                    `Didn't find the expected amount of ${times} matching events, found ${foundTimes}.`,\n                    \"expectToHaveSentEventTimes - count mismatch\"\n                );\n            }\n\n            dumpEventsIfEnabled(context, \"expectToHaveSentEventTimes - no match\");\n            expect(lastNonMatch).toMatchObject(expectedEvent);\n        },\n\n        clearEvents(): void {\n            events.length = 0;\n            beacons.length = 0;\n        },\n\n        toMatchEventSnapshot(options: SnapshotOptions = {}): Promise<void> {\n            const {\n                ignoreEvents = [...DEFAULT_IGNORED_EVENTS],\n                ignoredFields = [...DEFAULT_IGNORED_FIELDS],\n                removedFields = [...DEFAULT_REMOVED_FIELDS],\n                name\n            } = options;\n\n            const snapshotName = name ?? \"events\";\n            const snapshotPath = testInfo.snapshotPath(`${snapshotName}.events.snap`);\n            const updateSnapshots = testInfo.config.updateSnapshots === \"all\";\n\n            const filteredEvents = filterIgnoredEvents(events, ignoreEvents);\n            const processedEvents = processEventProperties(filteredEvents, ignoredFields, removedFields);\n            const snapshotResult = readSnapshot(snapshotPath);\n\n            if (!snapshotResult.exists || updateSnapshots) {\n                writeSnapshot(snapshotPath, processedEvents);\n                if (!snapshotResult.exists) {\n                    console.log(`[Dynatrace Testing] Created new snapshot: ${snapshotPath}`);\n                } else {\n                    console.log(`[Dynatrace Testing] Updated snapshot: ${snapshotPath}`);\n                }\n                return Promise.resolve();\n            }\n\n            const snapshotData = snapshotResult.data;\n            if (!snapshotData) {\n                return Promise.reject(new Error(\"[Dynatrace Testing] Snapshot file is empty\"));\n            }\n\n            // Sort both arrays by characteristics for order-independent comparison\n            const sortedActual = sortEventsByCharacteristics(processedEvents);\n            const sortedExpected = sortEventsByCharacteristics(snapshotData);\n\n            try {\n                // Use Playwright's expect for comparison - provides excellent diff output\n                expect(sortedActual).toStrictEqual(sortedExpected);\n                return Promise.resolve();\n            } catch (error) {\n                dumpEventsIfEnabled(context, \"toMatchEventSnapshot - mismatch\");\n                const errorMessage = `[Dynatrace Testing] Snapshot mismatch for \"${snapshotName}\".\\n\\n`\n                    + `Snapshot path: ${snapshotPath}\\n\\n`\n                    + `${error instanceof Error ? error.message : String(error)}\\n\\n`\n                    + `To update the snapshot, run with --update-snapshots flag.`;\n                return Promise.reject(new Error(errorMessage));\n            }\n        }\n    };\n}\n\n/**\n * Dumps received events to console if dumpEventsOnFail is enabled.\n *\n * @param context     The testing context containing events and beacons\n * @param description Additional context about why events are being dumped\n */\nfunction dumpEventsIfEnabled(context: TestingContext, description: string): void {\n    if (!context.dumpEventsOnFail) {\n        return;\n    }\n\n    console.log(`\\n[Dynatrace Testing] Event Dump (${description})`);\n    console.log(`Total events received: ${context.events.length}`);\n    console.log(`Total beacons received: ${context.beacons.length}`);\n\n    if (context.events.length === 0) {\n        console.log(\"No events were received.\");\n    } else {\n        console.log(\"\\nReceived events:\");\n        context.events.forEach((event, index) => {\n            console.log(`\\nEvent ${index + 1}:`, JSON.stringify(event, null, 2));\n        });\n    }\n}\n\n/**\n * Throws an error with a custom message. If event dumping is enabled, logs the received Dynatrace events before\n * throwing the error.\n *\n * @param context     The testing context for event dumping\n * @param message     The error message to be thrown\n * @param description Context about the failure for event dumping\n */\nfunction throwWithEventDump(context: TestingContext, message: string, description?: string): never {\n    dumpEventsIfEnabled(context, description ?? \"assertion failure\");\n    throw new Error(message);\n}\n\n/**\n * Creates a helpful error for missing configuration fields.\n *\n * @param missingFields Array of field names that are missing or empty\n * @returns             Error with detailed setup instructions\n */\nfunction createMissingConfigError(missingFields: string[]): Error {\n    const missingDetails = missingFields\n        .map(field => `    - ${field}: ${FIELD_DESCRIPTIONS[field]}`)\n        .join(\"\\n\");\n\n    return new Error(\n        `[Dynatrace Testing] Missing required configuration field${missingFields.length > 1 ? \"s\" : \"\"}: ${missingFields.join(\", \")}\n\n${missingDetails}\n\nConfigure via test.use():\n    test.use({\n        dynatraceConfig: {\n            endpointUrl: process.env.DT_ENDPOINT_URL!,\n            appId: process.env.DT_APP_ID!,\n            token: process.env.DT_TOKEN!\n        }\n    });\n\nSee testing.md for detailed setup instructions.`\n    );\n}\n\n/**\n * Validates the Dynatrace configuration and throws a helpful error if invalid.\n *\n * @param config The configuration to validate\n * @throws Error with detailed setup instructions if configuration is invalid\n */\nfunction validateDynatraceConfig(config: DynatraceConfig): void {\n    const missingFields: string[] = [];\n\n    if (!config.endpointUrl || config.endpointUrl.trim() === \"\") {\n        missingFields.push(\"endpointUrl\");\n    }\n\n    if (!config.appId || config.appId.trim() === \"\") {\n        missingFields.push(\"appId\");\n    }\n\n    if (!config.token || config.token.trim() === \"\") {\n        missingFields.push(\"token\");\n    }\n\n    if (missingFields.length > 0) {\n        throw createMissingConfigError(missingFields);\n    }\n}\n\n/**\n * Fetches the RUM JavaScript content from the Dynatrace API endpoint.\n *\n * @param endpointUrl The URL of the Dynatrace API endpoint\n * @param appId       The application ID to identify the correct RUM JavaScript\n * @param token       The authentication token for the installation\n * @returns           A promise that resolves to the RUM JavaScript content\n */\nasync function fetchRumJavaScriptContent(endpointUrl: string, appId: string, token: string): Promise<string> {\n    const rumApiResult = await fetchRumJavaScript(endpointUrl, appId, token);\n    if (!rumApiResult.includes(\"_complete.js\")) {\n        throw new Error(`Dynatrace received unexpected RUM JavaScript when requesting the JavaScript Tag via API: ${rumApiResult}`);\n    }\n\n    // Extract the src URL from the script tag\n    const srcMatch = rumApiResult.match(/src=\"([^\"]+)\"/);\n    if (!srcMatch) {\n        throw new Error(\"Failed to extract src URL from RUM JavaScript tag.\");\n    }\n    const scriptUrl = srcMatch[1];\n\n    // Fetch the actual JavaScript content\n    const scriptResponse = await fetch(scriptUrl);\n    return scriptResponse.text();\n}\n\n/**\n * Extracts the beacon URI from the RUM JavaScript content.\n *\n * @param rum The RUM JavaScript content to extract the beacon URI from\n * @returns   The extracted beacon URI\n */\nfunction extractBeaconUri(rum: string): string {\n    // Extract the beaconUri from the JavaScript content\n    const beaconUriMatch = rum.match(/\"beaconUri\":\"([^\"]+)\"/);\n    if (!beaconUriMatch) {\n        throw new Error(\"Failed to extract beaconUri from RUM JavaScript content.\");\n    }\n    return beaconUriMatch[1];\n}\n\n/**\n * Waits for a given time and resolves upon timeout.\n *\n * @param time The time to wait in milliseconds\n * @returns    A resolved promise when the time ran out\n */\nfunction wait(time: number): Promise<void> {\n    return new Promise((resolve) => {\n        setTimeout(resolve, time);\n    });\n}\n\n/**\n * Extracts the body of a beacon request, optionally decompressing it if the URL indicates compressed data.\n *\n * @param request The request object containing the beacon data\n * @param url     The URL of the request to check for specific compression indicators\n * @returns       The extracted body of the beacon request, or null if no request body is present\n */\nfunction extractBeaconBody(request: Request, url: string): string | null {\n    const buffer = request.postDataBuffer();\n    if (url.includes(\"co=snappy\") && buffer) {\n        const uint8Data = uncompress(buffer);\n        return decoder.decode(uint8Data);\n    }\n    return request.postData();\n}\n"]}