1
- import type { File , Task , TaskResultPack } from '@vitest/runner'
1
+ import type { File , Task } from '@vitest/runner'
2
2
import type { ErrorWithDiff , UserConsoleLog } from '../../types/general'
3
3
import type { Vitest } from '../core'
4
4
import type { Reporter } from '../types/reporter'
5
- import type { TestCase , TestModule , TestResult , TestSuite } from './reported-tasks'
5
+ import type { TestCase , TestCollection , TestModule , TestModuleState , TestResult , TestSuite , TestSuiteState } from './reported-tasks'
6
6
import { performance } from 'node:perf_hooks'
7
7
import { getFullName , getSuites , getTestName , getTests , hasFailed } from '@vitest/runner/utils'
8
8
import { toArray } from '@vitest/utils'
@@ -24,7 +24,7 @@ export abstract class BaseReporter implements Reporter {
24
24
start = 0
25
25
end = 0
26
26
watchFilters ?: string [ ]
27
- failedUnwatchedFiles : Task [ ] = [ ]
27
+ failedUnwatchedFiles : TestModule [ ] = [ ]
28
28
isTTY : boolean
29
29
ctx : Vitest = undefined !
30
30
renderSucceed = false
@@ -83,6 +83,8 @@ export abstract class BaseReporter implements Reporter {
83
83
if ( testModule . state ( ) === 'failed' ) {
84
84
this . logFailedTask ( testModule . task )
85
85
}
86
+
87
+ this . printTestModule ( testModule )
86
88
}
87
89
88
90
private logFailedTask ( task : Task ) {
@@ -93,121 +95,153 @@ export abstract class BaseReporter implements Reporter {
93
95
}
94
96
}
95
97
96
- onTaskUpdate ( packs : TaskResultPack [ ] ) : void {
97
- for ( const pack of packs ) {
98
- const task = this . ctx . state . idMap . get ( pack [ 0 ] )
99
-
100
- if ( task ) {
101
- this . printTask ( task )
102
- }
103
- }
104
- }
105
-
106
- /**
107
- * Callback invoked with a single `Task` from `onTaskUpdate`
108
- */
109
- protected printTask ( task : Task ) : void {
110
- if (
111
- ! ( 'filepath' in task )
112
- || ! task . result ?. state
113
- || task . result ?. state === 'run'
114
- || task . result ?. state === 'queued' ) {
98
+ protected printTestModule ( testModule : TestModule ) : void {
99
+ const moduleState = testModule . state ( )
100
+ if ( moduleState === 'queued' || moduleState === 'pending' ) {
115
101
return
116
102
}
117
103
118
- const suites = getSuites ( task )
119
- const allTests = getTests ( task )
120
- const failed = allTests . filter ( t => t . result ?. state === 'fail' )
121
- const skipped = allTests . filter ( t => t . mode === 'skip' || t . mode === 'todo' )
104
+ let testsCount = 0
105
+ let failedCount = 0
106
+ let skippedCount = 0
122
107
123
- let state = c . dim ( `${ allTests . length } test${ allTests . length > 1 ? 's' : '' } ` )
108
+ // delaying logs to calculate the test stats first
109
+ // which minimizes the amount of for loops
110
+ const logs : string [ ] = [ ]
111
+ const originalLog = this . log . bind ( this )
112
+ this . log = ( msg : string ) => logs . push ( msg )
124
113
125
- if ( failed . length ) {
126
- state += c . dim ( ' | ' ) + c . red ( `${ failed . length } failed` )
114
+ const visit = ( suiteState : TestSuiteState , children : TestCollection ) => {
115
+ for ( const child of children ) {
116
+ if ( child . type === 'suite' ) {
117
+ const suiteState = child . state ( )
118
+
119
+ // Skipped suites are hidden when --hideSkippedTests, print otherwise
120
+ if ( ! this . ctx . config . hideSkippedTests || suiteState !== 'skipped' ) {
121
+ this . printTestSuite ( child )
122
+ }
123
+
124
+ visit ( suiteState , child . children )
125
+ }
126
+ else {
127
+ const testResult = child . result ( )
128
+
129
+ testsCount ++
130
+ if ( testResult . state === 'failed' ) {
131
+ failedCount ++
132
+ }
133
+ else if ( testResult . state === 'skipped' ) {
134
+ skippedCount ++
135
+ }
136
+
137
+ if ( this . ctx . config . hideSkippedTests && suiteState === 'skipped' ) {
138
+ // Skipped suites are hidden when --hideSkippedTests
139
+ continue
140
+ }
141
+
142
+ this . printTestCase ( moduleState , child )
143
+ }
144
+ }
127
145
}
128
146
129
- if ( skipped . length ) {
130
- state += c . dim ( ' | ' ) + c . yellow ( `${ skipped . length } skipped` )
147
+ try {
148
+ visit ( moduleState , testModule . children )
149
+ }
150
+ finally {
151
+ this . log = originalLog
131
152
}
132
153
133
- let suffix = c . dim ( '(' ) + state + c . dim ( ')' ) + this . getDurationPrefix ( task )
154
+ this . log ( this . getModuleLog ( testModule , {
155
+ tests : testsCount ,
156
+ failed : failedCount ,
157
+ skipped : skippedCount ,
158
+ } ) )
159
+ logs . forEach ( log => this . log ( log ) )
160
+ }
134
161
135
- if ( this . ctx . config . logHeapUsage && task . result . heap != null ) {
136
- suffix += c . magenta ( ` ${ Math . floor ( task . result . heap / 1024 / 1024 ) } MB heap used` )
137
- }
162
+ protected printTestCase ( moduleState : TestModuleState , test : TestCase ) : void {
163
+ const testResult = test . result ( )
138
164
139
- let title = getStateSymbol ( task )
165
+ const { duration, retryCount, repeatCount } = test . diagnostic ( ) || { }
166
+ const padding = this . getTestIndentation ( test . task )
167
+ let suffix = this . getDurationPrefix ( test . task )
140
168
141
- if ( task . meta . typecheck ) {
142
- title += ` ${ c . bgBlue ( c . bold ( ' TS ' ) ) } `
169
+ if ( retryCount != null && retryCount > 0 ) {
170
+ suffix += c . yellow ( ` (retry x ${ retryCount } )` )
143
171
}
144
172
145
- if ( task . projectName ) {
146
- title += ` ${ formatProjectName ( task . projectName , '' ) } `
173
+ if ( repeatCount != null && repeatCount > 0 ) {
174
+ suffix += c . yellow ( ` (repeat x ${ repeatCount } )` )
147
175
}
148
176
149
- this . log ( ` ${ title } ${ task . name } ${ suffix } ` )
177
+ if ( testResult . state === 'failed' ) {
178
+ this . log ( c . red ( ` ${ padding } ${ taskFail } ${ this . getTestName ( test . task , c . dim ( ' > ' ) ) } ` ) + suffix )
150
179
151
- for ( const suite of suites ) {
152
- if ( this . ctx . config . hideSkippedTests && ( suite . mode === 'skip' || suite . result ?. state === 'skip' ) ) {
153
- // Skipped suites are hidden when --hideSkippedTests
154
- continue
155
- }
180
+ // print short errors, full errors will be at the end in summary
181
+ testResult . errors . forEach ( ( error ) => {
182
+ const message = this . formatShortError ( error )
156
183
157
- const tests = suite . tasks . filter ( task => task . type === 'test' )
184
+ if ( message ) {
185
+ this . log ( c . red ( ` ${ padding } ${ message } ` ) )
186
+ }
187
+ } )
188
+ }
158
189
159
- if ( ! ( 'filepath' in suite ) ) {
160
- this . printSuite ( suite )
161
- }
190
+ // also print slow tests
191
+ else if ( duration && duration > this . ctx . config . slowTestThreshold ) {
192
+ this . log ( ` ${ padding } ${ c . yellow ( c . dim ( F_CHECK ) ) } ${ this . getTestName ( test . task , c . dim ( ' > ' ) ) } ${ suffix } ` )
193
+ }
162
194
163
- for ( const test of tests ) {
164
- const { duration, retryCount, repeatCount } = test . result || { }
165
- const padding = this . getTestIndentation ( test )
166
- let suffix = this . getDurationPrefix ( test )
195
+ else if ( this . ctx . config . hideSkippedTests && ( testResult . state === 'skipped' ) ) {
196
+ // Skipped tests are hidden when --hideSkippedTests
197
+ }
167
198
168
- if ( retryCount != null && retryCount > 0 ) {
169
- suffix += c . yellow ( ` (retry x${ retryCount } )` )
170
- }
199
+ // also print skipped tests that have notes
200
+ else if ( testResult . state === 'skipped' && testResult . note ) {
201
+ this . log ( ` ${ padding } ${ getStateSymbol ( test . task ) } ${ this . getTestName ( test . task , c . dim ( ' > ' ) ) } ${ c . dim ( c . gray ( ` [${ testResult . note } ]` ) ) } ` )
202
+ }
171
203
172
- if ( repeatCount != null && repeatCount > 0 ) {
173
- suffix += c . yellow ( ` (repeat x${ repeatCount } )` )
174
- }
204
+ else if ( this . renderSucceed || moduleState === 'failed' ) {
205
+ this . log ( ` ${ padding } ${ getStateSymbol ( test . task ) } ${ this . getTestName ( test . task , c . dim ( ' > ' ) ) } ${ suffix } ` )
206
+ }
207
+ }
175
208
176
- if ( test . result ?. state === 'fail' ) {
177
- this . log ( c . red ( ` ${ padding } ${ taskFail } ${ this . getTestName ( test , c . dim ( ' > ' ) ) } ` ) + suffix )
209
+ private getModuleLog ( testModule : TestModule , counts : {
210
+ tests : number
211
+ failed : number
212
+ skipped : number
213
+ } ) : string {
214
+ let state = c . dim ( `${ counts . tests } test${ counts . tests > 1 ? 's' : '' } ` )
178
215
179
- // print short errors, full errors will be at the end in summary
180
- test . result ?. errors ?. forEach ( ( error ) => {
181
- const message = this . formatShortError ( error )
216
+ if ( counts . failed ) {
217
+ state += c . dim ( ' | ' ) + c . red ( ` ${ counts . failed } failed` )
218
+ }
182
219
183
- if ( message ) {
184
- this . log ( c . red ( ` ${ padding } ${ message } ` ) )
185
- }
186
- } )
187
- }
220
+ if ( counts . skipped ) {
221
+ state += c . dim ( ' | ' ) + c . yellow ( `${ counts . skipped } skipped` )
222
+ }
188
223
189
- // also print slow tests
190
- else if ( duration && duration > this . ctx . config . slowTestThreshold ) {
191
- this . log ( ` ${ padding } ${ c . yellow ( c . dim ( F_CHECK ) ) } ${ this . getTestName ( test , c . dim ( ' > ' ) ) } ${ suffix } ` )
192
- }
224
+ let suffix = c . dim ( '(' ) + state + c . dim ( ')' ) + this . getDurationPrefix ( testModule . task )
193
225
194
- else if ( this . ctx . config . hideSkippedTests && ( test . mode === 'skip' || test . result ?. state === 'skip' ) ) {
195
- // Skipped tests are hidden when --hideSkippedTests
196
- }
226
+ const diagnostic = testModule . diagnostic ( )
227
+ if ( diagnostic . heap != null ) {
228
+ suffix += c . magenta ( ` ${ Math . floor ( diagnostic . heap / 1024 / 1024 ) } MB heap used` )
229
+ }
197
230
198
- // also print skipped tests that have notes
199
- else if ( test . result ?. state === 'skip' && test . result . note ) {
200
- this . log ( ` ${ padding } ${ getStateSymbol ( test ) } ${ this . getTestName ( test ) } ${ c . dim ( c . gray ( ` [${ test . result . note } ]` ) ) } ` )
201
- }
231
+ let title = getStateSymbol ( testModule . task )
202
232
203
- else if ( this . renderSucceed || failed . length > 0 ) {
204
- this . log ( ` ${ padding } ${ getStateSymbol ( test ) } ${ this . getTestName ( test , c . dim ( ' > ' ) ) } ${ suffix } ` )
205
- }
206
- }
233
+ if ( testModule . meta ( ) . typecheck ) {
234
+ title += ` ${ c . bgBlue ( c . bold ( ' TS ' ) ) } `
207
235
}
236
+
237
+ if ( testModule . project . name ) {
238
+ title += ` ${ formatProjectName ( testModule . project . name , '' ) } `
239
+ }
240
+
241
+ return ` ${ title } ${ testModule . task . name } ${ suffix } `
208
242
}
209
243
210
- protected printSuite ( _task : Task ) : void {
244
+ protected printTestSuite ( _suite : TestSuite ) : void {
211
245
// Suite name is included in getTestName by default
212
246
}
213
247
@@ -262,8 +296,8 @@ export abstract class BaseReporter implements Reporter {
262
296
263
297
onWatcherRerun ( files : string [ ] , trigger ?: string ) : void {
264
298
this . watchFilters = files
265
- this . failedUnwatchedFiles = this . ctx . state . getFiles ( ) . filter ( file =>
266
- ! files . includes ( file . filepath ) && hasFailed ( file ) ,
299
+ this . failedUnwatchedFiles = this . ctx . state . getTestModules ( ) . filter ( testModule =>
300
+ ! files . includes ( testModule . task . filepath ) && testModule . state ( ) === 'failed' ,
267
301
)
268
302
269
303
// Update re-run count for each file
@@ -296,8 +330,8 @@ export abstract class BaseReporter implements Reporter {
296
330
297
331
this . log ( '' )
298
332
299
- for ( const task of this . failedUnwatchedFiles ) {
300
- this . printTask ( task )
333
+ for ( const testModule of this . failedUnwatchedFiles ) {
334
+ this . printTestModule ( testModule )
301
335
}
302
336
303
337
this . _timeStart = formatTimeString ( new Date ( ) )
@@ -405,7 +439,7 @@ export abstract class BaseReporter implements Reporter {
405
439
this . log ( )
406
440
407
441
const affectedFiles = [
408
- ...this . failedUnwatchedFiles ,
442
+ ...this . failedUnwatchedFiles . map ( m => m . task ) ,
409
443
...files ,
410
444
]
411
445
const tests = getTests ( affectedFiles )
0 commit comments