@dynatrace/rum-javascript-sdk 1.329.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.
Files changed (115) hide show
  1. package/LICENSE +174 -0
  2. package/README.md +402 -0
  3. package/dist/api/constants.d.ts +1 -0
  4. package/dist/api/constants.js +2 -0
  5. package/dist/api/dynatrace-error.d.ts +3 -0
  6. package/dist/api/dynatrace-error.js +7 -0
  7. package/dist/api/global-adapter.d.ts +1 -0
  8. package/dist/api/global-adapter.js +4 -0
  9. package/dist/api/index.d.ts +65 -0
  10. package/dist/api/index.js +110 -0
  11. package/dist/api/logging.d.ts +2 -0
  12. package/dist/api/logging.js +8 -0
  13. package/dist/api/promises/index.d.ts +85 -0
  14. package/dist/api/promises/index.js +104 -0
  15. package/dist/api/promises/user-actions.d.ts +35 -0
  16. package/dist/api/promises/user-actions.js +48 -0
  17. package/dist/api/promises/wait-for-agent.d.ts +2 -0
  18. package/dist/api/promises/wait-for-agent.js +77 -0
  19. package/dist/api/promises/wait-for-send-exception-event.d.ts +9 -0
  20. package/dist/api/promises/wait-for-send-exception-event.js +18 -0
  21. package/dist/api/promises/wait-for-user-actions.d.ts +9 -0
  22. package/dist/api/promises/wait-for-user-actions.js +18 -0
  23. package/dist/api/type-guards/is-valid-agent.d.ts +2 -0
  24. package/dist/api/type-guards/is-valid-agent.js +4 -0
  25. package/dist/api/user-actions.d.ts +62 -0
  26. package/dist/api/user-actions.js +99 -0
  27. package/dist/testing/install.d.ts +15 -0
  28. package/dist/testing/install.js +19 -0
  29. package/dist/testing/test.d.ts +111 -0
  30. package/dist/testing/test.js +229 -0
  31. package/dist/types/api/dynatrace-api-types.d.ts +319 -0
  32. package/dist/types/api/dynatrace-api-types.js +7 -0
  33. package/dist/types/api/index.d.ts +1 -0
  34. package/dist/types/api/index.js +2 -0
  35. package/dist/types/index-typedoc.d.ts +31 -0
  36. package/dist/types/index-typedoc.js +32 -0
  37. package/dist/types/index.d.ts +11 -0
  38. package/dist/types/index.js +5 -0
  39. package/dist/types/internal/beacon-query.d.ts +29 -0
  40. package/dist/types/internal/beacon-query.js +32 -0
  41. package/dist/types/internal/data-dt-config-scope.d.ts +10 -0
  42. package/dist/types/internal/data-dt-config-scope.js +12 -0
  43. package/dist/types/internal/feature-hashes.d.ts +19 -0
  44. package/dist/types/internal/feature-hashes.js +22 -0
  45. package/dist/types/internal/index.d.ts +3 -0
  46. package/dist/types/internal/index.js +4 -0
  47. package/dist/types/rum-events/event-context/event-context.d.ts +364 -0
  48. package/dist/types/rum-events/event-context/event-context.js +106 -0
  49. package/dist/types/rum-events/event-context/index.d.ts +1 -0
  50. package/dist/types/rum-events/event-context/index.js +2 -0
  51. package/dist/types/rum-events/event-updates.d.ts +4 -0
  52. package/dist/types/rum-events/event-updates.js +2 -0
  53. package/dist/types/rum-events/index.d.ts +19 -0
  54. package/dist/types/rum-events/index.js +20 -0
  55. package/dist/types/rum-events/json-event.d.ts +84 -0
  56. package/dist/types/rum-events/json-event.js +2 -0
  57. package/dist/types/rum-events/rum-biz-event.d.ts +10 -0
  58. package/dist/types/rum-events/rum-biz-event.js +15 -0
  59. package/dist/types/rum-events/rum-event.d.ts +13 -0
  60. package/dist/types/rum-events/rum-event.js +3 -0
  61. package/dist/types/rum-events/rum-internal-selfmonitoring-event.d.ts +114 -0
  62. package/dist/types/rum-events/rum-internal-selfmonitoring-event.js +99 -0
  63. package/dist/types/rum-events/rum-long-task-event.d.ts +18 -0
  64. package/dist/types/rum-events/rum-long-task-event.js +10 -0
  65. package/dist/types/rum-events/rum-page-summary-event.d.ts +370 -0
  66. package/dist/types/rum-events/rum-page-summary-event.js +184 -0
  67. package/dist/types/rum-events/rum-selfmonitoring-event.d.ts +39 -0
  68. package/dist/types/rum-events/rum-selfmonitoring-event.js +32 -0
  69. package/dist/types/rum-events/rum-session-properties-event.d.ts +9 -0
  70. package/dist/types/rum-events/rum-session-properties-event.js +3 -0
  71. package/dist/types/rum-events/rum-standalone-csp-rule-violation-event.d.ts +16 -0
  72. package/dist/types/rum-events/rum-standalone-csp-rule-violation-event.js +3 -0
  73. package/dist/types/rum-events/rum-standalone-exception-event.d.ts +16 -0
  74. package/dist/types/rum-events/rum-standalone-exception-event.js +2 -0
  75. package/dist/types/rum-events/rum-standalone-navigation-event.d.ts +23 -0
  76. package/dist/types/rum-events/rum-standalone-navigation-event.js +6 -0
  77. package/dist/types/rum-events/rum-user-action-event.d.ts +187 -0
  78. package/dist/types/rum-events/rum-user-action-event.js +78 -0
  79. package/dist/types/rum-events/rum-user-interaction-event.d.ts +302 -0
  80. package/dist/types/rum-events/rum-user-interaction-event.js +131 -0
  81. package/dist/types/rum-events/rum-visibility-change-event.d.ts +15 -0
  82. package/dist/types/rum-events/rum-visibility-change-event.js +6 -0
  83. package/dist/types/rum-events/rum-web-request-event.d.ts +269 -0
  84. package/dist/types/rum-events/rum-web-request-event.js +143 -0
  85. package/dist/types/rum-events/schema-versions.d.ts +4 -0
  86. package/dist/types/rum-events/schema-versions.js +5 -0
  87. package/dist/types/rum-events/shared-namespaces-and-fields/csp-fields.d.ts +38 -0
  88. package/dist/types/rum-events/shared-namespaces-and-fields/csp-fields.js +23 -0
  89. package/dist/types/rum-events/shared-namespaces-and-fields/exception-fields.d.ts +26 -0
  90. package/dist/types/rum-events/shared-namespaces-and-fields/exception-fields.js +19 -0
  91. package/dist/types/rum-events/shared-namespaces-and-fields/general-rum-error-fields.d.ts +66 -0
  92. package/dist/types/rum-events/shared-namespaces-and-fields/general-rum-error-fields.js +66 -0
  93. package/dist/types/rum-events/shared-namespaces-and-fields/general-rum-event-fields.d.ts +142 -0
  94. package/dist/types/rum-events/shared-namespaces-and-fields/general-rum-event-fields.js +98 -0
  95. package/dist/types/rum-events/shared-namespaces-and-fields/http-namespace.d.ts +16 -0
  96. package/dist/types/rum-events/shared-namespaces-and-fields/http-namespace.js +11 -0
  97. package/dist/types/rum-events/shared-namespaces-and-fields/index.d.ts +9 -0
  98. package/dist/types/rum-events/shared-namespaces-and-fields/index.js +10 -0
  99. package/dist/types/rum-events/shared-namespaces-and-fields/navigation-fields.d.ts +48 -0
  100. package/dist/types/rum-events/shared-namespaces-and-fields/navigation-fields.js +47 -0
  101. package/dist/types/rum-events/shared-namespaces-and-fields/page-source-fields.d.ts +9 -0
  102. package/dist/types/rum-events/shared-namespaces-and-fields/page-source-fields.js +5 -0
  103. package/dist/types/rum-events/shared-namespaces-and-fields/request-fields.d.ts +14 -0
  104. package/dist/types/rum-events/shared-namespaces-and-fields/request-fields.js +10 -0
  105. package/dist/types/rum-events/shared-namespaces-and-fields/view-source-fields.d.ts +14 -0
  106. package/dist/types/rum-events/shared-namespaces-and-fields/view-source-fields.js +9 -0
  107. package/dist/types/user-actions/index.d.ts +3 -0
  108. package/dist/types/user-actions/index.js +4 -0
  109. package/dist/types/user-actions/user-action-end-event.d.ts +28 -0
  110. package/dist/types/user-actions/user-action-end-event.js +2 -0
  111. package/dist/types/user-actions/user-action-start-options.d.ts +23 -0
  112. package/dist/types/user-actions/user-action-start-options.js +2 -0
  113. package/dist/types/user-actions/user-action-tracker.d.ts +72 -0
  114. package/dist/types/user-actions/user-action-tracker.js +2 -0
  115. package/package.json +115 -0
