Skip to content

Commit 0c01532

Browse files
committedNov 21, 2022
perf(@angular-devkit/build-angular): use worker pool for JavaScript transforms in esbuild builder
When using the experimental esbuild-based browser application builder, the JavaScript transformation steps of the build process will now be performed within a worker pool to allow for the steps to be executed in parallel when possible. This also moves the steps off of the main thread which provides more time for the build orchestration and esbuild integration code to execute.
1 parent 5ba6401 commit 0c01532

File tree

3 files changed

+200
-79
lines changed

3 files changed

+200
-79
lines changed
 

‎packages/angular_devkit/build_angular/src/builders/browser-esbuild/compiler-plugin.ts

+12-79
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,6 @@
77
*/
88

99
import type { NgtscProgram } from '@angular/compiler-cli';
10-
import { transformAsync } from '@babel/core';
1110
import type {
1211
OnStartResult,
1312
OutputFile,
@@ -17,15 +16,14 @@ import type {
1716
PluginBuild,
1817
} from 'esbuild';
1918
import * as assert from 'node:assert';
20-
import * as fs from 'node:fs/promises';
2119
import { platform } from 'node:os';
2220
import * as path from 'node:path';
2321
import { pathToFileURL } from 'node:url';
2422
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';
2724
import { loadEsmModule } from '../../utils/load-esm';
2825
import { createAngularCompilerHost, ensureSourceFileVersions } from './angular-host';
26+
import { JavaScriptTransformer } from './javascript-transformer';
2927
import {
3028
logCumulativeDurations,
3129
profileAsync,
@@ -175,6 +173,9 @@ export function createCompilerPlugin(
175173
async setup(build: PluginBuild): Promise<void> {
176174
let setupWarnings: PartialMessage[] | undefined;
177175

176+
// Initialize a worker pool for JavaScript transformations
177+
const javascriptTransformer = new JavaScriptTransformer(pluginOptions, maxWorkers);
178+
178179
// This uses a wrapped dynamic import to load `@angular/compiler-cli` which is ESM.
179180
// Once TypeScript provides support for retaining dynamic imports this workaround can be dropped.
180181
const { GLOBAL_DEFS_FOR_TERSER_WITH_AOT, NgtscProgram, OptimizeFor, readConfiguration } =
@@ -252,7 +253,6 @@ export function createCompilerPlugin(
252253

253254
let previousBuilder: ts.EmitAndSemanticDiagnosticsBuilderProgram | undefined;
254255
let previousAngularProgram: NgtscProgram | undefined;
255-
const babelDataCache = new Map<string, Uint8Array>();
256256
const diagnosticCache = new WeakMap<ts.SourceFile, ts.Diagnostic[]>();
257257

258258
build.onStart(async () => {
@@ -428,7 +428,7 @@ export function createCompilerPlugin(
428428

429429
if (contents === undefined) {
430430
const typescriptResult = await fileEmitter(request);
431-
if (!typescriptResult) {
431+
if (!typescriptResult?.content) {
432432
// No TS result indicates the file is not part of the TypeScript program.
433433
// If allowJs is enabled and the file is JS then defer to the next load hook.
434434
if (compilerOptions.allowJs && /\.[cm]?js$/.test(request)) {
@@ -447,17 +447,11 @@ export function createCompilerPlugin(
447447
};
448448
}
449449

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+
);
461455

462456
pluginOptions.sourceFileCache?.typeScriptFileCache.set(
463457
pathToFileURL(request).href,
@@ -484,9 +478,7 @@ export function createCompilerPlugin(
484478
// would need to be added to the key as well as a check for any change of content.
485479
let contents = pluginOptions.sourceFileCache?.babelFileCache.get(args.path);
486480
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);
490482
pluginOptions.sourceFileCache?.babelFileCache.set(args.path, contents);
491483
}
492484

@@ -540,65 +532,6 @@ function createFileEmitter(
540532
};
541533
}
542534

543-
async function transformWithBabel(
544-
filename: string,
545-
data: string,
546-
pluginOptions: CompilerPluginOptions,
547-
): Promise<string> {
548-
const forceAsyncTransformation =
549-
!/[\\/][_f]?esm2015[\\/]/.test(filename) && /async\s+function\s*\*/.test(data);
550-
const shouldLink = await requiresLinking(filename, data);
551-
const useInputSourcemap =
552-
pluginOptions.sourcemap &&
553-
(!!pluginOptions.thirdPartySourcemaps || !/[\\/]node_modules[\\/]/.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(/^\/\/# sourceMappingURL=[^\r\n]*/gm, '');
559-
}
560-
561-
const angularPackage = /[\\/]node_modules[\\/]@angular[\\/]/.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-
602535
function findAffectedFiles(
603536
builder: ts.EmitAndSemanticDiagnosticsBuilderProgram,
604537
{ ignoreForDiagnostics, ignoreForEmit, incrementalCompilation }: NgtscProgram['compiler'],
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,97 @@
1+
/**
2+
* @license
3+
* Copyright Google LLC All Rights Reserved.
4+
*
5+
* Use of this source code is governed by an MIT-style license that can be
6+
* found in the LICENSE file at https://angular.io/license
7+
*/
8+
9+
import { transformAsync } from '@babel/core';
10+
import { readFile } from 'node:fs/promises';
11+
import angularApplicationPreset from '../../babel/presets/application';
12+
import { requiresLinking } from '../../babel/webpack-loader';
13+
import { loadEsmModule } from '../../utils/load-esm';
14+
15+
interface JavaScriptTransformRequest {
16+
filename: string;
17+
data: string;
18+
sourcemap: boolean;
19+
thirdPartySourcemaps: boolean;
20+
advancedOptimizations: boolean;
21+
forceAsyncTransformation?: boolean;
22+
skipLinker: boolean;
23+
}
24+
25+
export default async function transformJavaScript(
26+
request: JavaScriptTransformRequest,
27+
): Promise<Uint8Array> {
28+
request.data ??= await readFile(request.filename, 'utf-8');
29+
const transformedData = await transformWithBabel(request);
30+
31+
return Buffer.from(transformedData, 'utf-8');
32+
}
33+
34+
let linkerPluginCreator:
35+
| typeof import('@angular/compiler-cli/linker/babel').createEs2015LinkerPlugin
36+
| undefined;
37+
38+
async function transformWithBabel({
39+
filename,
40+
data,
41+
...options
42+
}: JavaScriptTransformRequest): Promise<string> {
43+
const forceAsyncTransformation =
44+
options.forceAsyncTransformation ??
45+
(!/[\\/][_f]?esm2015[\\/]/.test(filename) && /async\s+function\s*\*/.test(data));
46+
const shouldLink = !options.skipLinker && (await requiresLinking(filename, data));
47+
const useInputSourcemap =
48+
options.sourcemap &&
49+
(!!options.thirdPartySourcemaps || !/[\\/]node_modules[\\/]/.test(filename));
50+
51+
// If no additional transformations are needed, return the data directly
52+
if (!forceAsyncTransformation && !options.advancedOptimizations && !shouldLink) {
53+
// Strip sourcemaps if they should not be used
54+
return useInputSourcemap ? data : data.replace(/^\/\/# sourceMappingURL=[^\r\n]*/gm, '');
55+
}
56+
57+
const angularPackage = /[\\/]node_modules[\\/]@angular[\\/]/.test(filename);
58+
59+
// Lazy load the linker plugin only when linking is required
60+
if (shouldLink) {
61+
linkerPluginCreator ??= (
62+
await loadEsmModule<typeof import('@angular/compiler-cli/linker/babel')>(
63+
'@angular/compiler-cli/linker/babel',
64+
)
65+
).createEs2015LinkerPlugin;
66+
}
67+
68+
const result = await transformAsync(data, {
69+
filename,
70+
inputSourceMap: (useInputSourcemap ? undefined : false) as undefined,
71+
sourceMaps: options.sourcemap ? 'inline' : false,
72+
compact: false,
73+
configFile: false,
74+
babelrc: false,
75+
browserslistConfigFile: false,
76+
plugins: [],
77+
presets: [
78+
[
79+
angularApplicationPreset,
80+
{
81+
angularLinker: linkerPluginCreator && {
82+
shouldLink,
83+
jitMode: false,
84+
linkerPluginCreator,
85+
},
86+
forceAsyncTransformation,
87+
optimize: options.advancedOptimizations && {
88+
looseEnums: angularPackage,
89+
pureTopLevel: angularPackage,
90+
},
91+
},
92+
],
93+
],
94+
});
95+
96+
return result?.code ?? data;
97+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,91 @@
1+
/**
2+
* @license
3+
* Copyright Google LLC All Rights Reserved.
4+
*
5+
* Use of this source code is governed by an MIT-style license that can be
6+
* found in the LICENSE file at https://angular.io/license
7+
*/
8+
9+
import Piscina from 'piscina';
10+
11+
/**
12+
* Transformation options that should apply to all transformed files and data.
13+
*/
14+
export interface JavaScriptTransformerOptions {
15+
sourcemap: boolean;
16+
thirdPartySourcemaps?: boolean;
17+
advancedOptimizations?: boolean;
18+
}
19+
20+
/**
21+
* A class that performs transformation of JavaScript files and raw data.
22+
* A worker pool is used to distribute the transformation actions and allow
23+
* parallel processing. Transformation behavior is based on the filename and
24+
* data. Transformations may include: async downleveling, Angular linking,
25+
* and advanced optimizations.
26+
*/
27+
export class JavaScriptTransformer {
28+
#workerPool: Piscina;
29+
30+
constructor(private options: JavaScriptTransformerOptions, maxThreads?: number) {
31+
this.#workerPool = new Piscina({
32+
filename: require.resolve('./javascript-transformer-worker'),
33+
maxThreads,
34+
});
35+
}
36+
37+
/**
38+
* Performs JavaScript transformations on a file from the filesystem.
39+
* If no transformations are required, the data for the original file will be returned.
40+
* @param filename The full path to the file.
41+
* @returns A promise that resolves to a UTF-8 encoded Uint8Array containing the result.
42+
*/
43+
transformFile(filename: string): Promise<Uint8Array> {
44+
// Always send the request to a worker. Files are almost always from node modules which measn
45+
// they may need linking. The data is also not yet available to perform most transformation checks.
46+
return this.#workerPool.run({
47+
filename,
48+
...this.options,
49+
});
50+
}
51+
52+
/**
53+
* Performs JavaScript transformations on the provided data of a file. The file does not need
54+
* to exist on the filesystem.
55+
* @param filename The full path of the file represented by the data.
56+
* @param data The data of the file that should be transformed.
57+
* @param skipLinker If true, bypass all Angular linker processing; if false, attempt linking.
58+
* @returns A promise that resolves to a UTF-8 encoded Uint8Array containing the result.
59+
*/
60+
async transformData(filename: string, data: string, skipLinker: boolean): Promise<Uint8Array> {
61+
// Perform a quick test to determine if the data needs any transformations.
62+
// This allows directly returning the data without the worker communication overhead.
63+
let forceAsyncTransformation;
64+
if (skipLinker && !this.options.advancedOptimizations) {
65+
// If the linker is being skipped and no optimizations are needed, only async transformation is left.
66+
// This checks for async generator functions. All other async transformation is handled by esbuild.
67+
forceAsyncTransformation = data.includes('async') && /async\s+function\s*\*/.test(data);
68+
69+
if (!forceAsyncTransformation) {
70+
return Buffer.from(data, 'utf-8');
71+
}
72+
}
73+
74+
return this.#workerPool.run({
75+
filename,
76+
data,
77+
// Send the async check result if present to avoid rechecking in the worker
78+
forceAsyncTransformation,
79+
skipLinker,
80+
...this.options,
81+
});
82+
}
83+
84+
/**
85+
* Stops all active transformation tasks and shuts down all workers.
86+
* @returns A void promise that resolves when closing is complete.
87+
*/
88+
close(): Promise<void> {
89+
return this.#workerPool.destroy();
90+
}
91+
}

0 commit comments

Comments
 (0)
Please sign in to comment.