@@ -22,18 +22,16 @@ import { provider } from 'std-env'
22
22
import createDebug from 'debug'
23
23
import { cleanUrl } from 'vite-node/utils'
24
24
import type { EncodedSourceMap , FetchResult } from 'vite-node'
25
- import {
26
- coverageConfigDefaults ,
27
- } from 'vitest/config'
25
+ import { coverageConfigDefaults } from 'vitest/config'
28
26
import { BaseCoverageProvider } from 'vitest/coverage'
27
+ import type { Vitest , WorkspaceProject } from 'vitest/node'
29
28
import type {
30
29
AfterSuiteRunMeta ,
31
30
CoverageProvider ,
32
31
CoverageV8Options ,
33
32
ReportContext ,
34
33
ResolvedCoverageOptions ,
35
34
} from 'vitest'
36
- import type { Vitest } from 'vitest/node'
37
35
// @ts -expect-error missing types
38
36
import _TestExclude from 'test-exclude'
39
37
@@ -65,6 +63,8 @@ type ProjectName =
65
63
| NonNullable < AfterSuiteRunMeta [ 'projectName' ] >
66
64
| typeof DEFAULT_PROJECT
67
65
66
+ type Entries < T > = [ keyof T , T [ keyof T ] ] [ ]
67
+
68
68
// TODO: vite-node should export this
69
69
const WRAPPER_LENGTH = 185
70
70
@@ -74,6 +74,7 @@ const VITE_EXPORTS_LINE_PATTERN
74
74
const DECORATOR_METADATA_PATTERN
75
75
= / _ t s _ m e t a d a t a \( " d e s i g n : p a r a m t y p e s " , \[ [ ^ \] ] * \] \) , * / g
76
76
const DEFAULT_PROJECT : unique symbol = Symbol . for ( 'default-project' )
77
+ const FILE_PROTOCOL = 'file://'
77
78
78
79
const debug = createDebug ( 'vitest:coverage' )
79
80
let uniqueId = 0
@@ -124,9 +125,7 @@ export class V8CoverageProvider extends BaseCoverageProvider implements Coverage
124
125
lines : config . thresholds [ '100' ] ? 100 : config . thresholds . lines ,
125
126
branches : config . thresholds [ '100' ] ? 100 : config . thresholds . branches ,
126
127
functions : config . thresholds [ '100' ] ? 100 : config . thresholds . functions ,
127
- statements : config . thresholds [ '100' ]
128
- ? 100
129
- : config . thresholds . statements ,
128
+ statements : config . thresholds [ '100' ] ? 100 : config . thresholds . statements ,
130
129
} ,
131
130
}
132
131
@@ -183,14 +182,14 @@ export class V8CoverageProvider extends BaseCoverageProvider implements Coverage
183
182
* backwards compatibility is a breaking change.
184
183
*/
185
184
onAfterSuiteRun ( { coverage, transformMode, projectName } : AfterSuiteRunMeta ) : void {
186
- if ( transformMode !== 'web' && transformMode !== 'ssr' ) {
185
+ if ( transformMode !== 'web' && transformMode !== 'ssr' && transformMode !== 'browser' ) {
187
186
throw new Error ( `Invalid transform mode: ${ transformMode } ` )
188
187
}
189
188
190
189
let entry = this . coverageFiles . get ( projectName || DEFAULT_PROJECT )
191
190
192
191
if ( ! entry ) {
193
- entry = { web : [ ] , ssr : [ ] }
192
+ entry = { web : [ ] , ssr : [ ] , browser : [ ] }
194
193
this . coverageFiles . set ( projectName || DEFAULT_PROJECT , entry )
195
194
}
196
195
@@ -212,36 +211,30 @@ export class V8CoverageProvider extends BaseCoverageProvider implements Coverage
212
211
await Promise . all ( this . pendingPromises )
213
212
this . pendingPromises = [ ]
214
213
215
- for ( const [
216
- projectName ,
217
- coveragePerProject ,
218
- ] of this . coverageFiles . entries ( ) ) {
219
- for ( const [ transformMode , filenames ] of Object . entries (
220
- coveragePerProject ,
221
- ) as [ AfterSuiteRunMeta [ 'transformMode' ] , Filename [ ] ] [ ] ) {
214
+ for ( const [ projectName , coveragePerProject ] of this . coverageFiles . entries ( ) ) {
215
+ for ( const [ transformMode , filenames ] of Object . entries ( coveragePerProject ) as Entries < CoverageFilesByTransformMode > ) {
222
216
let merged : RawCoverage = { result : [ ] }
223
217
224
- for ( const chunk of this . toSlices (
225
- filenames ,
226
- this . options . processingConcurrency ,
227
- ) ) {
218
+ const project = this . ctx . projects . find ( p => p . getName ( ) === projectName ) || this . ctx . getCoreWorkspaceProject ( )
219
+
220
+ for ( const chunk of this . toSlices ( filenames , this . options . processingConcurrency ) ) {
228
221
if ( debug . enabled ) {
229
222
index += chunk . length
230
223
debug ( 'Covered files %d/%d' , index , total )
231
224
}
232
225
233
- await Promise . all (
234
- chunk . map ( async ( filename ) => {
235
- const contents = await fs . readFile ( filename , 'utf-8' )
236
- const coverage = JSON . parse ( contents ) as RawCoverage
237
- merged = mergeProcessCovs ( [ merged , coverage ] )
238
- } ) ,
226
+ await Promise . all ( chunk . map ( async ( filename ) => {
227
+ const contents = await fs . readFile ( filename , 'utf-8' )
228
+ const coverage = JSON . parse ( contents ) as RawCoverage
229
+
230
+ merged = mergeProcessCovs ( [ merged , coverage ] )
231
+ } ) ,
239
232
)
240
233
}
241
234
242
235
const converted = await this . convertCoverage (
243
236
merged ,
244
- projectName ,
237
+ project ,
245
238
transformMode ,
246
239
)
247
240
@@ -404,6 +397,7 @@ export class V8CoverageProvider extends BaseCoverageProvider implements Coverage
404
397
const { originalSource } = await this . getSources (
405
398
filename . href ,
406
399
transformResults ,
400
+ file => this . ctx . vitenode . transformRequest ( file ) ,
407
401
)
408
402
409
403
const coverage = {
@@ -441,9 +435,10 @@ export class V8CoverageProvider extends BaseCoverageProvider implements Coverage
441
435
return merged
442
436
}
443
437
444
- private async getSources (
438
+ private async getSources < TransformResult extends ( FetchResult | Awaited < ReturnType < typeof this . ctx . vitenode . transformRequest > > ) > (
445
439
url : string ,
446
440
transformResults : TransformResults ,
441
+ onTransform : ( filepath : string ) => Promise < TransformResult > ,
447
442
functions : Profiler . FunctionCoverage [ ] = [ ] ,
448
443
) : Promise < {
449
444
source : string
@@ -454,16 +449,11 @@ export class V8CoverageProvider extends BaseCoverageProvider implements Coverage
454
449
const filePath = normalize ( fileURLToPath ( url ) )
455
450
456
451
let isExecuted = true
457
- let transformResult :
458
- | FetchResult
459
- | Awaited < ReturnType < typeof this . ctx . vitenode . transformRequest > >
460
- = transformResults . get ( filePath )
452
+ let transformResult : FetchResult | TransformResult | undefined = transformResults . get ( filePath )
461
453
462
454
if ( ! transformResult ) {
463
455
isExecuted = false
464
- transformResult = await this . ctx . vitenode
465
- . transformRequest ( filePath )
466
- . catch ( ( ) => null )
456
+ transformResult = await onTransform ( removeStartsWith ( url , FILE_PROTOCOL ) ) . catch ( ( ) => undefined )
467
457
}
468
458
469
459
const map = transformResult ?. map as EncodedSourceMap | undefined
@@ -513,27 +503,49 @@ export class V8CoverageProvider extends BaseCoverageProvider implements Coverage
513
503
514
504
private async convertCoverage (
515
505
coverage : RawCoverage ,
516
- projectName ?: ProjectName ,
517
- transformMode ?: 'web' | 'ssr' ,
506
+ project : WorkspaceProject = this . ctx . getCoreWorkspaceProject ( ) ,
507
+ transformMode ?: keyof CoverageFilesByTransformMode ,
518
508
) : Promise < CoverageMap > {
519
- const viteNode
520
- = this . ctx . projects . find ( project => project . getName ( ) === projectName )
521
- ?. vitenode || this . ctx . vitenode
522
- const fetchCache = transformMode
523
- ? viteNode . fetchCaches [ transformMode ]
524
- : viteNode . fetchCache
509
+ let fetchCache = project . vitenode . fetchCache
510
+
511
+ if ( transformMode ) {
512
+ fetchCache = transformMode === 'browser' ? new Map ( ) : project . vitenode . fetchCaches [ transformMode ]
513
+ }
514
+
525
515
const transformResults = normalizeTransformResults ( fetchCache )
526
516
527
- const scriptCoverages = coverage . result . filter ( result =>
528
- this . testExclude . shouldInstrument ( fileURLToPath ( result . url ) ) ,
529
- )
517
+ async function onTransform ( filepath : string ) {
518
+ if ( transformMode === 'browser' && project . browser ) {
519
+ const result = await project . browser . vite . transformRequest ( removeStartsWith ( filepath , project . config . root ) )
520
+
521
+ if ( result ) {
522
+ return { ...result , code : `${ result . code } // <inline-source-map>` }
523
+ }
524
+ }
525
+ return project . vitenode . transformRequest ( filepath )
526
+ }
527
+
528
+ const scriptCoverages = [ ]
529
+
530
+ for ( const result of coverage . result ) {
531
+ if ( transformMode === 'browser' ) {
532
+ if ( result . url . startsWith ( '/@fs' ) ) {
533
+ result . url = `${ FILE_PROTOCOL } ${ removeStartsWith ( result . url , '/@fs' ) } `
534
+ }
535
+ else {
536
+ result . url = `${ FILE_PROTOCOL } ${ project . config . root } ${ result . url } `
537
+ }
538
+ }
539
+
540
+ if ( this . testExclude . shouldInstrument ( fileURLToPath ( result . url ) ) ) {
541
+ scriptCoverages . push ( result )
542
+ }
543
+ }
544
+
530
545
const coverageMap = libCoverage . createCoverageMap ( { } )
531
546
let index = 0
532
547
533
- for ( const chunk of this . toSlices (
534
- scriptCoverages ,
535
- this . options . processingConcurrency ,
536
- ) ) {
548
+ for ( const chunk of this . toSlices ( scriptCoverages , this . options . processingConcurrency ) ) {
537
549
if ( debug . enabled ) {
538
550
index += chunk . length
539
551
debug ( 'Converting %d/%d' , index , scriptCoverages . length )
@@ -544,6 +556,7 @@ export class V8CoverageProvider extends BaseCoverageProvider implements Coverage
544
556
const sources = await this . getSources (
545
557
url ,
546
558
transformResults ,
559
+ onTransform ,
547
560
functions ,
548
561
)
549
562
@@ -645,3 +658,11 @@ function normalizeTransformResults(
645
658
646
659
return normalized
647
660
}
661
+
662
+ function removeStartsWith ( filepath : string , start : string ) {
663
+ if ( filepath . startsWith ( start ) ) {
664
+ return filepath . slice ( start . length )
665
+ }
666
+
667
+ return filepath
668
+ }
0 commit comments