@@ -699,6 +699,18 @@ namespace ts {
699
699
} ;
700
700
}
701
701
702
+ enum BuildStep {
703
+ CreateProgram ,
704
+ SyntaxDiagnostics ,
705
+ SemanticDiagnostics ,
706
+ Emit ,
707
+ EmitBundle ,
708
+ EmitBuildInfo ,
709
+ BuildInvalidatedProjectOfBundle ,
710
+ QueueReferencingProjects ,
711
+ Done
712
+ }
713
+
702
714
function createBuildOrUpdateInvalidedProject < T extends BuilderProgram > (
703
715
kind : InvalidatedProjectKind . Build | InvalidatedProjectKind . UpdateBundle ,
704
716
state : SolutionBuilderState < T > ,
@@ -708,18 +720,7 @@ namespace ts {
708
720
config : ParsedCommandLine ,
709
721
buildOrder : readonly ResolvedConfigFileName [ ] ,
710
722
) : BuildInvalidedProject < T > | UpdateBundleProject < T > {
711
- enum Step {
712
- CreateProgram ,
713
- SyntaxDiagnostics ,
714
- SemanticDiagnostics ,
715
- Emit ,
716
- EmitBundle ,
717
- BuildInvalidatedProjectOfBundle ,
718
- QueueReferencingProjects ,
719
- Done
720
- }
721
-
722
- let step = kind === InvalidatedProjectKind . Build ? Step . CreateProgram : Step . EmitBundle ;
723
+ let step = kind === InvalidatedProjectKind . Build ? BuildStep . CreateProgram : BuildStep . EmitBundle ;
723
724
let program : T | undefined ;
724
725
let buildResult : BuildResultFlags | undefined ;
725
726
let invalidatedProjectOfBundle : BuildInvalidedProject < T > | undefined ;
@@ -781,8 +782,11 @@ namespace ts {
781
782
program => program . emit ( targetSourceFile , writeFile , cancellationToken , emitOnlyDtsFiles , customTransformers )
782
783
) ;
783
784
}
784
- executeSteps ( Step . SemanticDiagnostics , cancellationToken ) ;
785
- if ( step !== Step . Emit ) return undefined ;
785
+ executeSteps ( BuildStep . SemanticDiagnostics , cancellationToken ) ;
786
+ if ( step === BuildStep . EmitBuildInfo ) {
787
+ return emitBuildInfo ( writeFile , cancellationToken ) ;
788
+ }
789
+ if ( step !== BuildStep . Emit ) return undefined ;
786
790
return emit ( writeFile , cancellationToken , customTransformers ) ;
787
791
} ,
788
792
done
@@ -795,19 +799,19 @@ namespace ts {
795
799
getCompilerOptions : ( ) => config . options ,
796
800
getCurrentDirectory : ( ) => state . currentDirectory ,
797
801
emit : ( writeFile : WriteFileCallback | undefined , customTransformers : CustomTransformers | undefined ) => {
798
- if ( step !== Step . EmitBundle ) return invalidatedProjectOfBundle ;
802
+ if ( step !== BuildStep . EmitBundle ) return invalidatedProjectOfBundle ;
799
803
return emitBundle ( writeFile , customTransformers ) ;
800
804
} ,
801
805
done,
802
806
} ;
803
807
804
808
function done ( cancellationToken ?: CancellationToken , writeFile ?: WriteFileCallback , customTransformers ?: CustomTransformers ) {
805
- executeSteps ( Step . Done , cancellationToken , writeFile , customTransformers ) ;
809
+ executeSteps ( BuildStep . Done , cancellationToken , writeFile , customTransformers ) ;
806
810
return doneInvalidatedProject ( state , projectPath ) ;
807
811
}
808
812
809
813
function withProgramOrUndefined < U > ( action : ( program : T ) => U | undefined ) : U | undefined {
810
- executeSteps ( Step . CreateProgram ) ;
814
+ executeSteps ( BuildStep . CreateProgram ) ;
811
815
return program && action ( program ) ;
812
816
}
813
817
@@ -821,7 +825,7 @@ namespace ts {
821
825
if ( state . options . dry ) {
822
826
reportStatus ( state , Diagnostics . A_non_dry_build_would_build_project_0 , project ) ;
823
827
buildResult = BuildResultFlags . Success ;
824
- step = Step . QueueReferencingProjects ;
828
+ step = BuildStep . QueueReferencingProjects ;
825
829
return ;
826
830
}
827
831
@@ -831,7 +835,7 @@ namespace ts {
831
835
reportAndStoreErrors ( state , projectPath , getConfigFileParsingDiagnostics ( config ) ) ;
832
836
// Nothing to build - must be a solution file, basically
833
837
buildResult = BuildResultFlags . None ;
834
- step = Step . QueueReferencingProjects ;
838
+ step = BuildStep . QueueReferencingProjects ;
835
839
return ;
836
840
}
837
841
@@ -854,16 +858,15 @@ namespace ts {
854
858
855
859
function handleDiagnostics ( diagnostics : readonly Diagnostic [ ] , errorFlags : BuildResultFlags , errorType : string ) {
856
860
if ( diagnostics . length ) {
857
- buildResult = buildErrors (
861
+ ( { buildResult, step } = buildErrors (
858
862
state ,
859
863
projectPath ,
860
864
program ,
861
865
config ,
862
866
diagnostics ,
863
867
errorFlags ,
864
868
errorType
865
- ) ;
866
- step = Step . QueueReferencingProjects ;
869
+ ) ) ;
867
870
}
868
871
else {
869
872
step ++ ;
@@ -894,7 +897,7 @@ namespace ts {
894
897
895
898
function emit ( writeFileCallback ?: WriteFileCallback , cancellationToken ?: CancellationToken , customTransformers ?: CustomTransformers ) : EmitResult {
896
899
Debug . assertIsDefined ( program ) ;
897
- Debug . assert ( step === Step . Emit ) ;
900
+ Debug . assert ( step === BuildStep . Emit ) ;
898
901
// Before emitting lets backup state, so we can revert it back if there are declaration errors to handle emit and declaration errors correctly
899
902
program . backupState ( ) ;
900
903
let declDiagnostics : Diagnostic [ ] | undefined ;
@@ -913,16 +916,15 @@ namespace ts {
913
916
// Don't emit .d.ts if there are decl file errors
914
917
if ( declDiagnostics ) {
915
918
program . restoreState ( ) ;
916
- buildResult = buildErrors (
919
+ ( { buildResult, step } = buildErrors (
917
920
state ,
918
921
projectPath ,
919
922
program ,
920
923
config ,
921
924
declDiagnostics ,
922
925
BuildResultFlags . DeclarationEmitErrors ,
923
926
"Declaration file"
924
- ) ;
925
- step = Step . QueueReferencingProjects ;
927
+ ) ) ;
926
928
return {
927
929
emitSkipped : true ,
928
930
diagnostics : emitResult . diagnostics
@@ -967,6 +969,24 @@ namespace ts {
967
969
return emitResult ;
968
970
}
969
971
972
+ function emitBuildInfo ( writeFileCallback ?: WriteFileCallback , cancellationToken ?: CancellationToken ) : EmitResult {
973
+ Debug . assertIsDefined ( program ) ;
974
+ Debug . assert ( step === BuildStep . EmitBuildInfo ) ;
975
+ const emitResult = program . emitBuildInfo ( writeFileCallback , cancellationToken ) ;
976
+ if ( emitResult . diagnostics . length ) {
977
+ reportErrors ( state , emitResult . diagnostics ) ;
978
+ state . diagnostics . set ( projectPath , [ ...state . diagnostics . get ( projectPath ) ! , ...emitResult . diagnostics ] ) ;
979
+ buildResult = BuildResultFlags . EmitErrors & buildResult ! ;
980
+ }
981
+
982
+ if ( emitResult . emittedFiles && state . writeFileName ) {
983
+ emitResult . emittedFiles . forEach ( name => listEmittedFile ( state , config , name ) ) ;
984
+ }
985
+ afterProgramDone ( state , projectPath , program , config ) ;
986
+ step = BuildStep . QueueReferencingProjects ;
987
+ return emitResult ;
988
+ }
989
+
970
990
function finishEmit (
971
991
emitterDiagnostics : DiagnosticCollection ,
972
992
emittedOutputs : FileMap < string > ,
@@ -977,22 +997,20 @@ namespace ts {
977
997
) {
978
998
const emitDiagnostics = emitterDiagnostics . getDiagnostics ( ) ;
979
999
if ( emitDiagnostics . length ) {
980
- buildResult = buildErrors (
1000
+ ( { buildResult, step } = buildErrors (
981
1001
state ,
982
1002
projectPath ,
983
1003
program ,
984
1004
config ,
985
1005
emitDiagnostics ,
986
1006
BuildResultFlags . EmitErrors ,
987
1007
"Emit"
988
- ) ;
989
- step = Step . QueueReferencingProjects ;
1008
+ ) ) ;
990
1009
return emitDiagnostics ;
991
1010
}
992
1011
993
1012
if ( state . writeFileName ) {
994
1013
emittedOutputs . forEach ( name => listEmittedFile ( state , config , name ) ) ;
995
- if ( program ) listFiles ( program , state . writeFileName ) ;
996
1014
}
997
1015
998
1016
// Update time stamps for rest of the outputs
@@ -1006,8 +1024,7 @@ namespace ts {
1006
1024
oldestOutputFileName
1007
1025
} ) ;
1008
1026
afterProgramDone ( state , projectPath , program , config ) ;
1009
- state . projectCompilerOptions = state . baseCompilerOptions ;
1010
- step = Step . QueueReferencingProjects ;
1027
+ step = BuildStep . QueueReferencingProjects ;
1011
1028
buildResult = resultFlags ;
1012
1029
return emitDiagnostics ;
1013
1030
}
@@ -1017,7 +1034,7 @@ namespace ts {
1017
1034
if ( state . options . dry ) {
1018
1035
reportStatus ( state , Diagnostics . A_non_dry_build_would_update_output_of_project_0 , project ) ;
1019
1036
buildResult = BuildResultFlags . Success ;
1020
- step = Step . QueueReferencingProjects ;
1037
+ step = BuildStep . QueueReferencingProjects ;
1021
1038
return undefined ;
1022
1039
}
1023
1040
@@ -1038,7 +1055,7 @@ namespace ts {
1038
1055
1039
1056
if ( isString ( outputFiles ) ) {
1040
1057
reportStatus ( state , Diagnostics . Cannot_update_output_of_project_0_because_there_was_error_reading_file_1 , project , relName ( state , outputFiles ) ) ;
1041
- step = Step . BuildInvalidatedProjectOfBundle ;
1058
+ step = BuildStep . BuildInvalidatedProjectOfBundle ;
1042
1059
return invalidatedProjectOfBundle = createBuildOrUpdateInvalidedProject (
1043
1060
InvalidatedProjectKind . Build ,
1044
1061
state ,
@@ -1070,44 +1087,48 @@ namespace ts {
1070
1087
return { emitSkipped : false , diagnostics : emitDiagnostics } ;
1071
1088
}
1072
1089
1073
- function executeSteps ( till : Step , cancellationToken ?: CancellationToken , writeFile ?: WriteFileCallback , customTransformers ?: CustomTransformers ) {
1074
- while ( step <= till && step < Step . Done ) {
1090
+ function executeSteps ( till : BuildStep , cancellationToken ?: CancellationToken , writeFile ?: WriteFileCallback , customTransformers ?: CustomTransformers ) {
1091
+ while ( step <= till && step < BuildStep . Done ) {
1075
1092
const currentStep = step ;
1076
1093
switch ( step ) {
1077
- case Step . CreateProgram :
1094
+ case BuildStep . CreateProgram :
1078
1095
createProgram ( ) ;
1079
1096
break ;
1080
1097
1081
- case Step . SyntaxDiagnostics :
1098
+ case BuildStep . SyntaxDiagnostics :
1082
1099
getSyntaxDiagnostics ( cancellationToken ) ;
1083
1100
break ;
1084
1101
1085
- case Step . SemanticDiagnostics :
1102
+ case BuildStep . SemanticDiagnostics :
1086
1103
getSemanticDiagnostics ( cancellationToken ) ;
1087
1104
break ;
1088
1105
1089
- case Step . Emit :
1106
+ case BuildStep . Emit :
1090
1107
emit ( writeFile , cancellationToken , customTransformers ) ;
1091
1108
break ;
1092
1109
1093
- case Step . EmitBundle :
1110
+ case BuildStep . EmitBuildInfo :
1111
+ emitBuildInfo ( writeFile , cancellationToken ) ;
1112
+ break ;
1113
+
1114
+ case BuildStep . EmitBundle :
1094
1115
emitBundle ( writeFile , customTransformers ) ;
1095
1116
break ;
1096
1117
1097
- case Step . BuildInvalidatedProjectOfBundle :
1118
+ case BuildStep . BuildInvalidatedProjectOfBundle :
1098
1119
Debug . checkDefined ( invalidatedProjectOfBundle ) . done ( cancellationToken ) ;
1099
- step = Step . Done ;
1120
+ step = BuildStep . Done ;
1100
1121
break ;
1101
1122
1102
- case Step . QueueReferencingProjects :
1123
+ case BuildStep . QueueReferencingProjects :
1103
1124
queueReferencingProjects ( state , project , projectPath , projectIndex , config , buildOrder , Debug . checkDefined ( buildResult ) ) ;
1104
1125
step ++ ;
1105
1126
break ;
1106
1127
1107
1128
// Should never be done
1108
- case Step . Done :
1129
+ case BuildStep . Done :
1109
1130
default :
1110
- assertType < Step . Done > ( step ) ;
1131
+ assertType < BuildStep . Done > ( step ) ;
1111
1132
1112
1133
}
1113
1134
Debug . assert ( step > currentStep ) ;
@@ -1247,23 +1268,25 @@ namespace ts {
1247
1268
}
1248
1269
1249
1270
function afterProgramDone < T extends BuilderProgram > (
1250
- { host , watch , builderPrograms } : SolutionBuilderState < T > ,
1271
+ state : SolutionBuilderState < T > ,
1251
1272
proj : ResolvedConfigFilePath ,
1252
1273
program : T | undefined ,
1253
1274
config : ParsedCommandLine
1254
1275
) {
1255
1276
if ( program ) {
1256
- if ( host . afterProgramEmitAndDiagnostics ) {
1257
- host . afterProgramEmitAndDiagnostics ( program ) ;
1277
+ if ( program && state . writeFileName ) listFiles ( program , state . writeFileName ) ;
1278
+ if ( state . host . afterProgramEmitAndDiagnostics ) {
1279
+ state . host . afterProgramEmitAndDiagnostics ( program ) ;
1258
1280
}
1259
- if ( watch ) {
1281
+ if ( state . watch ) {
1260
1282
program . releaseProgram ( ) ;
1261
- builderPrograms . set ( proj , program ) ;
1283
+ state . builderPrograms . set ( proj , program ) ;
1262
1284
}
1263
1285
}
1264
- else if ( host . afterEmitBundle ) {
1265
- host . afterEmitBundle ( config ) ;
1286
+ else if ( state . host . afterEmitBundle ) {
1287
+ state . host . afterEmitBundle ( config ) ;
1266
1288
}
1289
+ state . projectCompilerOptions = state . baseCompilerOptions ;
1267
1290
}
1268
1291
1269
1292
function buildErrors < T extends BuilderProgram > (
@@ -1272,16 +1295,17 @@ namespace ts {
1272
1295
program : T | undefined ,
1273
1296
config : ParsedCommandLine ,
1274
1297
diagnostics : readonly Diagnostic [ ] ,
1275
- errorFlags : BuildResultFlags ,
1276
- errorType : string
1298
+ buildResult : BuildResultFlags ,
1299
+ errorType : string ,
1277
1300
) {
1301
+ const canEmitBuildInfo = ! ( buildResult & BuildResultFlags . SyntaxErrors ) && program && ! outFile ( program . getCompilerOptions ( ) ) ;
1302
+
1278
1303
reportAndStoreErrors ( state , resolvedPath , diagnostics ) ;
1279
1304
// List files if any other build error using program (emit errors already report files)
1280
- if ( program && state . writeFileName ) listFiles ( program , state . writeFileName ) ;
1281
1305
state . projectStatus . set ( resolvedPath , { type : UpToDateStatusType . Unbuildable , reason : `${ errorType } errors` } ) ;
1306
+ if ( canEmitBuildInfo ) return { buildResult, step : BuildStep . EmitBuildInfo } ;
1282
1307
afterProgramDone ( state , resolvedPath , program , config ) ;
1283
- state . projectCompilerOptions = state . baseCompilerOptions ;
1284
- return errorFlags ;
1308
+ return { buildResult, step : BuildStep . QueueReferencingProjects } ;
1285
1309
}
1286
1310
1287
1311
function updateModuleResolutionCache (
0 commit comments