@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.
@@ -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"]}
@@ -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;