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.SecondaryEntryPointsMigration = void 0;
|
---|
11 | const schematics_1 = require("@angular/cdk/schematics");
|
---|
12 | const ts = require("typescript");
|
---|
13 | const module_specifiers_1 = require("../../../ng-update/typescript/module-specifiers");
|
---|
14 | const ONLY_SUBPACKAGE_FAILURE_STR = `Importing from "@angular/material" is deprecated. ` +
|
---|
15 | `Instead import from the entry-point the symbol belongs to.`;
|
---|
16 | const NO_IMPORT_NAMED_SYMBOLS_FAILURE_STR = `Imports from Angular Material should import ` +
|
---|
17 | `specific symbols rather than importing the entire library.`;
|
---|
18 | /**
|
---|
19 | * Regex for testing file paths against to determine if the file is from the
|
---|
20 | * Angular Material library.
|
---|
21 | */
|
---|
22 | const ANGULAR_MATERIAL_FILEPATH_REGEX = new RegExp(`${module_specifiers_1.materialModuleSpecifier}/(.*?)/`);
|
---|
23 | /**
|
---|
24 | * Mapping of Material symbol names to their module names. Used as a fallback if
|
---|
25 | * we didn't manage to resolve the module name of a symbol using the type checker.
|
---|
26 | */
|
---|
27 | const ENTRY_POINT_MAPPINGS = require('./material-symbols.json');
|
---|
28 | /**
|
---|
29 | * Migration that updates imports which refer to the primary Angular Material
|
---|
30 | * entry-point to use the appropriate secondary entry points (e.g. @angular/material/button).
|
---|
31 | */
|
---|
32 | class SecondaryEntryPointsMigration extends schematics_1.Migration {
|
---|
33 | constructor() {
|
---|
34 | super(...arguments);
|
---|
35 | this.printer = ts.createPrinter();
|
---|
36 | // Only enable this rule if the migration targets version 8. The primary
|
---|
37 | // entry-point of Material has been marked as deprecated in version 8.
|
---|
38 | this.enabled = this.targetVersion === schematics_1.TargetVersion.V8 || this.targetVersion === schematics_1.TargetVersion.V9;
|
---|
39 | }
|
---|
40 | visitNode(declaration) {
|
---|
41 | // Only look at import declarations.
|
---|
42 | if (!ts.isImportDeclaration(declaration) ||
|
---|
43 | !ts.isStringLiteralLike(declaration.moduleSpecifier)) {
|
---|
44 | return;
|
---|
45 | }
|
---|
46 | const importLocation = declaration.moduleSpecifier.text;
|
---|
47 | // If the import module is not @angular/material, skip the check.
|
---|
48 | if (importLocation !== module_specifiers_1.materialModuleSpecifier) {
|
---|
49 | return;
|
---|
50 | }
|
---|
51 | // If no import clause is found, or nothing is named as a binding in the
|
---|
52 | // import, add failure saying to import symbols in clause.
|
---|
53 | if (!declaration.importClause || !declaration.importClause.namedBindings) {
|
---|
54 | this.createFailureAtNode(declaration, NO_IMPORT_NAMED_SYMBOLS_FAILURE_STR);
|
---|
55 | return;
|
---|
56 | }
|
---|
57 | // All named bindings in import clauses must be named symbols, otherwise add
|
---|
58 | // failure saying to import symbols in clause.
|
---|
59 | if (!ts.isNamedImports(declaration.importClause.namedBindings)) {
|
---|
60 | this.createFailureAtNode(declaration, NO_IMPORT_NAMED_SYMBOLS_FAILURE_STR);
|
---|
61 | return;
|
---|
62 | }
|
---|
63 | // If no symbols are in the named bindings then add failure saying to
|
---|
64 | // import symbols in clause.
|
---|
65 | if (!declaration.importClause.namedBindings.elements.length) {
|
---|
66 | this.createFailureAtNode(declaration, NO_IMPORT_NAMED_SYMBOLS_FAILURE_STR);
|
---|
67 | return;
|
---|
68 | }
|
---|
69 | // Whether the existing import declaration is using a single quote module specifier.
|
---|
70 | const singleQuoteImport = declaration.moduleSpecifier.getText()[0] === `'`;
|
---|
71 | // Map which consists of secondary entry-points and import specifiers which are used
|
---|
72 | // within the current import declaration.
|
---|
73 | const importMap = new Map();
|
---|
74 | // Determine the subpackage each symbol in the namedBinding comes from.
|
---|
75 | for (const element of declaration.importClause.namedBindings.elements) {
|
---|
76 | const elementName = element.propertyName ? element.propertyName : element.name;
|
---|
77 | // Try to resolve the module name via the type checker, and if it fails, fall back to
|
---|
78 | // resolving it from our list of symbol to entry point mappings. Using the type checker is
|
---|
79 | // more accurate and doesn't require us to keep a list of symbols, but it won't work if
|
---|
80 | // the symbols don't exist anymore (e.g. after we remove the top-level @angular/material).
|
---|
81 | const moduleName = resolveModuleName(elementName, this.typeChecker) ||
|
---|
82 | ENTRY_POINT_MAPPINGS[elementName.text] || null;
|
---|
83 | if (!moduleName) {
|
---|
84 | this.createFailureAtNode(element, `"${element.getText()}" was not found in the Material library.`);
|
---|
85 | return;
|
---|
86 | }
|
---|
87 | // The module name where the symbol is defined e.g. card, dialog. The
|
---|
88 | // first capture group is contains the module name.
|
---|
89 | if (importMap.has(moduleName)) {
|
---|
90 | importMap.get(moduleName).push(element);
|
---|
91 | }
|
---|
92 | else {
|
---|
93 | importMap.set(moduleName, [element]);
|
---|
94 | }
|
---|
95 | }
|
---|
96 | // Transforms the import declaration into multiple import declarations that import
|
---|
97 | // the given symbols from the individual secondary entry-points. For example:
|
---|
98 | // import {MatCardModule, MatCardTitle} from '@angular/material/card';
|
---|
99 | // import {MatRadioModule} from '@angular/material/radio';
|
---|
100 | const newImportStatements = Array.from(importMap.entries())
|
---|
101 | .sort()
|
---|
102 | .map(([name, elements]) => {
|
---|
103 | const newImport = ts.createImportDeclaration(undefined, undefined, ts.createImportClause(undefined, ts.createNamedImports(elements)), createStringLiteral(`${module_specifiers_1.materialModuleSpecifier}/${name}`, singleQuoteImport));
|
---|
104 | return this.printer.printNode(ts.EmitHint.Unspecified, newImport, declaration.getSourceFile());
|
---|
105 | })
|
---|
106 | .join('\n');
|
---|
107 | // Without any import statements that were generated, we can assume that this was an empty
|
---|
108 | // import declaration. We still want to add a failure in order to make developers aware that
|
---|
109 | // importing from "@angular/material" is deprecated.
|
---|
110 | if (!newImportStatements) {
|
---|
111 | this.createFailureAtNode(declaration.moduleSpecifier, ONLY_SUBPACKAGE_FAILURE_STR);
|
---|
112 | return;
|
---|
113 | }
|
---|
114 | const filePath = this.fileSystem.resolve(declaration.moduleSpecifier.getSourceFile().fileName);
|
---|
115 | const recorder = this.fileSystem.edit(filePath);
|
---|
116 | // Perform the replacement that switches the primary entry-point import to
|
---|
117 | // the individual secondary entry-point imports.
|
---|
118 | recorder.remove(declaration.getStart(), declaration.getWidth());
|
---|
119 | recorder.insertRight(declaration.getStart(), newImportStatements);
|
---|
120 | }
|
---|
121 | }
|
---|
122 | exports.SecondaryEntryPointsMigration = SecondaryEntryPointsMigration;
|
---|
123 | /**
|
---|
124 | * Creates a string literal from the specified text.
|
---|
125 | * @param text Text of the string literal.
|
---|
126 | * @param singleQuotes Whether single quotes should be used when printing the literal node.
|
---|
127 | */
|
---|
128 | function createStringLiteral(text, singleQuotes) {
|
---|
129 | const literal = ts.createStringLiteral(text);
|
---|
130 | // See: https://github.com/microsoft/TypeScript/blob/master/src/compiler/utilities.ts#L584-L590
|
---|
131 | literal.singleQuote = singleQuotes;
|
---|
132 | return literal;
|
---|
133 | }
|
---|
134 | /** Gets the symbol that contains the value declaration of the given node. */
|
---|
135 | function getDeclarationSymbolOfNode(node, checker) {
|
---|
136 | const symbol = checker.getSymbolAtLocation(node);
|
---|
137 | // Symbols can be aliases of the declaration symbol. e.g. in named import specifiers.
|
---|
138 | // We need to resolve the aliased symbol back to the declaration symbol.
|
---|
139 | // tslint:disable-next-line:no-bitwise
|
---|
140 | if (symbol && (symbol.flags & ts.SymbolFlags.Alias) !== 0) {
|
---|
141 | return checker.getAliasedSymbol(symbol);
|
---|
142 | }
|
---|
143 | return symbol;
|
---|
144 | }
|
---|
145 | /** Tries to resolve the name of the Material module that a node is imported from. */
|
---|
146 | function resolveModuleName(node, typeChecker) {
|
---|
147 | var _a;
|
---|
148 | // Get the symbol for the named binding element. Note that we cannot determine the
|
---|
149 | // value declaration based on the type of the element as types are not necessarily
|
---|
150 | // specific to a given secondary entry-point (e.g. exports with the type of "string")
|
---|
151 | // would resolve to the module types provided by TypeScript itself.
|
---|
152 | const symbol = getDeclarationSymbolOfNode(node, typeChecker);
|
---|
153 | // If the symbol can't be found, or no declaration could be found within
|
---|
154 | // the symbol, add failure to report that the given symbol can't be found.
|
---|
155 | if (!symbol ||
|
---|
156 | !(symbol.valueDeclaration || (symbol.declarations && symbol.declarations.length !== 0))) {
|
---|
157 | return null;
|
---|
158 | }
|
---|
159 | // The filename for the source file of the node that contains the
|
---|
160 | // first declaration of the symbol. All symbol declarations must be
|
---|
161 | // part of a defining node, so parent can be asserted to be defined.
|
---|
162 | const resolvedNode = symbol.valueDeclaration || ((_a = symbol.declarations) === null || _a === void 0 ? void 0 : _a[0]);
|
---|
163 | if (resolvedNode === undefined) {
|
---|
164 | return null;
|
---|
165 | }
|
---|
166 | const sourceFile = resolvedNode.getSourceFile().fileName;
|
---|
167 | // File the module the symbol belongs to from a regex match of the
|
---|
168 | // filename. This will always match since only "@angular/material"
|
---|
169 | // elements are analyzed.
|
---|
170 | const matches = sourceFile.match(ANGULAR_MATERIAL_FILEPATH_REGEX);
|
---|
171 | return matches ? matches[1] : null;
|
---|
172 | }
|
---|
173 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"secondary-entry-points-migration.js","sourceRoot":"","sources":["../../../../../../../../../src/material/schematics/ng-update/migrations/package-imports-v8/secondary-entry-points-migration.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 {Migration, TargetVersion} from '@angular/cdk/schematics';\nimport * as ts from 'typescript';\nimport {materialModuleSpecifier} from '../../../ng-update/typescript/module-specifiers';\n\nconst ONLY_SUBPACKAGE_FAILURE_STR = `Importing from \"@angular/material\" is deprecated. ` +\n    `Instead import from the entry-point the symbol belongs to.`;\n\nconst NO_IMPORT_NAMED_SYMBOLS_FAILURE_STR = `Imports from Angular Material should import ` +\n    `specific symbols rather than importing the entire library.`;\n\n/**\n * Regex for testing file paths against to determine if the file is from the\n * Angular Material library.\n */\nconst ANGULAR_MATERIAL_FILEPATH_REGEX = new RegExp(`${materialModuleSpecifier}/(.*?)/`);\n\n/**\n * Mapping of Material symbol names to their module names. Used as a fallback if\n * we didn't manage to resolve the module name of a symbol using the type checker.\n */\nconst ENTRY_POINT_MAPPINGS: {[name: string]: string} = require('./material-symbols.json');\n\n/**\n * Migration that updates imports which refer to the primary Angular Material\n * entry-point to use the appropriate secondary entry points (e.g. @angular/material/button).\n */\nexport class SecondaryEntryPointsMigration extends Migration<null> {\n  printer = ts.createPrinter();\n\n  // Only enable this rule if the migration targets version 8. The primary\n  // entry-point of Material has been marked as deprecated in version 8.\n  enabled = this.targetVersion === TargetVersion.V8 || this.targetVersion === TargetVersion.V9;\n\n  override visitNode(declaration: ts.Node): void {\n    // Only look at import declarations.\n    if (!ts.isImportDeclaration(declaration) ||\n        !ts.isStringLiteralLike(declaration.moduleSpecifier)) {\n      return;\n    }\n\n    const importLocation = declaration.moduleSpecifier.text;\n    // If the import module is not @angular/material, skip the check.\n    if (importLocation !== materialModuleSpecifier) {\n      return;\n    }\n\n    // If no import clause is found, or nothing is named as a binding in the\n    // import, add failure saying to import symbols in clause.\n    if (!declaration.importClause || !declaration.importClause.namedBindings) {\n      this.createFailureAtNode(declaration, NO_IMPORT_NAMED_SYMBOLS_FAILURE_STR);\n      return;\n    }\n\n    // All named bindings in import clauses must be named symbols, otherwise add\n    // failure saying to import symbols in clause.\n    if (!ts.isNamedImports(declaration.importClause.namedBindings)) {\n      this.createFailureAtNode(declaration, NO_IMPORT_NAMED_SYMBOLS_FAILURE_STR);\n      return;\n    }\n\n    // If no symbols are in the named bindings then add failure saying to\n    // import symbols in clause.\n    if (!declaration.importClause.namedBindings.elements.length) {\n      this.createFailureAtNode(declaration, NO_IMPORT_NAMED_SYMBOLS_FAILURE_STR);\n      return;\n    }\n\n    // Whether the existing import declaration is using a single quote module specifier.\n    const singleQuoteImport = declaration.moduleSpecifier.getText()[0] === `'`;\n\n    // Map which consists of secondary entry-points and import specifiers which are used\n    // within the current import declaration.\n    const importMap = new Map<string, ts.ImportSpecifier[]>();\n\n    // Determine the subpackage each symbol in the namedBinding comes from.\n    for (const element of declaration.importClause.namedBindings.elements) {\n      const elementName = element.propertyName ? element.propertyName : element.name;\n\n      // Try to resolve the module name via the type checker, and if it fails, fall back to\n      // resolving it from our list of symbol to entry point mappings. Using the type checker is\n      // more accurate and doesn't require us to keep a list of symbols, but it won't work if\n      // the symbols don't exist anymore (e.g. after we remove the top-level @angular/material).\n      const moduleName = resolveModuleName(elementName, this.typeChecker) ||\n          ENTRY_POINT_MAPPINGS[elementName.text] || null;\n\n      if (!moduleName) {\n        this.createFailureAtNode(\n          element, `\"${element.getText()}\" was not found in the Material library.`);\n        return;\n      }\n\n        // The module name where the symbol is defined e.g. card, dialog. The\n        // first capture group is contains the module name.\n        if (importMap.has(moduleName)) {\n          importMap.get(moduleName)!.push(element);\n        } else {\n          importMap.set(moduleName, [element]);\n        }\n    }\n\n    // Transforms the import declaration into multiple import declarations that import\n    // the given symbols from the individual secondary entry-points. For example:\n    // import {MatCardModule, MatCardTitle} from '@angular/material/card';\n    // import {MatRadioModule} from '@angular/material/radio';\n    const newImportStatements =\n        Array.from(importMap.entries())\n            .sort()\n            .map(([name, elements]) => {\n              const newImport = ts.createImportDeclaration(\n                  undefined, undefined,\n                  ts.createImportClause(undefined, ts.createNamedImports(elements)),\n                  createStringLiteral(`${materialModuleSpecifier}/${name}`, singleQuoteImport));\n              return this.printer.printNode(\n                  ts.EmitHint.Unspecified, newImport, declaration.getSourceFile());\n            })\n            .join('\\n');\n\n    // Without any import statements that were generated, we can assume that this was an empty\n    // import declaration. We still want to add a failure in order to make developers aware that\n    // importing from \"@angular/material\" is deprecated.\n    if (!newImportStatements) {\n      this.createFailureAtNode(declaration.moduleSpecifier, ONLY_SUBPACKAGE_FAILURE_STR);\n      return;\n    }\n\n    const filePath = this.fileSystem.resolve(\n        declaration.moduleSpecifier.getSourceFile().fileName);\n    const recorder = this.fileSystem.edit(filePath);\n\n    // Perform the replacement that switches the primary entry-point import to\n    // the individual secondary entry-point imports.\n    recorder.remove(declaration.getStart(), declaration.getWidth());\n    recorder.insertRight(declaration.getStart(), newImportStatements);\n  }\n}\n\n/**\n * Creates a string literal from the specified text.\n * @param text Text of the string literal.\n * @param singleQuotes Whether single quotes should be used when printing the literal node.\n */\nfunction createStringLiteral(text: string, singleQuotes: boolean): ts.StringLiteral {\n  const literal = ts.createStringLiteral(text);\n  // See: https://github.com/microsoft/TypeScript/blob/master/src/compiler/utilities.ts#L584-L590\n  (literal as any).singleQuote = singleQuotes;\n  return literal;\n}\n\n/** Gets the symbol that contains the value declaration of the given node. */\nfunction getDeclarationSymbolOfNode(node: ts.Node, checker: ts.TypeChecker): ts.Symbol|undefined {\n  const symbol = checker.getSymbolAtLocation(node);\n\n  // Symbols can be aliases of the declaration symbol. e.g. in named import specifiers.\n  // We need to resolve the aliased symbol back to the declaration symbol.\n  // tslint:disable-next-line:no-bitwise\n  if (symbol && (symbol.flags & ts.SymbolFlags.Alias) !== 0) {\n    return checker.getAliasedSymbol(symbol);\n  }\n  return symbol;\n}\n\n\n/** Tries to resolve the name of the Material module that a node is imported from. */\nfunction resolveModuleName(node: ts.Identifier, typeChecker: ts.TypeChecker): string|null {\n  // Get the symbol for the named binding element. Note that we cannot determine the\n  // value declaration based on the type of the element as types are not necessarily\n  // specific to a given secondary entry-point (e.g. exports with the type of \"string\")\n  // would resolve to the module types provided by TypeScript itself.\n  const symbol = getDeclarationSymbolOfNode(node, typeChecker);\n\n  // If the symbol can't be found, or no declaration could be found within\n  // the symbol, add failure to report that the given symbol can't be found.\n  if (!symbol ||\n      !(symbol.valueDeclaration || (symbol.declarations && symbol.declarations.length !== 0))) {\n    return null;\n  }\n\n  // The filename for the source file of the node that contains the\n  // first declaration of the symbol. All symbol declarations must be\n  // part of a defining node, so parent can be asserted to be defined.\n  const resolvedNode = symbol.valueDeclaration || symbol.declarations?.[0];\n\n  if (resolvedNode === undefined) {\n    return null;\n  }\n\n  const sourceFile = resolvedNode.getSourceFile().fileName;\n\n  // File the module the symbol belongs to from a regex match of the\n  // filename. This will always match since only \"@angular/material\"\n  // elements are analyzed.\n  const matches = sourceFile.match(ANGULAR_MATERIAL_FILEPATH_REGEX);\n  return matches ? matches[1] : null;\n}\n"]} |
---|