UNPKG

@ngtools/webpack

Version:

Webpack plugin that AoT compiles your Angular components and modules.

158 lines • 18 kB
"use strict"; /** * @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 __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.elideImports = void 0; const ts = __importStar(require("typescript")); // Remove imports for which all identifiers have been removed. // Needs type checker, and works even if it's not the first transformer. // Works by removing imports for symbols whose identifiers have all been removed. // Doesn't use the `symbol.declarations` because that previous transforms might have removed nodes // but the type checker doesn't know. // See https://github.com/Microsoft/TypeScript/issues/17552 for more information. function elideImports(sourceFile, removedNodes, getTypeChecker, compilerOptions) { const importNodeRemovals = new Set(); if (removedNodes.length === 0) { return importNodeRemovals; } const typeChecker = getTypeChecker(); // Collect all imports and used identifiers const usedSymbols = new Set(); const imports = []; ts.forEachChild(sourceFile, function visit(node) { var _a; // Skip removed nodes. if (removedNodes.includes(node)) { return; } // Consider types for 'implements' as unused. // A HeritageClause token can also be an 'AbstractKeyword' // which in that case we should not elide the import. if (ts.isHeritageClause(node) && node.token === ts.SyntaxKind.ImplementsKeyword) { return; } // Record import and skip if (ts.isImportDeclaration(node)) { if (!((_a = node.importClause) === null || _a === void 0 ? void 0 : _a.isTypeOnly)) { imports.push(node); } return; } // Type reference imports do not need to be emitted when emitDecoratorMetadata is disabled. if (ts.isTypeReferenceNode(node) && !compilerOptions.emitDecoratorMetadata) { return; } let symbol; switch (node.kind) { case ts.SyntaxKind.Identifier: const parent = node.parent; if (parent && ts.isShorthandPropertyAssignment(parent)) { const shorthandSymbol = typeChecker.getShorthandAssignmentValueSymbol(parent); if (shorthandSymbol) { symbol = shorthandSymbol; } } else { symbol = typeChecker.getSymbolAtLocation(node); } break; case ts.SyntaxKind.ExportSpecifier: symbol = typeChecker.getExportSpecifierLocalTargetSymbol(node); break; case ts.SyntaxKind.ShorthandPropertyAssignment: symbol = typeChecker.getShorthandAssignmentValueSymbol(node); break; } if (symbol) { usedSymbols.add(symbol); } ts.forEachChild(node, visit); }); if (imports.length === 0) { return importNodeRemovals; } const isUnused = (node) => { // Do not remove JSX factory imports if (node.text === compilerOptions.jsxFactory) { return false; } const symbol = typeChecker.getSymbolAtLocation(node); return symbol && !usedSymbols.has(symbol); }; for (const node of imports) { if (!node.importClause) { // "import 'abc';" continue; } const namedBindings = node.importClause.namedBindings; if (namedBindings && ts.isNamespaceImport(namedBindings)) { // "import * as XYZ from 'abc';" if (isUnused(namedBindings.name)) { importNodeRemovals.add(node); } } else { const specifierNodeRemovals = []; let clausesCount = 0; // "import { XYZ, ... } from 'abc';" if (namedBindings && ts.isNamedImports(namedBindings)) { let removedClausesCount = 0; clausesCount += namedBindings.elements.length; for (const specifier of namedBindings.elements) { if (specifier.isTypeOnly || isUnused(specifier.name)) { removedClausesCount++; // in case we don't have any more namedImports we should remove the parent ie the {} const nodeToRemove = clausesCount === removedClausesCount ? specifier.parent : specifier; specifierNodeRemovals.push(nodeToRemove); } } } // "import XYZ from 'abc';" if (node.importClause.name) { clausesCount++; if (node.importClause.isTypeOnly || isUnused(node.importClause.name)) { specifierNodeRemovals.push(node.importClause.name); } } if (specifierNodeRemovals.length === clausesCount) { importNodeRemovals.add(node); } else { for (const specifierNodeRemoval of specifierNodeRemovals) { importNodeRemovals.add(specifierNodeRemoval); } } } } return importNodeRemovals; } exports.elideImports = elideImports; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elide_imports.js","sourceRoot":"","sources":["../../../../../../../../packages/ngtools/webpack/src/transformers/elide_imports.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 * as ts from 'typescript';\n\n// Remove imports for which all identifiers have been removed.\n// Needs type checker, and works even if it's not the first transformer.\n// Works by removing imports for symbols whose identifiers have all been removed.\n// Doesn't use the `symbol.declarations` because that previous transforms might have removed nodes\n// but the type checker doesn't know.\n// See https://github.com/Microsoft/TypeScript/issues/17552 for more information.\nexport function elideImports(\n  sourceFile: ts.SourceFile,\n  removedNodes: ts.Node[],\n  getTypeChecker: () => ts.TypeChecker,\n  compilerOptions: ts.CompilerOptions,\n): Set<ts.Node> {\n  const importNodeRemovals = new Set<ts.Node>();\n\n  if (removedNodes.length === 0) {\n    return importNodeRemovals;\n  }\n\n  const typeChecker = getTypeChecker();\n\n  // Collect all imports and used identifiers\n  const usedSymbols = new Set<ts.Symbol>();\n  const imports: ts.ImportDeclaration[] = [];\n\n  ts.forEachChild(sourceFile, function visit(node) {\n    // Skip removed nodes.\n    if (removedNodes.includes(node)) {\n      return;\n    }\n\n    // Consider types for 'implements' as unused.\n    // A HeritageClause token can also be an 'AbstractKeyword'\n    // which in that case we should not elide the import.\n    if (ts.isHeritageClause(node) && node.token === ts.SyntaxKind.ImplementsKeyword) {\n      return;\n    }\n\n    // Record import and skip\n    if (ts.isImportDeclaration(node)) {\n      if (!node.importClause?.isTypeOnly) {\n        imports.push(node);\n      }\n\n      return;\n    }\n\n    // Type reference imports do not need to be emitted when emitDecoratorMetadata is disabled.\n    if (ts.isTypeReferenceNode(node) && !compilerOptions.emitDecoratorMetadata) {\n      return;\n    }\n\n    let symbol: ts.Symbol | undefined;\n    switch (node.kind) {\n      case ts.SyntaxKind.Identifier:\n        const parent = node.parent;\n        if (parent && ts.isShorthandPropertyAssignment(parent)) {\n          const shorthandSymbol = typeChecker.getShorthandAssignmentValueSymbol(parent);\n          if (shorthandSymbol) {\n            symbol = shorthandSymbol;\n          }\n        } else {\n          symbol = typeChecker.getSymbolAtLocation(node);\n        }\n        break;\n      case ts.SyntaxKind.ExportSpecifier:\n        symbol = typeChecker.getExportSpecifierLocalTargetSymbol(node as ts.ExportSpecifier);\n        break;\n      case ts.SyntaxKind.ShorthandPropertyAssignment:\n        symbol = typeChecker.getShorthandAssignmentValueSymbol(node);\n        break;\n    }\n\n    if (symbol) {\n      usedSymbols.add(symbol);\n    }\n\n    ts.forEachChild(node, visit);\n  });\n\n  if (imports.length === 0) {\n    return importNodeRemovals;\n  }\n\n  const isUnused = (node: ts.Identifier) => {\n    // Do not remove JSX factory imports\n    if (node.text === compilerOptions.jsxFactory) {\n      return false;\n    }\n\n    const symbol = typeChecker.getSymbolAtLocation(node);\n\n    return symbol && !usedSymbols.has(symbol);\n  };\n\n  for (const node of imports) {\n    if (!node.importClause) {\n      // \"import 'abc';\"\n      continue;\n    }\n\n    const namedBindings = node.importClause.namedBindings;\n\n    if (namedBindings && ts.isNamespaceImport(namedBindings)) {\n      // \"import * as XYZ from 'abc';\"\n      if (isUnused(namedBindings.name)) {\n        importNodeRemovals.add(node);\n      }\n    } else {\n      const specifierNodeRemovals = [];\n      let clausesCount = 0;\n\n      // \"import { XYZ, ... } from 'abc';\"\n      if (namedBindings && ts.isNamedImports(namedBindings)) {\n        let removedClausesCount = 0;\n        clausesCount += namedBindings.elements.length;\n\n        for (const specifier of namedBindings.elements) {\n          if (specifier.isTypeOnly || isUnused(specifier.name)) {\n            removedClausesCount++;\n            // in case we don't have any more namedImports we should remove the parent ie the {}\n            const nodeToRemove =\n              clausesCount === removedClausesCount ? specifier.parent : specifier;\n\n            specifierNodeRemovals.push(nodeToRemove);\n          }\n        }\n      }\n\n      // \"import XYZ from 'abc';\"\n      if (node.importClause.name) {\n        clausesCount++;\n\n        if (node.importClause.isTypeOnly || isUnused(node.importClause.name)) {\n          specifierNodeRemovals.push(node.importClause.name);\n        }\n      }\n\n      if (specifierNodeRemovals.length === clausesCount) {\n        importNodeRemovals.add(node);\n      } else {\n        for (const specifierNodeRemoval of specifierNodeRemovals) {\n          importNodeRemovals.add(specifierNodeRemoval);\n        }\n      }\n    }\n  }\n\n  return importNodeRemovals;\n}\n"]}