@serenity-js/jasmine 3.38.0 → 3.39.0
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/CHANGELOG.md +11 -0
- package/README.md +1 -0
- package/esm/SerenityReporterForJasmine.d.ts +11 -1
- package/esm/SerenityReporterForJasmine.d.ts.map +1 -1
- package/esm/SerenityReporterForJasmine.js +39 -6
- package/esm/SerenityReporterForJasmine.js.map +1 -1
- package/esm/bootstrap.d.ts +7 -2
- package/esm/bootstrap.d.ts.map +1 -1
- package/esm/bootstrap.js +99 -13
- package/esm/bootstrap.js.map +1 -1
- package/esm/jasmine/SpecResult.d.ts +10 -1
- package/esm/jasmine/SpecResult.d.ts.map +1 -1
- package/esm/jasmine/SuiteResult.d.ts +10 -1
- package/esm/jasmine/SuiteResult.d.ts.map +1 -1
- package/esm/jasmine/index.d.ts +1 -0
- package/esm/jasmine/index.d.ts.map +1 -1
- package/esm/jasmine/index.js +1 -0
- package/esm/jasmine/index.js.map +1 -1
- package/esm/monkeyPatched.d.ts +4 -0
- package/esm/monkeyPatched.d.ts.map +1 -1
- package/esm/monkeyPatched.js +36 -2
- package/esm/monkeyPatched.js.map +1 -1
- package/lib/SerenityReporterForJasmine.d.ts +11 -1
- package/lib/SerenityReporterForJasmine.d.ts.map +1 -1
- package/lib/SerenityReporterForJasmine.js +39 -6
- package/lib/SerenityReporterForJasmine.js.map +1 -1
- package/lib/bootstrap.d.ts +7 -2
- package/lib/bootstrap.d.ts.map +1 -1
- package/lib/bootstrap.js +99 -13
- package/lib/bootstrap.js.map +1 -1
- package/lib/jasmine/SpecResult.d.ts +10 -1
- package/lib/jasmine/SpecResult.d.ts.map +1 -1
- package/lib/jasmine/SuiteResult.d.ts +10 -1
- package/lib/jasmine/SuiteResult.d.ts.map +1 -1
- package/lib/jasmine/index.d.ts +1 -0
- package/lib/jasmine/index.d.ts.map +1 -1
- package/lib/jasmine/index.js +1 -0
- package/lib/jasmine/index.js.map +1 -1
- package/lib/monkeyPatched.d.ts +4 -0
- package/lib/monkeyPatched.d.ts.map +1 -1
- package/lib/monkeyPatched.js +36 -2
- package/lib/monkeyPatched.js.map +1 -1
- package/package.json +5 -5
- package/src/SerenityReporterForJasmine.ts +55 -9
- package/src/bootstrap.ts +112 -13
- package/src/jasmine/SpecResult.ts +11 -1
- package/src/jasmine/SuiteResult.ts +11 -1
- package/src/jasmine/index.ts +1 -0
- package/src/monkeyPatched.ts +39 -2
package/src/bootstrap.ts
CHANGED
|
@@ -5,8 +5,13 @@ import { monkeyPatched } from './monkeyPatched.js';
|
|
|
5
5
|
import { SerenityReporterForJasmine } from './SerenityReporterForJasmine.js';
|
|
6
6
|
|
|
7
7
|
/**
|
|
8
|
-
*
|
|
9
|
-
*
|
|
8
|
+
* Bootstraps the Serenity/JS reporter for Jasmine.
|
|
9
|
+
*
|
|
10
|
+
* This function monkey-patches Jasmine's Suite and Spec constructors
|
|
11
|
+
* so that they provide more accurate location information.
|
|
12
|
+
*
|
|
13
|
+
* For Jasmine 5.x, Suite and Spec are on the jasmine object directly.
|
|
14
|
+
* For Jasmine 6.x, Suite and Spec are in jasmine.private.
|
|
10
15
|
*
|
|
11
16
|
* ## Registering the reporter from the command line
|
|
12
17
|
*
|
|
@@ -27,20 +32,42 @@ import { SerenityReporterForJasmine } from './SerenityReporterForJasmine.js';
|
|
|
27
32
|
* the global.jasmine instance
|
|
28
33
|
*/
|
|
29
34
|
export function bootstrap(config: SerenityReporterForJasmineConfig = {}, jasmine = (global as any).jasmine): SerenityReporterForJasmine {
|
|
30
|
-
const wrappers = {
|
|
31
|
-
expectationResultFactory: originalExpectationResultFactory => ((attributes: { passed: boolean, error: Error }) => {
|
|
32
|
-
const result = originalExpectationResultFactory(attributes);
|
|
33
35
|
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
36
|
+
// Jasmine 6+ moved Suite and Spec to jasmine.private,
|
|
37
|
+
// so we check both locations for backwards compatibility
|
|
38
|
+
const Suite = jasmine.Suite || jasmine.private?.Suite;
|
|
39
|
+
const Spec = jasmine.Spec || jasmine.private?.Spec;
|
|
37
40
|
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
+
if (Suite && Spec) {
|
|
42
|
+
// Monkey-patch Suite and Spec to capture location info
|
|
43
|
+
const wrappers = {
|
|
44
|
+
expectationResultFactory: originalExpectationResultFactory => ((attributes: { passed: boolean, error: Error }) => {
|
|
45
|
+
const result = originalExpectationResultFactory(attributes);
|
|
46
|
+
|
|
47
|
+
if (! attributes.passed && attributes.error instanceof RuntimeError) {
|
|
48
|
+
result.stack = attributes.error.stack;
|
|
49
|
+
}
|
|
41
50
|
|
|
42
|
-
|
|
43
|
-
|
|
51
|
+
return result;
|
|
52
|
+
}),
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
// Jasmine 5.x: Suite and Spec are on jasmine directly
|
|
56
|
+
if (jasmine.Suite && jasmine.Spec) {
|
|
57
|
+
jasmine.Suite = monkeyPatched(jasmine.Suite, wrappers);
|
|
58
|
+
jasmine.Spec = monkeyPatched(jasmine.Spec, wrappers);
|
|
59
|
+
}
|
|
60
|
+
// Jasmine 6+: Suite and Spec are in jasmine.private
|
|
61
|
+
else if (jasmine.private?.Suite && jasmine.private?.Spec) {
|
|
62
|
+
jasmine.private.Suite = monkeyPatched(jasmine.private.Suite, wrappers);
|
|
63
|
+
jasmine.private.Spec = monkeyPatched(jasmine.private.Spec, wrappers);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// For Jasmine 6+, patch the Expector and buildExpectationResult to restore expected/actual values
|
|
68
|
+
// This is needed because Jasmine 6 removed these from expectation results
|
|
69
|
+
patchBuildExpectationResultForExpectedActual(jasmine);
|
|
70
|
+
patchExpectorForExpectedActual(jasmine);
|
|
44
71
|
|
|
45
72
|
const cwd = Path.from(process.cwd());
|
|
46
73
|
const requirementsHierarchy = new RequirementsHierarchy(
|
|
@@ -51,6 +78,78 @@ export function bootstrap(config: SerenityReporterForJasmineConfig = {}, jasmine
|
|
|
51
78
|
return new SerenityReporterForJasmine(serenity, requirementsHierarchy);
|
|
52
79
|
}
|
|
53
80
|
|
|
81
|
+
/**
|
|
82
|
+
* Patches Jasmine's buildExpectationResult to include expected/actual values
|
|
83
|
+
* in the expectation result. This restores functionality that was removed in Jasmine 6.
|
|
84
|
+
*
|
|
85
|
+
* In Jasmine 6, the buildExpectationResult function no longer includes expected/actual
|
|
86
|
+
* values in the result object. This patch wraps the function to preserve these values
|
|
87
|
+
* when they are provided.
|
|
88
|
+
*
|
|
89
|
+
* @param jasmine - The global jasmine instance
|
|
90
|
+
*/
|
|
91
|
+
function patchBuildExpectationResultForExpectedActual(jasmine: any): void {
|
|
92
|
+
// Access the private buildExpectationResult function
|
|
93
|
+
const originalBuildExpectationResult = jasmine?.private?.buildExpectationResult;
|
|
94
|
+
|
|
95
|
+
if (!originalBuildExpectationResult) {
|
|
96
|
+
// buildExpectationResult not available, skip
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
jasmine.private.buildExpectationResult = function(options: any): any {
|
|
101
|
+
const result = originalBuildExpectationResult(options);
|
|
102
|
+
|
|
103
|
+
// Restore expected/actual values that Jasmine 6 removed
|
|
104
|
+
if (options.expected !== undefined) {
|
|
105
|
+
result.expected = options.expected;
|
|
106
|
+
}
|
|
107
|
+
if (options.actual !== undefined) {
|
|
108
|
+
result.actual = options.actual;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
return result;
|
|
112
|
+
};
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* Patches Jasmine's Expector.prototype.processResult to include expected/actual values
|
|
117
|
+
* when calling addExpectationResult. This ensures the values are passed to
|
|
118
|
+
* buildExpectationResult.
|
|
119
|
+
*
|
|
120
|
+
* @param jasmine - The global jasmine instance
|
|
121
|
+
*/
|
|
122
|
+
function patchExpectorForExpectedActual(jasmine: any): void {
|
|
123
|
+
// Access the private Expector class through jasmine's internal structure
|
|
124
|
+
// In Jasmine 6+, Expector is in jasmine.private.Expector
|
|
125
|
+
const Expector = jasmine?.private?.Expector;
|
|
126
|
+
|
|
127
|
+
if (!Expector || !Expector.prototype || !Expector.prototype.processResult) {
|
|
128
|
+
// Expector not available, skip
|
|
129
|
+
return;
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
Expector.prototype.processResult = function(result: any, errorForStack?: Error): void {
|
|
133
|
+
const message = this.buildMessage(result);
|
|
134
|
+
|
|
135
|
+
let expected = this.expected;
|
|
136
|
+
if (Array.isArray(expected) && expected.length === 1) {
|
|
137
|
+
expected = expected[0];
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
this.addExpectationResult(result.pass, {
|
|
141
|
+
matcherName: this.matcherName,
|
|
142
|
+
passed: result.pass,
|
|
143
|
+
message: message,
|
|
144
|
+
error: errorForStack ? undefined : result.error,
|
|
145
|
+
errorForStack: errorForStack || undefined,
|
|
146
|
+
// Include expected/actual values that Jasmine 6 removed
|
|
147
|
+
expected: expected,
|
|
148
|
+
actual: this.actual,
|
|
149
|
+
});
|
|
150
|
+
};
|
|
151
|
+
}
|
|
152
|
+
|
|
54
153
|
export interface SerenityReporterForJasmineConfig {
|
|
55
154
|
specDirectory?: string;
|
|
56
155
|
}
|
|
@@ -29,5 +29,15 @@ export interface SpecResult {
|
|
|
29
29
|
/** The time in ms used by the spec execution, including any before/afterEach. */
|
|
30
30
|
duration: number | null;
|
|
31
31
|
|
|
32
|
-
|
|
32
|
+
/**
|
|
33
|
+
* The file system location of the spec.
|
|
34
|
+
* Available in Jasmine 5.x and earlier when monkey-patched by Serenity/JS.
|
|
35
|
+
*/
|
|
36
|
+
location?: Location;
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* The filename where the spec is defined.
|
|
40
|
+
* Available in Jasmine 6.x and later.
|
|
41
|
+
*/
|
|
42
|
+
filename?: string;
|
|
33
43
|
}
|
|
@@ -28,5 +28,15 @@ export interface SuiteResult extends Result {
|
|
|
28
28
|
/** The time in ms for Suite execution, including any before/afterAll, before/afterEach. */
|
|
29
29
|
duration: number | null;
|
|
30
30
|
|
|
31
|
-
|
|
31
|
+
/**
|
|
32
|
+
* The file system location of the suite.
|
|
33
|
+
* Available in Jasmine 5.x and earlier when monkey-patched by Serenity/JS.
|
|
34
|
+
*/
|
|
35
|
+
location?: Location;
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* The filename where the suite is defined.
|
|
39
|
+
* Available in Jasmine 6.x and later.
|
|
40
|
+
*/
|
|
41
|
+
filename?: string;
|
|
32
42
|
}
|
package/src/jasmine/index.ts
CHANGED
|
@@ -2,6 +2,7 @@ export * from './Expectation.js';
|
|
|
2
2
|
export * from './JasmineDoneInfo.js';
|
|
3
3
|
export * from './JasmineReporter.js';
|
|
4
4
|
export * from './JasmineStartedInfo.js';
|
|
5
|
+
export * from './Location.js';
|
|
5
6
|
export * from './Order.js';
|
|
6
7
|
export * from './SpecResult.js';
|
|
7
8
|
export * from './SuiteResult.js';
|
package/src/monkeyPatched.ts
CHANGED
|
@@ -2,7 +2,7 @@ import { ErrorStackParser } from '@serenity-js/core/lib/errors/index.js';
|
|
|
2
2
|
|
|
3
3
|
const parser = new ErrorStackParser();
|
|
4
4
|
|
|
5
|
-
/* eslint-disable @typescript-eslint/explicit-module-boundary-types
|
|
5
|
+
/* eslint-disable @typescript-eslint/explicit-module-boundary-types */
|
|
6
6
|
|
|
7
7
|
/**
|
|
8
8
|
* Monkey-patches Jasmine domain model constructors so that they
|
|
@@ -10,6 +10,10 @@ const parser = new ErrorStackParser();
|
|
|
10
10
|
*
|
|
11
11
|
* This helps to make reporting more accurate.
|
|
12
12
|
*
|
|
13
|
+
* For Jasmine 5.x, the location is stored on instance.result.location.
|
|
14
|
+
* For Jasmine 6.x, the location is stored on instance._serenityLocation and
|
|
15
|
+
* injected into the result events via patched startedEvent/doneEvent methods.
|
|
16
|
+
*
|
|
13
17
|
* @param jasmineConstructor - A Jasmine constructor function to be patched
|
|
14
18
|
* @param {object} wrappers - Attributes to wrap when the monkey-patched Jasmine constructor is invoked
|
|
15
19
|
*/
|
|
@@ -23,7 +27,40 @@ export function monkeyPatched(
|
|
|
23
27
|
});
|
|
24
28
|
|
|
25
29
|
const instance = new jasmineConstructor(attrs);
|
|
26
|
-
|
|
30
|
+
const location = callerLocation();
|
|
31
|
+
|
|
32
|
+
// Jasmine 5.x: instance.result exists, store location there
|
|
33
|
+
if (instance.result) {
|
|
34
|
+
instance.result.location = location;
|
|
35
|
+
}
|
|
36
|
+
// Jasmine 6.x: store location on instance and patch event methods
|
|
37
|
+
else {
|
|
38
|
+
instance._serenityLocation = location;
|
|
39
|
+
|
|
40
|
+
// Patch startedEvent to include location
|
|
41
|
+
if (typeof instance.startedEvent === 'function') {
|
|
42
|
+
const originalStartedEvent = instance.startedEvent.bind(instance);
|
|
43
|
+
instance.startedEvent = function() {
|
|
44
|
+
const event = originalStartedEvent();
|
|
45
|
+
if (event && instance._serenityLocation) {
|
|
46
|
+
event.location = instance._serenityLocation;
|
|
47
|
+
}
|
|
48
|
+
return event;
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
// Patch doneEvent to include location
|
|
53
|
+
if (typeof instance.doneEvent === 'function') {
|
|
54
|
+
const originalDoneEvent = instance.doneEvent.bind(instance);
|
|
55
|
+
instance.doneEvent = function() {
|
|
56
|
+
const event = originalDoneEvent();
|
|
57
|
+
if (event && instance._serenityLocation) {
|
|
58
|
+
event.location = instance._serenityLocation;
|
|
59
|
+
}
|
|
60
|
+
return event;
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
}
|
|
27
64
|
|
|
28
65
|
return instance;
|
|
29
66
|
};
|