@angular-devkit/build-angular 15.2.0-next.3 → 15.2.0-rc.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/package.json +13 -12
- package/src/babel/plugins/adjust-static-class-members.js +64 -19
- package/src/builders/browser-esbuild/angular/jit-compilation.d.ts +23 -0
- package/src/builders/browser-esbuild/angular/jit-compilation.js +99 -0
- package/src/builders/browser-esbuild/angular/jit-plugin-callbacks.d.ts +21 -0
- package/src/builders/browser-esbuild/angular/jit-plugin-callbacks.js +115 -0
- package/src/builders/browser-esbuild/angular/jit-resource-transformer.d.ts +17 -0
- package/src/builders/browser-esbuild/angular/jit-resource-transformer.js +186 -0
- package/src/builders/browser-esbuild/angular/uri.d.ts +54 -0
- package/src/builders/browser-esbuild/angular/uri.js +75 -0
- package/src/builders/browser-esbuild/commonjs-checker.js +4 -1
- package/src/builders/browser-esbuild/compiler-plugin.d.ts +1 -0
- package/src/builders/browser-esbuild/compiler-plugin.js +17 -11
- package/src/builders/browser-esbuild/experimental-warnings.js +1 -4
- package/src/builders/browser-esbuild/index.js +4 -9
- package/src/builders/browser-esbuild/javascript-transformer-worker.d.ts +1 -0
- package/src/builders/browser-esbuild/javascript-transformer-worker.js +2 -2
- package/src/builders/browser-esbuild/javascript-transformer.d.ts +1 -0
- package/src/builders/browser-esbuild/javascript-transformer.js +3 -2
- package/src/builders/browser-esbuild/less-plugin.d.ts +14 -0
- package/src/builders/browser-esbuild/less-plugin.js +108 -0
- package/src/builders/browser-esbuild/options.d.ts +1 -0
- package/src/builders/browser-esbuild/options.js +3 -2
- package/src/builders/browser-esbuild/stylesheets.d.ts +1 -1
- package/src/builders/browser-esbuild/stylesheets.js +18 -7
- package/src/builders/karma/find-tests-plugin.js +12 -8
- package/src/builders/karma/index.js +13 -1
- package/src/typings.d.ts +8 -0
- package/src/utils/esbuild-targets.js +2 -2
- package/src/utils/index-file/html-rewriting-stream.d.ts +1 -1
- package/src/utils/index-file/html-rewriting-stream.js +4 -25
- package/src/utils/service-worker.js +2 -2
- package/src/webpack/plugins/karma/karma-context.html +2 -2
- package/src/webpack/plugins/karma/karma-debug.html +2 -2
- package/src/webpack/plugins/scripts-webpack-plugin.js +10 -6
- package/src/webpack/plugins/typescript.js +3 -2
|
@@ -0,0 +1,17 @@
|
|
|
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 ts from 'typescript';
|
|
9
|
+
/**
|
|
10
|
+
* Creates a TypeScript Transformer to transform Angular Component resource references into
|
|
11
|
+
* static import statements. This transformer is used in Angular's JIT compilation mode to
|
|
12
|
+
* support processing of component resources. When in AOT mode, the Angular AOT compiler handles
|
|
13
|
+
* this processing and this transformer is not used.
|
|
14
|
+
* @param getTypeChecker A function that returns a TypeScript TypeChecker instance for the program.
|
|
15
|
+
* @returns A TypeScript transformer factory.
|
|
16
|
+
*/
|
|
17
|
+
export declare function createJitResourceTransformer(getTypeChecker: () => ts.TypeChecker): ts.TransformerFactory<ts.SourceFile>;
|
|
@@ -0,0 +1,186 @@
|
|
|
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
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
10
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
11
|
+
};
|
|
12
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
13
|
+
exports.createJitResourceTransformer = void 0;
|
|
14
|
+
const typescript_1 = __importDefault(require("typescript"));
|
|
15
|
+
const uri_1 = require("./uri");
|
|
16
|
+
/**
|
|
17
|
+
* Creates a TypeScript Transformer to transform Angular Component resource references into
|
|
18
|
+
* static import statements. This transformer is used in Angular's JIT compilation mode to
|
|
19
|
+
* support processing of component resources. When in AOT mode, the Angular AOT compiler handles
|
|
20
|
+
* this processing and this transformer is not used.
|
|
21
|
+
* @param getTypeChecker A function that returns a TypeScript TypeChecker instance for the program.
|
|
22
|
+
* @returns A TypeScript transformer factory.
|
|
23
|
+
*/
|
|
24
|
+
function createJitResourceTransformer(getTypeChecker) {
|
|
25
|
+
return (context) => {
|
|
26
|
+
const typeChecker = getTypeChecker();
|
|
27
|
+
const nodeFactory = context.factory;
|
|
28
|
+
const resourceImportDeclarations = [];
|
|
29
|
+
const visitNode = (node) => {
|
|
30
|
+
var _a;
|
|
31
|
+
if (typescript_1.default.isClassDeclaration(node)) {
|
|
32
|
+
const decorators = typescript_1.default.getDecorators(node);
|
|
33
|
+
if (!decorators || decorators.length === 0) {
|
|
34
|
+
return node;
|
|
35
|
+
}
|
|
36
|
+
return nodeFactory.updateClassDeclaration(node, [
|
|
37
|
+
...decorators.map((current) => visitDecorator(nodeFactory, current, typeChecker, resourceImportDeclarations)),
|
|
38
|
+
...((_a = typescript_1.default.getModifiers(node)) !== null && _a !== void 0 ? _a : []),
|
|
39
|
+
], node.name, node.typeParameters, node.heritageClauses, node.members);
|
|
40
|
+
}
|
|
41
|
+
return typescript_1.default.visitEachChild(node, visitNode, context);
|
|
42
|
+
};
|
|
43
|
+
return (sourceFile) => {
|
|
44
|
+
const updatedSourceFile = typescript_1.default.visitEachChild(sourceFile, visitNode, context);
|
|
45
|
+
if (resourceImportDeclarations.length > 0) {
|
|
46
|
+
return nodeFactory.updateSourceFile(updatedSourceFile, typescript_1.default.setTextRange(nodeFactory.createNodeArray([...resourceImportDeclarations, ...updatedSourceFile.statements], updatedSourceFile.statements.hasTrailingComma), updatedSourceFile.statements), updatedSourceFile.isDeclarationFile, updatedSourceFile.referencedFiles, updatedSourceFile.typeReferenceDirectives, updatedSourceFile.hasNoDefaultLib, updatedSourceFile.libReferenceDirectives);
|
|
47
|
+
}
|
|
48
|
+
else {
|
|
49
|
+
return updatedSourceFile;
|
|
50
|
+
}
|
|
51
|
+
};
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
exports.createJitResourceTransformer = createJitResourceTransformer;
|
|
55
|
+
function visitDecorator(nodeFactory, node, typeChecker, resourceImportDeclarations) {
|
|
56
|
+
const origin = getDecoratorOrigin(node, typeChecker);
|
|
57
|
+
if (!origin || origin.module !== '@angular/core' || origin.name !== 'Component') {
|
|
58
|
+
return node;
|
|
59
|
+
}
|
|
60
|
+
if (!typescript_1.default.isCallExpression(node.expression)) {
|
|
61
|
+
return node;
|
|
62
|
+
}
|
|
63
|
+
const decoratorFactory = node.expression;
|
|
64
|
+
const args = decoratorFactory.arguments;
|
|
65
|
+
if (args.length !== 1 || !typescript_1.default.isObjectLiteralExpression(args[0])) {
|
|
66
|
+
// Unsupported component metadata
|
|
67
|
+
return node;
|
|
68
|
+
}
|
|
69
|
+
const objectExpression = args[0];
|
|
70
|
+
const styleReplacements = [];
|
|
71
|
+
// visit all properties
|
|
72
|
+
let properties = typescript_1.default.visitNodes(objectExpression.properties, (node) => typescript_1.default.isObjectLiteralElementLike(node)
|
|
73
|
+
? visitComponentMetadata(nodeFactory, node, styleReplacements, resourceImportDeclarations)
|
|
74
|
+
: node);
|
|
75
|
+
// replace properties with updated properties
|
|
76
|
+
if (styleReplacements.length > 0) {
|
|
77
|
+
const styleProperty = nodeFactory.createPropertyAssignment(nodeFactory.createIdentifier('styles'), nodeFactory.createArrayLiteralExpression(styleReplacements));
|
|
78
|
+
properties = nodeFactory.createNodeArray([...properties, styleProperty]);
|
|
79
|
+
}
|
|
80
|
+
return nodeFactory.updateDecorator(node, nodeFactory.updateCallExpression(decoratorFactory, decoratorFactory.expression, decoratorFactory.typeArguments, [nodeFactory.updateObjectLiteralExpression(objectExpression, properties)]));
|
|
81
|
+
}
|
|
82
|
+
function visitComponentMetadata(nodeFactory, node, styleReplacements, resourceImportDeclarations) {
|
|
83
|
+
if (!typescript_1.default.isPropertyAssignment(node) || typescript_1.default.isComputedPropertyName(node.name)) {
|
|
84
|
+
return node;
|
|
85
|
+
}
|
|
86
|
+
switch (node.name.text) {
|
|
87
|
+
case 'templateUrl':
|
|
88
|
+
// Only analyze string literals
|
|
89
|
+
if (!typescript_1.default.isStringLiteral(node.initializer) &&
|
|
90
|
+
!typescript_1.default.isNoSubstitutionTemplateLiteral(node.initializer)) {
|
|
91
|
+
return node;
|
|
92
|
+
}
|
|
93
|
+
const url = node.initializer.text;
|
|
94
|
+
if (!url) {
|
|
95
|
+
return node;
|
|
96
|
+
}
|
|
97
|
+
return nodeFactory.updatePropertyAssignment(node, nodeFactory.createIdentifier('template'), createResourceImport(nodeFactory, (0, uri_1.generateJitFileUri)(url, 'template'), resourceImportDeclarations));
|
|
98
|
+
case 'styles':
|
|
99
|
+
if (!typescript_1.default.isArrayLiteralExpression(node.initializer)) {
|
|
100
|
+
return node;
|
|
101
|
+
}
|
|
102
|
+
const inlineStyles = typescript_1.default.visitNodes(node.initializer.elements, (node) => {
|
|
103
|
+
if (!typescript_1.default.isStringLiteral(node) && !typescript_1.default.isNoSubstitutionTemplateLiteral(node)) {
|
|
104
|
+
return node;
|
|
105
|
+
}
|
|
106
|
+
const contents = node.text;
|
|
107
|
+
if (!contents) {
|
|
108
|
+
// An empty inline style is equivalent to not having a style element
|
|
109
|
+
return undefined;
|
|
110
|
+
}
|
|
111
|
+
return createResourceImport(nodeFactory, (0, uri_1.generateJitInlineUri)(contents, 'style'), resourceImportDeclarations);
|
|
112
|
+
});
|
|
113
|
+
// Inline styles should be placed first
|
|
114
|
+
styleReplacements.unshift(...inlineStyles);
|
|
115
|
+
// The inline styles will be added afterwards in combination with any external styles
|
|
116
|
+
return undefined;
|
|
117
|
+
case 'styleUrls':
|
|
118
|
+
if (!typescript_1.default.isArrayLiteralExpression(node.initializer)) {
|
|
119
|
+
return node;
|
|
120
|
+
}
|
|
121
|
+
const externalStyles = typescript_1.default.visitNodes(node.initializer.elements, (node) => {
|
|
122
|
+
if (!typescript_1.default.isStringLiteral(node) && !typescript_1.default.isNoSubstitutionTemplateLiteral(node)) {
|
|
123
|
+
return node;
|
|
124
|
+
}
|
|
125
|
+
const url = node.text;
|
|
126
|
+
if (!url) {
|
|
127
|
+
return node;
|
|
128
|
+
}
|
|
129
|
+
return createResourceImport(nodeFactory, (0, uri_1.generateJitFileUri)(url, 'style'), resourceImportDeclarations);
|
|
130
|
+
});
|
|
131
|
+
// External styles are applied after any inline styles
|
|
132
|
+
styleReplacements.push(...externalStyles);
|
|
133
|
+
// The external styles will be added afterwards in combination with any inline styles
|
|
134
|
+
return undefined;
|
|
135
|
+
default:
|
|
136
|
+
// All other elements are passed through
|
|
137
|
+
return node;
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
function createResourceImport(nodeFactory, url, resourceImportDeclarations) {
|
|
141
|
+
const urlLiteral = nodeFactory.createStringLiteral(url);
|
|
142
|
+
const importName = nodeFactory.createIdentifier(`__NG_CLI_RESOURCE__${resourceImportDeclarations.length}`);
|
|
143
|
+
resourceImportDeclarations.push(nodeFactory.createImportDeclaration(undefined, nodeFactory.createImportClause(false, importName, undefined), urlLiteral));
|
|
144
|
+
return importName;
|
|
145
|
+
}
|
|
146
|
+
function getDecoratorOrigin(decorator, typeChecker) {
|
|
147
|
+
if (!typescript_1.default.isCallExpression(decorator.expression)) {
|
|
148
|
+
return null;
|
|
149
|
+
}
|
|
150
|
+
let identifier;
|
|
151
|
+
let name = '';
|
|
152
|
+
if (typescript_1.default.isPropertyAccessExpression(decorator.expression.expression)) {
|
|
153
|
+
identifier = decorator.expression.expression.expression;
|
|
154
|
+
name = decorator.expression.expression.name.text;
|
|
155
|
+
}
|
|
156
|
+
else if (typescript_1.default.isIdentifier(decorator.expression.expression)) {
|
|
157
|
+
identifier = decorator.expression.expression;
|
|
158
|
+
}
|
|
159
|
+
else {
|
|
160
|
+
return null;
|
|
161
|
+
}
|
|
162
|
+
// NOTE: resolver.getReferencedImportDeclaration would work as well but is internal
|
|
163
|
+
const symbol = typeChecker.getSymbolAtLocation(identifier);
|
|
164
|
+
if (symbol && symbol.declarations && symbol.declarations.length > 0) {
|
|
165
|
+
const declaration = symbol.declarations[0];
|
|
166
|
+
let module;
|
|
167
|
+
if (typescript_1.default.isImportSpecifier(declaration)) {
|
|
168
|
+
name = (declaration.propertyName || declaration.name).text;
|
|
169
|
+
module = declaration.parent.parent.parent.moduleSpecifier.text;
|
|
170
|
+
}
|
|
171
|
+
else if (typescript_1.default.isNamespaceImport(declaration)) {
|
|
172
|
+
// Use the name from the decorator namespace property access
|
|
173
|
+
module = declaration.parent.parent.moduleSpecifier.text;
|
|
174
|
+
}
|
|
175
|
+
else if (typescript_1.default.isImportClause(declaration)) {
|
|
176
|
+
name = declaration.name.text;
|
|
177
|
+
module = declaration.parent.moduleSpecifier.text;
|
|
178
|
+
}
|
|
179
|
+
else {
|
|
180
|
+
return null;
|
|
181
|
+
}
|
|
182
|
+
return { name, module };
|
|
183
|
+
}
|
|
184
|
+
return null;
|
|
185
|
+
}
|
|
186
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"jit-resource-transformer.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/builders/browser-esbuild/angular/jit-resource-transformer.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 ts from 'typescript';\nimport { generateJitFileUri, generateJitInlineUri } from './uri';\n\n/**\n * Creates a TypeScript Transformer to transform Angular Component resource references into\n * static import statements. This transformer is used in Angular's JIT compilation mode to\n * support processing of component resources. When in AOT mode, the Angular AOT compiler handles\n * this processing and this transformer is not used.\n * @param getTypeChecker A function that returns a TypeScript TypeChecker instance for the program.\n * @returns A TypeScript transformer factory.\n */\nexport function createJitResourceTransformer(\n  getTypeChecker: () => ts.TypeChecker,\n): ts.TransformerFactory<ts.SourceFile> {\n  return (context: ts.TransformationContext) => {\n    const typeChecker = getTypeChecker();\n    const nodeFactory = context.factory;\n    const resourceImportDeclarations: ts.ImportDeclaration[] = [];\n\n    const visitNode: ts.Visitor = (node: ts.Node) => {\n      if (ts.isClassDeclaration(node)) {\n        const decorators = ts.getDecorators(node);\n\n        if (!decorators || decorators.length === 0) {\n          return node;\n        }\n\n        return nodeFactory.updateClassDeclaration(\n          node,\n          [\n            ...decorators.map((current) =>\n              visitDecorator(nodeFactory, current, typeChecker, resourceImportDeclarations),\n            ),\n            ...(ts.getModifiers(node) ?? []),\n          ],\n          node.name,\n          node.typeParameters,\n          node.heritageClauses,\n          node.members,\n        );\n      }\n\n      return ts.visitEachChild(node, visitNode, context);\n    };\n\n    return (sourceFile) => {\n      const updatedSourceFile = ts.visitEachChild(sourceFile, visitNode, context);\n\n      if (resourceImportDeclarations.length > 0) {\n        return nodeFactory.updateSourceFile(\n          updatedSourceFile,\n          ts.setTextRange(\n            nodeFactory.createNodeArray(\n              [...resourceImportDeclarations, ...updatedSourceFile.statements],\n              updatedSourceFile.statements.hasTrailingComma,\n            ),\n            updatedSourceFile.statements,\n          ),\n          updatedSourceFile.isDeclarationFile,\n          updatedSourceFile.referencedFiles,\n          updatedSourceFile.typeReferenceDirectives,\n          updatedSourceFile.hasNoDefaultLib,\n          updatedSourceFile.libReferenceDirectives,\n        );\n      } else {\n        return updatedSourceFile;\n      }\n    };\n  };\n}\n\nfunction visitDecorator(\n  nodeFactory: ts.NodeFactory,\n  node: ts.Decorator,\n  typeChecker: ts.TypeChecker,\n  resourceImportDeclarations: ts.ImportDeclaration[],\n): ts.Decorator {\n  const origin = getDecoratorOrigin(node, typeChecker);\n  if (!origin || origin.module !== '@angular/core' || origin.name !== 'Component') {\n    return node;\n  }\n\n  if (!ts.isCallExpression(node.expression)) {\n    return node;\n  }\n\n  const decoratorFactory = node.expression;\n  const args = decoratorFactory.arguments;\n  if (args.length !== 1 || !ts.isObjectLiteralExpression(args[0])) {\n    // Unsupported component metadata\n    return node;\n  }\n\n  const objectExpression = args[0] as ts.ObjectLiteralExpression;\n  const styleReplacements: ts.Expression[] = [];\n\n  // visit all properties\n  let properties = ts.visitNodes(objectExpression.properties, (node) =>\n    ts.isObjectLiteralElementLike(node)\n      ? visitComponentMetadata(nodeFactory, node, styleReplacements, resourceImportDeclarations)\n      : node,\n  );\n\n  // replace properties with updated properties\n  if (styleReplacements.length > 0) {\n    const styleProperty = nodeFactory.createPropertyAssignment(\n      nodeFactory.createIdentifier('styles'),\n      nodeFactory.createArrayLiteralExpression(styleReplacements),\n    );\n\n    properties = nodeFactory.createNodeArray([...properties, styleProperty]);\n  }\n\n  return nodeFactory.updateDecorator(\n    node,\n    nodeFactory.updateCallExpression(\n      decoratorFactory,\n      decoratorFactory.expression,\n      decoratorFactory.typeArguments,\n      [nodeFactory.updateObjectLiteralExpression(objectExpression, properties)],\n    ),\n  );\n}\n\nfunction visitComponentMetadata(\n  nodeFactory: ts.NodeFactory,\n  node: ts.ObjectLiteralElementLike,\n  styleReplacements: ts.Expression[],\n  resourceImportDeclarations: ts.ImportDeclaration[],\n): ts.ObjectLiteralElementLike | undefined {\n  if (!ts.isPropertyAssignment(node) || ts.isComputedPropertyName(node.name)) {\n    return node;\n  }\n\n  switch (node.name.text) {\n    case 'templateUrl':\n      // Only analyze string literals\n      if (\n        !ts.isStringLiteral(node.initializer) &&\n        !ts.isNoSubstitutionTemplateLiteral(node.initializer)\n      ) {\n        return node;\n      }\n\n      const url = node.initializer.text;\n      if (!url) {\n        return node;\n      }\n\n      return nodeFactory.updatePropertyAssignment(\n        node,\n        nodeFactory.createIdentifier('template'),\n        createResourceImport(\n          nodeFactory,\n          generateJitFileUri(url, 'template'),\n          resourceImportDeclarations,\n        ),\n      );\n    case 'styles':\n      if (!ts.isArrayLiteralExpression(node.initializer)) {\n        return node;\n      }\n\n      const inlineStyles = ts.visitNodes(node.initializer.elements, (node) => {\n        if (!ts.isStringLiteral(node) && !ts.isNoSubstitutionTemplateLiteral(node)) {\n          return node;\n        }\n\n        const contents = node.text;\n        if (!contents) {\n          // An empty inline style is equivalent to not having a style element\n          return undefined;\n        }\n\n        return createResourceImport(\n          nodeFactory,\n          generateJitInlineUri(contents, 'style'),\n          resourceImportDeclarations,\n        );\n      });\n\n      // Inline styles should be placed first\n      styleReplacements.unshift(...inlineStyles);\n\n      // The inline styles will be added afterwards in combination with any external styles\n      return undefined;\n    case 'styleUrls':\n      if (!ts.isArrayLiteralExpression(node.initializer)) {\n        return node;\n      }\n\n      const externalStyles = ts.visitNodes(node.initializer.elements, (node) => {\n        if (!ts.isStringLiteral(node) && !ts.isNoSubstitutionTemplateLiteral(node)) {\n          return node;\n        }\n\n        const url = node.text;\n        if (!url) {\n          return node;\n        }\n\n        return createResourceImport(\n          nodeFactory,\n          generateJitFileUri(url, 'style'),\n          resourceImportDeclarations,\n        );\n      });\n\n      // External styles are applied after any inline styles\n      styleReplacements.push(...externalStyles);\n\n      // The external styles will be added afterwards in combination with any inline styles\n      return undefined;\n    default:\n      // All other elements are passed through\n      return node;\n  }\n}\n\nfunction createResourceImport(\n  nodeFactory: ts.NodeFactory,\n  url: string,\n  resourceImportDeclarations: ts.ImportDeclaration[],\n): ts.Identifier {\n  const urlLiteral = nodeFactory.createStringLiteral(url);\n\n  const importName = nodeFactory.createIdentifier(\n    `__NG_CLI_RESOURCE__${resourceImportDeclarations.length}`,\n  );\n  resourceImportDeclarations.push(\n    nodeFactory.createImportDeclaration(\n      undefined,\n      nodeFactory.createImportClause(false, importName, undefined),\n      urlLiteral,\n    ),\n  );\n\n  return importName;\n}\n\nfunction getDecoratorOrigin(\n  decorator: ts.Decorator,\n  typeChecker: ts.TypeChecker,\n): { name: string; module: string } | null {\n  if (!ts.isCallExpression(decorator.expression)) {\n    return null;\n  }\n\n  let identifier: ts.Node;\n  let name = '';\n\n  if (ts.isPropertyAccessExpression(decorator.expression.expression)) {\n    identifier = decorator.expression.expression.expression;\n    name = decorator.expression.expression.name.text;\n  } else if (ts.isIdentifier(decorator.expression.expression)) {\n    identifier = decorator.expression.expression;\n  } else {\n    return null;\n  }\n\n  // NOTE: resolver.getReferencedImportDeclaration would work as well but is internal\n  const symbol = typeChecker.getSymbolAtLocation(identifier);\n  if (symbol && symbol.declarations && symbol.declarations.length > 0) {\n    const declaration = symbol.declarations[0];\n    let module: string;\n\n    if (ts.isImportSpecifier(declaration)) {\n      name = (declaration.propertyName || declaration.name).text;\n      module = (declaration.parent.parent.parent.moduleSpecifier as ts.StringLiteral).text;\n    } else if (ts.isNamespaceImport(declaration)) {\n      // Use the name from the decorator namespace property access\n      module = (declaration.parent.parent.moduleSpecifier as ts.StringLiteral).text;\n    } else if (ts.isImportClause(declaration)) {\n      name = (declaration.name as ts.Identifier).text;\n      module = (declaration.parent.moduleSpecifier as ts.StringLiteral).text;\n    } else {\n      return null;\n    }\n\n    return { name, module };\n  }\n\n  return null;\n}\n"]}
|
|
@@ -0,0 +1,54 @@
|
|
|
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
|
+
* A string value representing the namespace for Angular JIT mode related imports for
|
|
10
|
+
* Component styles. This namespace is used for both inline (`styles`) and external
|
|
11
|
+
* (`styleUrls`) styles.
|
|
12
|
+
*/
|
|
13
|
+
export declare const JIT_STYLE_NAMESPACE: "angular:jit:style";
|
|
14
|
+
/**
|
|
15
|
+
* A string value representing the namespace for Angular JIT mode related imports for
|
|
16
|
+
* Component templates. This namespace is currently only used for external (`templateUrl`)
|
|
17
|
+
* templates.
|
|
18
|
+
*/
|
|
19
|
+
export declare const JIT_TEMPLATE_NAMESPACE: "angular:jit:template";
|
|
20
|
+
/**
|
|
21
|
+
* A regular expression that can be used to match a Angular JIT mode namespace URI.
|
|
22
|
+
* It contains capture groups for the type (template/style), origin (file/inline), and specifier.
|
|
23
|
+
* The {@link parseJitUri} function can be used to parse and return an object representation of a JIT URI.
|
|
24
|
+
*/
|
|
25
|
+
export declare const JIT_NAMESPACE_REGEXP: RegExp;
|
|
26
|
+
/**
|
|
27
|
+
* Generates an Angular JIT mode namespace URI for a given file.
|
|
28
|
+
* @param file The path of the file to be included.
|
|
29
|
+
* @param type The type of the file (`style` or `template`).
|
|
30
|
+
* @returns A string containing the full JIT namespace URI.
|
|
31
|
+
*/
|
|
32
|
+
export declare function generateJitFileUri(file: string, type: 'style' | 'template'): string;
|
|
33
|
+
/**
|
|
34
|
+
* Generates an Angular JIT mode namespace URI for a given inline style or template.
|
|
35
|
+
* The provided content is base64 encoded and included in the URI.
|
|
36
|
+
* @param data The content to encode within the URI.
|
|
37
|
+
* @param type The type of the content (`style` or `template`).
|
|
38
|
+
* @returns A string containing the full JIT namespace URI.
|
|
39
|
+
*/
|
|
40
|
+
export declare function generateJitInlineUri(data: string | Uint8Array, type: 'style' | 'template'): string;
|
|
41
|
+
/**
|
|
42
|
+
* Parses a string containing a JIT namespace URI.
|
|
43
|
+
* JIT namespace URIs are used to encode the information for an Angular component's stylesheets
|
|
44
|
+
* and templates when compiled in JIT mode.
|
|
45
|
+
* @param uri The URI to parse into its underlying components.
|
|
46
|
+
* @returns An object containing the namespace, type, origin, and specifier of the URI;
|
|
47
|
+
* `undefined` if not a JIT namespace URI.
|
|
48
|
+
*/
|
|
49
|
+
export declare function parseJitUri(uri: string): {
|
|
50
|
+
namespace: string;
|
|
51
|
+
type: "style" | "template";
|
|
52
|
+
origin: "inline" | "file";
|
|
53
|
+
specifier: string;
|
|
54
|
+
} | undefined;
|
|
@@ -0,0 +1,75 @@
|
|
|
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.parseJitUri = exports.generateJitInlineUri = exports.generateJitFileUri = exports.JIT_NAMESPACE_REGEXP = exports.JIT_TEMPLATE_NAMESPACE = exports.JIT_STYLE_NAMESPACE = void 0;
|
|
11
|
+
/**
|
|
12
|
+
* A string value representing the base namespace for Angular JIT mode related imports.
|
|
13
|
+
*/
|
|
14
|
+
const JIT_BASE_NAMESPACE = 'angular:jit';
|
|
15
|
+
/**
|
|
16
|
+
* A string value representing the namespace for Angular JIT mode related imports for
|
|
17
|
+
* Component styles. This namespace is used for both inline (`styles`) and external
|
|
18
|
+
* (`styleUrls`) styles.
|
|
19
|
+
*/
|
|
20
|
+
exports.JIT_STYLE_NAMESPACE = `${JIT_BASE_NAMESPACE}:style`;
|
|
21
|
+
/**
|
|
22
|
+
* A string value representing the namespace for Angular JIT mode related imports for
|
|
23
|
+
* Component templates. This namespace is currently only used for external (`templateUrl`)
|
|
24
|
+
* templates.
|
|
25
|
+
*/
|
|
26
|
+
exports.JIT_TEMPLATE_NAMESPACE = `${JIT_BASE_NAMESPACE}:template`;
|
|
27
|
+
/**
|
|
28
|
+
* A regular expression that can be used to match a Angular JIT mode namespace URI.
|
|
29
|
+
* It contains capture groups for the type (template/style), origin (file/inline), and specifier.
|
|
30
|
+
* The {@link parseJitUri} function can be used to parse and return an object representation of a JIT URI.
|
|
31
|
+
*/
|
|
32
|
+
exports.JIT_NAMESPACE_REGEXP = new RegExp(`^${JIT_BASE_NAMESPACE}:(template|style):(file|inline);(.*)$`);
|
|
33
|
+
/**
|
|
34
|
+
* Generates an Angular JIT mode namespace URI for a given file.
|
|
35
|
+
* @param file The path of the file to be included.
|
|
36
|
+
* @param type The type of the file (`style` or `template`).
|
|
37
|
+
* @returns A string containing the full JIT namespace URI.
|
|
38
|
+
*/
|
|
39
|
+
function generateJitFileUri(file, type) {
|
|
40
|
+
return `${JIT_BASE_NAMESPACE}:${type}:file;${file}`;
|
|
41
|
+
}
|
|
42
|
+
exports.generateJitFileUri = generateJitFileUri;
|
|
43
|
+
/**
|
|
44
|
+
* Generates an Angular JIT mode namespace URI for a given inline style or template.
|
|
45
|
+
* The provided content is base64 encoded and included in the URI.
|
|
46
|
+
* @param data The content to encode within the URI.
|
|
47
|
+
* @param type The type of the content (`style` or `template`).
|
|
48
|
+
* @returns A string containing the full JIT namespace URI.
|
|
49
|
+
*/
|
|
50
|
+
function generateJitInlineUri(data, type) {
|
|
51
|
+
return `${JIT_BASE_NAMESPACE}:${type}:inline;${Buffer.from(data).toString('base64')}`;
|
|
52
|
+
}
|
|
53
|
+
exports.generateJitInlineUri = generateJitInlineUri;
|
|
54
|
+
/**
|
|
55
|
+
* Parses a string containing a JIT namespace URI.
|
|
56
|
+
* JIT namespace URIs are used to encode the information for an Angular component's stylesheets
|
|
57
|
+
* and templates when compiled in JIT mode.
|
|
58
|
+
* @param uri The URI to parse into its underlying components.
|
|
59
|
+
* @returns An object containing the namespace, type, origin, and specifier of the URI;
|
|
60
|
+
* `undefined` if not a JIT namespace URI.
|
|
61
|
+
*/
|
|
62
|
+
function parseJitUri(uri) {
|
|
63
|
+
const matches = exports.JIT_NAMESPACE_REGEXP.exec(uri);
|
|
64
|
+
if (!matches) {
|
|
65
|
+
return undefined;
|
|
66
|
+
}
|
|
67
|
+
return {
|
|
68
|
+
namespace: `${JIT_BASE_NAMESPACE}:${matches[1]}`,
|
|
69
|
+
type: matches[1],
|
|
70
|
+
origin: matches[2],
|
|
71
|
+
specifier: matches[3],
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
exports.parseJitUri = parseJitUri;
|
|
75
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXJpLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvYW5ndWxhcl9kZXZraXQvYnVpbGRfYW5ndWxhci9zcmMvYnVpbGRlcnMvYnJvd3Nlci1lc2J1aWxkL2FuZ3VsYXIvdXJpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7O0dBTUc7OztBQUVIOztHQUVHO0FBQ0gsTUFBTSxrQkFBa0IsR0FBRyxhQUFhLENBQUM7QUFFekM7Ozs7R0FJRztBQUNVLFFBQUEsbUJBQW1CLEdBQUcsR0FBRyxrQkFBa0IsUUFBaUIsQ0FBQztBQUUxRTs7OztHQUlHO0FBQ1UsUUFBQSxzQkFBc0IsR0FBRyxHQUFHLGtCQUFrQixXQUFvQixDQUFDO0FBRWhGOzs7O0dBSUc7QUFDVSxRQUFBLG9CQUFvQixHQUFHLElBQUksTUFBTSxDQUM1QyxJQUFJLGtCQUFrQix1Q0FBdUMsQ0FDOUQsQ0FBQztBQUVGOzs7OztHQUtHO0FBQ0gsU0FBZ0Isa0JBQWtCLENBQUMsSUFBWSxFQUFFLElBQTBCO0lBQ3pFLE9BQU8sR0FBRyxrQkFBa0IsSUFBSSxJQUFJLFNBQVMsSUFBSSxFQUFFLENBQUM7QUFDdEQsQ0FBQztBQUZELGdEQUVDO0FBRUQ7Ozs7OztHQU1HO0FBQ0gsU0FBZ0Isb0JBQW9CLENBQUMsSUFBeUIsRUFBRSxJQUEwQjtJQUN4RixPQUFPLEdBQUcsa0JBQWtCLElBQUksSUFBSSxXQUFXLE1BQU0sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsUUFBUSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUM7QUFDeEYsQ0FBQztBQUZELG9EQUVDO0FBRUQ7Ozs7Ozs7R0FPRztBQUNILFNBQWdCLFdBQVcsQ0FBQyxHQUFXO0lBQ3JDLE1BQU0sT0FBTyxHQUFHLDRCQUFvQixDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUMvQyxJQUFJLENBQUMsT0FBTyxFQUFFO1FBQ1osT0FBTyxTQUFTLENBQUM7S0FDbEI7SUFFRCxPQUFPO1FBQ0wsU0FBUyxFQUFFLEdBQUcsa0JBQWtCLElBQUksT0FBTyxDQUFDLENBQUMsQ0FBQyxFQUFFO1FBQ2hELElBQUksRUFBRSxPQUFPLENBQUMsQ0FBQyxDQUF5QjtRQUN4QyxNQUFNLEVBQUUsT0FBTyxDQUFDLENBQUMsQ0FBc0I7UUFDdkMsU0FBUyxFQUFFLE9BQU8sQ0FBQyxDQUFDLENBQUM7S0FDdEIsQ0FBQztBQUNKLENBQUM7QUFaRCxrQ0FZQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQGxpY2Vuc2VcbiAqIENvcHlyaWdodCBHb29nbGUgTExDIEFsbCBSaWdodHMgUmVzZXJ2ZWQuXG4gKlxuICogVXNlIG9mIHRoaXMgc291cmNlIGNvZGUgaXMgZ292ZXJuZWQgYnkgYW4gTUlULXN0eWxlIGxpY2Vuc2UgdGhhdCBjYW4gYmVcbiAqIGZvdW5kIGluIHRoZSBMSUNFTlNFIGZpbGUgYXQgaHR0cHM6Ly9hbmd1bGFyLmlvL2xpY2Vuc2VcbiAqL1xuXG4vKipcbiAqIEEgc3RyaW5nIHZhbHVlIHJlcHJlc2VudGluZyB0aGUgYmFzZSBuYW1lc3BhY2UgZm9yIEFuZ3VsYXIgSklUIG1vZGUgcmVsYXRlZCBpbXBvcnRzLlxuICovXG5jb25zdCBKSVRfQkFTRV9OQU1FU1BBQ0UgPSAnYW5ndWxhcjpqaXQnO1xuXG4vKipcbiAqIEEgc3RyaW5nIHZhbHVlIHJlcHJlc2VudGluZyB0aGUgbmFtZXNwYWNlIGZvciBBbmd1bGFyIEpJVCBtb2RlIHJlbGF0ZWQgaW1wb3J0cyBmb3JcbiAqIENvbXBvbmVudCBzdHlsZXMuIFRoaXMgbmFtZXNwYWNlIGlzIHVzZWQgZm9yIGJvdGggaW5saW5lIChgc3R5bGVzYCkgYW5kIGV4dGVybmFsXG4gKiAoYHN0eWxlVXJsc2ApIHN0eWxlcy5cbiAqL1xuZXhwb3J0IGNvbnN0IEpJVF9TVFlMRV9OQU1FU1BBQ0UgPSBgJHtKSVRfQkFTRV9OQU1FU1BBQ0V9OnN0eWxlYCBhcyBjb25zdDtcblxuLyoqXG4gKiBBIHN0cmluZyB2YWx1ZSByZXByZXNlbnRpbmcgdGhlIG5hbWVzcGFjZSBmb3IgQW5ndWxhciBKSVQgbW9kZSByZWxhdGVkIGltcG9ydHMgZm9yXG4gKiBDb21wb25lbnQgdGVtcGxhdGVzLiBUaGlzIG5hbWVzcGFjZSBpcyBjdXJyZW50bHkgb25seSB1c2VkIGZvciBleHRlcm5hbCAoYHRlbXBsYXRlVXJsYClcbiAqIHRlbXBsYXRlcy5cbiAqL1xuZXhwb3J0IGNvbnN0IEpJVF9URU1QTEFURV9OQU1FU1BBQ0UgPSBgJHtKSVRfQkFTRV9OQU1FU1BBQ0V9OnRlbXBsYXRlYCBhcyBjb25zdDtcblxuLyoqXG4gKiBBIHJlZ3VsYXIgZXhwcmVzc2lvbiB0aGF0IGNhbiBiZSB1c2VkIHRvIG1hdGNoIGEgQW5ndWxhciBKSVQgbW9kZSBuYW1lc3BhY2UgVVJJLlxuICogSXQgY29udGFpbnMgY2FwdHVyZSBncm91cHMgZm9yIHRoZSB0eXBlICh0ZW1wbGF0ZS9zdHlsZSksIG9yaWdpbiAoZmlsZS9pbmxpbmUpLCBhbmQgc3BlY2lmaWVyLlxuICogVGhlIHtAbGluayBwYXJzZUppdFVyaX0gZnVuY3Rpb24gY2FuIGJlIHVzZWQgdG8gcGFyc2UgYW5kIHJldHVybiBhbiBvYmplY3QgcmVwcmVzZW50YXRpb24gb2YgYSBKSVQgVVJJLlxuICovXG5leHBvcnQgY29uc3QgSklUX05BTUVTUEFDRV9SRUdFWFAgPSBuZXcgUmVnRXhwKFxuICBgXiR7SklUX0JBU0VfTkFNRVNQQUNFfToodGVtcGxhdGV8c3R5bGUpOihmaWxlfGlubGluZSk7KC4qKSRgLFxuKTtcblxuLyoqXG4gKiBHZW5lcmF0ZXMgYW4gQW5ndWxhciBKSVQgbW9kZSBuYW1lc3BhY2UgVVJJIGZvciBhIGdpdmVuIGZpbGUuXG4gKiBAcGFyYW0gZmlsZSBUaGUgcGF0aCBvZiB0aGUgZmlsZSB0byBiZSBpbmNsdWRlZC5cbiAqIEBwYXJhbSB0eXBlIFRoZSB0eXBlIG9mIHRoZSBmaWxlIChgc3R5bGVgIG9yIGB0ZW1wbGF0ZWApLlxuICogQHJldHVybnMgQSBzdHJpbmcgY29udGFpbmluZyB0aGUgZnVsbCBKSVQgbmFtZXNwYWNlIFVSSS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdlbmVyYXRlSml0RmlsZVVyaShmaWxlOiBzdHJpbmcsIHR5cGU6ICdzdHlsZScgfCAndGVtcGxhdGUnKSB7XG4gIHJldHVybiBgJHtKSVRfQkFTRV9OQU1FU1BBQ0V9OiR7dHlwZX06ZmlsZTske2ZpbGV9YDtcbn1cblxuLyoqXG4gKiBHZW5lcmF0ZXMgYW4gQW5ndWxhciBKSVQgbW9kZSBuYW1lc3BhY2UgVVJJIGZvciBhIGdpdmVuIGlubGluZSBzdHlsZSBvciB0ZW1wbGF0ZS5cbiAqIFRoZSBwcm92aWRlZCBjb250ZW50IGlzIGJhc2U2NCBlbmNvZGVkIGFuZCBpbmNsdWRlZCBpbiB0aGUgVVJJLlxuICogQHBhcmFtIGRhdGEgVGhlIGNvbnRlbnQgdG8gZW5jb2RlIHdpdGhpbiB0aGUgVVJJLlxuICogQHBhcmFtIHR5cGUgVGhlIHR5cGUgb2YgdGhlIGNvbnRlbnQgKGBzdHlsZWAgb3IgYHRlbXBsYXRlYCkuXG4gKiBAcmV0dXJucyBBIHN0cmluZyBjb250YWluaW5nIHRoZSBmdWxsIEpJVCBuYW1lc3BhY2UgVVJJLlxuICovXG5leHBvcnQgZnVuY3Rpb24gZ2VuZXJhdGVKaXRJbmxpbmVVcmkoZGF0YTogc3RyaW5nIHwgVWludDhBcnJheSwgdHlwZTogJ3N0eWxlJyB8ICd0ZW1wbGF0ZScpIHtcbiAgcmV0dXJuIGAke0pJVF9CQVNFX05BTUVTUEFDRX06JHt0eXBlfTppbmxpbmU7JHtCdWZmZXIuZnJvbShkYXRhKS50b1N0cmluZygnYmFzZTY0Jyl9YDtcbn1cblxuLyoqXG4gKiBQYXJzZXMgYSBzdHJpbmcgY29udGFpbmluZyBhIEpJVCBuYW1lc3BhY2UgVVJJLlxuICogSklUIG5hbWVzcGFjZSBVUklzIGFyZSB1c2VkIHRvIGVuY29kZSB0aGUgaW5mb3JtYXRpb24gZm9yIGFuIEFuZ3VsYXIgY29tcG9uZW50J3Mgc3R5bGVzaGVldHNcbiAqIGFuZCB0ZW1wbGF0ZXMgd2hlbiBjb21waWxlZCBpbiBKSVQgbW9kZS5cbiAqIEBwYXJhbSB1cmkgVGhlIFVSSSB0byBwYXJzZSBpbnRvIGl0cyB1bmRlcmx5aW5nIGNvbXBvbmVudHMuXG4gKiBAcmV0dXJucyBBbiBvYmplY3QgY29udGFpbmluZyB0aGUgbmFtZXNwYWNlLCB0eXBlLCBvcmlnaW4sIGFuZCBzcGVjaWZpZXIgb2YgdGhlIFVSSTtcbiAqIGB1bmRlZmluZWRgIGlmIG5vdCBhIEpJVCBuYW1lc3BhY2UgVVJJLlxuICovXG5leHBvcnQgZnVuY3Rpb24gcGFyc2VKaXRVcmkodXJpOiBzdHJpbmcpIHtcbiAgY29uc3QgbWF0Y2hlcyA9IEpJVF9OQU1FU1BBQ0VfUkVHRVhQLmV4ZWModXJpKTtcbiAgaWYgKCFtYXRjaGVzKSB7XG4gICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgfVxuXG4gIHJldHVybiB7XG4gICAgbmFtZXNwYWNlOiBgJHtKSVRfQkFTRV9OQU1FU1BBQ0V9OiR7bWF0Y2hlc1sxXX1gLFxuICAgIHR5cGU6IG1hdGNoZXNbMV0gYXMgJ3N0eWxlJyB8ICd0ZW1wbGF0ZScsXG4gICAgb3JpZ2luOiBtYXRjaGVzWzJdIGFzICdmaWxlJyB8ICdpbmxpbmUnLFxuICAgIHNwZWNpZmllcjogbWF0Y2hlc1szXSxcbiAgfTtcbn1cbiJdfQ==
|
|
@@ -32,6 +32,9 @@ function checkCommonJSModules(metafile, allowedCommonJsDependencies) {
|
|
|
32
32
|
const allowedRequests = new Set(allowedCommonJsDependencies);
|
|
33
33
|
// Ignore Angular locale definitions which are currently UMD
|
|
34
34
|
allowedRequests.add('@angular/common/locales');
|
|
35
|
+
// Ignore zone.js due to it currently being built with a UMD like structure.
|
|
36
|
+
// Once the build output is updated to be fully ESM, this can be removed.
|
|
37
|
+
allowedRequests.add('zone.js');
|
|
35
38
|
// Find all entry points that contain code (JS/TS)
|
|
36
39
|
const files = [];
|
|
37
40
|
for (const { entryPoint } of Object.values(metafile.outputs)) {
|
|
@@ -119,4 +122,4 @@ function createCommonJSModuleError(request, importer) {
|
|
|
119
122
|
};
|
|
120
123
|
return error;
|
|
121
124
|
}
|
|
122
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
125
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"commonjs-checker.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/browser-esbuild/commonjs-checker.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 { Metafile, PartialMessage } from 'esbuild';\n\n/**\n * Checks the input files of a build to determine if any of the files included\n * in the build are not ESM. ESM files can be tree-shaken and otherwise optimized\n * in ways that CommonJS and other module formats cannot. The esbuild metafile\n * information is used as the basis for the analysis as it contains information\n * for each input file including its respective format.\n *\n * If any allowed dependencies are provided via the `allowedCommonJsDependencies`\n * parameter, both the direct import and any deep imports will be ignored and no\n * diagnostic will be generated.\n *\n * If a module has been issued a diagnostic message, then all descendant modules\n * will not be checked. This prevents a potential massive amount of inactionable\n * messages since the initial module import is the cause of the problem.\n *\n * @param metafile An esbuild metafile object to check.\n * @param allowedCommonJsDependencies An optional list of allowed dependencies.\n * @returns Zero or more diagnostic messages for any non-ESM modules.\n */\nexport function checkCommonJSModules(\n  metafile: Metafile,\n  allowedCommonJsDependencies?: string[],\n): PartialMessage[] {\n  const messages: PartialMessage[] = [];\n  const allowedRequests = new Set(allowedCommonJsDependencies);\n\n  // Ignore Angular locale definitions which are currently UMD\n  allowedRequests.add('@angular/common/locales');\n\n  // Ignore zone.js due to it currently being built with a UMD like structure.\n  // Once the build output is updated to be fully ESM, this can be removed.\n  allowedRequests.add('zone.js');\n\n  // Find all entry points that contain code (JS/TS)\n  const files: string[] = [];\n  for (const { entryPoint } of Object.values(metafile.outputs)) {\n    if (!entryPoint) {\n      continue;\n    }\n    if (!isPathCode(entryPoint)) {\n      continue;\n    }\n\n    files.push(entryPoint);\n  }\n\n  // Track seen files so they are only analyzed once.\n  // Bundler runtime code is also ignored since it cannot be actionable.\n  const seenFiles = new Set<string>(['<runtime>']);\n\n  // Analyze the files present by walking the import graph\n  let currentFile: string | undefined;\n  while ((currentFile = files.shift())) {\n    const input = metafile.inputs[currentFile];\n\n    for (const imported of input.imports) {\n      // Ignore imports that were already seen or not originally in the code (bundler injected)\n      if (!imported.original || seenFiles.has(imported.path)) {\n        continue;\n      }\n      seenFiles.add(imported.path);\n\n      // Only check actual code files\n      if (!isPathCode(imported.path)) {\n        continue;\n      }\n\n      // Check if the import is ESM format and issue a diagnostic if the file is not allowed\n      if (metafile.inputs[imported.path].format !== 'esm') {\n        const request = imported.original;\n\n        let notAllowed = true;\n        if (allowedRequests.has(request)) {\n          notAllowed = false;\n        } else {\n          // Check for deep imports of allowed requests\n          for (const allowed of allowedRequests) {\n            if (request.startsWith(allowed + '/')) {\n              notAllowed = false;\n              break;\n            }\n          }\n        }\n\n        if (notAllowed) {\n          // Issue a diagnostic message and skip all descendants since they are also most\n          // likely not ESM but solved by addressing this import.\n          messages.push(createCommonJSModuleError(request, currentFile));\n          continue;\n        }\n      }\n\n      // Add the path so that its imports can be checked\n      files.push(imported.path);\n    }\n  }\n\n  return messages;\n}\n\n/**\n * Determines if a file path has an extension that is a JavaScript or TypeScript\n * code file.\n *\n * @param name A path to check for code file extensions.\n * @returns True, if a code file path; false, otherwise.\n */\nfunction isPathCode(name: string): boolean {\n  return /\\.[cm]?[jt]sx?$/.test(name);\n}\n\n/**\n * Creates an esbuild diagnostic message for a given non-ESM module request.\n *\n * @param request The requested non-ESM module name.\n * @param importer The path of the file containing the import.\n * @returns A message representing the diagnostic.\n */\nfunction createCommonJSModuleError(request: string, importer: string): PartialMessage {\n  const error = {\n    text: `Module '${request}' used by '${importer}' is not ESM`,\n    notes: [\n      {\n        text:\n          'CommonJS or AMD dependencies can cause optimization bailouts.\\n' +\n          'For more information see: https://angular.io/guide/build#configuring-commonjs-dependencies',\n      },\n    ],\n  };\n\n  return error;\n}\n"]}
|
|
@@ -17,6 +17,7 @@ export declare class SourceFileCache extends Map<string, ts.SourceFile> {
|
|
|
17
17
|
export interface CompilerPluginOptions {
|
|
18
18
|
sourcemap: boolean;
|
|
19
19
|
tsconfig: string;
|
|
20
|
+
jit?: boolean;
|
|
20
21
|
advancedOptimizations?: boolean;
|
|
21
22
|
thirdPartySourcemaps?: boolean;
|
|
22
23
|
fileReplacements?: Record<string, string>;
|