1
1
'use strict' ;
2
2
const {
3
3
ArrayPrototypeForEach,
4
- FunctionPrototypeBind,
5
4
SafeMap,
5
+ SafeWeakSet,
6
6
} = primordials ;
7
7
const {
8
8
createHook,
@@ -13,13 +13,18 @@ const {
13
13
ERR_TEST_FAILURE ,
14
14
} ,
15
15
} = require ( 'internal/errors' ) ;
16
+ const { kEmptyObject } = require ( 'internal/util' ) ;
16
17
const { getOptionValue } = require ( 'internal/options' ) ;
17
18
const { kCancelledByParent, Test, ItTest, Suite } = require ( 'internal/test_runner/test' ) ;
19
+ const { bigint : hrtime } = process . hrtime ;
18
20
19
- const isTestRunner = getOptionValue ( '--test' ) ;
21
+ const isTestRunnerCli = getOptionValue ( '--test' ) ;
20
22
const testResources = new SafeMap ( ) ;
21
- const root = new Test ( { __proto__ : null , name : '<root>' } ) ;
22
- let wasRootSetup = false ;
23
+ const wasRootSetup = new SafeWeakSet ( ) ;
24
+
25
+ function createTestTree ( options = kEmptyObject ) {
26
+ return setup ( new Test ( { __proto__ : null , ...options , name : '<root>' } ) ) ;
27
+ }
23
28
24
29
function createProcessEventHandler ( eventName , rootTest ) {
25
30
return ( err ) => {
@@ -48,7 +53,7 @@ function createProcessEventHandler(eventName, rootTest) {
48
53
}
49
54
50
55
function setup ( root ) {
51
- if ( wasRootSetup ) {
56
+ if ( wasRootSetup . has ( root ) ) {
52
57
return root ;
53
58
}
54
59
const hook = createHook ( {
@@ -81,52 +86,9 @@ function setup(root) {
81
86
'Promise resolution is still pending but the event loop has already resolved' ,
82
87
kCancelledByParent ) ) ;
83
88
84
- let passCount = 0 ;
85
- let failCount = 0 ;
86
- let skipCount = 0 ;
87
- let todoCount = 0 ;
88
- let cancelledCount = 0 ;
89
-
90
- for ( let i = 0 ; i < root . subtests . length ; i ++ ) {
91
- const test = root . subtests [ i ] ;
92
-
93
- // Check SKIP and TODO tests first, as those should not be counted as
94
- // failures.
95
- if ( test . skipped ) {
96
- skipCount ++ ;
97
- } else if ( test . isTodo ) {
98
- todoCount ++ ;
99
- } else if ( test . cancelled ) {
100
- cancelledCount ++ ;
101
- } else if ( ! test . passed ) {
102
- failCount ++ ;
103
- } else {
104
- passCount ++ ;
105
- }
106
- }
107
-
108
- root . reporter . plan ( root . indent , root . subtests . length ) ;
109
-
110
- for ( let i = 0 ; i < root . diagnostics . length ; i ++ ) {
111
- root . reporter . diagnostic ( root . indent , root . diagnostics [ i ] ) ;
112
- }
113
-
114
- root . reporter . diagnostic ( root . indent , `tests ${ root . subtests . length } ` ) ;
115
- root . reporter . diagnostic ( root . indent , `pass ${ passCount } ` ) ;
116
- root . reporter . diagnostic ( root . indent , `fail ${ failCount } ` ) ;
117
- root . reporter . diagnostic ( root . indent , `cancelled ${ cancelledCount } ` ) ;
118
- root . reporter . diagnostic ( root . indent , `skipped ${ skipCount } ` ) ;
119
- root . reporter . diagnostic ( root . indent , `todo ${ todoCount } ` ) ;
120
- root . reporter . diagnostic ( root . indent , `duration_ms ${ process . uptime ( ) } ` ) ;
121
-
122
- root . reporter . push ( null ) ;
123
89
hook . disable ( ) ;
124
90
process . removeListener ( 'unhandledRejection' , rejectionHandler ) ;
125
91
process . removeListener ( 'uncaughtException' , exceptionHandler ) ;
126
-
127
- if ( failCount > 0 || cancelledCount > 0 ) {
128
- process . exitCode = 1 ;
129
- }
130
92
} ;
131
93
132
94
const terminationHandler = ( ) => {
@@ -137,29 +99,41 @@ function setup(root) {
137
99
process . on ( 'uncaughtException' , exceptionHandler ) ;
138
100
process . on ( 'unhandledRejection' , rejectionHandler ) ;
139
101
process . on ( 'beforeExit' , exitHandler ) ;
140
- // TODO(MoLow): Make it configurable to hook when isTestRunner === false.
141
- if ( isTestRunner ) {
102
+ // TODO(MoLow): Make it configurable to hook when isTestRunnerCli === false.
103
+ if ( isTestRunnerCli ) {
142
104
process . on ( 'SIGINT' , terminationHandler ) ;
143
105
process . on ( 'SIGTERM' , terminationHandler ) ;
144
106
}
145
107
146
- root . reporter . pipe ( process . stdout ) ;
108
+ root . startTime = hrtime ( ) ;
147
109
root . reporter . version ( ) ;
148
110
149
- wasRootSetup = true ;
111
+ wasRootSetup . add ( root ) ;
150
112
return root ;
151
113
}
152
114
115
+ let globalRoot ;
116
+ function getGlobalRoot ( ) {
117
+ if ( ! globalRoot ) {
118
+ globalRoot = createTestTree ( ) ;
119
+ globalRoot . reporter . pipe ( process . stdout ) ;
120
+ globalRoot . reporter . once ( 'test:fail' , ( ) => {
121
+ process . exitCode = 1 ;
122
+ } ) ;
123
+ }
124
+ return globalRoot ;
125
+ }
126
+
153
127
function test ( name , options , fn ) {
154
- const subtest = setup ( root ) . createSubtest ( Test , name , options , fn ) ;
128
+ const subtest = getGlobalRoot ( ) . createSubtest ( Test , name , options , fn ) ;
155
129
return subtest . start ( ) ;
156
130
}
157
131
158
132
function runInParentContext ( Factory ) {
159
133
function run ( name , options , fn , overrides ) {
160
- const parent = testResources . get ( executionAsyncId ( ) ) || setup ( root ) ;
134
+ const parent = testResources . get ( executionAsyncId ( ) ) || getGlobalRoot ( ) ;
161
135
const subtest = parent . createSubtest ( Factory , name , options , fn , overrides ) ;
162
- if ( parent === root ) {
136
+ if ( parent === getGlobalRoot ( ) ) {
163
137
subtest . start ( ) ;
164
138
}
165
139
}
@@ -178,13 +152,14 @@ function runInParentContext(Factory) {
178
152
179
153
function hook ( hook ) {
180
154
return ( fn , options ) => {
181
- const parent = testResources . get ( executionAsyncId ( ) ) || setup ( root ) ;
155
+ const parent = testResources . get ( executionAsyncId ( ) ) || getGlobalRoot ( ) ;
182
156
parent . createHook ( hook , fn , options ) ;
183
157
} ;
184
158
}
185
159
186
160
module . exports = {
187
- test : FunctionPrototypeBind ( test , root ) ,
161
+ createTestTree,
162
+ test,
188
163
describe : runInParentContext ( Suite ) ,
189
164
it : runInParentContext ( ItTest ) ,
190
165
before : hook ( 'before' ) ,
0 commit comments