@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.
- package/README.md +8 -390
- package/dist/testing/index.d.ts +4 -0
- package/dist/testing/index.js +3 -0
- package/dist/testing/snapshot.d.ts +129 -0
- package/dist/testing/snapshot.js +385 -0
- package/dist/testing/test.d.ts +9 -0
- package/dist/testing/test.js +237 -160
- package/dist/types/index-typedoc.d.ts +3 -0
- package/dist/types/index-typedoc.js +4 -1
- package/dist/types/rum-events/rum-event-keys.d.ts +9 -0
- package/dist/types/rum-events/rum-event-keys.js +11 -1
- package/dist/types/rum-events/rum-internal-selfmonitoring-event.d.ts +5 -1
- package/dist/types/rum-events/rum-internal-selfmonitoring-event.js +5 -1
- package/dist/types/rum-events/rum-web-request-event.d.ts +2 -1
- package/dist/types/rum-events/rum-web-request-event.js +2 -1
- package/dist/types/rum-events/shared-namespaces-and-fields/general-rum-event-fields.d.ts +1 -0
- package/dist/types/rum-events/shared-namespaces-and-fields/general-rum-event-fields.js +1 -1
- package/docs/1-overview.md +405 -0
- package/docs/2-testing.md +424 -0
- package/docs/{types.md → 3-types.md} +8 -2
- package/docs/{USERACTIONS.md → 4-useractions.md} +10 -8
- package/package.json +46 -9
- package/docs/testing.md +0 -215
|
@@ -0,0 +1,385 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Snapshot testing utilities for Dynatrace RUM event verification.
|
|
3
|
+
* Provides mechanisms to compare captured events against stored snapshots
|
|
4
|
+
* with support for masking volatile fields.
|
|
5
|
+
*/
|
|
6
|
+
import { existsSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
|
|
7
|
+
import { dirname } from "node:path";
|
|
8
|
+
/**
|
|
9
|
+
* Default properties that are ignored (replaced with "[IGNORED]") in snapshots to prevent flaky tests.
|
|
10
|
+
* These fields typically contain volatile values like timestamps, IDs, and durations
|
|
11
|
+
* that change between test runs.
|
|
12
|
+
*/
|
|
13
|
+
export const DEFAULT_IGNORED_FIELDS = [
|
|
14
|
+
// Base timing fields
|
|
15
|
+
"start_time",
|
|
16
|
+
"duration",
|
|
17
|
+
"timestamp",
|
|
18
|
+
// Session/Instance identity fields
|
|
19
|
+
"dt.rum.sid",
|
|
20
|
+
"dt.rum.browser.sid",
|
|
21
|
+
"dt.rum.instance.id",
|
|
22
|
+
"user_session.id",
|
|
23
|
+
"user.anonymous_id",
|
|
24
|
+
// View/Page
|
|
25
|
+
"view.foreground_time",
|
|
26
|
+
"view.instance_id",
|
|
27
|
+
"page.instance_id",
|
|
28
|
+
// User action
|
|
29
|
+
"user_action.instance_id",
|
|
30
|
+
"user_action.interrupted_by_instance_id",
|
|
31
|
+
"user_action.interrupted_instance_id",
|
|
32
|
+
"positions",
|
|
33
|
+
// Browser/Frame instance fields
|
|
34
|
+
"browser.frame.instance_id",
|
|
35
|
+
"browser.frame.parent_instance_id",
|
|
36
|
+
"browser.tab.instance_id",
|
|
37
|
+
// Distributed tracing fields
|
|
38
|
+
"trace.id",
|
|
39
|
+
"span.id",
|
|
40
|
+
// Self-monitoring
|
|
41
|
+
"dt.rum.sfm_events",
|
|
42
|
+
"dt.rum.agent.version",
|
|
43
|
+
"dt.support.last_user_input"
|
|
44
|
+
];
|
|
45
|
+
/**
|
|
46
|
+
* Default properties that are removed entirely from snapshots to prevent flaky tests.
|
|
47
|
+
* These fields vary in count or presence between test runs.
|
|
48
|
+
*/
|
|
49
|
+
export const DEFAULT_REMOVED_FIELDS = [
|
|
50
|
+
// Performance timing fields (W3C Navigation/Resource Timing)
|
|
51
|
+
"performance.*",
|
|
52
|
+
// Web Vitals timing fields
|
|
53
|
+
"fcp.*",
|
|
54
|
+
"fp.*",
|
|
55
|
+
"ttfb.*",
|
|
56
|
+
"lcp.*",
|
|
57
|
+
"fid.*",
|
|
58
|
+
"inp.*",
|
|
59
|
+
"cls.*",
|
|
60
|
+
// Long tasks
|
|
61
|
+
"long_task.*",
|
|
62
|
+
// Viewport/screen dimensions (vary by test environment)
|
|
63
|
+
"browser.window.*",
|
|
64
|
+
"device.screen.*",
|
|
65
|
+
"device.battery.level"
|
|
66
|
+
];
|
|
67
|
+
/**
|
|
68
|
+
* Default event patterns that are ignored during snapshot comparison.
|
|
69
|
+
* These events are volatile and may or may not be recorded depending on browser performance.
|
|
70
|
+
*/
|
|
71
|
+
export const DEFAULT_IGNORED_EVENTS = [
|
|
72
|
+
{ "characteristics.has_long_task": true },
|
|
73
|
+
{ "characteristics.is_self_monitoring": true }
|
|
74
|
+
];
|
|
75
|
+
/**
|
|
76
|
+
* Placeholder value used to replace ignored property values.
|
|
77
|
+
*/
|
|
78
|
+
export const IGNORED_PLACEHOLDER = "[IGNORED]";
|
|
79
|
+
/**
|
|
80
|
+
* Processes events by ignoring and removing specified properties.
|
|
81
|
+
* - Ignored properties have their values replaced with "[IGNORED]" placeholder
|
|
82
|
+
* - Removed properties are deleted entirely from the event
|
|
83
|
+
*
|
|
84
|
+
* @param events Array of event objects to process
|
|
85
|
+
* @param ignoredFields Array of field paths to ignore (values replaced with placeholder)
|
|
86
|
+
* @param removedFields Array of field paths to remove entirely
|
|
87
|
+
* @returns Deep copy of events with specified fields processed
|
|
88
|
+
*/
|
|
89
|
+
export function processEventProperties(events, ignoredFields = DEFAULT_IGNORED_FIELDS, removedFields = DEFAULT_REMOVED_FIELDS) {
|
|
90
|
+
return events.map(event => processSingleEvent(event, ignoredFields, removedFields));
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* Filters out events that match any of the ignore patterns.
|
|
94
|
+
* An event matches a pattern if all properties in the pattern exist in the event with the same values.
|
|
95
|
+
*
|
|
96
|
+
* @param events Array of events to filter
|
|
97
|
+
* @param ignorePatterns Array of patterns to match against
|
|
98
|
+
* @returns Filtered array excluding events that match any pattern
|
|
99
|
+
*/
|
|
100
|
+
export function filterIgnoredEvents(events, ignorePatterns = DEFAULT_IGNORED_EVENTS) {
|
|
101
|
+
if (ignorePatterns.length === 0) {
|
|
102
|
+
return events;
|
|
103
|
+
}
|
|
104
|
+
return events.filter(event => !matchesAnyPattern(event, ignorePatterns));
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* Checks if an event matches any of the provided patterns.
|
|
108
|
+
*
|
|
109
|
+
* @param event Event to check
|
|
110
|
+
* @param patterns Patterns to match against
|
|
111
|
+
* @returns True if the event matches any pattern
|
|
112
|
+
*/
|
|
113
|
+
function matchesAnyPattern(event, patterns) {
|
|
114
|
+
return patterns.some(pattern => matchesPattern(event, pattern));
|
|
115
|
+
}
|
|
116
|
+
/**
|
|
117
|
+
* Checks if an event matches a single pattern.
|
|
118
|
+
* A match occurs when all properties in the pattern exist in the event with equal values.
|
|
119
|
+
*
|
|
120
|
+
* @param event Event to check
|
|
121
|
+
* @param pattern Pattern to match against
|
|
122
|
+
* @returns True if all pattern properties match
|
|
123
|
+
*/
|
|
124
|
+
function matchesPattern(event, pattern) {
|
|
125
|
+
return Object.keys(pattern).every(key => key in event && event[key] === pattern[key]);
|
|
126
|
+
}
|
|
127
|
+
/**
|
|
128
|
+
* Processes a single event by ignoring and removing specified properties.
|
|
129
|
+
* - Ignored properties: values replaced with "[IGNORED]" placeholder
|
|
130
|
+
* - Removed properties: fields deleted entirely
|
|
131
|
+
* Both support exact field names and wildcard patterns (`prefix.*`).
|
|
132
|
+
*
|
|
133
|
+
* @param event Event object to process
|
|
134
|
+
* @param ignoredFields Array of field names/patterns to ignore (replace values)
|
|
135
|
+
* @param removedFields Array of field names/patterns to remove entirely
|
|
136
|
+
* @returns Deep copy of event with fields processed
|
|
137
|
+
*/
|
|
138
|
+
function processSingleEvent(event, ignoredFields, removedFields) {
|
|
139
|
+
const result = deepClone(event);
|
|
140
|
+
// Process ignored properties (replace values with placeholder)
|
|
141
|
+
for (const fieldPath of ignoredFields) {
|
|
142
|
+
if (fieldPath.endsWith(".*")) {
|
|
143
|
+
const prefix = fieldPath.slice(0, -1); // Remove "*", keep the dot
|
|
144
|
+
ignoreFieldsWithPrefix(result, prefix);
|
|
145
|
+
}
|
|
146
|
+
else if (fieldPath in result) {
|
|
147
|
+
result[fieldPath] = IGNORED_PLACEHOLDER;
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
// Process removed properties (delete entirely)
|
|
151
|
+
for (const fieldPath of removedFields) {
|
|
152
|
+
if (fieldPath.endsWith(".*")) {
|
|
153
|
+
const prefix = fieldPath.slice(0, -1); // Remove "*", keep the dot
|
|
154
|
+
removeFieldsWithPrefix(result, prefix);
|
|
155
|
+
}
|
|
156
|
+
else {
|
|
157
|
+
delete result[fieldPath];
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
return result;
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* Replaces values of all fields in an object that start with the given prefix with the ignored placeholder.
|
|
164
|
+
* Used for wildcard patterns like `inp.*` to ignore `inp.start_time`, `inp.duration`, etc.
|
|
165
|
+
*
|
|
166
|
+
* @param obj Object to modify
|
|
167
|
+
* @param prefix Prefix to match (including trailing dot, e.g., "inp.")
|
|
168
|
+
*/
|
|
169
|
+
function ignoreFieldsWithPrefix(obj, prefix) {
|
|
170
|
+
for (const key of Object.keys(obj)) {
|
|
171
|
+
if (key.startsWith(prefix)) {
|
|
172
|
+
obj[key] = IGNORED_PLACEHOLDER;
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
/**
|
|
177
|
+
* Removes all fields in an object that start with the given prefix.
|
|
178
|
+
* Used for wildcard patterns like `inp.*` to remove `inp.start_time`, `inp.duration`, etc.
|
|
179
|
+
*
|
|
180
|
+
* @param obj Object to modify
|
|
181
|
+
* @param prefix Prefix to match (including trailing dot, e.g., "inp.")
|
|
182
|
+
*/
|
|
183
|
+
function removeFieldsWithPrefix(obj, prefix) {
|
|
184
|
+
for (const key of Object.keys(obj)) {
|
|
185
|
+
if (key.startsWith(prefix)) {
|
|
186
|
+
delete obj[key];
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
/**
|
|
191
|
+
* Type guard to check if a value is a Record<string, unknown>.
|
|
192
|
+
*
|
|
193
|
+
* @param value Value to check
|
|
194
|
+
* @returns True if value is a non-null object
|
|
195
|
+
*/
|
|
196
|
+
function isRecord(value) {
|
|
197
|
+
return typeof value === "object" && value !== null && !Array.isArray(value);
|
|
198
|
+
}
|
|
199
|
+
/**
|
|
200
|
+
* Type guard to check if a value is an array of Record<string, unknown>.
|
|
201
|
+
*
|
|
202
|
+
* @param value Value to check
|
|
203
|
+
* @returns True if value is an array where all elements are records
|
|
204
|
+
*/
|
|
205
|
+
function isRecordArray(value) {
|
|
206
|
+
return Array.isArray(value) && value.every(isRecord);
|
|
207
|
+
}
|
|
208
|
+
/**
|
|
209
|
+
* Creates a deep clone of an object.
|
|
210
|
+
*
|
|
211
|
+
* @param obj Object to clone
|
|
212
|
+
* @returns Deep clone of the object
|
|
213
|
+
*/
|
|
214
|
+
function deepClone(obj) {
|
|
215
|
+
return JSON.parse(JSON.stringify(obj));
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* Reads a snapshot file from disk.
|
|
219
|
+
* Returns an object indicating whether the file exists and its parsed contents.
|
|
220
|
+
*
|
|
221
|
+
* @param snapshotPath Absolute path to the snapshot file
|
|
222
|
+
* @returns Object with exists flag and parsed data
|
|
223
|
+
* @throws Error if the file contains invalid JSON or is not an array of records
|
|
224
|
+
*/
|
|
225
|
+
export function readSnapshot(snapshotPath) {
|
|
226
|
+
if (!existsSync(snapshotPath)) {
|
|
227
|
+
return { exists: false };
|
|
228
|
+
}
|
|
229
|
+
const content = readFileSync(snapshotPath, "utf8");
|
|
230
|
+
const parsed = safeParseJson(content, snapshotPath);
|
|
231
|
+
if (!isRecordArray(parsed)) {
|
|
232
|
+
throw new Error(`Invalid snapshot format in "${snapshotPath}": expected an array of objects`);
|
|
233
|
+
}
|
|
234
|
+
return { exists: true, data: parsed };
|
|
235
|
+
}
|
|
236
|
+
/**
|
|
237
|
+
* Safely parses a JSON string, wrapping errors with context.
|
|
238
|
+
*
|
|
239
|
+
* @param content JSON string to parse
|
|
240
|
+
* @param filePath Path to the file (for error messages)
|
|
241
|
+
* @returns Parsed value
|
|
242
|
+
* @throws Error with context if parsing fails
|
|
243
|
+
*/
|
|
244
|
+
function safeParseJson(content, filePath) {
|
|
245
|
+
try {
|
|
246
|
+
return JSON.parse(content);
|
|
247
|
+
}
|
|
248
|
+
catch (error) {
|
|
249
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
250
|
+
throw new Error(`Failed to parse snapshot file "${filePath}": ${message}`);
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
/**
|
|
254
|
+
* Writes snapshot data to disk.
|
|
255
|
+
* Creates parent directories if they don't exist.
|
|
256
|
+
* Events are sorted by characteristics for consistent ordering.
|
|
257
|
+
* Uses deterministic JSON serialization with sorted keys for consistent diffs.
|
|
258
|
+
*
|
|
259
|
+
* @param snapshotPath Absolute path to the snapshot file
|
|
260
|
+
* @param data Array of events to write
|
|
261
|
+
*/
|
|
262
|
+
export function writeSnapshot(snapshotPath, data) {
|
|
263
|
+
const dir = dirname(snapshotPath);
|
|
264
|
+
if (!existsSync(dir)) {
|
|
265
|
+
mkdirSync(dir, { recursive: true });
|
|
266
|
+
}
|
|
267
|
+
const sortedData = sortEventsByCharacteristics(data);
|
|
268
|
+
const serialized = serializeSnapshot(sortedData);
|
|
269
|
+
writeFileSync(snapshotPath, serialized, "utf8");
|
|
270
|
+
}
|
|
271
|
+
/**
|
|
272
|
+
* Serializes snapshot data to a deterministic JSON string.
|
|
273
|
+
* Sorts object keys to ensure consistent output across different Node.js versions.
|
|
274
|
+
*
|
|
275
|
+
* @param data Array of events to serialize
|
|
276
|
+
* @returns Pretty-printed JSON string with sorted keys
|
|
277
|
+
*/
|
|
278
|
+
function serializeSnapshot(data) {
|
|
279
|
+
return JSON.stringify(data, sortedReplacer, 2) + "\n";
|
|
280
|
+
}
|
|
281
|
+
/**
|
|
282
|
+
* JSON replacer function that sorts object keys for deterministic serialization.
|
|
283
|
+
*
|
|
284
|
+
* @param key Current key (unused)
|
|
285
|
+
* @param value Current value
|
|
286
|
+
* @returns Value with sorted keys if it's an object
|
|
287
|
+
*/
|
|
288
|
+
function sortedReplacer(key, value) {
|
|
289
|
+
if (!isRecord(value)) {
|
|
290
|
+
return value;
|
|
291
|
+
}
|
|
292
|
+
const sortedKeys = Object.keys(value).sort();
|
|
293
|
+
const sorted = {};
|
|
294
|
+
for (const sortedKey of sortedKeys) {
|
|
295
|
+
sorted[sortedKey] = value[sortedKey];
|
|
296
|
+
}
|
|
297
|
+
return sorted;
|
|
298
|
+
}
|
|
299
|
+
/**
|
|
300
|
+
* Sorts events by their characteristics signature for order-independent comparison.
|
|
301
|
+
* Events with the same characteristics are further sorted by other stable properties.
|
|
302
|
+
* This ensures consistent ordering regardless of event arrival order.
|
|
303
|
+
*
|
|
304
|
+
* @param events Array of events to sort
|
|
305
|
+
* @returns New array with events sorted by characteristics
|
|
306
|
+
*/
|
|
307
|
+
export function sortEventsByCharacteristics(events) {
|
|
308
|
+
return [...events].sort((a, b) => {
|
|
309
|
+
const sigA = getCharacteristicsSignature(a);
|
|
310
|
+
const sigB = getCharacteristicsSignature(b);
|
|
311
|
+
if (sigA !== sigB) {
|
|
312
|
+
return sigA.localeCompare(sigB);
|
|
313
|
+
}
|
|
314
|
+
// If characteristics are the same, sort by other stable properties
|
|
315
|
+
return getSecondarySignature(a).localeCompare(getSecondarySignature(b));
|
|
316
|
+
});
|
|
317
|
+
}
|
|
318
|
+
/**
|
|
319
|
+
* Generates a signature string from an event's characteristics fields.
|
|
320
|
+
* Characteristics fields are boolean flags like "characteristics.has_navigation".
|
|
321
|
+
*
|
|
322
|
+
* @param event Event object
|
|
323
|
+
* @returns Signature string representing the event's characteristics
|
|
324
|
+
*/
|
|
325
|
+
function getCharacteristicsSignature(event) {
|
|
326
|
+
const characteristicsKeys = Object.keys(event)
|
|
327
|
+
.filter(key => key.startsWith("characteristics.") && event[key] === true)
|
|
328
|
+
.sort();
|
|
329
|
+
return characteristicsKeys.join(",");
|
|
330
|
+
}
|
|
331
|
+
/**
|
|
332
|
+
* Generates a secondary signature for events with the same characteristics.
|
|
333
|
+
* Uses stable identifying properties like name, type, url, etc.
|
|
334
|
+
*
|
|
335
|
+
* @param event Event object
|
|
336
|
+
* @returns Secondary signature string
|
|
337
|
+
*/
|
|
338
|
+
function getSecondarySignature(event) {
|
|
339
|
+
// Keys must match RumEventKeys enum values for consistency
|
|
340
|
+
const stableKeys = [
|
|
341
|
+
"url.full",
|
|
342
|
+
"view.sequence.number",
|
|
343
|
+
"view.url.full",
|
|
344
|
+
"page.url.full",
|
|
345
|
+
"user_action.type",
|
|
346
|
+
"interaction.name",
|
|
347
|
+
"navigation.type",
|
|
348
|
+
"exception.type",
|
|
349
|
+
"exception.message"
|
|
350
|
+
];
|
|
351
|
+
return stableKeys
|
|
352
|
+
.map((key) => {
|
|
353
|
+
const value = getValueByKey(event, key);
|
|
354
|
+
if (value !== void 0) {
|
|
355
|
+
return `${key}:${String(value)}`;
|
|
356
|
+
}
|
|
357
|
+
return void 0;
|
|
358
|
+
})
|
|
359
|
+
.filter(Boolean)
|
|
360
|
+
.join("|");
|
|
361
|
+
}
|
|
362
|
+
/**
|
|
363
|
+
* Gets a value from an object, trying both literal key and nested path.
|
|
364
|
+
*
|
|
365
|
+
* @param obj Object to get value from
|
|
366
|
+
* @param key Key (may be literal or dot-notation path)
|
|
367
|
+
* @returns Value or undefined
|
|
368
|
+
*/
|
|
369
|
+
function getValueByKey(obj, key) {
|
|
370
|
+
// First try literal key (e.g., "request.url" as a single key)
|
|
371
|
+
if (key in obj) {
|
|
372
|
+
return obj[key];
|
|
373
|
+
}
|
|
374
|
+
// Then try nested path (e.g., obj.request.url)
|
|
375
|
+
const parts = key.split(".");
|
|
376
|
+
let current = obj;
|
|
377
|
+
for (const part of parts) {
|
|
378
|
+
if (!isRecord(current) || !(part in current)) {
|
|
379
|
+
return void 0;
|
|
380
|
+
}
|
|
381
|
+
current = current[part];
|
|
382
|
+
}
|
|
383
|
+
return current;
|
|
384
|
+
}
|
|
385
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshot.js","sourceRoot":"","sources":["../../source/testing/snapshot.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Snapshot testing utilities for Dynatrace RUM event verification.\n * Provides mechanisms to compare captured events against stored snapshots\n * with support for masking volatile fields.\n */\n\nimport {\n    existsSync,\n    mkdirSync,\n    readFileSync,\n    writeFileSync\n} from \"node:fs\";\nimport { dirname } from \"node:path\";\n\n/**\n * Result of reading a snapshot file.\n */\ninterface SnapshotReadResult {\n    /**\n     * Whether the snapshot file exists.\n     */\n    exists: boolean;\n\n    /**\n     * The parsed snapshot data, or undefined if the file doesn't exist.\n     */\n    data?: Record<string, unknown>[];\n}\n\n/**\n * Pattern to match events that should be completely ignored during snapshot comparison.\n * Events matching any of these patterns will be filtered out before comparison.\n */\nexport type EventIgnorePattern = Record<string, unknown>;\n\n/**\n * Options for configuring snapshot comparison behavior.\n */\nexport interface SnapshotOptions {\n    /**\n     * Properties to ignore before comparison. Values are replaced with \"[IGNORED]\" placeholder.\n     * Supports exact field names and wildcard patterns (`prefix.*`).\n     *\n     * Examples:\n     * - `\"start_time\"` → value replaced with \"[IGNORED]\"\n     * - `\"inp.*\"` → all fields starting with `inp.` have values replaced with \"[IGNORED]\"\n     *\n     * Default: DEFAULT_IGNORED_FIELDS\n     */\n    ignoredFields?: string[];\n\n    /**\n     * Properties to remove entirely before comparison.\n     * Supports exact field names and wildcard patterns (`prefix.*`).\n     *\n     * Examples:\n     * - `\"start_time\"` → field is removed from the event\n     * - `\"inp.*\"` → all fields starting with `inp.` are removed\n     *\n     * Default: DEFAULT_REMOVED_FIELDS\n     */\n    removedFields?: string[];\n\n    /**\n     * Event patterns to completely ignore during snapshot comparison.\n     * Events matching any of these patterns will be filtered out.\n     * Default: DEFAULT_IGNORED_EVENTS\n     */\n    ignoreEvents?: EventIgnorePattern[];\n\n    /**\n     * Custom snapshot name. If not provided, auto-generated from test name.\n     */\n    name?: string;\n}\n\n/**\n * Default properties that are ignored (replaced with \"[IGNORED]\") in snapshots to prevent flaky tests.\n * These fields typically contain volatile values like timestamps, IDs, and durations\n * that change between test runs.\n */\nexport const DEFAULT_IGNORED_FIELDS: readonly string[] = [\n    // Base timing fields\n    \"start_time\",\n    \"duration\",\n    \"timestamp\",\n\n    // Session/Instance identity fields\n    \"dt.rum.sid\",\n    \"dt.rum.browser.sid\",\n    \"dt.rum.instance.id\",\n    \"user_session.id\",\n    \"user.anonymous_id\",\n\n    // View/Page\n    \"view.foreground_time\",\n    \"view.instance_id\",\n    \"page.instance_id\",\n\n    // User action\n    \"user_action.instance_id\",\n    \"user_action.interrupted_by_instance_id\",\n    \"user_action.interrupted_instance_id\",\n    \"positions\",\n\n    // Browser/Frame instance fields\n    \"browser.frame.instance_id\",\n    \"browser.frame.parent_instance_id\",\n    \"browser.tab.instance_id\",\n\n    // Distributed tracing fields\n    \"trace.id\",\n    \"span.id\",\n\n    // Self-monitoring\n    \"dt.rum.sfm_events\",\n    \"dt.rum.agent.version\",\n    \"dt.support.last_user_input\"\n] as const;\n\n/**\n * Default properties that are removed entirely from snapshots to prevent flaky tests.\n * These fields vary in count or presence between test runs.\n */\nexport const DEFAULT_REMOVED_FIELDS: readonly string[] = [\n    // Performance timing fields (W3C Navigation/Resource Timing)\n    \"performance.*\",\n\n    // Web Vitals timing fields\n    \"fcp.*\",\n    \"fp.*\",\n    \"ttfb.*\",\n    \"lcp.*\",\n    \"fid.*\",\n    \"inp.*\",\n    \"cls.*\",\n\n    // Long tasks\n    \"long_task.*\",\n\n    // Viewport/screen dimensions (vary by test environment)\n    \"browser.window.*\",\n    \"device.screen.*\",\n    \"device.battery.level\"\n] as const;\n\n/**\n * Default event patterns that are ignored during snapshot comparison.\n * These events are volatile and may or may not be recorded depending on browser performance.\n */\nexport const DEFAULT_IGNORED_EVENTS: readonly EventIgnorePattern[] = [\n    { \"characteristics.has_long_task\": true },\n    { \"characteristics.is_self_monitoring\": true }\n];\n\n/**\n * Placeholder value used to replace ignored property values.\n */\nexport const IGNORED_PLACEHOLDER = \"[IGNORED]\";\n\n/**\n * Processes events by ignoring and removing specified properties.\n * - Ignored properties have their values replaced with \"[IGNORED]\" placeholder\n * - Removed properties are deleted entirely from the event\n *\n * @param events        Array of event objects to process\n * @param ignoredFields Array of field paths to ignore (values replaced with placeholder)\n * @param removedFields Array of field paths to remove entirely\n * @returns             Deep copy of events with specified fields processed\n */\nexport function processEventProperties(\n    events: Record<string, unknown>[],\n    ignoredFields: readonly string[] = DEFAULT_IGNORED_FIELDS,\n    removedFields: readonly string[] = DEFAULT_REMOVED_FIELDS\n): Record<string, unknown>[] {\n    return events.map(event => processSingleEvent(event, ignoredFields, removedFields));\n}\n\n/**\n * Filters out events that match any of the ignore patterns.\n * An event matches a pattern if all properties in the pattern exist in the event with the same values.\n *\n * @param events         Array of events to filter\n * @param ignorePatterns Array of patterns to match against\n * @returns              Filtered array excluding events that match any pattern\n */\nexport function filterIgnoredEvents(\n    events: Record<string, unknown>[],\n    ignorePatterns: readonly EventIgnorePattern[] = DEFAULT_IGNORED_EVENTS\n): Record<string, unknown>[] {\n    if (ignorePatterns.length === 0) {\n        return events;\n    }\n\n    return events.filter(event => !matchesAnyPattern(event, ignorePatterns));\n}\n\n/**\n * Checks if an event matches any of the provided patterns.\n *\n * @param event    Event to check\n * @param patterns Patterns to match against\n * @returns        True if the event matches any pattern\n */\nfunction matchesAnyPattern(\n    event: Record<string, unknown>,\n    patterns: readonly EventIgnorePattern[]\n): boolean {\n    return patterns.some(pattern => matchesPattern(event, pattern));\n}\n\n/**\n * Checks if an event matches a single pattern.\n * A match occurs when all properties in the pattern exist in the event with equal values.\n *\n * @param event   Event to check\n * @param pattern Pattern to match against\n * @returns       True if all pattern properties match\n */\nfunction matchesPattern(\n    event: Record<string, unknown>,\n    pattern: EventIgnorePattern\n): boolean {\n    return Object.keys(pattern).every(key => key in event && event[key] === pattern[key]);\n}\n\n/**\n * Processes a single event by ignoring and removing specified properties.\n * - Ignored properties: values replaced with \"[IGNORED]\" placeholder\n * - Removed properties: fields deleted entirely\n * Both support exact field names and wildcard patterns (`prefix.*`).\n *\n * @param event         Event object to process\n * @param ignoredFields Array of field names/patterns to ignore (replace values)\n * @param removedFields Array of field names/patterns to remove entirely\n * @returns             Deep copy of event with fields processed\n */\nfunction processSingleEvent(\n    event: Record<string, unknown>,\n    ignoredFields: readonly string[],\n    removedFields: readonly string[]\n): Record<string, unknown> {\n    const result = deepClone(event);\n\n    // Process ignored properties (replace values with placeholder)\n    for (const fieldPath of ignoredFields) {\n        if (fieldPath.endsWith(\".*\")) {\n            const prefix = fieldPath.slice(0, -1); // Remove \"*\", keep the dot\n            ignoreFieldsWithPrefix(result, prefix);\n        } else if (fieldPath in result) {\n            result[fieldPath] = IGNORED_PLACEHOLDER;\n        }\n    }\n\n    // Process removed properties (delete entirely)\n    for (const fieldPath of removedFields) {\n        if (fieldPath.endsWith(\".*\")) {\n            const prefix = fieldPath.slice(0, -1); // Remove \"*\", keep the dot\n            removeFieldsWithPrefix(result, prefix);\n        } else {\n            delete result[fieldPath];\n        }\n    }\n\n    return result;\n}\n\n/**\n * Replaces values of all fields in an object that start with the given prefix with the ignored placeholder.\n * Used for wildcard patterns like `inp.*` to ignore `inp.start_time`, `inp.duration`, etc.\n *\n * @param obj    Object to modify\n * @param prefix Prefix to match (including trailing dot, e.g., \"inp.\")\n */\nfunction ignoreFieldsWithPrefix(obj: Record<string, unknown>, prefix: string): void {\n    for (const key of Object.keys(obj)) {\n        if (key.startsWith(prefix)) {\n            obj[key] = IGNORED_PLACEHOLDER;\n        }\n    }\n}\n\n/**\n * Removes all fields in an object that start with the given prefix.\n * Used for wildcard patterns like `inp.*` to remove `inp.start_time`, `inp.duration`, etc.\n *\n * @param obj    Object to modify\n * @param prefix Prefix to match (including trailing dot, e.g., \"inp.\")\n */\nfunction removeFieldsWithPrefix(obj: Record<string, unknown>, prefix: string): void {\n    for (const key of Object.keys(obj)) {\n        if (key.startsWith(prefix)) {\n            delete obj[key];\n        }\n    }\n}\n\n/**\n * Type guard to check if a value is a Record<string, unknown>.\n *\n * @param value Value to check\n * @returns     True if value is a non-null object\n */\nfunction isRecord(value: unknown): value is Record<string, unknown> {\n    return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\n/**\n * Type guard to check if a value is an array of Record<string, unknown>.\n *\n * @param value Value to check\n * @returns     True if value is an array where all elements are records\n */\nfunction isRecordArray(value: unknown): value is Record<string, unknown>[] {\n    return Array.isArray(value) && value.every(isRecord);\n}\n\n/**\n * Creates a deep clone of an object.\n *\n * @param obj Object to clone\n * @returns   Deep clone of the object\n */\nfunction deepClone<T>(obj: T): T {\n    return JSON.parse(JSON.stringify(obj));\n}\n\n/**\n * Reads a snapshot file from disk.\n * Returns an object indicating whether the file exists and its parsed contents.\n *\n * @param snapshotPath Absolute path to the snapshot file\n * @returns            Object with exists flag and parsed data\n * @throws             Error if the file contains invalid JSON or is not an array of records\n */\nexport function readSnapshot(snapshotPath: string): SnapshotReadResult {\n    if (!existsSync(snapshotPath)) {\n        return { exists: false };\n    }\n\n    const content = readFileSync(snapshotPath, \"utf8\");\n    const parsed = safeParseJson(content, snapshotPath);\n\n    if (!isRecordArray(parsed)) {\n        throw new Error(\n            `Invalid snapshot format in \"${snapshotPath}\": expected an array of objects`\n        );\n    }\n\n    return { exists: true, data: parsed };\n}\n\n/**\n * Safely parses a JSON string, wrapping errors with context.\n *\n * @param content  JSON string to parse\n * @param filePath Path to the file (for error messages)\n * @returns        Parsed value\n * @throws         Error with context if parsing fails\n */\nfunction safeParseJson(content: string, filePath: string): unknown {\n    try {\n        return JSON.parse(content);\n    } catch (error) {\n        const message = error instanceof Error ? error.message : String(error);\n        throw new Error(`Failed to parse snapshot file \"${filePath}\": ${message}`);\n    }\n}\n\n/**\n * Writes snapshot data to disk.\n * Creates parent directories if they don't exist.\n * Events are sorted by characteristics for consistent ordering.\n * Uses deterministic JSON serialization with sorted keys for consistent diffs.\n *\n * @param snapshotPath Absolute path to the snapshot file\n * @param data         Array of events to write\n */\nexport function writeSnapshot(snapshotPath: string, data: Record<string, unknown>[]): void {\n    const dir = dirname(snapshotPath);\n    if (!existsSync(dir)) {\n        mkdirSync(dir, { recursive: true });\n    }\n\n    const sortedData = sortEventsByCharacteristics(data);\n    const serialized = serializeSnapshot(sortedData);\n    writeFileSync(snapshotPath, serialized, \"utf8\");\n}\n\n/**\n * Serializes snapshot data to a deterministic JSON string.\n * Sorts object keys to ensure consistent output across different Node.js versions.\n *\n * @param data Array of events to serialize\n * @returns    Pretty-printed JSON string with sorted keys\n */\nfunction serializeSnapshot(data: Record<string, unknown>[]): string {\n    return JSON.stringify(data, sortedReplacer, 2) + \"\\n\";\n}\n\n/**\n * JSON replacer function that sorts object keys for deterministic serialization.\n *\n * @param key   Current key (unused)\n * @param value Current value\n * @returns     Value with sorted keys if it's an object\n */\nfunction sortedReplacer(key: string, value: unknown): unknown {\n    if (!isRecord(value)) {\n        return value;\n    }\n\n    const sortedKeys = Object.keys(value).sort();\n    const sorted: Record<string, unknown> = {};\n    for (const sortedKey of sortedKeys) {\n        sorted[sortedKey] = value[sortedKey];\n    }\n    return sorted;\n}\n\n/**\n * Sorts events by their characteristics signature for order-independent comparison.\n * Events with the same characteristics are further sorted by other stable properties.\n * This ensures consistent ordering regardless of event arrival order.\n *\n * @param events Array of events to sort\n * @returns      New array with events sorted by characteristics\n */\nexport function sortEventsByCharacteristics(events: Record<string, unknown>[]): Record<string, unknown>[] {\n    return [...events].sort((a, b) => {\n        const sigA = getCharacteristicsSignature(a);\n        const sigB = getCharacteristicsSignature(b);\n\n        if (sigA !== sigB) {\n            return sigA.localeCompare(sigB);\n        }\n\n        // If characteristics are the same, sort by other stable properties\n        return getSecondarySignature(a).localeCompare(getSecondarySignature(b));\n    });\n}\n\n/**\n * Generates a signature string from an event's characteristics fields.\n * Characteristics fields are boolean flags like \"characteristics.has_navigation\".\n *\n * @param event Event object\n * @returns     Signature string representing the event's characteristics\n */\nfunction getCharacteristicsSignature(event: Record<string, unknown>): string {\n    const characteristicsKeys = Object.keys(event)\n        .filter(key => key.startsWith(\"characteristics.\") && event[key] === true)\n        .sort();\n\n    return characteristicsKeys.join(\",\");\n}\n\n/**\n * Generates a secondary signature for events with the same characteristics.\n * Uses stable identifying properties like name, type, url, etc.\n *\n * @param event Event object\n * @returns     Secondary signature string\n */\nfunction getSecondarySignature(event: Record<string, unknown>): string {\n    // Keys must match RumEventKeys enum values for consistency\n    const stableKeys = [\n        \"url.full\",\n        \"view.sequence.number\",\n        \"view.url.full\",\n        \"page.url.full\",\n        \"user_action.type\",\n        \"interaction.name\",\n        \"navigation.type\",\n        \"exception.type\",\n        \"exception.message\"\n    ];\n\n    return stableKeys\n        .map((key) => {\n            const value = getValueByKey(event, key);\n            if (value !== void 0) {\n                return `${key}:${String(value)}`;\n            }\n            return void 0;\n        })\n        .filter(Boolean)\n        .join(\"|\");\n}\n\n/**\n * Gets a value from an object, trying both literal key and nested path.\n *\n * @param obj Object to get value from\n * @param key Key (may be literal or dot-notation path)\n * @returns   Value or undefined\n */\nfunction getValueByKey(obj: Record<string, unknown>, key: string): unknown {\n    // First try literal key (e.g., \"request.url\" as a single key)\n    if (key in obj) {\n        return obj[key];\n    }\n\n    // Then try nested path (e.g., obj.request.url)\n    const parts = key.split(\".\");\n    let current: unknown = obj;\n\n    for (const part of parts) {\n        if (!isRecord(current) || !(part in current)) {\n            return void 0;\n        }\n        current = current[part];\n    }\n\n    return current;\n}\n"]}
|
package/dist/testing/test.d.ts
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import type { SnapshotOptions } from "./snapshot.js";
|
|
1
2
|
export interface WaitForBeaconsOptions {
|
|
2
3
|
/**
|
|
3
4
|
* The minimum number of beacon requests to wait for
|
|
@@ -50,6 +51,14 @@ export interface DynatraceTesting {
|
|
|
50
51
|
* ```
|
|
51
52
|
*/
|
|
52
53
|
clearEvents(): void;
|
|
54
|
+
/**
|
|
55
|
+
* Compares captured events against a stored snapshot file.
|
|
56
|
+
* On first run, creates the snapshot. On subsequent runs, compares and fails if different.
|
|
57
|
+
* Volatile fields (timestamps, IDs, etc.) are masked by default to prevent flaky tests.
|
|
58
|
+
*
|
|
59
|
+
* @param options Configuration for snapshot comparison
|
|
60
|
+
*/
|
|
61
|
+
toMatchEventSnapshot(options?: SnapshotOptions): Promise<void>;
|
|
53
62
|
}
|
|
54
63
|
export interface BeaconRequest {
|
|
55
64
|
url: string;
|