@@ -0,0 +1,4 @@
1
+ export function isValidAgent(agent) {
2
+ return typeof agent === "object" && !!agent && "sendEvent" in agent;
3
+ }
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaXMtdmFsaWQtYWdlbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zb3VyY2UvYXBpL3R5cGUtZ3VhcmRzL2lzLXZhbGlkLWFnZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBLE1BQU0sVUFBVSxZQUFZLENBQUMsS0FBYztJQUN2QyxPQUFPLE9BQU8sS0FBSyxLQUFLLFFBQVEsSUFBSSxDQUFDLENBQUMsS0FBSyxJQUFJLFdBQVcsSUFBSSxLQUFLLENBQUM7QUFDeEUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgZHluYXRyYWNlIGFzIER0QXBpIH0gZnJvbSBcIi4uLy4uL3R5cGVzL2FwaS9keW5hdHJhY2UtYXBpLXR5cGVzLmpzXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBpc1ZhbGlkQWdlbnQoYWdlbnQ6IHVua25vd24pOiBhZ2VudCBpcyB0eXBlb2YgRHRBcGkge1xuICAgIHJldHVybiB0eXBlb2YgYWdlbnQgPT09IFwib2JqZWN0XCIgJiYgISFhZ2VudCAmJiBcInNlbmRFdmVudFwiIGluIGFnZW50O1xufVxuIl19
@@ -0,0 +1,62 @@
1
+ import type { UserActionStartOptions, UserActionTracker } from "../types/index.js";
2
+ /**
3
+ * Safe wrapper for {@link dynatrace.userActions.create} which reverts to a noop if RUM JavaScript or the user actions module is not available.
4
+ *
5
+ * @param options An optional options object to configure the user action.
6
+ * @returns A `UserActionTracker` object that can be started and finished, or `undefined` if the user actions module is not enabled.
7
+ *
8
+ * @example
9
+ * ```typescript
10
+ * import * as userActions from "@dynatrace/rum-javascript-sdk/api/user-actions";
11
+ * const userAction = userActions.create({ autoClose: false });
12
+ * // perform work
13
+ * userAction?.finish();
14
+ * ```
15
+ */
16
+ export declare function create(options?: UserActionStartOptions): UserActionTracker | undefined;
17
+ /**
18
+ * Safe wrapper for {@link dynatrace.userActions.subscribe} which reverts to a noop if RUM JavaScript or the user actions module is not available.
19
+ *
20
+ * @param subscriber A callback function that is called whenever Dynatrace creates a new user action.
21
+ * @returns An unsubscriber function to remove the subscription, or undefined if the user actions module is not enabled.
22
+ *
23
+ * @example
24
+ * ```typescript
25
+ * import * as userActions from "@dynatrace/rum-javascript-sdk/api/user-actions";
26
+ * const unsubscribe = userActions.subscribe((userAction) => {
27
+ * console.log('New user action created');
28
+ * userAction.autoClose = false;
29
+ * });
30
+ * // later
31
+ * unsubscribe?.();
32
+ * ```
33
+ */
34
+ export declare function subscribe(subscriber: (userAction: UserActionTracker) => void): (() => void) | undefined;
35
+ /**
36
+ * Safe wrapper for {@link dynatrace.userActions.setAutomaticDetection} which reverts to a noop if RUM JavaScript or the user actions module is not available.
37
+ *
38
+ * @param automaticDetection If false, disables automatic user action detection, otherwise enables it.
39
+ *
40
+ * @example
41
+ * ```typescript
42
+ * import * as userActions from "@dynatrace/rum-javascript-sdk/api/user-actions";
43
+ * userActions.setAutomaticDetection(false);
44
+ * // manual user action handling
45
+ * ```
46
+ */
47
+ export declare function setAutomaticDetection(automaticDetection: boolean): void;
48
+ /**
49
+ * Safe wrapper for {@link dynatrace.userActions.current} which returns undefined if RUM JavaScript or the user actions module is not available.
50
+ *
51
+ * @returns The current user action, or `undefined` if no user action is in progress or the user actions module is not enabled.
52
+ *
53
+ * @example
54
+ * ```typescript
55
+ * import * as userActions from "@dynatrace/rum-javascript-sdk/api/user-actions";
56
+ * const currentAction = userActions.getCurrent();
57
+ * if (currentAction) {
58
+ * currentAction.autoClose = false;
59
+ * }
60
+ * ```
61
+ */
62
+ export declare function getCurrent(): UserActionTracker | undefined;
@@ -0,0 +1,99 @@
1
+ import { getGlobal } from "./global-adapter.js";
2
+ import { isValidAgent } from "./type-guards/is-valid-agent.js";
3
+ import { log } from "./logging.js";
4
+ /**
5
+ * Safe wrapper for {@link dynatrace.userActions.create} which reverts to a noop if RUM JavaScript or the user actions module is not available.
6
+ *
7
+ * @param options An optional options object to configure the user action.
8
+ * @returns A `UserActionTracker` object that can be started and finished, or `undefined` if the user actions module is not enabled.
9
+ *
10
+ * @example
11
+ * ```typescript
12
+ * import * as userActions from "@dynatrace/rum-javascript-sdk/api/user-actions";
13
+ * const userAction = userActions.create({ autoClose: false });
14
+ * // perform work
15
+ * userAction?.finish();
16
+ * ```
17
+ */
18
+ export function create(options) {
19
+ return guardUserActions((userActions) => {
20
+ return userActions.create(options);
21
+ });
22
+ }
23
+ /**
24
+ * Safe wrapper for {@link dynatrace.userActions.subscribe} which reverts to a noop if RUM JavaScript or the user actions module is not available.
25
+ *
26
+ * @param subscriber A callback function that is called whenever Dynatrace creates a new user action.
27
+ * @returns An unsubscriber function to remove the subscription, or undefined if the user actions module is not enabled.
28
+ *
29
+ * @example
30
+ * ```typescript
31
+ * import * as userActions from "@dynatrace/rum-javascript-sdk/api/user-actions";
32
+ * const unsubscribe = userActions.subscribe((userAction) => {
33
+ * console.log('New user action created');
34
+ * userAction.autoClose = false;
35
+ * });
36
+ * // later
37
+ * unsubscribe?.();
38
+ * ```
39
+ */
40
+ export function subscribe(subscriber) {
41
+ return guardUserActions((userActions) => {
42
+ return userActions.subscribe(subscriber);
43
+ });
44
+ }
45
+ /**
46
+ * Safe wrapper for {@link dynatrace.userActions.setAutomaticDetection} which reverts to a noop if RUM JavaScript or the user actions module is not available.
47
+ *
48
+ * @param automaticDetection If false, disables automatic user action detection, otherwise enables it.
49
+ *
50
+ * @example
51
+ * ```typescript
52
+ * import * as userActions from "@dynatrace/rum-javascript-sdk/api/user-actions";
53
+ * userActions.setAutomaticDetection(false);
54
+ * // manual user action handling
55
+ * ```
56
+ */
57
+ export function setAutomaticDetection(automaticDetection) {
58
+ guardUserActions((userActions) => {
59
+ userActions.setAutomaticDetection(automaticDetection);
60
+ });
61
+ }
62
+ /**
63
+ * Safe wrapper for {@link dynatrace.userActions.current} which returns undefined if RUM JavaScript or the user actions module is not available.
64
+ *
65
+ * @returns The current user action, or `undefined` if no user action is in progress or the user actions module is not enabled.
66
+ *
67
+ * @example
68
+ * ```typescript
69
+ * import * as userActions from "@dynatrace/rum-javascript-sdk/api/user-actions";
70
+ * const currentAction = userActions.getCurrent();
71
+ * if (currentAction) {
72
+ * currentAction.autoClose = false;
73
+ * }
74
+ * ```
75
+ */
76
+ export function getCurrent() {
77
+ return guardUserActions((userActions) => {
78
+ return userActions.current;
79
+ });
80
+ }
81
+ /**
82
+ * Safe wrapper for various user actions API functions which reverts to a noop if RUM JavaScript or the user actions module is not available.
83
+ *
84
+ * @param fn The function to be executed if the user actions module is available.
85
+ * @returns The result of the provided function call if the user actions module is available, otherwise undefined.
86
+ */
87
+ function guardUserActions(fn) {
88
+ const globalDt = getGlobal("dynatrace");
89
+ if (!isValidAgent(globalDt)) {
90
+ log("dynatrace is not available");
91
+ return void 0;
92
+ }
93
+ if (!globalDt.userActions) {
94
+ log("dynatrace.userActions module is not available");
95
+ return void 0;
96
+ }
97
+ return fn(globalDt.userActions);
98
+ }
99
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXNlci1hY3Rpb25zLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc291cmNlL2FwaS91c2VyLWFjdGlvbnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBS0EsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ2hELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUMvRCxPQUFPLEVBQUUsR0FBRyxFQUFFLE1BQU0sY0FBYyxDQUFDO0FBRW5DOzs7Ozs7Ozs7Ozs7O0dBYUc7QUFDSCxNQUFNLFVBQVUsTUFBTSxDQUFDLE9BQWdDO0lBQ25ELE9BQU8sZ0JBQWdCLENBQUMsQ0FBQyxXQUFXLEVBQUUsRUFBRTtRQUNwQyxPQUFPLFdBQVcsQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDdkMsQ0FBQyxDQUFDLENBQUM7QUFDUCxDQUFDO0FBRUQ7Ozs7Ozs7Ozs7Ozs7Ozs7R0FnQkc7QUFDSCxNQUFNLFVBQVUsU0FBUyxDQUFDLFVBQW1EO0lBQ3pFLE9BQU8sZ0JBQWdCLENBQUMsQ0FBQyxXQUFXLEVBQUUsRUFBRTtRQUNwQyxPQUFPLFdBQVcsQ0FBQyxTQUFTLENBQUMsVUFBVSxDQUFDLENBQUM7SUFDN0MsQ0FBQyxDQUFDLENBQUM7QUFDUCxDQUFDO0FBRUQ7Ozs7Ozs7Ozs7O0dBV0c7QUFDSCxNQUFNLFVBQVUscUJBQXFCLENBQUMsa0JBQTJCO0lBQzdELGdCQUFnQixDQUFDLENBQUMsV0FBVyxFQUFFLEVBQUU7UUFDN0IsV0FBVyxDQUFDLHFCQUFxQixDQUFDLGtCQUFrQixDQUFDLENBQUM7SUFDMUQsQ0FBQyxDQUFDLENBQUM7QUFDUCxDQUFDO0FBRUQ7Ozs7Ozs7Ozs7Ozs7R0FhRztBQUNILE1BQU0sVUFBVSxVQUFVO0lBQ3RCLE9BQU8sZ0JBQWdCLENBQUMsQ0FBQyxXQUFXLEVBQUUsRUFBRTtRQUNwQyxPQUFPLFdBQVcsQ0FBQyxPQUFPLENBQUM7SUFDL0IsQ0FBQyxDQUFDLENBQUM7QUFDUCxDQUFDO0FBRUQ7Ozs7O0dBS0c7QUFDSCxTQUFTLGdCQUFnQixDQUE4QyxFQUFLO0lBQ3hFLE1BQU0sUUFBUSxHQUFHLFNBQVMsQ0FBQyxXQUFXLENBQUMsQ0FBQztJQUN4QyxJQUFJLENBQUMsWUFBWSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUM7UUFDMUIsR0FBRyxDQUFDLDRCQUE0QixDQUFDLENBQUM7UUFDbEMsT0FBTyxLQUFLLENBQUMsQ0FBQztJQUNsQixDQUFDO0lBRUQsSUFBSSxDQUFDLFFBQVEsQ0FBQyxXQUFXLEVBQUUsQ0FBQztRQUN4QixHQUFHLENBQUMsK0NBQStDLENBQUMsQ0FBQztRQUNyRCxPQUFPLEtBQUssQ0FBQyxDQUFDO0lBQ2xCLENBQUM7SUFFRCxPQUFPLEVBQUUsQ0FBQyxRQUFRLENBQUMsV0FBVyxDQUFDLENBQUM7QUFDcEMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHtcbiAgICBVc2VyQWN0aW9ucyxcbiAgICBVc2VyQWN0aW9uU3RhcnRPcHRpb25zLFxuICAgIFVzZXJBY3Rpb25UcmFja2VyXG59IGZyb20gXCIuLi90eXBlcy9pbmRleC5qc1wiO1xuaW1wb3J0IHsgZ2V0R2xvYmFsIH0gZnJvbSBcIi4vZ2xvYmFsLWFkYXB0ZXIuanNcIjtcbmltcG9ydCB7IGlzVmFsaWRBZ2VudCB9IGZyb20gXCIuL3R5cGUtZ3VhcmRzL2lzLXZhbGlkLWFnZW50LmpzXCI7XG5pbXBvcnQgeyBsb2cgfSBmcm9tIFwiLi9sb2dnaW5nLmpzXCI7XG5cbi8qKlxuICogU2FmZSB3cmFwcGVyIGZvciB7QGxpbmsgZHluYXRyYWNlLnVzZXJBY3Rpb25zLmNyZWF0ZX0gd2hpY2ggcmV2ZXJ0cyB0byBhIG5vb3AgaWYgUlVNIEphdmFTY3JpcHQgb3IgdGhlIHVzZXIgYWN0aW9ucyBtb2R1bGUgaXMgbm90IGF2YWlsYWJsZS5cbiAqXG4gKiBAcGFyYW0gb3B0aW9ucyBBbiBvcHRpb25hbCBvcHRpb25zIG9iamVjdCB0byBjb25maWd1cmUgdGhlIHVzZXIgYWN0aW9uLlxuICogQHJldHVybnMgICAgICAgQSBgVXNlckFjdGlvblRyYWNrZXJgIG9iamVjdCB0aGF0IGNhbiBiZSBzdGFydGVkIGFuZCBmaW5pc2hlZCwgb3IgYHVuZGVmaW5lZGAgaWYgdGhlIHVzZXIgYWN0aW9ucyBtb2R1bGUgaXMgbm90IGVuYWJsZWQuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCAqIGFzIHVzZXJBY3Rpb25zIGZyb20gXCJAZHluYXRyYWNlL3J1bS1qYXZhc2NyaXB0LXNkay9hcGkvdXNlci1hY3Rpb25zXCI7XG4gKiBjb25zdCB1c2VyQWN0aW9uID0gdXNlckFjdGlvbnMuY3JlYXRlKHsgYXV0b0Nsb3NlOiBmYWxzZSB9KTtcbiAqIC8vIHBlcmZvcm0gd29ya1xuICogdXNlckFjdGlvbj8uZmluaXNoKCk7XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZShvcHRpb25zPzogVXNlckFjdGlvblN0YXJ0T3B0aW9ucyk6IFVzZXJBY3Rpb25UcmFja2VyIHwgdW5kZWZpbmVkIHtcbiAgICByZXR1cm4gZ3VhcmRVc2VyQWN0aW9ucygodXNlckFjdGlvbnMpID0+IHtcbiAgICAgICAgcmV0dXJuIHVzZXJBY3Rpb25zLmNyZWF0ZShvcHRpb25zKTtcbiAgICB9KTtcbn1cblxuLyoqXG4gKiBTYWZlIHdyYXBwZXIgZm9yIHtAbGluayBkeW5hdHJhY2UudXNlckFjdGlvbnMuc3Vic2NyaWJlfSB3aGljaCByZXZlcnRzIHRvIGEgbm9vcCBpZiBSVU0gSmF2YVNjcmlwdCBvciB0aGUgdXNlciBhY3Rpb25zIG1vZHVsZSBpcyBub3QgYXZhaWxhYmxlLlxuICpcbiAqIEBwYXJhbSBzdWJzY3JpYmVyIEEgY2FsbGJhY2sgZnVuY3Rpb24gdGhhdCBpcyBjYWxsZWQgd2hlbmV2ZXIgRHluYXRyYWNlIGNyZWF0ZXMgYSBuZXcgdXNlciBhY3Rpb24uXG4gKiBAcmV0dXJucyAgICAgICAgICBBbiB1bnN1YnNjcmliZXIgZnVuY3Rpb24gdG8gcmVtb3ZlIHRoZSBzdWJzY3JpcHRpb24sIG9yIHVuZGVmaW5lZCBpZiB0aGUgdXNlciBhY3Rpb25zIG1vZHVsZSBpcyBub3QgZW5hYmxlZC5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogaW1wb3J0ICogYXMgdXNlckFjdGlvbnMgZnJvbSBcIkBkeW5hdHJhY2UvcnVtLWphdmFzY3JpcHQtc2RrL2FwaS91c2VyLWFjdGlvbnNcIjtcbiAqIGNvbnN0IHVuc3Vic2NyaWJlID0gdXNlckFjdGlvbnMuc3Vic2NyaWJlKCh1c2VyQWN0aW9uKSA9PiB7XG4gKiAgICAgY29uc29sZS5sb2coJ05ldyB1c2VyIGFjdGlvbiBjcmVhdGVkJyk7XG4gKiAgICAgdXNlckFjdGlvbi5hdXRvQ2xvc2UgPSBmYWxzZTtcbiAqIH0pO1xuICogLy8gbGF0ZXJcbiAqIHVuc3Vic2NyaWJlPy4oKTtcbiAqIGBgYFxuICovXG5leHBvcnQgZnVuY3Rpb24gc3Vic2NyaWJlKHN1YnNjcmliZXI6ICh1c2VyQWN0aW9uOiBVc2VyQWN0aW9uVHJhY2tlcikgPT4gdm9pZCk6ICgoKSA9PiB2b2lkKSB8IHVuZGVmaW5lZCB7XG4gICAgcmV0dXJuIGd1YXJkVXNlckFjdGlvbnMoKHVzZXJBY3Rpb25zKSA9PiB7XG4gICAgICAgIHJldHVybiB1c2VyQWN0aW9ucy5zdWJzY3JpYmUoc3Vic2NyaWJlcik7XG4gICAgfSk7XG59XG5cbi8qKlxuICogU2FmZSB3cmFwcGVyIGZvciB7QGxpbmsgZHluYXRyYWNlLnVzZXJBY3Rpb25zLnNldEF1dG9tYXRpY0RldGVjdGlvbn0gd2hpY2ggcmV2ZXJ0cyB0byBhIG5vb3AgaWYgUlVNIEphdmFTY3JpcHQgb3IgdGhlIHVzZXIgYWN0aW9ucyBtb2R1bGUgaXMgbm90IGF2YWlsYWJsZS5cbiAqXG4gKiBAcGFyYW0gYXV0b21hdGljRGV0ZWN0aW9uIElmIGZhbHNlLCBkaXNhYmxlcyBhdXRvbWF0aWMgdXNlciBhY3Rpb24gZGV0ZWN0aW9uLCBvdGhlcndpc2UgZW5hYmxlcyBpdC5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogaW1wb3J0ICogYXMgdXNlckFjdGlvbnMgZnJvbSBcIkBkeW5hdHJhY2UvcnVtLWphdmFzY3JpcHQtc2RrL2FwaS91c2VyLWFjdGlvbnNcIjtcbiAqIHVzZXJBY3Rpb25zLnNldEF1dG9tYXRpY0RldGVjdGlvbihmYWxzZSk7XG4gKiAvLyBtYW51YWwgdXNlciBhY3Rpb24gaGFuZGxpbmdcbiAqIGBgYFxuICovXG5leHBvcnQgZnVuY3Rpb24gc2V0QXV0b21hdGljRGV0ZWN0aW9uKGF1dG9tYXRpY0RldGVjdGlvbjogYm9vbGVhbik6IHZvaWQge1xuICAgIGd1YXJkVXNlckFjdGlvbnMoKHVzZXJBY3Rpb25zKSA9PiB7XG4gICAgICAgIHVzZXJBY3Rpb25zLnNldEF1dG9tYXRpY0RldGVjdGlvbihhdXRvbWF0aWNEZXRlY3Rpb24pO1xuICAgIH0pO1xufVxuXG4vKipcbiAqIFNhZmUgd3JhcHBlciBmb3Ige0BsaW5rIGR5bmF0cmFjZS51c2VyQWN0aW9ucy5jdXJyZW50fSB3aGljaCByZXR1cm5zIHVuZGVmaW5lZCBpZiBSVU0gSmF2YVNjcmlwdCBvciB0aGUgdXNlciBhY3Rpb25zIG1vZHVsZSBpcyBub3QgYXZhaWxhYmxlLlxuICpcbiAqIEByZXR1cm5zIFRoZSBjdXJyZW50IHVzZXIgYWN0aW9uLCBvciBgdW5kZWZpbmVkYCBpZiBubyB1c2VyIGFjdGlvbiBpcyBpbiBwcm9ncmVzcyBvciB0aGUgdXNlciBhY3Rpb25zIG1vZHVsZSBpcyBub3QgZW5hYmxlZC5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogaW1wb3J0ICogYXMgdXNlckFjdGlvbnMgZnJvbSBcIkBkeW5hdHJhY2UvcnVtLWphdmFzY3JpcHQtc2RrL2FwaS91c2VyLWFjdGlvbnNcIjtcbiAqIGNvbnN0IGN1cnJlbnRBY3Rpb24gPSB1c2VyQWN0aW9ucy5nZXRDdXJyZW50KCk7XG4gKiBpZiAoY3VycmVudEFjdGlvbikge1xuICogICAgIGN1cnJlbnRBY3Rpb24uYXV0b0Nsb3NlID0gZmFsc2U7XG4gKiB9XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldEN1cnJlbnQoKTogVXNlckFjdGlvblRyYWNrZXIgfCB1bmRlZmluZWQge1xuICAgIHJldHVybiBndWFyZFVzZXJBY3Rpb25zKCh1c2VyQWN0aW9ucykgPT4ge1xuICAgICAgICByZXR1cm4gdXNlckFjdGlvbnMuY3VycmVudDtcbiAgICB9KTtcbn1cblxuLyoqXG4gKiBTYWZlIHdyYXBwZXIgZm9yIHZhcmlvdXMgdXNlciBhY3Rpb25zIEFQSSBmdW5jdGlvbnMgd2hpY2ggcmV2ZXJ0cyB0byBhIG5vb3AgaWYgUlVNIEphdmFTY3JpcHQgb3IgdGhlIHVzZXIgYWN0aW9ucyBtb2R1bGUgaXMgbm90IGF2YWlsYWJsZS5cbiAqXG4gKiBAcGFyYW0gZm4gVGhlIGZ1bmN0aW9uIHRvIGJlIGV4ZWN1dGVkIGlmIHRoZSB1c2VyIGFjdGlvbnMgbW9kdWxlIGlzIGF2YWlsYWJsZS5cbiAqIEByZXR1cm5zICBUaGUgcmVzdWx0IG9mIHRoZSBwcm92aWRlZCBmdW5jdGlvbiBjYWxsIGlmIHRoZSB1c2VyIGFjdGlvbnMgbW9kdWxlIGlzIGF2YWlsYWJsZSwgb3RoZXJ3aXNlIHVuZGVmaW5lZC5cbiAqL1xuZnVuY3Rpb24gZ3VhcmRVc2VyQWN0aW9uczxUIGV4dGVuZHMgKHVzZXJBY3Rpb25zOiBVc2VyQWN0aW9ucykgPT4gYW55PihmbjogVCk6IFJldHVyblR5cGU8VD4gfCB1bmRlZmluZWQge1xuICAgIGNvbnN0IGdsb2JhbER0ID0gZ2V0R2xvYmFsKFwiZHluYXRyYWNlXCIpO1xuICAgIGlmICghaXNWYWxpZEFnZW50KGdsb2JhbER0KSkge1xuICAgICAgICBsb2coXCJkeW5hdHJhY2UgaXMgbm90IGF2YWlsYWJsZVwiKTtcbiAgICAgICAgcmV0dXJuIHZvaWQgMDtcbiAgICB9XG5cbiAgICBpZiAoIWdsb2JhbER0LnVzZXJBY3Rpb25zKSB7XG4gICAgICAgIGxvZyhcImR5bmF0cmFjZS51c2VyQWN0aW9ucyBtb2R1bGUgaXMgbm90IGF2YWlsYWJsZVwiKTtcbiAgICAgICAgcmV0dXJuIHZvaWQgMDtcbiAgICB9XG5cbiAgICByZXR1cm4gZm4oZ2xvYmFsRHQudXNlckFjdGlvbnMpO1xufVxuIl19
@@ -0,0 +1,15 @@
1
+ /**
2
+ * Fetches the RUM JavaScript from the given environment.
3
+ *
4
+ * @param endpointUrl The location to fetch the script from, e.g. "https://{your-environment-id}.live.dynatrace.com" or "https://{your-activegate-domain}:9999/e/{your-environment-id}".
5
+ * @param appId The RUM application ID to fetch the configuration.
6
+ * @param token The authentication token to access the API.
7
+ * @returns A promise that resolves with an HTML string containing the RUM JavaScript for insertion into a page.
8
+ *
9
+ * @example
10
+ * const rumJavaScriptHtml = await fetchRumJavaScript("https://myenv.live.dynatrace.com","APPLICATION-EA7C4B59F27D43EB", "dt0c01.abcd.efghijk");
11
+ * // use the result to write your javascript, e.g. with cheerio:
12
+ * const $ = cheerio.load(myHtml);
13
+ * $("head").append(rumJavaScriptHtml);
14
+ */
15
+ export declare function fetchRumJavaScript(endpointUrl: string, appId: string, token: string): Promise<string>;
@@ -0,0 +1,19 @@
1
+ /**
2
+ * Fetches the RUM JavaScript from the given environment.
3
+ *
4
+ * @param endpointUrl The location to fetch the script from, e.g. "https://{your-environment-id}.live.dynatrace.com" or "https://{your-activegate-domain}:9999/e/{your-environment-id}".
5
+ * @param appId The RUM application ID to fetch the configuration.
6
+ * @param token The authentication token to access the API.
7
+ * @returns A promise that resolves with an HTML string containing the RUM JavaScript for insertion into a page.
8
+ *
9
+ * @example
10
+ * const rumJavaScriptHtml = await fetchRumJavaScript("https://myenv.live.dynatrace.com","APPLICATION-EA7C4B59F27D43EB", "dt0c01.abcd.efghijk");
11
+ * // use the result to write your javascript, e.g. with cheerio:
12
+ * const $ = cheerio.load(myHtml);
13
+ * $("head").append(rumJavaScriptHtml);
14
+ */
15
+ export async function fetchRumJavaScript(endpointUrl, appId, token) {
16
+ const response = await fetch(`${endpointUrl}/api/v2/rum/javaScriptTag/${appId}`, { headers: { Authorization: `Api-Token ${token}` } });
17
+ return response.text();
18
+ }
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5zdGFsbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NvdXJjZS90ZXN0aW5nL2luc3RhbGwudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7Ozs7R0FhRztBQUNILE1BQU0sQ0FBQyxLQUFLLFVBQVUsa0JBQWtCLENBQUMsV0FBbUIsRUFBRSxLQUFhLEVBQUUsS0FBYTtJQUN0RixNQUFNLFFBQVEsR0FBRyxNQUFNLEtBQUssQ0FDeEIsR0FBRyxXQUFXLDZCQUE2QixLQUFLLEVBQUUsRUFDbEQsRUFBRSxPQUFPLEVBQUUsRUFBRSxhQUFhLEVBQUUsYUFBYSxLQUFLLEVBQUUsRUFBRSxFQUFFLENBQUMsQ0FBQztJQUMxRCxPQUFPLFFBQVEsQ0FBQyxJQUFJLEVBQUUsQ0FBQztBQUMzQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBGZXRjaGVzIHRoZSBSVU0gSmF2YVNjcmlwdCBmcm9tIHRoZSBnaXZlbiBlbnZpcm9ubWVudC5cbiAqXG4gKiBAcGFyYW0gZW5kcG9pbnRVcmwgVGhlIGxvY2F0aW9uIHRvIGZldGNoIHRoZSBzY3JpcHQgZnJvbSwgZS5nLiBcImh0dHBzOi8ve3lvdXItZW52aXJvbm1lbnQtaWR9LmxpdmUuZHluYXRyYWNlLmNvbVwiIG9yIFwiaHR0cHM6Ly97eW91ci1hY3RpdmVnYXRlLWRvbWFpbn06OTk5OS9lL3t5b3VyLWVudmlyb25tZW50LWlkfVwiLlxuICogQHBhcmFtIGFwcElkICAgICAgIFRoZSBSVU0gYXBwbGljYXRpb24gSUQgdG8gZmV0Y2ggdGhlIGNvbmZpZ3VyYXRpb24uXG4gKiBAcGFyYW0gdG9rZW4gICAgICAgVGhlIGF1dGhlbnRpY2F0aW9uIHRva2VuIHRvIGFjY2VzcyB0aGUgQVBJLlxuICogQHJldHVybnMgICAgICAgICAgIEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHdpdGggYW4gSFRNTCBzdHJpbmcgY29udGFpbmluZyB0aGUgUlVNIEphdmFTY3JpcHQgZm9yIGluc2VydGlvbiBpbnRvIGEgcGFnZS5cbiAqXG4gKiBAZXhhbXBsZVxuICogY29uc3QgcnVtSmF2YVNjcmlwdEh0bWwgPSBhd2FpdCBmZXRjaFJ1bUphdmFTY3JpcHQoXCJodHRwczovL215ZW52LmxpdmUuZHluYXRyYWNlLmNvbVwiLFwiQVBQTElDQVRJT04tRUE3QzRCNTlGMjdENDNFQlwiLCBcImR0MGMwMS5hYmNkLmVmZ2hpamtcIik7XG4gKiAvLyB1c2UgdGhlIHJlc3VsdCB0byB3cml0ZSB5b3VyIGphdmFzY3JpcHQsIGUuZy4gd2l0aCBjaGVlcmlvOlxuICogY29uc3QgJCA9IGNoZWVyaW8ubG9hZChteUh0bWwpO1xuICogJChcImhlYWRcIikuYXBwZW5kKHJ1bUphdmFTY3JpcHRIdG1sKTtcbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGZldGNoUnVtSmF2YVNjcmlwdChlbmRwb2ludFVybDogc3RyaW5nLCBhcHBJZDogc3RyaW5nLCB0b2tlbjogc3RyaW5nKTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IGZldGNoKFxuICAgICAgICBgJHtlbmRwb2ludFVybH0vYXBpL3YyL3J1bS9qYXZhU2NyaXB0VGFnLyR7YXBwSWR9YCxcbiAgICAgICAgeyBoZWFkZXJzOiB7IEF1dGhvcml6YXRpb246IGBBcGktVG9rZW4gJHt0b2tlbn1gIH0gfSk7XG4gICAgcmV0dXJuIHJlc3BvbnNlLnRleHQoKTtcbn1cbiJdfQ==
@@ -0,0 +1,111 @@
1
+ export interface WaitForBeaconsOptions {
2
+ /**
3
+ * The minimum number of beacon requests to wait for
4
+ */
5
+ minCount?: number;
6
+ /**
7
+ * The maximum time to wait for beacon requests, in milliseconds - Default: 5_000
8
+ */
9
+ timeout?: number;
10
+ }
11
+ export interface ExpectOptions {
12
+ /**
13
+ * The maximum time to wait for the event to be sent, in milliseconds - Default: 5_000
14
+ */
15
+ timeout?: number;
16
+ }
17
+ export interface DynatraceTesting {
18
+ /**
19
+ * Waits for a specified number of beacon requests or until a timeout occurs.
20
+ *
21
+ * @param options Configuration options for waiting for beacons
22
+ * @returns A promise that resolves with an array of beacon requests
23
+ */
24
+ waitForBeacons(options?: WaitForBeaconsOptions): Promise<BeaconRequest[]>;
25
+ /**
26
+ * Verifies that a specific event has been sent, optionally within a timeout period.
27
+ *
28
+ * @param event The event to check, represented as a key-value pair object. This uses Playwrights expect(event).toMatchObject.
29
+ * @param options Configuration options for the verification
30
+ * @returns A promise that resolves when the event is confirmed to have been sent
31
+ */
32
+ expectToHaveSentEvent(event: Record<string, unknown>, options?: ExpectOptions): Promise<void>;
33
+ /**
34
+ * Verifies that a specific event has been sent a specified number of times, optionally within a timeout period.
35
+ *
36
+ * @param event The event to check, represented as a key-value pair object. This uses Playwrights expect(event).toMatchObject.
37
+ * @param times The exact number of times the event is expected to have been sent
38
+ * @param options Configuration options for the verification
39
+ * @returns A promise that resolves when the event is confirmed to have been sent the specified number of times
40
+ */
41
+ expectToHaveSentEventTimes(event: Record<string, unknown>, times: number, options?: ExpectOptions): Promise<void>;
42
+ /**
43
+ * Clears all events and beacons, allowing for subsequent expectations to ignore events that have been sent
44
+ * in the past. Example:
45
+ * ```
46
+ * sendFooEvent();
47
+ * await dynatraceTesting.expectToHaveSentEventTimes({ foo: "bar" }, 1);
48
+ * dynatraceTesting.clearEvents();
49
+ * await dynatraceTesting.expectToHaveSentEventTimes({ foo: "bar" }, 1);
50
+ * ```
51
+ */
52
+ clearEvents(): void;
53
+ }
54
+ export interface BeaconRequest {
55
+ url: string;
56
+ body: Record<string, unknown>;
57
+ }
58
+ export interface DynatraceTestingFixture {
59
+ /**
60
+ * The testing API providing expect methods.
61
+ */
62
+ dynatraceTesting: DynatraceTesting;
63
+ /**
64
+ * Configuration to enable Dynatrace testing with a given environment.
65
+ */
66
+ dynatraceConfig: DynatraceConfig;
67
+ }
68
+ export interface DynatraceConfig {
69
+ /**
70
+ * The URL of the Dynatrace API endpoint to connect to, see
71
+ * [RUM manual insertion tags API](https://docs.dynatrace.com/docs/discover-dynatrace/references/dynatrace-api/environment-api/rum/rum-manual-insertion-tags)
72
+ * for details. Example: https://{your-environment-id}.live.dynatrace.com or
73
+ * https://{your-activegate-domain}/e/{your-environment-id}
74
+ */
75
+ endpointUrl: string;
76
+ /**
77
+ * The application ID to identify the correct RUM JavaScript to fetch. You can find
78
+ * this in the URL if you open the Experience Vitals app and select the frontend you
79
+ * want to test. Example: APPLICATION-ABCDEF0123456789
80
+ */
81
+ appId: string;
82
+ /**
83
+ * The authentication token for the installation. See
84
+ * [Tokens and authentication](https://docs.dynatrace.com/docs/discover-dynatrace/references/dynatrace-api/basics/dynatrace-api-authentication).
85
+ */
86
+ token: string;
87
+ /**
88
+ * Accepts an array of regular expressions to match against Dynatrace-related warnings. By default, these warnings
89
+ * trigger test fails. If you anticipate warnings and don't want your tests to fail, use this setting to ignore them
90
+ * by message. Example:
91
+ * ["invalid property my_app_data\.\\w+[0-9_]*"]
92
+ */
93
+ ignoreWarnings?: string[];
94
+ /**
95
+ * Dump the array of Dynatrace events into the console in case an assertion fails.
96
+ */
97
+ dumpEventsOnFail?: boolean;
98
+ }
99
+ /**
100
+ * Worker-scoped fixtures for caching RUM JavaScript across tests in the same worker.
101
+ * This improves performance by avoiding redundant network requests.
102
+ */
103
+ export interface DynatraceTestingWorkerFixture {
104
+ /**
105
+ * Cache for RUM JavaScript snippets, scoped to the worker.
106
+ *
107
+ * @internal
108
+ */
109
+ rumJavaScriptCache: Map<string, string>;
110
+ }
111
+ export declare const test: import("@playwright/test").TestType<import("@playwright/test").PlaywrightTestArgs & import("@playwright/test").PlaywrightTestOptions & DynatraceTestingFixture, import("@playwright/test").PlaywrightWorkerArgs & import("@playwright/test").PlaywrightWorkerOptions & DynatraceTestingWorkerFixture>;
@@ -0,0 +1,229 @@
1
+ import { expect, test as base } from "@playwright/test";
2
+ import { fetchRumJavaScript } from "./install.js";
3
+ import { uncompress } from "snappyjs";
4
+ const DEFAULT_TIMEOUT = 5_000;
5
+ const decoder = new TextDecoder();
6
+ export const test = base.extend({
7
+ dynatraceConfig: [{
8
+ appId: "",
9
+ endpointUrl: "",
10
+ token: ""
11
+ }, { option: true }],
12
+ // eslint-disable-next-line no-empty-pattern -- Playwright requires object destructuring even when no dependencies are needed
13
+ rumJavaScriptCache: [async ({}, use) => {
14
+ const cache = new Map();
15
+ await use(cache);
16
+ }, { scope: "worker", auto: true, box: true }],
17
+ dynatraceTesting: async ({ page, dynatraceConfig, rumJavaScriptCache }, use) => {
18
+ const { endpointUrl, appId, token, ignoreWarnings = [], dumpEventsOnFail = false } = dynatraceConfig;
19
+ const beacons = [];
20
+ const events = [];
21
+ const warnings = [];
22
+ const ignoreRegexes = ignoreWarnings.map(x => new RegExp(x, "i"));
23
+ page.on("console", (msg) => {
24
+ if (msg.text().includes("ynatrace") && !ignoreRegexes.some(regex => regex.test(msg.text()))) {
25
+ warnings.push(msg.text());
26
+ }
27
+ });
28
+ await setupRumJavaScript();
29
+ await use({
30
+ async waitForBeacons(options = {}) {
31
+ const { minCount = 0, timeout = DEFAULT_TIMEOUT } = options;
32
+ const start = Date.now();
33
+ while (beacons.length < minCount && (Date.now() - start) <= timeout) {
34
+ if (warnings.length > 0) {
35
+ throwWithEventDump("Unexpected Dynatrace API warnings: " + warnings.join("\n"));
36
+ }
37
+ await wait(100);
38
+ }
39
+ if (beacons.length < minCount) {
40
+ throwWithEventDump(`Found only ${beacons.length} Dynatrace beacons after timeout of ${timeout}ms, but expected at least ${minCount}.`);
41
+ }
42
+ return beacons;
43
+ },
44
+ async expectToHaveSentEvent(expectedEvent, options = {}) {
45
+ const { timeout = DEFAULT_TIMEOUT } = options;
46
+ const start = Date.now();
47
+ let lastNonMatch = void 0;
48
+ let nextCheckIndex = 0;
49
+ while ((Date.now() - start) <= timeout) {
50
+ for (let i = nextCheckIndex; i < events.length; i++) {
51
+ const beaconEvent = events[i];
52
+ try {
53
+ expect(beaconEvent).toMatchObject(expectedEvent);
54
+ return;
55
+ }
56
+ catch {
57
+ lastNonMatch = beaconEvent;
58
+ }
59
+ nextCheckIndex = i + 1;
60
+ }
61
+ await wait(100);
62
+ }
63
+ if (!lastNonMatch) {
64
+ throwWithEventDump("Dynatrace didn't send any events.");
65
+ }
66
+ console.log(`[Dynatrace Testing] Didn't find matching events.`);
67
+ expect(lastNonMatch).toMatchObject(expectedEvent);
68
+ },
69
+ async expectToHaveSentEventTimes(expectedEvent, times, options = {}) {
70
+ const { timeout = DEFAULT_TIMEOUT } = options;
71
+ const start = Date.now();
72
+ let lastNonMatch = void 0;
73
+ let nextCheckIndex = 0;
74
+ let foundTimes = 0;
75
+ while ((Date.now() - start) <= timeout) {
76
+ for (let i = nextCheckIndex; i < events.length; i++) {
77
+ const beaconEvent = events[i];
78
+ try {
79
+ expect(beaconEvent).toMatchObject(expectedEvent);
80
+ foundTimes++;
81
+ }
82
+ catch {
83
+ lastNonMatch = beaconEvent;
84
+ }
85
+ nextCheckIndex = i + 1;
86
+ }
87
+ if (foundTimes === times) {
88
+ return;
89
+ }
90
+ if (foundTimes > times) {
91
+ throwWithEventDump(`Expected ${times} event occurrences, found ${foundTimes}.`);
92
+ }
93
+ await wait(100);
94
+ }
95
+ if (!lastNonMatch) {
96
+ throwWithEventDump("Dynatrace didn't send any events");
97
+ }
98
+ if (foundTimes < times) {
99
+ throwWithEventDump(`Didn't find the expected amount of ${times} matching events, found ${foundTimes}.`);
100
+ }
101
+ expect(lastNonMatch).toMatchObject(expectedEvent);
102
+ },
103
+ clearEvents() {
104
+ events.length = 0;
105
+ beacons.length = 0;
106
+ }
107
+ });
108
+ /**
109
+ * Throws an error with a custom message. If event dumping is enabled, logs the received Dynatrace events before
110
+ * throwing the error.
111
+ *
112
+ * @param message The error message to be thrown
113
+ */
114
+ function throwWithEventDump(message) {
115
+ if (dumpEventsOnFail) {
116
+ console.log("Received Dynatrace events:", events);
117
+ }
118
+ throw new Error(message);
119
+ }
120
+ async function setupRumJavaScript() {
121
+ // Fetch RUM JavaScript once and cache it
122
+ const cacheKey = `${endpointUrl}:${appId}`;
123
+ let rum = rumJavaScriptCache.get(cacheKey);
124
+ let beaconUri;
125
+ if (!rum) {
126
+ rum = await fetchRumJavaScriptContent(endpointUrl, appId, token);
127
+ beaconUri = extractBeaconUri(rum);
128
+ rumJavaScriptCache.set(cacheKey, rum);
129
+ rumJavaScriptCache.set(`${cacheKey}:beaconUri`, beaconUri);
130
+ }
131
+ else {
132
+ beaconUri = rumJavaScriptCache.get(`${cacheKey}:beaconUri`);
133
+ if (!beaconUri) {
134
+ throw new Error("Failed to retrieve beaconUri from cache.");
135
+ }
136
+ }
137
+ // Listen to network requests to capture beacon data
138
+ // This approach doesn't interfere with user-defined routes
139
+ page.on("request", (request) => {
140
+ const url = request.url();
141
+ // Check if this is a beacon request matching the extracted beaconUri
142
+ if (!(beaconUri && url.startsWith(beaconUri) && url.includes("ty=js"))) {
143
+ return;
144
+ }
145
+ const beaconBodyString = extractBeaconBody(request, url);
146
+ if (!beaconBodyString) {
147
+ return;
148
+ }
149
+ try {
150
+ const beaconBody = JSON.parse(beaconBodyString);
151
+ beacons.push({
152
+ url: url,
153
+ body: beaconBody
154
+ });
155
+ events.push(...beaconBody.data.events);
156
+ }
157
+ catch (error) {
158
+ console.error("[Dynatrace Testing] Failed to parse beacon body:", error);
159
+ }
160
+ });
161
+ // Use addInitScript to inject RUM script tag before any page scripts execute
162
+ await page.addInitScript(rum);
163
+ }
164
+ }
165
+ });
166
+ /**
167
+ * Fetches the RUM JavaScript content from the Dynatrace API endpoint.
168
+ *
169
+ * @param endpointUrl The URL of the Dynatrace API endpoint
170
+ * @param appId The application ID to identify the correct RUM JavaScript
171
+ * @param token The authentication token for the installation
172
+ * @returns A promise that resolves to the RUM JavaScript content
173
+ */
174
+ async function fetchRumJavaScriptContent(endpointUrl, appId, token) {
175
+ const rumApiResult = await fetchRumJavaScript(endpointUrl, appId, token);
176
+ if (!rumApiResult.includes("_complete.js")) {
177
+ throw new Error("Dynatrace received unexpected RUM JavaScript when requesting the JavaScript Tag via API.");
178
+ }
179
+ // Extract the src URL from the script tag
180
+ const srcMatch = rumApiResult.match(/src="([^"]+)"/);
181
+ if (!srcMatch) {
182
+ throw new Error("Failed to extract src URL from RUM JavaScript tag.");
183
+ }
184
+ const scriptUrl = srcMatch[1];
185
+ // Fetch the actual JavaScript content
186
+ const scriptResponse = await fetch(scriptUrl);
187
+ return scriptResponse.text();
188
+ }
189
+ /**
190
+ * Extracts the beacon URI from the RUM JavaScript content.
191
+ *
192
+ * @param rum The RUM JavaScript content to extract the beacon URI from
193
+ * @returns The extracted beacon URI
194
+ */
195
+ function extractBeaconUri(rum) {
196
+ // Extract the beaconUri from the JavaScript content
197
+ const beaconUriMatch = rum.match(/"beaconUri":"([^"]+)"/);
198
+ if (!beaconUriMatch) {
199
+ throw new Error("Failed to extract beaconUri from RUM JavaScript content.");
200
+ }
201
+ return beaconUriMatch[1];
202
+ }
203
+ /**
204
+ * Waits for a given time and resolves upon timeout.
205
+ *
206
+ * @param time The time to wait in milliseconds
207
+ * @returns A resolved promise when the time ran out
208
+ */
209
+ function wait(time) {
210
+ return new Promise((resolve) => {
211
+ setTimeout(resolve, time);
212
+ });
213
+ }
214
+ /**
215
+ * Extracts the body of a beacon request, optionally decompressing it if the URL indicates compressed data.
216
+ *
217
+ * @param request The request object containing the beacon data
218
+ * @param url The URL of the request to check for specific compression indicators
219
+ * @returns The extracted body of the beacon request, or null if no request body is present
220
+ */
221
+ function extractBeaconBody(request, url) {
222
+ const buffer = request.postDataBuffer();
223
+ if (url.includes("co=snappy") && buffer) {
224
+ const uint8Data = uncompress(buffer);
225
+ return decoder.decode(uint8Data);
226
+ }
227
+ return request.postData();
228
+ }
229
+ //# 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 { 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\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        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(\"Unexpected Dynatrace API warnings: \" + warnings.join(\"\\n\"));\n                    }\n                    await wait(100);\n                }\n\n                if (beacons.length < minCount) {\n                    throwWithEventDump(`Found only ${beacons.length} Dynatrace beacons after timeout of ${timeout}ms, but expected at least ${minCount}.`);\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(\"Dynatrace didn't send any events.\");\n                }\n\n                console.log(`[Dynatrace Testing] Didn't find matching events.`);\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(`Expected ${times} event occurrences, found ${foundTimes}.`);\n                    }\n                    await wait(100);\n                }\n\n                if (!lastNonMatch) {\n                    throwWithEventDump(\"Dynatrace didn't send any events\");\n                }\n\n                if (foundTimes < times) {\n                    throwWithEventDump(`Didn't find the expected amount of ${\n                        times\n                    } matching events, found ${foundTimes}.`);\n                }\n\n                expect(lastNonMatch).toMatchObject(expectedEvent);\n            },\n\n            clearEvents(): void {\n                events.length = 0;\n                beacons.length = 0;\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         */\n        function throwWithEventDump(message: string): never {\n            if (dumpEventsOnFail) {\n                console.log(\"Received Dynatrace events:\", events);\n            }\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                // Check if this is a beacon request matching the extracted beaconUri\n                if (!(beaconUri && url.startsWith(beaconUri) && 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 * 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.\");\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"]}