7
7
*/
8
8
9
9
import type { NgtscProgram } from '@angular/compiler-cli' ;
10
- import { transformAsync } from '@babel/core' ;
11
10
import type {
12
11
OnStartResult ,
13
12
OutputFile ,
@@ -17,15 +16,14 @@ import type {
17
16
PluginBuild ,
18
17
} from 'esbuild' ;
19
18
import * as assert from 'node:assert' ;
20
- import * as fs from 'node:fs/promises' ;
21
19
import { platform } from 'node:os' ;
22
20
import * as path from 'node:path' ;
23
21
import { pathToFileURL } from 'node:url' ;
24
22
import ts from 'typescript' ;
25
- import angularApplicationPreset from '../../babel/presets/application' ;
26
- import { requiresLinking } from '../../babel/webpack-loader' ;
23
+ import { maxWorkers } from '../../utils/environment-options' ;
27
24
import { loadEsmModule } from '../../utils/load-esm' ;
28
25
import { createAngularCompilerHost , ensureSourceFileVersions } from './angular-host' ;
26
+ import { JavaScriptTransformer } from './javascript-transformer' ;
29
27
import {
30
28
logCumulativeDurations ,
31
29
profileAsync ,
@@ -175,6 +173,9 @@ export function createCompilerPlugin(
175
173
async setup ( build : PluginBuild ) : Promise < void > {
176
174
let setupWarnings : PartialMessage [ ] | undefined ;
177
175
176
+ // Initialize a worker pool for JavaScript transformations
177
+ const javascriptTransformer = new JavaScriptTransformer ( pluginOptions , maxWorkers ) ;
178
+
178
179
// This uses a wrapped dynamic import to load `@angular/compiler-cli` which is ESM.
179
180
// Once TypeScript provides support for retaining dynamic imports this workaround can be dropped.
180
181
const { GLOBAL_DEFS_FOR_TERSER_WITH_AOT , NgtscProgram, OptimizeFor, readConfiguration } =
@@ -252,7 +253,6 @@ export function createCompilerPlugin(
252
253
253
254
let previousBuilder : ts . EmitAndSemanticDiagnosticsBuilderProgram | undefined ;
254
255
let previousAngularProgram : NgtscProgram | undefined ;
255
- const babelDataCache = new Map < string , Uint8Array > ( ) ;
256
256
const diagnosticCache = new WeakMap < ts . SourceFile , ts . Diagnostic [ ] > ( ) ;
257
257
258
258
build . onStart ( async ( ) => {
@@ -428,7 +428,7 @@ export function createCompilerPlugin(
428
428
429
429
if ( contents === undefined ) {
430
430
const typescriptResult = await fileEmitter ( request ) ;
431
- if ( ! typescriptResult ) {
431
+ if ( ! typescriptResult ?. content ) {
432
432
// No TS result indicates the file is not part of the TypeScript program.
433
433
// If allowJs is enabled and the file is JS then defer to the next load hook.
434
434
if ( compilerOptions . allowJs && / \. [ c m ] ? j s $ / . test ( request ) ) {
@@ -447,17 +447,11 @@ export function createCompilerPlugin(
447
447
} ;
448
448
}
449
449
450
- const data = typescriptResult . content ?? '' ;
451
- // The pre-transformed data is used as a cache key. Since the cache is memory only,
452
- // the options cannot change and do not need to be represented in the key. If the
453
- // cache is later stored to disk, then the options that affect transform output
454
- // would need to be added to the key as well.
455
- contents = babelDataCache . get ( data ) ;
456
- if ( contents === undefined ) {
457
- const transformedData = await transformWithBabel ( request , data , pluginOptions ) ;
458
- contents = Buffer . from ( transformedData , 'utf-8' ) ;
459
- babelDataCache . set ( data , contents ) ;
460
- }
450
+ contents = await javascriptTransformer . transformData (
451
+ request ,
452
+ typescriptResult . content ,
453
+ true /* skipLinker */ ,
454
+ ) ;
461
455
462
456
pluginOptions . sourceFileCache ?. typeScriptFileCache . set (
463
457
pathToFileURL ( request ) . href ,
@@ -484,9 +478,7 @@ export function createCompilerPlugin(
484
478
// would need to be added to the key as well as a check for any change of content.
485
479
let contents = pluginOptions . sourceFileCache ?. babelFileCache . get ( args . path ) ;
486
480
if ( contents === undefined ) {
487
- const data = await fs . readFile ( args . path , 'utf-8' ) ;
488
- const transformedData = await transformWithBabel ( args . path , data , pluginOptions ) ;
489
- contents = Buffer . from ( transformedData , 'utf-8' ) ;
481
+ contents = await javascriptTransformer . transformFile ( args . path ) ;
490
482
pluginOptions . sourceFileCache ?. babelFileCache . set ( args . path , contents ) ;
491
483
}
492
484
@@ -540,65 +532,6 @@ function createFileEmitter(
540
532
} ;
541
533
}
542
534
543
- async function transformWithBabel (
544
- filename : string ,
545
- data : string ,
546
- pluginOptions : CompilerPluginOptions ,
547
- ) : Promise < string > {
548
- const forceAsyncTransformation =
549
- ! / [ \\ / ] [ _ f ] ? e s m 2 0 1 5 [ \\ / ] / . test ( filename ) && / a s y n c \s + f u n c t i o n \s * \* / . test ( data ) ;
550
- const shouldLink = await requiresLinking ( filename , data ) ;
551
- const useInputSourcemap =
552
- pluginOptions . sourcemap &&
553
- ( ! ! pluginOptions . thirdPartySourcemaps || ! / [ \\ / ] n o d e _ m o d u l e s [ \\ / ] / . test ( filename ) ) ;
554
-
555
- // If no additional transformations are needed, return the data directly
556
- if ( ! forceAsyncTransformation && ! pluginOptions . advancedOptimizations && ! shouldLink ) {
557
- // Strip sourcemaps if they should not be used
558
- return useInputSourcemap ? data : data . replace ( / ^ \/ \/ # s o u r c e M a p p i n g U R L = [ ^ \r \n ] * / gm, '' ) ;
559
- }
560
-
561
- const angularPackage = / [ \\ / ] n o d e _ m o d u l e s [ \\ / ] @ a n g u l a r [ \\ / ] / . test ( filename ) ;
562
-
563
- const linkerPluginCreator = shouldLink
564
- ? (
565
- await loadEsmModule < typeof import ( '@angular/compiler-cli/linker/babel' ) > (
566
- '@angular/compiler-cli/linker/babel' ,
567
- )
568
- ) . createEs2015LinkerPlugin
569
- : undefined ;
570
-
571
- const result = await transformAsync ( data , {
572
- filename,
573
- inputSourceMap : ( useInputSourcemap ? undefined : false ) as undefined ,
574
- sourceMaps : pluginOptions . sourcemap ? 'inline' : false ,
575
- compact : false ,
576
- configFile : false ,
577
- babelrc : false ,
578
- browserslistConfigFile : false ,
579
- plugins : [ ] ,
580
- presets : [
581
- [
582
- angularApplicationPreset ,
583
- {
584
- angularLinker : {
585
- shouldLink,
586
- jitMode : false ,
587
- linkerPluginCreator,
588
- } ,
589
- forceAsyncTransformation,
590
- optimize : pluginOptions . advancedOptimizations && {
591
- looseEnums : angularPackage ,
592
- pureTopLevel : angularPackage ,
593
- } ,
594
- } ,
595
- ] ,
596
- ] ,
597
- } ) ;
598
-
599
- return result ?. code ?? data ;
600
- }
601
-
602
535
function findAffectedFiles (
603
536
builder : ts . EmitAndSemanticDiagnosticsBuilderProgram ,
604
537
{ ignoreForDiagnostics, ignoreForEmit, incrementalCompilation } : NgtscProgram [ 'compiler' ] ,
0 commit comments