@angular-devkit/build-angular
Version:
Angular Webpack Build Facade
199 lines • 28.3 kB
JavaScript
;
/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.requiresLinking = void 0;
const node_assert_1 = __importDefault(require("node:assert"));
const node_fs_1 = __importDefault(require("node:fs"));
const node_path_1 = __importDefault(require("node:path"));
const load_esm_1 = require("../../../utils/load-esm");
/**
* Cached instance of the compiler-cli linker's needsLinking function.
*/
let needsLinking;
/**
* List of browsers which are affected by a WebKit bug where class field
* initializers might have incorrect variable scopes.
*
* See: https://github.com/angular/angular-cli/issues/24355#issuecomment-1333477033
* See: https://github.com/WebKit/WebKit/commit/e8788a34b3d5f5b4edd7ff6450b80936bff396f2
*/
let safariClassFieldScopeBugBrowsers;
function createI18nDiagnostics(reporter) {
const diagnostics = new (class {
constructor() {
this.messages = [];
this.hasErrors = false;
}
add(type, message) {
if (type === 'ignore') {
return;
}
this.messages.push({ type, message });
this.hasErrors || (this.hasErrors = type === 'error');
reporter?.(type, message);
}
error(message) {
this.add('error', message);
}
warn(message) {
this.add('warning', message);
}
merge(other) {
for (const diagnostic of other.messages) {
this.add(diagnostic.type, diagnostic.message);
}
}
formatDiagnostics() {
node_assert_1.default.fail('@angular/localize Diagnostics formatDiagnostics should not be called from within babel.');
}
})();
return diagnostics;
}
function createI18nPlugins(locale, translation, missingTranslationBehavior, diagnosticReporter, pluginCreators) {
const diagnostics = createI18nDiagnostics(diagnosticReporter);
const plugins = [];
const { makeEs2015TranslatePlugin, makeLocalePlugin } = pluginCreators;
if (translation) {
plugins.push(makeEs2015TranslatePlugin(diagnostics, translation, {
missingTranslation: missingTranslationBehavior,
}));
}
plugins.push(makeLocalePlugin(locale));
return plugins;
}
function createNgtscLogger(reporter) {
return {
level: 1,
debug(...args) { },
info(...args) {
reporter?.('info', args.join());
},
warn(...args) {
reporter?.('warning', args.join());
},
error(...args) {
reporter?.('error', args.join());
},
};
}
function default_1(api, options) {
const presets = [];
const plugins = [];
let needRuntimeTransform = false;
if (options.angularLinker?.shouldLink) {
plugins.push(options.angularLinker.linkerPluginCreator({
linkerJitMode: options.angularLinker.jitMode,
// This is a workaround until https://github.com/angular/angular/issues/42769 is fixed.
sourceMapping: false,
logger: createNgtscLogger(options.diagnosticReporter),
fileSystem: {
resolve: node_path_1.default.resolve,
exists: node_fs_1.default.existsSync,
dirname: node_path_1.default.dirname,
relative: node_path_1.default.relative,
readFile: node_fs_1.default.readFileSync,
// Node.JS types don't overlap the Compiler types.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
},
}));
}
// Applications code ES version can be controlled using TypeScript's `target` option.
// However, this doesn't effect libraries and hence we use preset-env to downlevel ES features
// based on the supported browsers in browserslist.
if (options.supportedBrowsers) {
const includePlugins = [];
if (safariClassFieldScopeBugBrowsers === undefined) {
const browserslist = require('browserslist');
safariClassFieldScopeBugBrowsers = new Set(browserslist([
// Safari <15 is technically not supported via https://angular.io/guide/browser-support,
// but we apply the workaround if forcibly selected.
'Safari <=15',
'iOS <=15',
]));
}
// If a Safari browser affected by the class field scope bug is selected, we
// downlevel class properties by ensuring the class properties Babel plugin
// is always included- regardless of the preset-env targets.
if (options.supportedBrowsers.some((b) => safariClassFieldScopeBugBrowsers.has(b))) {
includePlugins.push('@babel/plugin-proposal-class-properties', '@babel/plugin-proposal-private-methods');
}
presets.push([
require('@babel/preset-env').default,
{
bugfixes: true,
modules: false,
targets: options.supportedBrowsers,
include: includePlugins,
exclude: ['transform-typeof-symbol'],
},
]);
needRuntimeTransform = true;
}
if (options.i18n) {
const { locale, missingTranslationBehavior, pluginCreators, translation } = options.i18n;
const i18nPlugins = createI18nPlugins(locale, translation, missingTranslationBehavior || 'ignore', options.diagnosticReporter, pluginCreators);
plugins.push(...i18nPlugins);
}
if (options.forceAsyncTransformation) {
// Always transform async/await to support Zone.js
plugins.push(require('@babel/plugin-transform-async-to-generator').default, require('@babel/plugin-proposal-async-generator-functions').default);
needRuntimeTransform = true;
}
if (options.optimize) {
if (options.optimize.pureTopLevel) {
plugins.push(require('../plugins/pure-toplevel-functions').default);
}
plugins.push(require('../plugins/elide-angular-metadata').default, [require('../plugins/adjust-typescript-enums').default, { loose: true }], [
require('../plugins/adjust-static-class-members').default,
{ wrapDecorators: options.optimize.wrapDecorators },
]);
}
if (options.instrumentCode) {
plugins.push([
require('babel-plugin-istanbul').default,
{
inputSourceMap: options.instrumentCode.inputSourceMap ?? false,
cwd: options.instrumentCode.includedBasePath,
},
]);
}
if (needRuntimeTransform) {
// Babel equivalent to TypeScript's `importHelpers` option
plugins.push([
require('@babel/plugin-transform-runtime').default,
{
useESModules: true,
version: require('@babel/runtime/package.json').version,
absoluteRuntime: node_path_1.default.dirname(require.resolve('@babel/runtime/package.json')),
},
]);
}
return { presets, plugins };
}
exports.default = default_1;
async function requiresLinking(path, source) {
// @angular/core and @angular/compiler will cause false positives
// Also, TypeScript files do not require linking
if (/[\\/]@angular[\\/](?:compiler|core)|\.tsx?$/.test(path)) {
return false;
}
if (!needsLinking) {
// Load ESM `@angular/compiler-cli/linker` using the TypeScript dynamic import workaround.
// Once TypeScript provides support for keeping the dynamic import this workaround can be
// changed to a direct dynamic import.
const linkerModule = await (0, load_esm_1.loadEsmModule)('@angular/compiler-cli/linker');
needsLinking = linkerModule.needsLinking;
}
return needsLinking(path, source);
}
exports.requiresLinking = requiresLinking;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"application.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/tools/babel/presets/application.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 type { ɵParsedTranslation } from '@angular/localize';\nimport type {\n  DiagnosticHandlingStrategy,\n  Diagnostics,\n  makeEs2015TranslatePlugin,\n  makeLocalePlugin,\n} from '@angular/localize/tools';\nimport assert from 'node:assert';\nimport fs from 'node:fs';\nimport path from 'node:path';\nimport { loadEsmModule } from '../../../utils/load-esm';\n\n/**\n * Cached instance of the compiler-cli linker's needsLinking function.\n */\nlet needsLinking: typeof import('@angular/compiler-cli/linker').needsLinking | undefined;\n\n/**\n * List of browsers which are affected by a WebKit bug where class field\n * initializers might have incorrect variable scopes.\n *\n * See: https://github.com/angular/angular-cli/issues/24355#issuecomment-1333477033\n * See: https://github.com/WebKit/WebKit/commit/e8788a34b3d5f5b4edd7ff6450b80936bff396f2\n */\nlet safariClassFieldScopeBugBrowsers: ReadonlySet<string>;\n\nexport type DiagnosticReporter = (type: 'error' | 'warning' | 'info', message: string) => void;\n\n/**\n * An interface representing the factory functions for the `@angular/localize` translation Babel plugins.\n * This must be provided for the ESM imports since dynamic imports are required to be asynchronous and\n * Babel presets currently can only be synchronous.\n *\n */\nexport interface I18nPluginCreators {\n  makeEs2015TranslatePlugin: typeof makeEs2015TranslatePlugin;\n  makeLocalePlugin: typeof makeLocalePlugin;\n}\n\nexport interface ApplicationPresetOptions {\n  i18n?: {\n    locale: string;\n    missingTranslationBehavior?: 'error' | 'warning' | 'ignore';\n    translation?: Record<string, ɵParsedTranslation>;\n    translationFiles?: string[];\n    pluginCreators: I18nPluginCreators;\n  };\n\n  angularLinker?: {\n    shouldLink: boolean;\n    jitMode: boolean;\n    linkerPluginCreator: typeof import('@angular/compiler-cli/linker/babel').createEs2015LinkerPlugin;\n  };\n\n  forceAsyncTransformation?: boolean;\n  instrumentCode?: {\n    includedBasePath: string;\n    inputSourceMap: unknown;\n  };\n  optimize?: {\n    pureTopLevel: boolean;\n    wrapDecorators: boolean;\n  };\n\n  supportedBrowsers?: string[];\n  diagnosticReporter?: DiagnosticReporter;\n}\n\n// Extract Logger type from the linker function to avoid deep importing to access the type\ntype NgtscLogger = Parameters<\n  typeof import('@angular/compiler-cli/linker/babel').createEs2015LinkerPlugin\n>[0]['logger'];\n\nfunction createI18nDiagnostics(reporter: DiagnosticReporter | undefined): Diagnostics {\n  const diagnostics: Diagnostics = new (class {\n    readonly messages: Diagnostics['messages'] = [];\n    hasErrors = false;\n\n    add(type: DiagnosticHandlingStrategy, message: string): void {\n      if (type === 'ignore') {\n        return;\n      }\n\n      this.messages.push({ type, message });\n      this.hasErrors ||= type === 'error';\n      reporter?.(type, message);\n    }\n\n    error(message: string): void {\n      this.add('error', message);\n    }\n\n    warn(message: string): void {\n      this.add('warning', message);\n    }\n\n    merge(other: Diagnostics): void {\n      for (const diagnostic of other.messages) {\n        this.add(diagnostic.type, diagnostic.message);\n      }\n    }\n\n    formatDiagnostics(): never {\n      assert.fail(\n        '@angular/localize Diagnostics formatDiagnostics should not be called from within babel.',\n      );\n    }\n  })();\n\n  return diagnostics;\n}\n\nfunction createI18nPlugins(\n  locale: string,\n  translation: Record<string, ɵParsedTranslation> | undefined,\n  missingTranslationBehavior: 'error' | 'warning' | 'ignore',\n  diagnosticReporter: DiagnosticReporter | undefined,\n  pluginCreators: I18nPluginCreators,\n) {\n  const diagnostics = createI18nDiagnostics(diagnosticReporter);\n  const plugins = [];\n\n  const { makeEs2015TranslatePlugin, makeLocalePlugin } = pluginCreators;\n\n  if (translation) {\n    plugins.push(\n      makeEs2015TranslatePlugin(diagnostics, translation, {\n        missingTranslation: missingTranslationBehavior,\n      }),\n    );\n  }\n\n  plugins.push(makeLocalePlugin(locale));\n\n  return plugins;\n}\n\nfunction createNgtscLogger(reporter: DiagnosticReporter | undefined): NgtscLogger {\n  return {\n    level: 1, // Info level\n    debug(...args: string[]) {},\n    info(...args: string[]) {\n      reporter?.('info', args.join());\n    },\n    warn(...args: string[]) {\n      reporter?.('warning', args.join());\n    },\n    error(...args: string[]) {\n      reporter?.('error', args.join());\n    },\n  };\n}\n\nexport default function (api: unknown, options: ApplicationPresetOptions) {\n  const presets = [];\n  const plugins = [];\n  let needRuntimeTransform = false;\n\n  if (options.angularLinker?.shouldLink) {\n    plugins.push(\n      options.angularLinker.linkerPluginCreator({\n        linkerJitMode: options.angularLinker.jitMode,\n        // This is a workaround until https://github.com/angular/angular/issues/42769 is fixed.\n        sourceMapping: false,\n        logger: createNgtscLogger(options.diagnosticReporter),\n        fileSystem: {\n          resolve: path.resolve,\n          exists: fs.existsSync,\n          dirname: path.dirname,\n          relative: path.relative,\n          readFile: fs.readFileSync,\n          // Node.JS types don't overlap the Compiler types.\n          // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        } as any,\n      }),\n    );\n  }\n\n  // Applications code ES version can be controlled using TypeScript's `target` option.\n  // However, this doesn't effect libraries and hence we use preset-env to downlevel ES features\n  // based on the supported browsers in browserslist.\n  if (options.supportedBrowsers) {\n    const includePlugins: string[] = [];\n\n    if (safariClassFieldScopeBugBrowsers === undefined) {\n      const browserslist = require('browserslist');\n      safariClassFieldScopeBugBrowsers = new Set(\n        browserslist([\n          // Safari <15 is technically not supported via https://angular.io/guide/browser-support,\n          // but we apply the workaround if forcibly selected.\n          'Safari <=15',\n          'iOS <=15',\n        ]),\n      );\n    }\n\n    // If a Safari browser affected by the class field scope bug is selected, we\n    // downlevel class properties by ensuring the class properties Babel plugin\n    // is always included- regardless of the preset-env targets.\n    if (options.supportedBrowsers.some((b) => safariClassFieldScopeBugBrowsers.has(b))) {\n      includePlugins.push(\n        '@babel/plugin-proposal-class-properties',\n        '@babel/plugin-proposal-private-methods',\n      );\n    }\n\n    presets.push([\n      require('@babel/preset-env').default,\n      {\n        bugfixes: true,\n        modules: false,\n        targets: options.supportedBrowsers,\n        include: includePlugins,\n        exclude: ['transform-typeof-symbol'],\n      },\n    ]);\n    needRuntimeTransform = true;\n  }\n\n  if (options.i18n) {\n    const { locale, missingTranslationBehavior, pluginCreators, translation } = options.i18n;\n    const i18nPlugins = createI18nPlugins(\n      locale,\n      translation,\n      missingTranslationBehavior || 'ignore',\n      options.diagnosticReporter,\n      pluginCreators,\n    );\n\n    plugins.push(...i18nPlugins);\n  }\n\n  if (options.forceAsyncTransformation) {\n    // Always transform async/await to support Zone.js\n    plugins.push(\n      require('@babel/plugin-transform-async-to-generator').default,\n      require('@babel/plugin-proposal-async-generator-functions').default,\n    );\n    needRuntimeTransform = true;\n  }\n\n  if (options.optimize) {\n    if (options.optimize.pureTopLevel) {\n      plugins.push(require('../plugins/pure-toplevel-functions').default);\n    }\n\n    plugins.push(\n      require('../plugins/elide-angular-metadata').default,\n      [require('../plugins/adjust-typescript-enums').default, { loose: true }],\n      [\n        require('../plugins/adjust-static-class-members').default,\n        { wrapDecorators: options.optimize.wrapDecorators },\n      ],\n    );\n  }\n\n  if (options.instrumentCode) {\n    plugins.push([\n      require('babel-plugin-istanbul').default,\n      {\n        inputSourceMap: options.instrumentCode.inputSourceMap ?? false,\n        cwd: options.instrumentCode.includedBasePath,\n      },\n    ]);\n  }\n\n  if (needRuntimeTransform) {\n    // Babel equivalent to TypeScript's `importHelpers` option\n    plugins.push([\n      require('@babel/plugin-transform-runtime').default,\n      {\n        useESModules: true,\n        version: require('@babel/runtime/package.json').version,\n        absoluteRuntime: path.dirname(require.resolve('@babel/runtime/package.json')),\n      },\n    ]);\n  }\n\n  return { presets, plugins };\n}\n\nexport async function requiresLinking(path: string, source: string): Promise<boolean> {\n  // @angular/core and @angular/compiler will cause false positives\n  // Also, TypeScript files do not require linking\n  if (/[\\\\/]@angular[\\\\/](?:compiler|core)|\\.tsx?$/.test(path)) {\n    return false;\n  }\n\n  if (!needsLinking) {\n    // Load ESM `@angular/compiler-cli/linker` using the TypeScript dynamic import workaround.\n    // Once TypeScript provides support for keeping the dynamic import this workaround can be\n    // changed to a direct dynamic import.\n    const linkerModule = await loadEsmModule<typeof import('@angular/compiler-cli/linker')>(\n      '@angular/compiler-cli/linker',\n    );\n    needsLinking = linkerModule.needsLinking;\n  }\n\n  return needsLinking(path, source);\n}\n"]}