@@ -8,8 +8,10 @@ import mergeWith from 'lodash.mergewith';
8
8
import stripIndent from 'strip-indent' ;
9
9
10
10
import { $type } from './symbols' ;
11
+
11
12
import {
12
13
runPluginUnderTestHere ,
14
+ runPresetUnderTestHere ,
13
15
type ResultFormatter ,
14
16
type PluginTesterOptions ,
15
17
type TestObject ,
@@ -20,7 +22,8 @@ import {
20
22
type PluginTesterTestFixtureConfig ,
21
23
type PluginTesterTestObjectConfig ,
22
24
type MaybePluginTesterTestObjectConfig ,
23
- type MaybePluginTesterTestFixtureConfig
25
+ type MaybePluginTesterTestFixtureConfig ,
26
+ type PartialPluginTesterBaseConfig
24
27
} from '.' ;
25
28
26
29
import type { Class } from 'type-fest' ;
@@ -91,27 +94,22 @@ export function pluginTester(options: PluginTesterOptions = {}) {
91
94
mergeCustomizer
92
95
) ;
93
96
94
- // TODO: implement support for preset testing configuration
95
-
96
- // ? Need to do this here instead of in the validation function since plugin
97
- // name inference relies on plugin being defined
98
- if ( ! rawBaseConfig . plugin ) {
99
- throw new TypeError ( 'plugin is a required parameter' ) ;
97
+ if (
98
+ ( rawBaseConfig . plugin &&
99
+ ( rawBaseConfig . preset ||
100
+ rawBaseConfig . presetName ||
101
+ rawBaseConfig . presetOptions ) ) ||
102
+ ( rawBaseConfig . preset &&
103
+ ( rawBaseConfig . plugin || rawBaseConfig . pluginName || rawBaseConfig . pluginOptions ) )
104
+ ) {
105
+ throw new Error (
106
+ 'failed to validate configuration: cannot test a plugin and a preset simultaneously. Specify one set of options or the other'
107
+ ) ;
100
108
}
101
109
102
- const pluginName =
103
- rawBaseConfig . pluginName || tryInferPluginName ( ) || 'unknown plugin' ;
104
-
105
- const baseConfig = {
106
- plugin : rawBaseConfig . plugin ,
107
- pluginName,
108
- basePluginOptions : rawBaseConfig . pluginOptions || { } ,
109
- preset : undefined ,
110
- presetName : undefined ,
111
- basePresetOptions : undefined ,
110
+ const baseConfig : PartialPluginTesterBaseConfig = {
112
111
babel : rawBaseConfig . babel || require ( '@babel/core' ) ,
113
112
baseBabelOptions : rawBaseConfig . babelOptions ,
114
- describeBlockTitle : rawBaseConfig . title ?? pluginName ,
115
113
// TODO: implement default filepath inference using Error stack trace
116
114
filepath : rawBaseConfig . filepath ?? rawBaseConfig . filename ,
117
115
endOfLine : rawBaseConfig . endOfLine ,
@@ -125,7 +123,25 @@ export function pluginTester(options: PluginTesterOptions = {}) {
125
123
tests : rawBaseConfig . tests || [ ]
126
124
} ;
127
125
128
- return baseConfig ;
126
+ if ( rawBaseConfig . plugin ) {
127
+ baseConfig . plugin = rawBaseConfig . plugin ;
128
+ baseConfig . pluginName =
129
+ rawBaseConfig . pluginName || tryInferPluginName ( ) || 'unknown plugin' ;
130
+ baseConfig . basePluginOptions = rawBaseConfig . pluginOptions || { } ;
131
+ } else if ( rawBaseConfig . preset ) {
132
+ baseConfig . preset = rawBaseConfig . preset ;
133
+ baseConfig . presetName = rawBaseConfig . presetName || 'unknown preset' ;
134
+ baseConfig . basePresetOptions = rawBaseConfig . presetOptions ;
135
+ } else {
136
+ throw new TypeError (
137
+ 'failed to validate configuration: must provide either `plugin` or `preset` option'
138
+ ) ;
139
+ }
140
+
141
+ baseConfig . describeBlockTitle =
142
+ rawBaseConfig . title ?? baseConfig . pluginName ?? baseConfig . presetName ;
143
+
144
+ return baseConfig as PluginTesterBaseConfig ;
129
145
130
146
function tryInferPluginName ( ) {
131
147
try {
@@ -298,7 +314,8 @@ export function pluginTester(options: PluginTesterOptions = {}) {
298
314
const {
299
315
plugin,
300
316
basePluginOptions,
301
- // TODO: use this preset, TODO: use this basePresetOptions,
317
+ preset,
318
+ basePresetOptions,
302
319
baseBabelOptions,
303
320
endOfLine,
304
321
baseFormatResult,
@@ -309,7 +326,7 @@ export function pluginTester(options: PluginTesterOptions = {}) {
309
326
const {
310
327
babelOptions,
311
328
pluginOptions,
312
- // TODO: use this presetOptions,
329
+ presetOptions,
313
330
title,
314
331
only,
315
332
skip,
@@ -363,16 +380,6 @@ export function pluginTester(options: PluginTesterOptions = {}) {
363
380
} ,
364
381
{ babelOptions } ,
365
382
{
366
- babelOptions : {
367
- // ? Ensure `localOptions` comes before `babelOptions.plugins` ?
368
- // to preserve default plugin run order
369
- plugins : [
370
- [
371
- plugin ,
372
- mergeWith ( { } , basePluginOptions , pluginOptions , mergeCustomizer )
373
- ]
374
- ]
375
- } ,
376
383
testBlockTitle : `${ currentTestNumber ++ } . ${ title || blockTitle } ` ,
377
384
only,
378
385
skip,
@@ -388,17 +395,28 @@ export function pluginTester(options: PluginTesterOptions = {}) {
388
395
exec,
389
396
execFixture : execPath
390
397
} ,
398
+ // ? This is last to ensure plugins/presets babelOptions are arrays
399
+ { babelOptions : { plugins : [ ] , presets : [ ] } } ,
391
400
mergeCustomizer
392
401
) ;
393
402
403
+ if ( plugin ) {
404
+ testConfig . babelOptions . plugins . push ( [
405
+ plugin ,
406
+ mergeWith ( { } , basePluginOptions , pluginOptions , mergeCustomizer )
407
+ ] ) ;
408
+ } else {
409
+ testConfig . babelOptions . presets . unshift ( [
410
+ preset ,
411
+ mergeWith ( { } , basePresetOptions , presetOptions , mergeCustomizer )
412
+ ] ) ;
413
+ }
414
+
394
415
finalizePluginAndPresetRunOrder ( testConfig . babelOptions ) ;
395
- // ? Ensures we have an actual PluginTesterTestFixtureConfig object
396
416
validateTestConfig ( testConfig ) ;
397
417
hasTests = true ;
398
418
399
- ( parentDescribeConfig ?. tests || testConfigs ) . push (
400
- testConfig as PluginTesterTestFixtureConfig
401
- ) ;
419
+ ( parentDescribeConfig ?. tests || testConfigs ) . push ( testConfig ) ;
402
420
}
403
421
} ) ;
404
422
}
@@ -412,8 +430,9 @@ export function pluginTester(options: PluginTesterOptions = {}) {
412
430
plugin,
413
431
pluginName,
414
432
basePluginOptions,
415
- // TODO: use this preset, TODO: use this presetName, TODO: use this
416
- //basePresetOptions,
433
+ preset,
434
+ presetName,
435
+ basePresetOptions,
417
436
baseBabelOptions,
418
437
endOfLine,
419
438
baseFormatResult,
@@ -423,7 +442,7 @@ export function pluginTester(options: PluginTesterOptions = {}) {
423
442
const {
424
443
babelOptions,
425
444
pluginOptions,
426
- // TODO: use this presetOptions,
445
+ presetOptions,
427
446
title,
428
447
only,
429
448
skip,
@@ -466,14 +485,7 @@ export function pluginTester(options: PluginTesterOptions = {}) {
466
485
{ babelOptions } ,
467
486
{
468
487
snapshot : snapshot ?? baseSnapshot ,
469
- // ? Ensure `rawFixtureConfig` comes before ? `babelOptions.plugins`
470
- // to preserve default plugin run order
471
- babelOptions : {
472
- plugins : [
473
- [ plugin , mergeWith ( { } , basePluginOptions , pluginOptions , mergeCustomizer ) ]
474
- ]
475
- } ,
476
- testBlockTitle : `${ currentTestNumber ++ } . ${ title || pluginName } ` ,
488
+ testBlockTitle : `${ currentTestNumber ++ } . ${ title || pluginName || presetName } ` ,
477
489
only,
478
490
skip,
479
491
expectedError : throws ?? error ,
@@ -491,15 +503,28 @@ export function pluginTester(options: PluginTesterOptions = {}) {
491
503
exec : exec ? trimAndFixLineEndings ( exec , endOfLine ) : undefined ,
492
504
execFixture
493
505
} ,
506
+ // ? This is last to ensure plugins/presets babelOptions are arrays
507
+ { babelOptions : { plugins : [ ] , presets : [ ] } } ,
494
508
mergeCustomizer
495
509
) ;
496
510
511
+ if ( plugin ) {
512
+ testConfig . babelOptions . plugins . push ( [
513
+ plugin ,
514
+ mergeWith ( { } , basePluginOptions , pluginOptions , mergeCustomizer )
515
+ ] ) ;
516
+ } else {
517
+ testConfig . babelOptions . presets . unshift ( [
518
+ preset ,
519
+ mergeWith ( { } , basePresetOptions , presetOptions , mergeCustomizer )
520
+ ] ) ;
521
+ }
522
+
497
523
finalizePluginAndPresetRunOrder ( testConfig . babelOptions ) ;
498
- // ? Ensures we have an actual PluginTesterTestObjectConfig object
499
524
validateTestConfig ( testConfig ) ;
500
525
hasTests = true ;
501
526
502
- return testConfig as PluginTesterTestObjectConfig ;
527
+ return testConfig ;
503
528
}
504
529
}
505
530
@@ -725,29 +750,29 @@ export function pluginTester(options: PluginTesterOptions = {}) {
725
750
expectedError
726
751
} = testConfig ;
727
752
728
- if ( testConfig [ $type ] == 'test-object' && testConfig . snapshot ) {
753
+ if ( testConfig [ $type ] == 'test-object' && testConfig . snapshot ) {
729
754
if ( ! globalContextExpectFnHasToMatchSnapshot ) {
730
755
throwTypeError (
731
756
'testing environment does not support `expect(...).toMatchSnapshot` method'
732
757
) ;
733
- }
758
+ }
734
759
735
- if ( output ) {
736
- throwTypeError (
737
- 'neither `output` nor `outputFixture` can be provided with `snapshot` enabled'
738
- ) ;
739
- }
760
+ if ( output ) {
761
+ throwTypeError (
762
+ 'neither `output` nor `outputFixture` can be provided with `snapshot` enabled'
763
+ ) ;
764
+ }
740
765
741
- if ( exec ) {
742
- throwTypeError (
743
- 'neither `exec` nor `execFixture` can be provided with `snapshot` enabled'
744
- ) ;
766
+ if ( exec ) {
767
+ throwTypeError (
768
+ 'neither `exec` nor `execFixture` can be provided with `snapshot` enabled'
769
+ ) ;
770
+ }
745
771
}
746
- }
747
772
748
- if ( skip && only ) {
749
- throwTypeError ( 'cannot enable both `skip` and `only` in the same test' ) ;
750
- }
773
+ if ( skip && only ) {
774
+ throwTypeError ( 'cannot enable both `skip` and `only` in the same test' ) ;
775
+ }
751
776
752
777
if ( skip && ! globalContextTestFnHasSkip ) {
753
778
throwTypeError ( 'testing environment does not support `it.skip(...)` method' ) ;
@@ -757,62 +782,62 @@ export function pluginTester(options: PluginTesterOptions = {}) {
757
782
throwTypeError ( 'testing environment does not support `it.only(...)` method' ) ;
758
783
}
759
784
760
- if ( output && expectedError !== undefined ) {
761
- throwTypeError (
762
- testConfig [ $type ] == 'test-object'
763
- ? 'neither `output` nor `outputFixture` can be provided with `throws` or `error`'
764
- : 'a fixture cannot be provided with `throws` or `error` and also contain an output file'
765
- ) ;
766
- }
785
+ if ( output && expectedError !== undefined ) {
786
+ throwTypeError (
787
+ testConfig [ $type ] == 'test-object'
788
+ ? 'neither `output` nor `outputFixture` can be provided with `throws` or `error`'
789
+ : 'a fixture cannot be provided with `throws` or `error` and also contain an output file'
790
+ ) ;
791
+ }
767
792
768
- if ( exec && expectedError !== undefined ) {
769
- `` ;
770
- throwTypeError (
771
- testConfig [ $type ] == 'test-object'
772
- ? 'neither `exec` nor `execFixture` can be provided with `throws` or `error`'
773
- : 'a fixture cannot be provided with `throws` or `error` and also contain an exec file'
774
- ) ;
775
- }
793
+ if ( exec && expectedError !== undefined ) {
794
+ `` ;
795
+ throwTypeError (
796
+ testConfig [ $type ] == 'test-object'
797
+ ? 'neither `exec` nor `execFixture` can be provided with `throws` or `error`'
798
+ : 'a fixture cannot be provided with `throws` or `error` and also contain an exec file'
799
+ ) ;
800
+ }
776
801
777
- if ( ! code && ! exec ) {
778
- throwTypeError (
779
- testConfig [ $type ] == 'test-object'
780
- ? 'a string or object with a `code`, `codeFixture`, `exec`, or `execFixture` must be provided'
781
- : 'a fixture must contain either a code file or an exec file'
782
- ) ;
783
- }
802
+ if ( ! code && ! exec ) {
803
+ throwTypeError (
804
+ testConfig [ $type ] == 'test-object'
805
+ ? 'a string or object with a `code`, `codeFixture`, `exec`, or `execFixture` must be provided'
806
+ : 'a fixture must contain either a code file or an exec file'
807
+ ) ;
808
+ }
784
809
785
- if ( ! ! ( code || output ) && ! ! exec ) {
786
- throwTypeError (
787
- testConfig [ $type ] == 'test-object'
788
- ? 'neither `code`, `codeFixture`, `output`, nor `outputFixture` can be provided with `exec` or `execFixture`'
789
- : 'a fixture cannot contain both an exec file and a code or output file'
790
- ) ;
791
- }
810
+ if ( ! ! ( code || output ) && ! ! exec ) {
811
+ throwTypeError (
812
+ testConfig [ $type ] == 'test-object'
813
+ ? 'neither `code`, `codeFixture`, `output`, nor `outputFixture` can be provided with `exec` or `execFixture`'
814
+ : 'a fixture cannot contain both an exec file and a code or output file'
815
+ ) ;
816
+ }
792
817
793
- if ( babelOptions . babelrc && ! babelOptions . filename ) {
794
- throwTypeError (
795
- '`babelOptions.babelrc` is enabled but `babelOptions.filename` was not provided'
796
- ) ;
797
- }
818
+ if ( babelOptions . babelrc && ! babelOptions . filename ) {
819
+ throwTypeError (
820
+ '`babelOptions.babelrc` is enabled but `babelOptions.filename` was not provided'
821
+ ) ;
822
+ }
798
823
799
- if (
800
- expectedError &&
801
- ! (
802
- [ 'function' , 'boolean' , 'string' ] . includes ( typeof expectedError ) ||
803
- expectedError instanceof RegExp
804
- )
805
- ) {
806
- throwTypeError (
807
- '`throws`/`error` must be a function, string, boolean, RegExp, or Error subtype'
808
- ) ;
809
- }
824
+ if (
825
+ expectedError &&
826
+ ! (
827
+ [ 'function' , 'boolean' , 'string' ] . includes ( typeof expectedError ) ||
828
+ expectedError instanceof RegExp
829
+ )
830
+ ) {
831
+ throwTypeError (
832
+ '`throws`/`error` must be a function, string, boolean, RegExp, or Error subtype'
833
+ ) ;
834
+ }
810
835
811
- function throwTypeError ( message : string ) {
812
- throw new TypeError (
813
- `failed to validate configuration for test "${ testBlockTitle } ": ${ message } `
814
- ) ;
815
- }
836
+ function throwTypeError ( message : string ) {
837
+ throw new TypeError (
838
+ `failed to validate configuration for test "${ testBlockTitle } ": ${ message } `
839
+ ) ;
840
+ }
816
841
}
817
842
}
818
843
@@ -880,16 +905,43 @@ function trimAndFixLineEndings(
880
905
}
881
906
}
882
907
908
+ /**
909
+ * Clears out nullish plugin/preset values and replaces symbols with their
910
+ * proper values.
911
+ */
883
912
function finalizePluginAndPresetRunOrder (
884
913
babelOptions : PluginTesterOptions [ 'babelOptions' ]
885
914
) {
886
- if ( babelOptions ?. plugins ?. includes ( runPluginUnderTestHere ) ) {
887
- babelOptions . plugins . splice (
888
- babelOptions . plugins . indexOf ( runPluginUnderTestHere ) ,
889
- 1 ,
890
- babelOptions . plugins . pop ( ) !
891
- ) ;
915
+ // TODO: debug statements here about replacing symbols and clearing nullish
916
+
917
+ if ( babelOptions ?. plugins ) {
918
+ babelOptions . plugins = babelOptions . plugins . filter ( ( p ) => {
919
+ // TODO: debug statement here
920
+ return Boolean ( p ) ;
921
+ } ) ;
922
+
923
+ if ( babelOptions . plugins . includes ( runPluginUnderTestHere ) ) {
924
+ babelOptions . plugins . splice (
925
+ babelOptions . plugins . indexOf ( runPluginUnderTestHere ) ,
926
+ 1 ,
927
+ babelOptions . plugins . pop ( ) !
928
+ ) ;
929
+ }
892
930
}
893
931
894
- // TODO: also finalize preset run order
932
+ if ( babelOptions ?. presets ) {
933
+ babelOptions . presets = babelOptions . presets . filter ( ( p ) => {
934
+ // TODO: debug statement here
935
+ return Boolean ( p ) ;
936
+ } ) ;
937
+
938
+ if ( babelOptions . presets . includes ( runPresetUnderTestHere ) ) {
939
+ babelOptions . presets . splice (
940
+ // ? -1 because we're shifting an element off the beginning afterwards
941
+ babelOptions . presets . indexOf ( runPresetUnderTestHere ) - 1 ,
942
+ 1 ,
943
+ babelOptions . presets . shift ( ) !
944
+ ) ;
945
+ }
946
+ }
895
947
}
0 commit comments