@@ -244,13 +244,16 @@ module.exports = function (api) {
244
244
name : "process.env.IS_PUBLISH" ,
245
245
value : bool ( process . env . IS_PUBLISH ) ,
246
246
} ,
247
+ "flag-IS_PUBLISH" ,
247
248
] ,
248
249
249
- process . env . STRIP_BABEL_8_FLAG && [
250
+ [
250
251
pluginToggleBooleanFlag ,
251
252
{
252
253
name : "process.env.BABEL_8_BREAKING" ,
253
- value : bool ( process . env . BABEL_8_BREAKING ) ,
254
+ value : process . env . STRIP_BABEL_8_FLAG
255
+ ? bool ( process . env . BABEL_8_BREAKING )
256
+ : null ,
254
257
} ,
255
258
"flag-BABEL_8_BREAKING" ,
256
259
] ,
@@ -572,39 +575,48 @@ function pluginPolyfillsOldNode({ template, types: t }) {
572
575
}
573
576
574
577
/**
578
+ * If `value` is null, it will leave the conditions in code. It will remove
579
+ * them in attributes by replacing them with <if true> || <if false>
580
+ *
575
581
* @param {import("@babel/core") } pluginAPI
576
582
* @returns {import("@babel/core").PluginObject }
577
583
*/
578
- function pluginToggleBooleanFlag ( { types : t } , { name, value } ) {
579
- if ( typeof value !== "boolean" ) throw new Error ( `.value must be a boolean` ) ;
584
+ function pluginToggleBooleanFlag ( { types : t , template } , { name, value } ) {
585
+ if ( typeof value !== "boolean" && value !== null ) {
586
+ throw new Error ( `.value must be a boolean, or null` ) ;
587
+ }
580
588
581
- function evaluate ( test ) {
589
+ function parseExpression ( source ) {
590
+ // eslint-disable-next-line regexp/no-useless-assertions
591
+ return template . expression ( source , { placeholderPattern : / (? ! .) ./ } ) ( { } ) ;
592
+ }
593
+
594
+ /**
595
+ * @param {import("@babel/core").types.Expression } test
596
+ */
597
+ function evaluate ( test , value ) {
582
598
const res = {
583
599
replace : replacement => ( { replacement, value : null , unrelated : false } ) ,
584
600
value : value => ( { replacement : null , value, unrelated : false } ) ,
585
- unrelated : ( ) => ( {
586
- replacement : test . node ,
587
- value : null ,
588
- unrelated : true ,
589
- } ) ,
601
+ unrelated : ( ) => ( { replacement : test , value : null , unrelated : true } ) ,
590
602
} ;
591
603
592
- if ( test . isIdentifier ( { name } ) || test . matchesPattern ( name ) ) {
604
+ if ( t . isIdentifier ( test , { name } ) || t . matchesPattern ( test , name ) ) {
593
605
return res . value ( value ) ;
594
606
}
595
607
596
- if ( test . isUnaryExpression ( { operator : "!" } ) ) {
597
- const arg = evaluate ( test . get ( " argument" ) ) ;
608
+ if ( t . isUnaryExpression ( test , { operator : "!" } ) ) {
609
+ const arg = evaluate ( test . argument , value ) ;
598
610
return arg . unrelated
599
611
? res . unrelated ( )
600
612
: arg . replacement
601
613
? res . replacement ( t . unaryExpression ( "!" , arg . replacement ) )
602
614
: res . value ( ! arg . value ) ;
603
615
}
604
616
605
- if ( test . isLogicalExpression ( { operator : "||" } ) ) {
606
- const left = evaluate ( test . get ( " left" ) ) ;
607
- const right = evaluate ( test . get ( " right" ) ) ;
617
+ if ( t . isLogicalExpression ( test , { operator : "||" } ) ) {
618
+ const left = evaluate ( test . left , value ) ;
619
+ const right = evaluate ( test . right , value ) ;
608
620
609
621
if ( left . value === true || right . value === true ) return res . value ( true ) ;
610
622
if ( left . value === false && right . value === false ) {
@@ -618,9 +630,9 @@ function pluginToggleBooleanFlag({ types: t }, { name, value }) {
618
630
) ;
619
631
}
620
632
621
- if ( test . isLogicalExpression ( { operator : "&&" } ) ) {
622
- const left = evaluate ( test . get ( " left" ) ) ;
623
- const right = evaluate ( test . get ( " right" ) ) ;
633
+ if ( t . isLogicalExpression ( test , { operator : "&&" } ) ) {
634
+ const left = evaluate ( test . left , value ) ;
635
+ const right = evaluate ( test . right , value ) ;
624
636
625
637
if ( left . value === true && right . value === true ) return res . value ( true ) ;
626
638
if ( left . value === false || right . value === false ) {
@@ -640,6 +652,8 @@ function pluginToggleBooleanFlag({ types: t }, { name, value }) {
640
652
return {
641
653
visitor : {
642
654
"IfStatement|ConditionalExpression" ( path ) {
655
+ if ( value === null ) return ;
656
+
643
657
let test = path . get ( "test" ) ;
644
658
645
659
// yarn-plugin-conditions injects bool(process.env.BABEL_8_BREAKING)
@@ -652,7 +666,7 @@ function pluginToggleBooleanFlag({ types: t }, { name, value }) {
652
666
test = test . get ( "arguments" ) [ 0 ] ;
653
667
}
654
668
655
- const res = evaluate ( test ) ;
669
+ const res = evaluate ( test . node , value ) ;
656
670
657
671
if ( res . unrelated ) return ;
658
672
if ( res . replacement ) {
@@ -666,7 +680,9 @@ function pluginToggleBooleanFlag({ types: t }, { name, value }) {
666
680
}
667
681
} ,
668
682
LogicalExpression ( path ) {
669
- const res = evaluate ( path ) ;
683
+ if ( value === null ) return ;
684
+
685
+ const res = evaluate ( path . node , value ) ;
670
686
if ( res . unrelated ) return ;
671
687
if ( res . replacement ) {
672
688
path . replaceWith ( res . replacement ) ;
@@ -675,15 +691,90 @@ function pluginToggleBooleanFlag({ types: t }, { name, value }) {
675
691
}
676
692
} ,
677
693
MemberExpression ( path ) {
694
+ if ( value === null ) return ;
695
+
678
696
if ( path . matchesPattern ( name ) ) {
679
697
throw path . buildCodeFrameError ( "This check could not be stripped." ) ;
680
698
}
681
699
} ,
682
700
ReferencedIdentifier ( path ) {
701
+ if ( value === null ) return ;
702
+
683
703
if ( path . node . name === name ) {
684
704
throw path . buildCodeFrameError ( "This check could not be stripped." ) ;
685
705
}
686
706
} ,
707
+ ImportDeclaration ( path ) {
708
+ if ( ! path . node . attributes ) return ;
709
+
710
+ /** @type {null | import("@babel/core").NodePath<import("@babel/core").types.ImportAttribute> } */
711
+ const ifAttribute = path
712
+ . get ( "attributes" )
713
+ . find ( attr => attr . node . key . name === "if" ) ;
714
+ if ( ifAttribute == null ) {
715
+ return ;
716
+ }
717
+
718
+ const condition = parseExpression ( ifAttribute . node . value . value ) ;
719
+
720
+ let res ;
721
+ res_block: if ( value !== null ) {
722
+ res = evaluate ( condition , value ) ;
723
+ } else {
724
+ const ifTrue = evaluate ( condition , true ) ;
725
+ if ( ifTrue . unrelated || ifTrue . value === true ) {
726
+ res = ifTrue ;
727
+ break res_block;
728
+ }
729
+ const ifFalse = evaluate ( condition , false ) ;
730
+ if ( ifFalse . unrelated ) throw new Error ( "Cannot be unrelated" ) ;
731
+ if ( ifFalse . value === true ) {
732
+ res = ifFalse ;
733
+ break res_block;
734
+ }
735
+
736
+ if ( ifTrue . value === false ) {
737
+ res = ifFalse ;
738
+ break res_block;
739
+ }
740
+ if ( ifTrue . value === false ) {
741
+ res = ifTrue ;
742
+ break res_block;
743
+ }
744
+
745
+ if ( ! ifTrue . replacement && ! ifFalse . replacement ) {
746
+ throw new Error ( "Expected two replacements" ) ;
747
+ }
748
+
749
+ res = {
750
+ replacement : t . binaryExpression (
751
+ "||" ,
752
+ ifTrue . replacement ,
753
+ ifFalse . replacement
754
+ ) ,
755
+ value : null ,
756
+ unrelated : false ,
757
+ } ;
758
+ }
759
+
760
+ if ( res . unrelated ) return ;
761
+ if ( res . replacement ) {
762
+ ifAttribute
763
+ . get ( "value" )
764
+ . replaceWith (
765
+ t . stringLiteral (
766
+ ifAttribute . toString . call ( { node : res . replacement } )
767
+ )
768
+ ) ;
769
+ } else if ( res . value === true ) {
770
+ ifAttribute . remove ( ) ;
771
+ if ( path . node . attributes . length === 0 ) {
772
+ path . node . attributes = null ;
773
+ }
774
+ } else if ( res . value === false ) {
775
+ path . remove ( ) ;
776
+ }
777
+ } ,
687
778
} ,
688
779
} ;
689
780
}
@@ -916,6 +1007,12 @@ function pluginImportMetaUrl({ types: t, template }) {
916
1007
917
1008
// Let's just remove this declaration to unshadow the "global" cjs require.
918
1009
path . remove ( ) ;
1010
+ path . scope . crawl ( ) ;
1011
+
1012
+ const createRequireBinding = path . scope . getBinding ( "createRequire" ) ;
1013
+ if ( ! createRequireBinding . referenced ) {
1014
+ createRequireBinding . path . remove ( ) ;
1015
+ }
919
1016
} ,
920
1017
921
1018
// import.meta.url
0 commit comments