@angular-devkit/build-angular 16.2.0 → 17.0.0-next.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +14 -10
- package/builders.json +10 -0
- package/package.json +30 -28
- package/src/builders/application/execute-build.js +2 -2
- package/src/builders/browser-esbuild/builder-status-warnings.js +2 -1
- package/src/builders/dev-server/vite-server.d.ts +1 -1
- package/src/builders/dev-server/vite-server.js +89 -52
- package/src/builders/extract-i18n/application-extraction.d.ts +17 -0
- package/src/builders/extract-i18n/application-extraction.js +137 -0
- package/src/builders/extract-i18n/builder.d.ts +17 -0
- package/src/builders/extract-i18n/builder.js +152 -0
- package/src/builders/extract-i18n/index.d.ts +4 -14
- package/src/builders/extract-i18n/index.js +4 -247
- package/src/builders/extract-i18n/options.d.ts +29 -0
- package/src/builders/extract-i18n/options.js +82 -0
- package/src/builders/extract-i18n/webpack-extraction.d.ts +21 -0
- package/src/builders/extract-i18n/webpack-extraction.js +100 -0
- package/src/builders/prerender/index.d.ts +20 -0
- package/src/builders/prerender/index.js +180 -0
- package/src/builders/prerender/render-worker.d.ts +30 -0
- package/src/builders/prerender/render-worker.js +126 -0
- package/src/builders/prerender/schema.d.ts +22 -0
- package/src/builders/prerender/schema.js +5 -0
- package/src/builders/prerender/schema.json +39 -0
- package/src/builders/prerender/utils.d.ts +22 -0
- package/src/builders/prerender/utils.js +79 -0
- package/src/builders/ssr-dev-server/index.d.ts +23 -0
- package/src/builders/ssr-dev-server/index.js +309 -0
- package/src/builders/ssr-dev-server/schema.d.ts +64 -0
- package/src/builders/ssr-dev-server/schema.js +5 -0
- package/src/builders/ssr-dev-server/schema.json +75 -0
- package/src/builders/ssr-dev-server/utils.d.ts +15 -0
- package/src/builders/ssr-dev-server/utils.js +75 -0
- package/src/tools/babel/webpack-loader.js +2 -2
- package/src/tools/esbuild/angular/compilation/angular-compilation.d.ts +7 -2
- package/src/tools/esbuild/angular/compilation/angular-compilation.js +21 -1
- package/src/tools/esbuild/angular/compiler-plugin.js +8 -13
- package/src/tools/esbuild/application-code-bundle.js +4 -6
- package/src/tools/esbuild/utils.js +5 -1
- package/src/tools/sass/rebasing-importer.js +15 -8
- package/src/utils/index-file/inline-critical-css.js +13 -26
- package/src/utils/index-file/inline-fonts.js +1 -2
- package/src/utils/package-chunk-sort.js +1 -1
- package/src/utils/server-rendering/esm-in-memory-file-loader.d.ts +4 -2
- package/src/utils/server-rendering/esm-in-memory-file-loader.js +28 -5
- package/src/utils/server-rendering/prerender.d.ts +1 -1
- package/src/utils/server-rendering/prerender.js +6 -3
- package/src/tools/esbuild/external-packages-plugin.d.ts +0 -17
- package/src/tools/esbuild/external-packages-plugin.js +0 -49
|
@@ -8,13 +8,23 @@
|
|
|
8
8
|
*/
|
|
9
9
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
10
10
|
exports.load = exports.resolve = void 0;
|
|
11
|
+
const node_path_1 = require("node:path");
|
|
11
12
|
const node_worker_threads_1 = require("node:worker_threads");
|
|
12
13
|
const url_1 = require("url");
|
|
14
|
+
const javascript_transformer_1 = require("../../tools/esbuild/javascript-transformer");
|
|
13
15
|
/**
|
|
14
16
|
* Node.js ESM loader to redirect imports to in memory files.
|
|
15
17
|
* @see: https://nodejs.org/api/esm.html#loaders for more information about loaders.
|
|
16
18
|
*/
|
|
17
|
-
const { outputFiles } = node_worker_threads_1.workerData;
|
|
19
|
+
const { outputFiles, workspaceRoot } = node_worker_threads_1.workerData;
|
|
20
|
+
const TRANSFORMED_FILES = {};
|
|
21
|
+
const CHUNKS_REGEXP = /file:\/\/\/(main\.server|chunk-\w+)\.mjs/;
|
|
22
|
+
const WORKSPACE_ROOT_FILE = new URL((0, node_path_1.join)(workspaceRoot, 'index.mjs'), 'file:').href;
|
|
23
|
+
const JAVASCRIPT_TRANSFORMER = new javascript_transformer_1.JavaScriptTransformer(
|
|
24
|
+
// Always enable JIT linking to support applications built with and without AOT.
|
|
25
|
+
// In a development environment the additional scope information does not
|
|
26
|
+
// have a negative effect unlike production where final output size is relevant.
|
|
27
|
+
{ sourcemap: true, jit: true }, 1);
|
|
18
28
|
function resolve(specifier, context, nextResolve) {
|
|
19
29
|
if (!isFileProtocol(specifier)) {
|
|
20
30
|
const normalizedSpecifier = specifier.replace(/^\.\//, '');
|
|
@@ -28,12 +38,16 @@ function resolve(specifier, context, nextResolve) {
|
|
|
28
38
|
}
|
|
29
39
|
// Defer to the next hook in the chain, which would be the
|
|
30
40
|
// Node.js default resolve if this is the last user-specified loader.
|
|
31
|
-
return nextResolve(specifier);
|
|
41
|
+
return nextResolve(specifier, isBundleEntryPointOrChunk(context) ? { ...context, parentURL: WORKSPACE_ROOT_FILE } : context);
|
|
32
42
|
}
|
|
33
43
|
exports.resolve = resolve;
|
|
34
|
-
function load(url, context, nextLoad) {
|
|
44
|
+
async function load(url, context, nextLoad) {
|
|
35
45
|
if (isFileProtocol(url)) {
|
|
36
|
-
const
|
|
46
|
+
const filePath = (0, url_1.fileURLToPath)(url);
|
|
47
|
+
let source = outputFiles[filePath.slice(1)] /* Remove leading slash */ ?? TRANSFORMED_FILES[filePath];
|
|
48
|
+
if (source === undefined) {
|
|
49
|
+
source = TRANSFORMED_FILES[filePath] = Buffer.from(await JAVASCRIPT_TRANSFORMER.transformFile(filePath)).toString('utf-8');
|
|
50
|
+
}
|
|
37
51
|
if (source !== undefined) {
|
|
38
52
|
const { format } = context;
|
|
39
53
|
return {
|
|
@@ -50,4 +64,13 @@ exports.load = load;
|
|
|
50
64
|
function isFileProtocol(url) {
|
|
51
65
|
return url.startsWith('file://');
|
|
52
66
|
}
|
|
53
|
-
|
|
67
|
+
function handleProcessExit() {
|
|
68
|
+
void JAVASCRIPT_TRANSFORMER.close();
|
|
69
|
+
}
|
|
70
|
+
function isBundleEntryPointOrChunk(context) {
|
|
71
|
+
return !!context.parentURL && CHUNKS_REGEXP.test(context.parentURL);
|
|
72
|
+
}
|
|
73
|
+
process.once('exit', handleProcessExit);
|
|
74
|
+
process.once('SIGINT', handleProcessExit);
|
|
75
|
+
process.once('uncaughtException', handleProcessExit);
|
|
76
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXNtLWluLW1lbW9yeS1maWxlLWxvYWRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2FuZ3VsYXJfZGV2a2l0L2J1aWxkX2FuZ3VsYXIvc3JjL3V0aWxzL3NlcnZlci1yZW5kZXJpbmcvZXNtLWluLW1lbW9yeS1maWxlLWxvYWRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7OztHQU1HOzs7QUFFSCx5Q0FBaUM7QUFDakMsNkRBQWlEO0FBQ2pELDZCQUFvQztBQUNwQyx1RkFBbUY7QUFFbkY7OztHQUdHO0FBRUgsTUFBTSxFQUFFLFdBQVcsRUFBRSxhQUFhLEVBQUUsR0FBRyxnQ0FHdEMsQ0FBQztBQUVGLE1BQU0saUJBQWlCLEdBQTJCLEVBQUUsQ0FBQztBQUNyRCxNQUFNLGFBQWEsR0FBRywwQ0FBMEMsQ0FBQztBQUNqRSxNQUFNLG1CQUFtQixHQUFHLElBQUksR0FBRyxDQUFDLElBQUEsZ0JBQUksRUFBQyxhQUFhLEVBQUUsV0FBVyxDQUFDLEVBQUUsT0FBTyxDQUFDLENBQUMsSUFBSSxDQUFDO0FBRXBGLE1BQU0sc0JBQXNCLEdBQUcsSUFBSSw4Q0FBcUI7QUFDdEQsZ0ZBQWdGO0FBQ2hGLHlFQUF5RTtBQUN6RSxnRkFBZ0Y7QUFDaEYsRUFBRSxTQUFTLEVBQUUsSUFBSSxFQUFFLEdBQUcsRUFBRSxJQUFJLEVBQUUsRUFDOUIsQ0FBQyxDQUNGLENBQUM7QUFFRixTQUFnQixPQUFPLENBQ3JCLFNBQWlCLEVBQ2pCLE9BQTBDLEVBQzFDLFdBQXFCO0lBRXJCLElBQUksQ0FBQyxjQUFjLENBQUMsU0FBUyxDQUFDLEVBQUU7UUFDOUIsTUFBTSxtQkFBbUIsR0FBRyxTQUFTLENBQUMsT0FBTyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsQ0FBQztRQUMzRCxJQUFJLG1CQUFtQixJQUFJLFdBQVcsRUFBRTtZQUN0QyxPQUFPO2dCQUNMLE1BQU0sRUFBRSxRQUFRO2dCQUNoQixZQUFZLEVBQUUsSUFBSTtnQkFDbEIsR0FBRyxFQUFFLElBQUksR0FBRyxDQUFDLG1CQUFtQixFQUFFLE9BQU8sQ0FBQyxDQUFDLElBQUk7YUFDaEQsQ0FBQztTQUNIO0tBQ0Y7SUFFRCwwREFBMEQ7SUFDMUQscUVBQXFFO0lBQ3JFLE9BQU8sV0FBVyxDQUNoQixTQUFTLEVBQ1QseUJBQXlCLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDLEVBQUUsR0FBRyxPQUFPLEVBQUUsU0FBUyxFQUFFLG1CQUFtQixFQUFFLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FDOUYsQ0FBQztBQUNKLENBQUM7QUF0QkQsMEJBc0JDO0FBRU0sS0FBSyxVQUFVLElBQUksQ0FBQyxHQUFXLEVBQUUsT0FBbUMsRUFBRSxRQUFrQjtJQUM3RixJQUFJLGNBQWMsQ0FBQyxHQUFHLENBQUMsRUFBRTtRQUN2QixNQUFNLFFBQVEsR0FBRyxJQUFBLG1CQUFhLEVBQUMsR0FBRyxDQUFDLENBQUM7UUFDcEMsSUFBSSxNQUFNLEdBQ1IsV0FBVyxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQywwQkFBMEIsSUFBSSxpQkFBaUIsQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUUzRixJQUFJLE1BQU0sS0FBSyxTQUFTLEVBQUU7WUFDeEIsTUFBTSxHQUFHLGlCQUFpQixDQUFDLFFBQVEsQ0FBQyxHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQ2hELE1BQU0sc0JBQXNCLENBQUMsYUFBYSxDQUFDLFFBQVEsQ0FBQyxDQUNyRCxDQUFDLFFBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQztTQUNyQjtRQUVELElBQUksTUFBTSxLQUFLLFNBQVMsRUFBRTtZQUN4QixNQUFNLEVBQUUsTUFBTSxFQUFFLEdBQUcsT0FBTyxDQUFDO1lBRTNCLE9BQU87Z0JBQ0wsTUFBTTtnQkFDTixZQUFZLEVBQUUsSUFBSTtnQkFDbEIsTUFBTTthQUNQLENBQUM7U0FDSDtLQUNGO0lBRUQscUNBQXFDO0lBQ3JDLE9BQU8sUUFBUSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0FBQ3ZCLENBQUM7QUF6QkQsb0JBeUJDO0FBRUQsU0FBUyxjQUFjLENBQUMsR0FBVztJQUNqQyxPQUFPLEdBQUcsQ0FBQyxVQUFVLENBQUMsU0FBUyxDQUFDLENBQUM7QUFDbkMsQ0FBQztBQUVELFNBQVMsaUJBQWlCO0lBQ3hCLEtBQUssc0JBQXNCLENBQUMsS0FBSyxFQUFFLENBQUM7QUFDdEMsQ0FBQztBQUVELFNBQVMseUJBQXlCLENBQUMsT0FBMEM7SUFDM0UsT0FBTyxDQUFDLENBQUMsT0FBTyxDQUFDLFNBQVMsSUFBSSxhQUFhLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxTQUFTLENBQUMsQ0FBQztBQUN0RSxDQUFDO0FBRUQsT0FBTyxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsaUJBQWlCLENBQUMsQ0FBQztBQUN4QyxPQUFPLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRSxpQkFBaUIsQ0FBQyxDQUFDO0FBQzFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsbUJBQW1CLEVBQUUsaUJBQWlCLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQGxpY2Vuc2VcbiAqIENvcHlyaWdodCBHb29nbGUgTExDIEFsbCBSaWdodHMgUmVzZXJ2ZWQuXG4gKlxuICogVXNlIG9mIHRoaXMgc291cmNlIGNvZGUgaXMgZ292ZXJuZWQgYnkgYW4gTUlULXN0eWxlIGxpY2Vuc2UgdGhhdCBjYW4gYmVcbiAqIGZvdW5kIGluIHRoZSBMSUNFTlNFIGZpbGUgYXQgaHR0cHM6Ly9hbmd1bGFyLmlvL2xpY2Vuc2VcbiAqL1xuXG5pbXBvcnQgeyBqb2luIH0gZnJvbSAnbm9kZTpwYXRoJztcbmltcG9ydCB7IHdvcmtlckRhdGEgfSBmcm9tICdub2RlOndvcmtlcl90aHJlYWRzJztcbmltcG9ydCB7IGZpbGVVUkxUb1BhdGggfSBmcm9tICd1cmwnO1xuaW1wb3J0IHsgSmF2YVNjcmlwdFRyYW5zZm9ybWVyIH0gZnJvbSAnLi4vLi4vdG9vbHMvZXNidWlsZC9qYXZhc2NyaXB0LXRyYW5zZm9ybWVyJztcblxuLyoqXG4gKiBOb2RlLmpzIEVTTSBsb2FkZXIgdG8gcmVkaXJlY3QgaW1wb3J0cyB0byBpbiBtZW1vcnkgZmlsZXMuXG4gKiBAc2VlOiBodHRwczovL25vZGVqcy5vcmcvYXBpL2VzbS5odG1sI2xvYWRlcnMgZm9yIG1vcmUgaW5mb3JtYXRpb24gYWJvdXQgbG9hZGVycy5cbiAqL1xuXG5jb25zdCB7IG91dHB1dEZpbGVzLCB3b3Jrc3BhY2VSb290IH0gPSB3b3JrZXJEYXRhIGFzIHtcbiAgb3V0cHV0RmlsZXM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz47XG4gIHdvcmtzcGFjZVJvb3Q6IHN0cmluZztcbn07XG5cbmNvbnN0IFRSQU5TRk9STUVEX0ZJTEVTOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0ge307XG5jb25zdCBDSFVOS1NfUkVHRVhQID0gL2ZpbGU6XFwvXFwvXFwvKG1haW5cXC5zZXJ2ZXJ8Y2h1bmstXFx3KylcXC5tanMvO1xuY29uc3QgV09SS1NQQUNFX1JPT1RfRklMRSA9IG5ldyBVUkwoam9pbih3b3Jrc3BhY2VSb290LCAnaW5kZXgubWpzJyksICdmaWxlOicpLmhyZWY7XG5cbmNvbnN0IEpBVkFTQ1JJUFRfVFJBTlNGT1JNRVIgPSBuZXcgSmF2YVNjcmlwdFRyYW5zZm9ybWVyKFxuICAvLyBBbHdheXMgZW5hYmxlIEpJVCBsaW5raW5nIHRvIHN1cHBvcnQgYXBwbGljYXRpb25zIGJ1aWx0IHdpdGggYW5kIHdpdGhvdXQgQU9ULlxuICAvLyBJbiBhIGRldmVsb3BtZW50IGVudmlyb25tZW50IHRoZSBhZGRpdGlvbmFsIHNjb3BlIGluZm9ybWF0aW9uIGRvZXMgbm90XG4gIC8vIGhhdmUgYSBuZWdhdGl2ZSBlZmZlY3QgdW5saWtlIHByb2R1Y3Rpb24gd2hlcmUgZmluYWwgb3V0cHV0IHNpemUgaXMgcmVsZXZhbnQuXG4gIHsgc291cmNlbWFwOiB0cnVlLCBqaXQ6IHRydWUgfSxcbiAgMSxcbik7XG5cbmV4cG9ydCBmdW5jdGlvbiByZXNvbHZlKFxuICBzcGVjaWZpZXI6IHN0cmluZyxcbiAgY29udGV4dDogeyBwYXJlbnRVUkw6IHVuZGVmaW5lZCB8IHN0cmluZyB9LFxuICBuZXh0UmVzb2x2ZTogRnVuY3Rpb24sXG4pIHtcbiAgaWYgKCFpc0ZpbGVQcm90b2NvbChzcGVjaWZpZXIpKSB7XG4gICAgY29uc3Qgbm9ybWFsaXplZFNwZWNpZmllciA9IHNwZWNpZmllci5yZXBsYWNlKC9eXFwuXFwvLywgJycpO1xuICAgIGlmIChub3JtYWxpemVkU3BlY2lmaWVyIGluIG91dHB1dEZpbGVzKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBmb3JtYXQ6ICdtb2R1bGUnLFxuICAgICAgICBzaG9ydENpcmN1aXQ6IHRydWUsXG4gICAgICAgIHVybDogbmV3IFVSTChub3JtYWxpemVkU3BlY2lmaWVyLCAnZmlsZTonKS5ocmVmLFxuICAgICAgfTtcbiAgICB9XG4gIH1cblxuICAvLyBEZWZlciB0byB0aGUgbmV4dCBob29rIGluIHRoZSBjaGFpbiwgd2hpY2ggd291bGQgYmUgdGhlXG4gIC8vIE5vZGUuanMgZGVmYXVsdCByZXNvbHZlIGlmIHRoaXMgaXMgdGhlIGxhc3QgdXNlci1zcGVjaWZpZWQgbG9hZGVyLlxuICByZXR1cm4gbmV4dFJlc29sdmUoXG4gICAgc3BlY2lmaWVyLFxuICAgIGlzQnVuZGxlRW50cnlQb2ludE9yQ2h1bmsoY29udGV4dCkgPyB7IC4uLmNvbnRleHQsIHBhcmVudFVSTDogV09SS1NQQUNFX1JPT1RfRklMRSB9IDogY29udGV4dCxcbiAgKTtcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGxvYWQodXJsOiBzdHJpbmcsIGNvbnRleHQ6IHsgZm9ybWF0Pzogc3RyaW5nIHwgbnVsbCB9LCBuZXh0TG9hZDogRnVuY3Rpb24pIHtcbiAgaWYgKGlzRmlsZVByb3RvY29sKHVybCkpIHtcbiAgICBjb25zdCBmaWxlUGF0aCA9IGZpbGVVUkxUb1BhdGgodXJsKTtcbiAgICBsZXQgc291cmNlID1cbiAgICAgIG91dHB1dEZpbGVzW2ZpbGVQYXRoLnNsaWNlKDEpXSAvKiBSZW1vdmUgbGVhZGluZyBzbGFzaCAqLyA/PyBUUkFOU0ZPUk1FRF9GSUxFU1tmaWxlUGF0aF07XG5cbiAgICBpZiAoc291cmNlID09PSB1bmRlZmluZWQpIHtcbiAgICAgIHNvdXJjZSA9IFRSQU5TRk9STUVEX0ZJTEVTW2ZpbGVQYXRoXSA9IEJ1ZmZlci5mcm9tKFxuICAgICAgICBhd2FpdCBKQVZBU0NSSVBUX1RSQU5TRk9STUVSLnRyYW5zZm9ybUZpbGUoZmlsZVBhdGgpLFxuICAgICAgKS50b1N0cmluZygndXRmLTgnKTtcbiAgICB9XG5cbiAgICBpZiAoc291cmNlICE9PSB1bmRlZmluZWQpIHtcbiAgICAgIGNvbnN0IHsgZm9ybWF0IH0gPSBjb250ZXh0O1xuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBmb3JtYXQsXG4gICAgICAgIHNob3J0Q2lyY3VpdDogdHJ1ZSxcbiAgICAgICAgc291cmNlLFxuICAgICAgfTtcbiAgICB9XG4gIH1cblxuICAvLyBMZXQgTm9kZS5qcyBoYW5kbGUgYWxsIG90aGVyIFVSTHMuXG4gIHJldHVybiBuZXh0TG9hZCh1cmwpO1xufVxuXG5mdW5jdGlvbiBpc0ZpbGVQcm90b2NvbCh1cmw6IHN0cmluZyk6IGJvb2xlYW4ge1xuICByZXR1cm4gdXJsLnN0YXJ0c1dpdGgoJ2ZpbGU6Ly8nKTtcbn1cblxuZnVuY3Rpb24gaGFuZGxlUHJvY2Vzc0V4aXQoKTogdm9pZCB7XG4gIHZvaWQgSkFWQVNDUklQVF9UUkFOU0ZPUk1FUi5jbG9zZSgpO1xufVxuXG5mdW5jdGlvbiBpc0J1bmRsZUVudHJ5UG9pbnRPckNodW5rKGNvbnRleHQ6IHsgcGFyZW50VVJMOiB1bmRlZmluZWQgfCBzdHJpbmcgfSk6IGJvb2xlYW4ge1xuICByZXR1cm4gISFjb250ZXh0LnBhcmVudFVSTCAmJiBDSFVOS1NfUkVHRVhQLnRlc3QoY29udGV4dC5wYXJlbnRVUkwpO1xufVxuXG5wcm9jZXNzLm9uY2UoJ2V4aXQnLCBoYW5kbGVQcm9jZXNzRXhpdCk7XG5wcm9jZXNzLm9uY2UoJ1NJR0lOVCcsIGhhbmRsZVByb2Nlc3NFeGl0KTtcbnByb2Nlc3Mub25jZSgndW5jYXVnaHRFeGNlcHRpb24nLCBoYW5kbGVQcm9jZXNzRXhpdCk7XG4iXX0=
|
|
@@ -14,7 +14,7 @@ interface PrerenderOptions {
|
|
|
14
14
|
interface AppShellOptions {
|
|
15
15
|
route?: string;
|
|
16
16
|
}
|
|
17
|
-
export declare function prerenderPages(tsConfigPath: string, appShellOptions: AppShellOptions | undefined, prerenderOptions: PrerenderOptions | undefined, outputFiles: Readonly<OutputFile[]>, document: string, inlineCriticalCss?: boolean, maxThreads?: number): Promise<{
|
|
17
|
+
export declare function prerenderPages(workspaceRoot: string, tsConfigPath: string, appShellOptions: AppShellOptions | undefined, prerenderOptions: PrerenderOptions | undefined, outputFiles: Readonly<OutputFile[]>, document: string, inlineCriticalCss?: boolean, maxThreads?: number): Promise<{
|
|
18
18
|
output: Record<string, string>;
|
|
19
19
|
warnings: string[];
|
|
20
20
|
errors: string[];
|
|
@@ -37,7 +37,7 @@ exports.prerenderPages = void 0;
|
|
|
37
37
|
const promises_1 = require("node:fs/promises");
|
|
38
38
|
const node_path_1 = require("node:path");
|
|
39
39
|
const piscina_1 = __importDefault(require("piscina"));
|
|
40
|
-
async function prerenderPages(tsConfigPath, appShellOptions = {}, prerenderOptions = {}, outputFiles, document, inlineCriticalCss, maxThreads = 1) {
|
|
40
|
+
async function prerenderPages(workspaceRoot, tsConfigPath, appShellOptions = {}, prerenderOptions = {}, outputFiles, document, inlineCriticalCss, maxThreads = 1) {
|
|
41
41
|
const allRoutes = await getAllRoutes(tsConfigPath, appShellOptions, prerenderOptions);
|
|
42
42
|
const outputFilesForWorker = {};
|
|
43
43
|
for (const { text, path } of outputFiles) {
|
|
@@ -52,6 +52,7 @@ async function prerenderPages(tsConfigPath, appShellOptions = {}, prerenderOptio
|
|
|
52
52
|
filename: require.resolve('./render-worker'),
|
|
53
53
|
maxThreads: Math.min(allRoutes.size, maxThreads),
|
|
54
54
|
workerData: {
|
|
55
|
+
workspaceRoot,
|
|
55
56
|
outputFiles: outputFilesForWorker,
|
|
56
57
|
inlineCriticalCss,
|
|
57
58
|
document,
|
|
@@ -73,7 +74,9 @@ async function prerenderPages(tsConfigPath, appShellOptions = {}, prerenderOptio
|
|
|
73
74
|
const render = renderWorker.run({ route, serverContext });
|
|
74
75
|
const renderResult = render.then(({ content, warnings, errors }) => {
|
|
75
76
|
if (content !== undefined) {
|
|
76
|
-
const outPath = isAppShellRoute
|
|
77
|
+
const outPath = isAppShellRoute
|
|
78
|
+
? 'index.html'
|
|
79
|
+
: node_path_1.posix.join(route.startsWith('/') ? route.slice(1) /* Remove leading slash */ : route, 'index.html');
|
|
77
80
|
output[outPath] = content;
|
|
78
81
|
}
|
|
79
82
|
if (warnings) {
|
|
@@ -124,4 +127,4 @@ async function getAllRoutes(tsConfigPath, appShellOptions, prerenderOptions) {
|
|
|
124
127
|
}
|
|
125
128
|
return routes;
|
|
126
129
|
}
|
|
127
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
130
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"prerender.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/utils/server-rendering/prerender.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport { OutputFile } from 'esbuild';\nimport { readFile } from 'node:fs/promises';\nimport { extname, posix } from 'node:path';\nimport Piscina from 'piscina';\nimport type { RenderResult, ServerContext } from './render-page';\nimport type { WorkerData } from './render-worker';\n\ninterface PrerenderOptions {\n  routesFile?: string;\n  discoverRoutes?: boolean;\n  routes?: string[];\n}\n\ninterface AppShellOptions {\n  route?: string;\n}\n\nexport async function prerenderPages(\n  workspaceRoot: string,\n  tsConfigPath: string,\n  appShellOptions: AppShellOptions = {},\n  prerenderOptions: PrerenderOptions = {},\n  outputFiles: Readonly<OutputFile[]>,\n  document: string,\n  inlineCriticalCss?: boolean,\n  maxThreads = 1,\n): Promise<{\n  output: Record<string, string>;\n  warnings: string[];\n  errors: string[];\n}> {\n  const allRoutes = await getAllRoutes(tsConfigPath, appShellOptions, prerenderOptions);\n  const outputFilesForWorker: Record<string, string> = {};\n\n  for (const { text, path } of outputFiles) {\n    switch (extname(path)) {\n      case '.mjs': // Contains the server runnable application code.\n      case '.css': // Global styles for critical CSS inlining.\n        outputFilesForWorker[path] = text;\n        break;\n    }\n  }\n\n  const renderWorker = new Piscina({\n    filename: require.resolve('./render-worker'),\n    maxThreads: Math.min(allRoutes.size, maxThreads),\n    workerData: {\n      workspaceRoot,\n      outputFiles: outputFilesForWorker,\n      inlineCriticalCss,\n      document,\n    } as WorkerData,\n    execArgv: [\n      '--no-warnings', // Suppress `ExperimentalWarning: Custom ESM Loaders is an experimental feature...`.\n      '--loader',\n      require.resolve('./esm-in-memory-file-loader.js'),\n    ],\n  });\n\n  const output: Record<string, string> = {};\n  const warnings: string[] = [];\n  const errors: string[] = [];\n\n  try {\n    const renderingPromises: Promise<void>[] = [];\n\n    for (const route of allRoutes) {\n      const isAppShellRoute = appShellOptions.route === route;\n      const serverContext: ServerContext = isAppShellRoute ? 'app-shell' : 'ssg';\n\n      const render: Promise<RenderResult> = renderWorker.run({ route, serverContext });\n      const renderResult: Promise<void> = render.then(({ content, warnings, errors }) => {\n        if (content !== undefined) {\n          const outPath = isAppShellRoute\n            ? 'index.html'\n            : posix.join(\n                route.startsWith('/') ? route.slice(1) /* Remove leading slash */ : route,\n                'index.html',\n              );\n          output[outPath] = content;\n        }\n\n        if (warnings) {\n          warnings.push(...warnings);\n        }\n\n        if (errors) {\n          errors.push(...errors);\n        }\n      });\n\n      renderingPromises.push(renderResult);\n    }\n\n    await Promise.all(renderingPromises);\n  } finally {\n    void renderWorker.destroy();\n  }\n\n  return {\n    errors,\n    warnings,\n    output,\n  };\n}\n\nasync function getAllRoutes(\n  tsConfigPath: string,\n  appShellOptions: AppShellOptions,\n  prerenderOptions: PrerenderOptions,\n): Promise<Set<string>> {\n  const { routesFile, discoverRoutes, routes: existingRoutes } = prerenderOptions;\n  const routes = new Set(existingRoutes);\n\n  const { route: appShellRoute } = appShellOptions;\n  if (appShellRoute !== undefined) {\n    routes.add(appShellRoute);\n  }\n\n  if (routesFile) {\n    const routesFromFile = (await readFile(routesFile, 'utf8')).split(/\\r?\\n/);\n    for (let route of routesFromFile) {\n      route = route.trim();\n      if (route) {\n        routes.add(route);\n      }\n    }\n  }\n\n  if (discoverRoutes) {\n    const { parseAngularRoutes } = await import('guess-parser');\n    for (const { path } of parseAngularRoutes(tsConfigPath)) {\n      // Exclude dynamic routes as these cannot be pre-rendered.\n      if (!/[*:]/.test(path)) {\n        routes.add(path);\n      }\n    }\n  }\n\n  return routes;\n}\n"]}
|
|
@@ -1,17 +0,0 @@
|
|
|
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
|
-
import type { Plugin } from 'esbuild';
|
|
9
|
-
/**
|
|
10
|
-
* Creates a plugin that marks any resolved path as external if it is within a node modules directory.
|
|
11
|
-
* This is used instead of the esbuild `packages` option to avoid marking bare specifiers that use
|
|
12
|
-
* tsconfig path mapping to resolve to a workspace relative path. This is common for monorepos that
|
|
13
|
-
* contain libraries that are built along with the application. These libraries should not be considered
|
|
14
|
-
*
|
|
15
|
-
* @returns An esbuild plugin.
|
|
16
|
-
*/
|
|
17
|
-
export declare function createExternalPackagesPlugin(): Plugin;
|
|
@@ -1,49 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
/**
|
|
3
|
-
* @license
|
|
4
|
-
* Copyright Google LLC All Rights Reserved.
|
|
5
|
-
*
|
|
6
|
-
* Use of this source code is governed by an MIT-style license that can be
|
|
7
|
-
* found in the LICENSE file at https://angular.io/license
|
|
8
|
-
*/
|
|
9
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
10
|
-
exports.createExternalPackagesPlugin = void 0;
|
|
11
|
-
const EXTERNAL_PACKAGE_RESOLUTION = Symbol('EXTERNAL_PACKAGE_RESOLUTION');
|
|
12
|
-
/**
|
|
13
|
-
* Creates a plugin that marks any resolved path as external if it is within a node modules directory.
|
|
14
|
-
* This is used instead of the esbuild `packages` option to avoid marking bare specifiers that use
|
|
15
|
-
* tsconfig path mapping to resolve to a workspace relative path. This is common for monorepos that
|
|
16
|
-
* contain libraries that are built along with the application. These libraries should not be considered
|
|
17
|
-
*
|
|
18
|
-
* @returns An esbuild plugin.
|
|
19
|
-
*/
|
|
20
|
-
function createExternalPackagesPlugin() {
|
|
21
|
-
return {
|
|
22
|
-
name: 'angular-external-packages',
|
|
23
|
-
setup(build) {
|
|
24
|
-
build.onResolve({ filter: /./ }, async (args) => {
|
|
25
|
-
if (args.pluginData?.[EXTERNAL_PACKAGE_RESOLUTION]) {
|
|
26
|
-
return null;
|
|
27
|
-
}
|
|
28
|
-
const { importer, kind, resolveDir, namespace, pluginData = {} } = args;
|
|
29
|
-
pluginData[EXTERNAL_PACKAGE_RESOLUTION] = true;
|
|
30
|
-
const result = await build.resolve(args.path, {
|
|
31
|
-
importer,
|
|
32
|
-
kind,
|
|
33
|
-
namespace,
|
|
34
|
-
pluginData,
|
|
35
|
-
resolveDir,
|
|
36
|
-
});
|
|
37
|
-
if (result.path && /[\\/]node_modules[\\/]/.test(result.path)) {
|
|
38
|
-
return {
|
|
39
|
-
path: args.path,
|
|
40
|
-
external: true,
|
|
41
|
-
};
|
|
42
|
-
}
|
|
43
|
-
return result;
|
|
44
|
-
});
|
|
45
|
-
},
|
|
46
|
-
};
|
|
47
|
-
}
|
|
48
|
-
exports.createExternalPackagesPlugin = createExternalPackagesPlugin;
|
|
49
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXh0ZXJuYWwtcGFja2FnZXMtcGx1Z2luLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvYW5ndWxhcl9kZXZraXQvYnVpbGRfYW5ndWxhci9zcmMvdG9vbHMvZXNidWlsZC9leHRlcm5hbC1wYWNrYWdlcy1wbHVnaW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7O0FBSUgsTUFBTSwyQkFBMkIsR0FBRyxNQUFNLENBQUMsNkJBQTZCLENBQUMsQ0FBQztBQUUxRTs7Ozs7OztHQU9HO0FBQ0gsU0FBZ0IsNEJBQTRCO0lBQzFDLE9BQU87UUFDTCxJQUFJLEVBQUUsMkJBQTJCO1FBQ2pDLEtBQUssQ0FBQyxLQUFLO1lBQ1QsS0FBSyxDQUFDLFNBQVMsQ0FBQyxFQUFFLE1BQU0sRUFBRSxHQUFHLEVBQUUsRUFBRSxLQUFLLEVBQUUsSUFBSSxFQUFFLEVBQUU7Z0JBQzlDLElBQUksSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFDLDJCQUEyQixDQUFDLEVBQUU7b0JBQ2xELE9BQU8sSUFBSSxDQUFDO2lCQUNiO2dCQUVELE1BQU0sRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLFVBQVUsRUFBRSxTQUFTLEVBQUUsVUFBVSxHQUFHLEVBQUUsRUFBRSxHQUFHLElBQUksQ0FBQztnQkFDeEUsVUFBVSxDQUFDLDJCQUEyQixDQUFDLEdBQUcsSUFBSSxDQUFDO2dCQUUvQyxNQUFNLE1BQU0sR0FBRyxNQUFNLEtBQUssQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksRUFBRTtvQkFDNUMsUUFBUTtvQkFDUixJQUFJO29CQUNKLFNBQVM7b0JBQ1QsVUFBVTtvQkFDVixVQUFVO2lCQUNYLENBQUMsQ0FBQztnQkFFSCxJQUFJLE1BQU0sQ0FBQyxJQUFJLElBQUksd0JBQXdCLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsRUFBRTtvQkFDN0QsT0FBTzt3QkFDTCxJQUFJLEVBQUUsSUFBSSxDQUFDLElBQUk7d0JBQ2YsUUFBUSxFQUFFLElBQUk7cUJBQ2YsQ0FBQztpQkFDSDtnQkFFRCxPQUFPLE1BQU0sQ0FBQztZQUNoQixDQUFDLENBQUMsQ0FBQztRQUNMLENBQUM7S0FDRixDQUFDO0FBQ0osQ0FBQztBQS9CRCxvRUErQkMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuaW1wb3J0IHR5cGUgeyBQbHVnaW4gfSBmcm9tICdlc2J1aWxkJztcblxuY29uc3QgRVhURVJOQUxfUEFDS0FHRV9SRVNPTFVUSU9OID0gU3ltYm9sKCdFWFRFUk5BTF9QQUNLQUdFX1JFU09MVVRJT04nKTtcblxuLyoqXG4gKiBDcmVhdGVzIGEgcGx1Z2luIHRoYXQgbWFya3MgYW55IHJlc29sdmVkIHBhdGggYXMgZXh0ZXJuYWwgaWYgaXQgaXMgd2l0aGluIGEgbm9kZSBtb2R1bGVzIGRpcmVjdG9yeS5cbiAqIFRoaXMgaXMgdXNlZCBpbnN0ZWFkIG9mIHRoZSBlc2J1aWxkIGBwYWNrYWdlc2Agb3B0aW9uIHRvIGF2b2lkIG1hcmtpbmcgYmFyZSBzcGVjaWZpZXJzIHRoYXQgdXNlXG4gKiB0c2NvbmZpZyBwYXRoIG1hcHBpbmcgdG8gcmVzb2x2ZSB0byBhIHdvcmtzcGFjZSByZWxhdGl2ZSBwYXRoLiBUaGlzIGlzIGNvbW1vbiBmb3IgbW9ub3JlcG9zIHRoYXRcbiAqIGNvbnRhaW4gbGlicmFyaWVzIHRoYXQgYXJlIGJ1aWx0IGFsb25nIHdpdGggdGhlIGFwcGxpY2F0aW9uLiBUaGVzZSBsaWJyYXJpZXMgc2hvdWxkIG5vdCBiZSBjb25zaWRlcmVkXG4gKlxuICogQHJldHVybnMgQW4gZXNidWlsZCBwbHVnaW4uXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVFeHRlcm5hbFBhY2thZ2VzUGx1Z2luKCk6IFBsdWdpbiB7XG4gIHJldHVybiB7XG4gICAgbmFtZTogJ2FuZ3VsYXItZXh0ZXJuYWwtcGFja2FnZXMnLFxuICAgIHNldHVwKGJ1aWxkKSB7XG4gICAgICBidWlsZC5vblJlc29sdmUoeyBmaWx0ZXI6IC8uLyB9LCBhc3luYyAoYXJncykgPT4ge1xuICAgICAgICBpZiAoYXJncy5wbHVnaW5EYXRhPy5bRVhURVJOQUxfUEFDS0FHRV9SRVNPTFVUSU9OXSkge1xuICAgICAgICAgIHJldHVybiBudWxsO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3QgeyBpbXBvcnRlciwga2luZCwgcmVzb2x2ZURpciwgbmFtZXNwYWNlLCBwbHVnaW5EYXRhID0ge30gfSA9IGFyZ3M7XG4gICAgICAgIHBsdWdpbkRhdGFbRVhURVJOQUxfUEFDS0FHRV9SRVNPTFVUSU9OXSA9IHRydWU7XG5cbiAgICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgYnVpbGQucmVzb2x2ZShhcmdzLnBhdGgsIHtcbiAgICAgICAgICBpbXBvcnRlcixcbiAgICAgICAgICBraW5kLFxuICAgICAgICAgIG5hbWVzcGFjZSxcbiAgICAgICAgICBwbHVnaW5EYXRhLFxuICAgICAgICAgIHJlc29sdmVEaXIsXG4gICAgICAgIH0pO1xuXG4gICAgICAgIGlmIChyZXN1bHQucGF0aCAmJiAvW1xcXFwvXW5vZGVfbW9kdWxlc1tcXFxcL10vLnRlc3QocmVzdWx0LnBhdGgpKSB7XG4gICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIHBhdGg6IGFyZ3MucGF0aCxcbiAgICAgICAgICAgIGV4dGVybmFsOiB0cnVlLFxuICAgICAgICAgIH07XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gcmVzdWx0O1xuICAgICAgfSk7XG4gICAgfSxcbiAgfTtcbn1cbiJdfQ==
|