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.ImportManager = void 0;
|
---|
11 | const path_1 = require("path");
|
---|
12 | const ts = require("typescript");
|
---|
13 | /** Checks whether an analyzed import has the given import flag set. */
|
---|
14 | const hasFlag = (data, flag) => (data.state & flag) !== 0;
|
---|
15 | /**
|
---|
16 | * Import manager that can be used to add or remove TypeScript imports within source
|
---|
17 | * files. The manager ensures that multiple transformations are applied properly
|
---|
18 | * without shifted offsets and that existing imports are re-used.
|
---|
19 | */
|
---|
20 | class ImportManager {
|
---|
21 | constructor(_fileSystem, _printer) {
|
---|
22 | this._fileSystem = _fileSystem;
|
---|
23 | this._printer = _printer;
|
---|
24 | /** Map of source-files and their previously used identifier names. */
|
---|
25 | this._usedIdentifierNames = new Map();
|
---|
26 | /** Map of source files and their analyzed imports. */
|
---|
27 | this._importCache = new Map();
|
---|
28 | }
|
---|
29 | /**
|
---|
30 | * Analyzes the import of the specified source file if needed. In order to perform
|
---|
31 | * modifications to imports of a source file, we store all imports in memory and
|
---|
32 | * update the source file once all changes have been made. This is essential to
|
---|
33 | * ensure that we can re-use newly added imports and not break file offsets.
|
---|
34 | */
|
---|
35 | _analyzeImportsIfNeeded(sourceFile) {
|
---|
36 | if (this._importCache.has(sourceFile)) {
|
---|
37 | return this._importCache.get(sourceFile);
|
---|
38 | }
|
---|
39 | const result = [];
|
---|
40 | for (let node of sourceFile.statements) {
|
---|
41 | if (!ts.isImportDeclaration(node) || !ts.isStringLiteral(node.moduleSpecifier)) {
|
---|
42 | continue;
|
---|
43 | }
|
---|
44 | const moduleName = node.moduleSpecifier.text;
|
---|
45 | // Handles side-effect imports which do neither have a name or
|
---|
46 | // specifiers. e.g. `import "my-package";`
|
---|
47 | if (!node.importClause) {
|
---|
48 | result.push({ moduleName, node, state: 0 /* UNMODIFIED */ });
|
---|
49 | continue;
|
---|
50 | }
|
---|
51 | // Handles imports resolving to default exports of a module.
|
---|
52 | // e.g. `import moment from "moment";`
|
---|
53 | if (!node.importClause.namedBindings) {
|
---|
54 | result.push({ moduleName, node, name: node.importClause.name, state: 0 /* UNMODIFIED */ });
|
---|
55 | continue;
|
---|
56 | }
|
---|
57 | // Handles imports with individual symbol specifiers.
|
---|
58 | // e.g. `import {A, B, C} from "my-module";`
|
---|
59 | if (ts.isNamedImports(node.importClause.namedBindings)) {
|
---|
60 | result.push({
|
---|
61 | moduleName,
|
---|
62 | node,
|
---|
63 | specifiers: node.importClause.namedBindings.elements.map(el => ({ name: el.name, propertyName: el.propertyName })),
|
---|
64 | state: 0 /* UNMODIFIED */,
|
---|
65 | });
|
---|
66 | }
|
---|
67 | else {
|
---|
68 | // Handles namespaced imports. e.g. `import * as core from "my-pkg";`
|
---|
69 | result.push({
|
---|
70 | moduleName,
|
---|
71 | node,
|
---|
72 | name: node.importClause.namedBindings.name,
|
---|
73 | namespace: true,
|
---|
74 | state: 0 /* UNMODIFIED */,
|
---|
75 | });
|
---|
76 | }
|
---|
77 | }
|
---|
78 | this._importCache.set(sourceFile, result);
|
---|
79 | return result;
|
---|
80 | }
|
---|
81 | /**
|
---|
82 | * Checks whether the given specifier, which can be relative to the base path,
|
---|
83 | * matches the passed module name.
|
---|
84 | */
|
---|
85 | _isModuleSpecifierMatching(basePath, specifier, moduleName) {
|
---|
86 | return specifier.startsWith('.') ?
|
---|
87 | path_1.resolve(basePath, specifier) === path_1.resolve(basePath, moduleName) :
|
---|
88 | specifier === moduleName;
|
---|
89 | }
|
---|
90 | /** Deletes a given named binding import from the specified source file. */
|
---|
91 | deleteNamedBindingImport(sourceFile, symbolName, moduleName) {
|
---|
92 | const sourceDir = path_1.dirname(sourceFile.fileName);
|
---|
93 | const fileImports = this._analyzeImportsIfNeeded(sourceFile);
|
---|
94 | for (let importData of fileImports) {
|
---|
95 | if (!this._isModuleSpecifierMatching(sourceDir, importData.moduleName, moduleName) ||
|
---|
96 | !importData.specifiers) {
|
---|
97 | continue;
|
---|
98 | }
|
---|
99 | const specifierIndex = importData.specifiers.findIndex(d => (d.propertyName || d.name).text === symbolName);
|
---|
100 | if (specifierIndex !== -1) {
|
---|
101 | importData.specifiers.splice(specifierIndex, 1);
|
---|
102 | // if the import does no longer contain any specifiers after the removal of the
|
---|
103 | // given symbol, we can just mark the whole import for deletion. Otherwise, we mark
|
---|
104 | // it as modified so that it will be re-printed.
|
---|
105 | if (importData.specifiers.length === 0) {
|
---|
106 | importData.state |= 8 /* DELETED */;
|
---|
107 | }
|
---|
108 | else {
|
---|
109 | importData.state |= 2 /* MODIFIED */;
|
---|
110 | }
|
---|
111 | }
|
---|
112 | }
|
---|
113 | }
|
---|
114 | /** Deletes the import that matches the given import declaration if found. */
|
---|
115 | deleteImportByDeclaration(declaration) {
|
---|
116 | const fileImports = this._analyzeImportsIfNeeded(declaration.getSourceFile());
|
---|
117 | for (let importData of fileImports) {
|
---|
118 | if (importData.node === declaration) {
|
---|
119 | importData.state |= 8 /* DELETED */;
|
---|
120 | }
|
---|
121 | }
|
---|
122 | }
|
---|
123 | /**
|
---|
124 | * Adds an import to the given source file and returns the TypeScript expression that
|
---|
125 | * can be used to access the newly imported symbol.
|
---|
126 | *
|
---|
127 | * Whenever an import is added to a source file, it's recommended that the returned
|
---|
128 | * expression is used to reference th symbol. This is necessary because the symbol
|
---|
129 | * could be aliased if it would collide with existing imports in source file.
|
---|
130 | *
|
---|
131 | * @param sourceFile Source file to which the import should be added.
|
---|
132 | * @param symbolName Name of the symbol that should be imported. Can be null if
|
---|
133 | * the default export is requested.
|
---|
134 | * @param moduleName Name of the module of which the symbol should be imported.
|
---|
135 | * @param typeImport Whether the symbol is a type.
|
---|
136 | * @param ignoreIdentifierCollisions List of identifiers which can be ignored when
|
---|
137 | * the import manager checks for import collisions.
|
---|
138 | */
|
---|
139 | addImportToSourceFile(sourceFile, symbolName, moduleName, typeImport = false, ignoreIdentifierCollisions = []) {
|
---|
140 | const sourceDir = path_1.dirname(sourceFile.fileName);
|
---|
141 | const fileImports = this._analyzeImportsIfNeeded(sourceFile);
|
---|
142 | let existingImport = null;
|
---|
143 | for (let importData of fileImports) {
|
---|
144 | if (!this._isModuleSpecifierMatching(sourceDir, importData.moduleName, moduleName)) {
|
---|
145 | continue;
|
---|
146 | }
|
---|
147 | // If no symbol name has been specified, the default import is requested. In that
|
---|
148 | // case we search for non-namespace and non-specifier imports.
|
---|
149 | if (!symbolName && !importData.namespace && !importData.specifiers) {
|
---|
150 | return ts.createIdentifier(importData.name.text);
|
---|
151 | }
|
---|
152 | // In case a "Type" symbol is imported, we can't use namespace imports
|
---|
153 | // because these only export symbols available at runtime (no types)
|
---|
154 | if (importData.namespace && !typeImport) {
|
---|
155 | return ts.createPropertyAccess(ts.createIdentifier(importData.name.text), ts.createIdentifier(symbolName || 'default'));
|
---|
156 | }
|
---|
157 | else if (importData.specifiers && symbolName) {
|
---|
158 | const existingSpecifier = importData.specifiers.find(s => s.propertyName ? s.propertyName.text === symbolName : s.name.text === symbolName);
|
---|
159 | if (existingSpecifier) {
|
---|
160 | return ts.createIdentifier(existingSpecifier.name.text);
|
---|
161 | }
|
---|
162 | // In case the symbol could not be found in an existing import, we
|
---|
163 | // keep track of the import declaration as it can be updated to include
|
---|
164 | // the specified symbol name without having to create a new import.
|
---|
165 | existingImport = importData;
|
---|
166 | }
|
---|
167 | }
|
---|
168 | // If there is an existing import that matches the specified module, we
|
---|
169 | // just update the import specifiers to also import the requested symbol.
|
---|
170 | if (existingImport) {
|
---|
171 | const propertyIdentifier = ts.createIdentifier(symbolName);
|
---|
172 | const generatedUniqueIdentifier = this._getUniqueIdentifier(sourceFile, symbolName, ignoreIdentifierCollisions);
|
---|
173 | const needsGeneratedUniqueName = generatedUniqueIdentifier.text !== symbolName;
|
---|
174 | const importName = needsGeneratedUniqueName ? generatedUniqueIdentifier : propertyIdentifier;
|
---|
175 | existingImport.specifiers.push({
|
---|
176 | name: importName,
|
---|
177 | propertyName: needsGeneratedUniqueName ? propertyIdentifier : undefined,
|
---|
178 | });
|
---|
179 | existingImport.state |= 2 /* MODIFIED */;
|
---|
180 | if (hasFlag(existingImport, 8 /* DELETED */)) {
|
---|
181 | // unset the deleted flag if the import is pending deletion, but
|
---|
182 | // can now be used for the new imported symbol.
|
---|
183 | existingImport.state &= ~8 /* DELETED */;
|
---|
184 | }
|
---|
185 | return importName;
|
---|
186 | }
|
---|
187 | let identifier = null;
|
---|
188 | let newImport = null;
|
---|
189 | if (symbolName) {
|
---|
190 | const propertyIdentifier = ts.createIdentifier(symbolName);
|
---|
191 | const generatedUniqueIdentifier = this._getUniqueIdentifier(sourceFile, symbolName, ignoreIdentifierCollisions);
|
---|
192 | const needsGeneratedUniqueName = generatedUniqueIdentifier.text !== symbolName;
|
---|
193 | identifier = needsGeneratedUniqueName ? generatedUniqueIdentifier : propertyIdentifier;
|
---|
194 | const newImportDecl = ts.createImportDeclaration(undefined, undefined, ts.createImportClause(undefined, ts.createNamedImports([])), ts.createStringLiteral(moduleName));
|
---|
195 | newImport = {
|
---|
196 | moduleName,
|
---|
197 | node: newImportDecl,
|
---|
198 | specifiers: [{
|
---|
199 | propertyName: needsGeneratedUniqueName ? propertyIdentifier : undefined,
|
---|
200 | name: identifier
|
---|
201 | }],
|
---|
202 | state: 4 /* ADDED */,
|
---|
203 | };
|
---|
204 | }
|
---|
205 | else {
|
---|
206 | identifier =
|
---|
207 | this._getUniqueIdentifier(sourceFile, 'defaultExport', ignoreIdentifierCollisions);
|
---|
208 | const newImportDecl = ts.createImportDeclaration(undefined, undefined, ts.createImportClause(identifier, undefined), ts.createStringLiteral(moduleName));
|
---|
209 | newImport = {
|
---|
210 | moduleName,
|
---|
211 | node: newImportDecl,
|
---|
212 | name: identifier,
|
---|
213 | state: 4 /* ADDED */,
|
---|
214 | };
|
---|
215 | }
|
---|
216 | fileImports.push(newImport);
|
---|
217 | return identifier;
|
---|
218 | }
|
---|
219 | /**
|
---|
220 | * Applies the recorded changes in the update recorders of the corresponding source files.
|
---|
221 | * The changes are applied separately after all changes have been recorded because otherwise
|
---|
222 | * file offsets will change and the source files would need to be re-parsed after each change.
|
---|
223 | */
|
---|
224 | recordChanges() {
|
---|
225 | this._importCache.forEach((fileImports, sourceFile) => {
|
---|
226 | const recorder = this._fileSystem.edit(this._fileSystem.resolve(sourceFile.fileName));
|
---|
227 | const lastUnmodifiedImport = fileImports.reverse().find(i => i.state === 0 /* UNMODIFIED */);
|
---|
228 | const importStartIndex = lastUnmodifiedImport ? this._getEndPositionOfNode(lastUnmodifiedImport.node) : 0;
|
---|
229 | fileImports.forEach(importData => {
|
---|
230 | if (importData.state === 0 /* UNMODIFIED */) {
|
---|
231 | return;
|
---|
232 | }
|
---|
233 | if (hasFlag(importData, 8 /* DELETED */)) {
|
---|
234 | // Imports which do not exist in source file, can be just skipped as
|
---|
235 | // we do not need any replacement to delete the import.
|
---|
236 | if (!hasFlag(importData, 4 /* ADDED */)) {
|
---|
237 | recorder.remove(importData.node.getFullStart(), importData.node.getFullWidth());
|
---|
238 | }
|
---|
239 | return;
|
---|
240 | }
|
---|
241 | if (importData.specifiers) {
|
---|
242 | const namedBindings = importData.node.importClause.namedBindings;
|
---|
243 | const importSpecifiers = importData.specifiers.map(s => ts.createImportSpecifier(s.propertyName, s.name));
|
---|
244 | const updatedBindings = ts.updateNamedImports(namedBindings, importSpecifiers);
|
---|
245 | // In case an import has been added newly, we need to print the whole import
|
---|
246 | // declaration and insert it at the import start index. Otherwise, we just
|
---|
247 | // update the named bindings to not re-print the whole import (which could
|
---|
248 | // cause unnecessary formatting changes)
|
---|
249 | if (hasFlag(importData, 4 /* ADDED */)) {
|
---|
250 | const updatedImport = ts.updateImportDeclaration(importData.node, undefined, undefined, ts.createImportClause(undefined, updatedBindings), ts.createStringLiteral(importData.moduleName));
|
---|
251 | const newImportText = this._printer.printNode(ts.EmitHint.Unspecified, updatedImport, sourceFile);
|
---|
252 | recorder.insertLeft(importStartIndex, importStartIndex === 0 ? `${newImportText}\n` : `\n${newImportText}`);
|
---|
253 | return;
|
---|
254 | }
|
---|
255 | else if (hasFlag(importData, 2 /* MODIFIED */)) {
|
---|
256 | const newNamedBindingsText = this._printer.printNode(ts.EmitHint.Unspecified, updatedBindings, sourceFile);
|
---|
257 | recorder.remove(namedBindings.getStart(), namedBindings.getWidth());
|
---|
258 | recorder.insertRight(namedBindings.getStart(), newNamedBindingsText);
|
---|
259 | return;
|
---|
260 | }
|
---|
261 | }
|
---|
262 | else if (hasFlag(importData, 4 /* ADDED */)) {
|
---|
263 | const newImportText = this._printer.printNode(ts.EmitHint.Unspecified, importData.node, sourceFile);
|
---|
264 | recorder.insertLeft(importStartIndex, importStartIndex === 0 ? `${newImportText}\n` : `\n${newImportText}`);
|
---|
265 | return;
|
---|
266 | }
|
---|
267 | // we should never hit this, but we rather want to print a custom exception
|
---|
268 | // instead of just skipping imports silently.
|
---|
269 | throw Error('Unexpected import modification.');
|
---|
270 | });
|
---|
271 | });
|
---|
272 | }
|
---|
273 | /**
|
---|
274 | * Corrects the line and character position of a given node. Since nodes of
|
---|
275 | * source files are immutable and we sometimes make changes to the containing
|
---|
276 | * source file, the node position might shift (e.g. if we add a new import before).
|
---|
277 | *
|
---|
278 | * This method can be used to retrieve a corrected position of the given node. This
|
---|
279 | * is helpful when printing out error messages which should reflect the new state of
|
---|
280 | * source files.
|
---|
281 | */
|
---|
282 | correctNodePosition(node, offset, position) {
|
---|
283 | const sourceFile = node.getSourceFile();
|
---|
284 | if (!this._importCache.has(sourceFile)) {
|
---|
285 | return position;
|
---|
286 | }
|
---|
287 | const newPosition = Object.assign({}, position);
|
---|
288 | const fileImports = this._importCache.get(sourceFile);
|
---|
289 | for (let importData of fileImports) {
|
---|
290 | const fullEnd = importData.node.getFullStart() + importData.node.getFullWidth();
|
---|
291 | // Subtract or add lines based on whether an import has been deleted or removed
|
---|
292 | // before the actual node offset.
|
---|
293 | if (offset > fullEnd && hasFlag(importData, 8 /* DELETED */)) {
|
---|
294 | newPosition.line--;
|
---|
295 | }
|
---|
296 | else if (offset > fullEnd && hasFlag(importData, 4 /* ADDED */)) {
|
---|
297 | newPosition.line++;
|
---|
298 | }
|
---|
299 | }
|
---|
300 | return newPosition;
|
---|
301 | }
|
---|
302 | /**
|
---|
303 | * Returns an unique identifier name for the specified symbol name.
|
---|
304 | * @param sourceFile Source file to check for identifier collisions.
|
---|
305 | * @param symbolName Name of the symbol for which we want to generate an unique name.
|
---|
306 | * @param ignoreIdentifierCollisions List of identifiers which should be ignored when
|
---|
307 | * checking for identifier collisions in the given source file.
|
---|
308 | */
|
---|
309 | _getUniqueIdentifier(sourceFile, symbolName, ignoreIdentifierCollisions) {
|
---|
310 | if (this._isUniqueIdentifierName(sourceFile, symbolName, ignoreIdentifierCollisions)) {
|
---|
311 | this._recordUsedIdentifier(sourceFile, symbolName);
|
---|
312 | return ts.createIdentifier(symbolName);
|
---|
313 | }
|
---|
314 | let name = null;
|
---|
315 | let counter = 1;
|
---|
316 | do {
|
---|
317 | name = `${symbolName}_${counter++}`;
|
---|
318 | } while (!this._isUniqueIdentifierName(sourceFile, name, ignoreIdentifierCollisions));
|
---|
319 | this._recordUsedIdentifier(sourceFile, name);
|
---|
320 | return ts.createIdentifier(name);
|
---|
321 | }
|
---|
322 | /**
|
---|
323 | * Checks whether the specified identifier name is used within the given source file.
|
---|
324 | * @param sourceFile Source file to check for identifier collisions.
|
---|
325 | * @param name Name of the identifier which is checked for its uniqueness.
|
---|
326 | * @param ignoreIdentifierCollisions List of identifiers which should be ignored when
|
---|
327 | * checking for identifier collisions in the given source file.
|
---|
328 | */
|
---|
329 | _isUniqueIdentifierName(sourceFile, name, ignoreIdentifierCollisions) {
|
---|
330 | if (this._usedIdentifierNames.has(sourceFile) &&
|
---|
331 | this._usedIdentifierNames.get(sourceFile).indexOf(name) !== -1) {
|
---|
332 | return false;
|
---|
333 | }
|
---|
334 | // Walk through the source file and search for an identifier matching
|
---|
335 | // the given name. In that case, it's not guaranteed that this name
|
---|
336 | // is unique in the given declaration scope and we just return false.
|
---|
337 | const nodeQueue = [sourceFile];
|
---|
338 | while (nodeQueue.length) {
|
---|
339 | const node = nodeQueue.shift();
|
---|
340 | if (ts.isIdentifier(node) && node.text === name &&
|
---|
341 | !ignoreIdentifierCollisions.includes(node)) {
|
---|
342 | return false;
|
---|
343 | }
|
---|
344 | nodeQueue.push(...node.getChildren());
|
---|
345 | }
|
---|
346 | return true;
|
---|
347 | }
|
---|
348 | /**
|
---|
349 | * Records that the given identifier is used within the specified source file. This
|
---|
350 | * is necessary since we do not apply changes to source files per change, but still
|
---|
351 | * want to avoid conflicts with newly imported symbols.
|
---|
352 | */
|
---|
353 | _recordUsedIdentifier(sourceFile, identifierName) {
|
---|
354 | this._usedIdentifierNames.set(sourceFile, (this._usedIdentifierNames.get(sourceFile) || []).concat(identifierName));
|
---|
355 | }
|
---|
356 | /**
|
---|
357 | * Determines the full end of a given node. By default the end position of a node is
|
---|
358 | * before all trailing comments. This could mean that generated imports shift comments.
|
---|
359 | */
|
---|
360 | _getEndPositionOfNode(node) {
|
---|
361 | const nodeEndPos = node.getEnd();
|
---|
362 | const commentRanges = ts.getTrailingCommentRanges(node.getSourceFile().text, nodeEndPos);
|
---|
363 | if (!commentRanges || !commentRanges.length) {
|
---|
364 | return nodeEndPos;
|
---|
365 | }
|
---|
366 | return commentRanges[commentRanges.length - 1].end;
|
---|
367 | }
|
---|
368 | }
|
---|
369 | exports.ImportManager = ImportManager;
|
---|
370 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"import-manager.js","sourceRoot":"","sources":["../../../../../../../../../src/material/schematics/ng-update/migrations/hammer-gestures-v9/import-manager.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 {FileSystem} from '@angular/cdk/schematics';\nimport {dirname, resolve} from 'path';\nimport * as ts from 'typescript';\n\n// tslint:disable:no-bitwise\n\n/** Enum describing the possible states of an analyzed import. */\nconst enum ImportState {\n  UNMODIFIED = 0b0,\n  MODIFIED = 0b10,\n  ADDED = 0b100,\n  DELETED = 0b1000,\n}\n\n/** Interface describing an import specifier. */\ninterface ImportSpecifier {\n  name: ts.Identifier;\n  propertyName?: ts.Identifier;\n}\n\n/** Interface describing an analyzed import. */\ninterface AnalyzedImport {\n  node: ts.ImportDeclaration;\n  moduleName: string;\n  name?: ts.Identifier;\n  specifiers?: ImportSpecifier[];\n  namespace?: boolean;\n  state: ImportState;\n}\n\n/** Checks whether an analyzed import has the given import flag set. */\nconst hasFlag = (data: AnalyzedImport, flag: ImportState) => (data.state & flag) !== 0;\n\n/**\n * Import manager that can be used to add or remove TypeScript imports within source\n * files. The manager ensures that multiple transformations are applied properly\n * without shifted offsets and that existing imports are re-used.\n */\nexport class ImportManager {\n  /** Map of source-files and their previously used identifier names. */\n  private _usedIdentifierNames = new Map<ts.SourceFile, string[]>();\n\n  /** Map of source files and their analyzed imports. */\n  private _importCache = new Map<ts.SourceFile, AnalyzedImport[]>();\n\n  constructor(\n      private _fileSystem: FileSystem,\n      private _printer: ts.Printer) {}\n\n  /**\n   * Analyzes the import of the specified source file if needed. In order to perform\n   * modifications to imports of a source file, we store all imports in memory and\n   * update the source file once all changes have been made. This is essential to\n   * ensure that we can re-use newly added imports and not break file offsets.\n   */\n  private _analyzeImportsIfNeeded(sourceFile: ts.SourceFile): AnalyzedImport[] {\n    if (this._importCache.has(sourceFile)) {\n      return this._importCache.get(sourceFile)!;\n    }\n\n    const result: AnalyzedImport[] = [];\n    for (let node of sourceFile.statements) {\n      if (!ts.isImportDeclaration(node) || !ts.isStringLiteral(node.moduleSpecifier)) {\n        continue;\n      }\n\n      const moduleName = node.moduleSpecifier.text;\n\n      // Handles side-effect imports which do neither have a name or\n      // specifiers. e.g. `import \"my-package\";`\n      if (!node.importClause) {\n        result.push({moduleName, node, state: ImportState.UNMODIFIED});\n        continue;\n      }\n\n      // Handles imports resolving to default exports of a module.\n      // e.g. `import moment from \"moment\";`\n      if (!node.importClause.namedBindings) {\n        result.push(\n            {moduleName, node, name: node.importClause.name, state: ImportState.UNMODIFIED});\n        continue;\n      }\n\n      // Handles imports with individual symbol specifiers.\n      // e.g. `import {A, B, C} from \"my-module\";`\n      if (ts.isNamedImports(node.importClause.namedBindings)) {\n        result.push({\n          moduleName,\n          node,\n          specifiers: node.importClause.namedBindings.elements.map(\n              el => ({name: el.name, propertyName: el.propertyName})),\n          state: ImportState.UNMODIFIED,\n        });\n      } else {\n        // Handles namespaced imports. e.g. `import * as core from \"my-pkg\";`\n        result.push({\n          moduleName,\n          node,\n          name: node.importClause.namedBindings.name,\n          namespace: true,\n          state: ImportState.UNMODIFIED,\n        });\n      }\n    }\n    this._importCache.set(sourceFile, result);\n    return result;\n  }\n\n  /**\n   * Checks whether the given specifier, which can be relative to the base path,\n   * matches the passed module name.\n   */\n  private _isModuleSpecifierMatching(basePath: string, specifier: string, moduleName: string):\n      boolean {\n    return specifier.startsWith('.') ?\n        resolve(basePath, specifier) === resolve(basePath, moduleName) :\n        specifier === moduleName;\n  }\n\n  /** Deletes a given named binding import from the specified source file. */\n  deleteNamedBindingImport(sourceFile: ts.SourceFile, symbolName: string, moduleName: string) {\n    const sourceDir = dirname(sourceFile.fileName);\n    const fileImports = this._analyzeImportsIfNeeded(sourceFile);\n\n    for (let importData of fileImports) {\n      if (!this._isModuleSpecifierMatching(sourceDir, importData.moduleName, moduleName) ||\n          !importData.specifiers) {\n        continue;\n      }\n\n      const specifierIndex =\n          importData.specifiers.findIndex(d => (d.propertyName || d.name).text === symbolName);\n      if (specifierIndex !== -1) {\n        importData.specifiers.splice(specifierIndex, 1);\n        // if the import does no longer contain any specifiers after the removal of the\n        // given symbol, we can just mark the whole import for deletion. Otherwise, we mark\n        // it as modified so that it will be re-printed.\n        if (importData.specifiers.length === 0) {\n          importData.state |= ImportState.DELETED;\n        } else {\n          importData.state |= ImportState.MODIFIED;\n        }\n      }\n    }\n  }\n\n  /** Deletes the import that matches the given import declaration if found. */\n  deleteImportByDeclaration(declaration: ts.ImportDeclaration) {\n    const fileImports = this._analyzeImportsIfNeeded(declaration.getSourceFile());\n    for (let importData of fileImports) {\n      if (importData.node === declaration) {\n        importData.state |= ImportState.DELETED;\n      }\n    }\n  }\n\n  /**\n   * Adds an import to the given source file and returns the TypeScript expression that\n   * can be used to access the newly imported symbol.\n   *\n   * Whenever an import is added to a source file, it's recommended that the returned\n   * expression is used to reference th symbol. This is necessary because the symbol\n   * could be aliased if it would collide with existing imports in source file.\n   *\n   * @param sourceFile Source file to which the import should be added.\n   * @param symbolName Name of the symbol that should be imported. Can be null if\n   *    the default export is requested.\n   * @param moduleName Name of the module of which the symbol should be imported.\n   * @param typeImport Whether the symbol is a type.\n   * @param ignoreIdentifierCollisions List of identifiers which can be ignored when\n   *    the import manager checks for import collisions.\n   */\n  addImportToSourceFile(\n      sourceFile: ts.SourceFile, symbolName: string|null, moduleName: string, typeImport = false,\n      ignoreIdentifierCollisions: ts.Identifier[] = []): ts.Expression {\n    const sourceDir = dirname(sourceFile.fileName);\n    const fileImports = this._analyzeImportsIfNeeded(sourceFile);\n\n    let existingImport: AnalyzedImport|null = null;\n    for (let importData of fileImports) {\n      if (!this._isModuleSpecifierMatching(sourceDir, importData.moduleName, moduleName)) {\n        continue;\n      }\n\n      // If no symbol name has been specified, the default import is requested. In that\n      // case we search for non-namespace and non-specifier imports.\n      if (!symbolName && !importData.namespace && !importData.specifiers) {\n        return ts.createIdentifier(importData.name!.text);\n      }\n\n      // In case a \"Type\" symbol is imported, we can't use namespace imports\n      // because these only export symbols available at runtime (no types)\n      if (importData.namespace && !typeImport) {\n        return ts.createPropertyAccess(\n            ts.createIdentifier(importData.name!.text),\n            ts.createIdentifier(symbolName || 'default'));\n      } else if (importData.specifiers && symbolName) {\n        const existingSpecifier = importData.specifiers.find(\n            s => s.propertyName ? s.propertyName.text === symbolName : s.name.text === symbolName);\n\n        if (existingSpecifier) {\n          return ts.createIdentifier(existingSpecifier.name.text);\n        }\n\n        // In case the symbol could not be found in an existing import, we\n        // keep track of the import declaration as it can be updated to include\n        // the specified symbol name without having to create a new import.\n        existingImport = importData;\n      }\n    }\n\n    // If there is an existing import that matches the specified module, we\n    // just update the import specifiers to also import the requested symbol.\n    if (existingImport) {\n      const propertyIdentifier = ts.createIdentifier(symbolName!);\n      const generatedUniqueIdentifier =\n          this._getUniqueIdentifier(sourceFile, symbolName!, ignoreIdentifierCollisions);\n      const needsGeneratedUniqueName = generatedUniqueIdentifier.text !== symbolName;\n      const importName = needsGeneratedUniqueName ? generatedUniqueIdentifier : propertyIdentifier;\n\n      existingImport.specifiers!.push({\n        name: importName,\n        propertyName: needsGeneratedUniqueName ? propertyIdentifier : undefined,\n      });\n      existingImport.state |= ImportState.MODIFIED;\n\n      if (hasFlag(existingImport, ImportState.DELETED)) {\n        // unset the deleted flag if the import is pending deletion, but\n        // can now be used for the new imported symbol.\n        existingImport.state &= ~ImportState.DELETED;\n      }\n\n      return importName;\n    }\n\n    let identifier: ts.Identifier|null = null;\n    let newImport: AnalyzedImport|null = null;\n\n    if (symbolName) {\n      const propertyIdentifier = ts.createIdentifier(symbolName);\n      const generatedUniqueIdentifier =\n          this._getUniqueIdentifier(sourceFile, symbolName, ignoreIdentifierCollisions);\n      const needsGeneratedUniqueName = generatedUniqueIdentifier.text !== symbolName;\n      identifier = needsGeneratedUniqueName ? generatedUniqueIdentifier : propertyIdentifier;\n\n      const newImportDecl = ts.createImportDeclaration(\n          undefined, undefined, ts.createImportClause(undefined, ts.createNamedImports([])),\n          ts.createStringLiteral(moduleName));\n\n      newImport = {\n        moduleName,\n        node: newImportDecl,\n        specifiers: [{\n          propertyName: needsGeneratedUniqueName ? propertyIdentifier : undefined,\n          name: identifier\n        }],\n        state: ImportState.ADDED,\n      };\n    } else {\n      identifier =\n          this._getUniqueIdentifier(sourceFile, 'defaultExport', ignoreIdentifierCollisions);\n      const newImportDecl = ts.createImportDeclaration(\n          undefined, undefined, ts.createImportClause(identifier, undefined),\n          ts.createStringLiteral(moduleName));\n      newImport = {\n        moduleName,\n        node: newImportDecl,\n        name: identifier,\n        state: ImportState.ADDED,\n      };\n    }\n    fileImports.push(newImport);\n    return identifier;\n  }\n\n  /**\n   * Applies the recorded changes in the update recorders of the corresponding source files.\n   * The changes are applied separately after all changes have been recorded because otherwise\n   * file offsets will change and the source files would need to be re-parsed after each change.\n   */\n  recordChanges() {\n    this._importCache.forEach((fileImports, sourceFile) => {\n      const recorder = this._fileSystem.edit(this._fileSystem.resolve(sourceFile.fileName));\n      const lastUnmodifiedImport =\n          fileImports.reverse().find(i => i.state === ImportState.UNMODIFIED);\n      const importStartIndex =\n          lastUnmodifiedImport ? this._getEndPositionOfNode(lastUnmodifiedImport.node) : 0;\n\n      fileImports.forEach(importData => {\n        if (importData.state === ImportState.UNMODIFIED) {\n          return;\n        }\n\n        if (hasFlag(importData, ImportState.DELETED)) {\n          // Imports which do not exist in source file, can be just skipped as\n          // we do not need any replacement to delete the import.\n          if (!hasFlag(importData, ImportState.ADDED)) {\n            recorder.remove(importData.node.getFullStart(), importData.node.getFullWidth());\n          }\n          return;\n        }\n\n        if (importData.specifiers) {\n          const namedBindings = importData.node.importClause!.namedBindings as ts.NamedImports;\n          const importSpecifiers =\n              importData.specifiers.map(s => ts.createImportSpecifier(s.propertyName, s.name));\n          const updatedBindings = ts.updateNamedImports(namedBindings, importSpecifiers);\n\n          // In case an import has been added newly, we need to print the whole import\n          // declaration and insert it at the import start index. Otherwise, we just\n          // update the named bindings to not re-print the whole import (which could\n          // cause unnecessary formatting changes)\n          if (hasFlag(importData, ImportState.ADDED)) {\n            const updatedImport = ts.updateImportDeclaration(\n                importData.node, undefined, undefined,\n                ts.createImportClause(undefined, updatedBindings),\n                ts.createStringLiteral(importData.moduleName));\n            const newImportText =\n                this._printer.printNode(ts.EmitHint.Unspecified, updatedImport, sourceFile);\n            recorder.insertLeft(\n                importStartIndex,\n                importStartIndex === 0 ? `${newImportText}\\n` : `\\n${newImportText}`);\n            return;\n          } else if (hasFlag(importData, ImportState.MODIFIED)) {\n            const newNamedBindingsText =\n                this._printer.printNode(ts.EmitHint.Unspecified, updatedBindings, sourceFile);\n            recorder.remove(namedBindings.getStart(), namedBindings.getWidth());\n            recorder.insertRight(namedBindings.getStart(), newNamedBindingsText);\n            return;\n          }\n        } else if (hasFlag(importData, ImportState.ADDED)) {\n          const newImportText =\n              this._printer.printNode(ts.EmitHint.Unspecified, importData.node, sourceFile);\n          recorder.insertLeft(\n              importStartIndex,\n              importStartIndex === 0 ? `${newImportText}\\n` : `\\n${newImportText}`);\n          return;\n        }\n\n        // we should never hit this, but we rather want to print a custom exception\n        // instead of just skipping imports silently.\n        throw Error('Unexpected import modification.');\n      });\n    });\n  }\n\n  /**\n   * Corrects the line and character position of a given node. Since nodes of\n   * source files are immutable and we sometimes make changes to the containing\n   * source file, the node position might shift (e.g. if we add a new import before).\n   *\n   * This method can be used to retrieve a corrected position of the given node. This\n   * is helpful when printing out error messages which should reflect the new state of\n   * source files.\n   */\n  correctNodePosition(node: ts.Node, offset: number, position: ts.LineAndCharacter) {\n    const sourceFile = node.getSourceFile();\n\n    if (!this._importCache.has(sourceFile)) {\n      return position;\n    }\n\n    const newPosition: ts.LineAndCharacter = {...position};\n    const fileImports = this._importCache.get(sourceFile)!;\n\n    for (let importData of fileImports) {\n      const fullEnd = importData.node.getFullStart() + importData.node.getFullWidth();\n      // Subtract or add lines based on whether an import has been deleted or removed\n      // before the actual node offset.\n      if (offset > fullEnd && hasFlag(importData, ImportState.DELETED)) {\n        newPosition.line--;\n      } else if (offset > fullEnd && hasFlag(importData, ImportState.ADDED)) {\n        newPosition.line++;\n      }\n    }\n    return newPosition;\n  }\n\n  /**\n   * Returns an unique identifier name for the specified symbol name.\n   * @param sourceFile Source file to check for identifier collisions.\n   * @param symbolName Name of the symbol for which we want to generate an unique name.\n   * @param ignoreIdentifierCollisions List of identifiers which should be ignored when\n   *    checking for identifier collisions in the given source file.\n   */\n  private _getUniqueIdentifier(\n      sourceFile: ts.SourceFile, symbolName: string,\n      ignoreIdentifierCollisions: ts.Identifier[]): ts.Identifier {\n    if (this._isUniqueIdentifierName(sourceFile, symbolName, ignoreIdentifierCollisions)) {\n      this._recordUsedIdentifier(sourceFile, symbolName);\n      return ts.createIdentifier(symbolName);\n    }\n\n    let name: string|null = null;\n    let counter = 1;\n    do {\n      name = `${symbolName}_${counter++}`;\n    } while (!this._isUniqueIdentifierName(sourceFile, name, ignoreIdentifierCollisions));\n\n    this._recordUsedIdentifier(sourceFile, name!);\n    return ts.createIdentifier(name!);\n  }\n\n  /**\n   * Checks whether the specified identifier name is used within the given source file.\n   * @param sourceFile Source file to check for identifier collisions.\n   * @param name Name of the identifier which is checked for its uniqueness.\n   * @param ignoreIdentifierCollisions List of identifiers which should be ignored when\n   *    checking for identifier collisions in the given source file.\n   */\n  private _isUniqueIdentifierName(\n      sourceFile: ts.SourceFile, name: string, ignoreIdentifierCollisions: ts.Identifier[]) {\n    if (this._usedIdentifierNames.has(sourceFile) &&\n        this._usedIdentifierNames.get(sourceFile)!.indexOf(name) !== -1) {\n      return false;\n    }\n\n    // Walk through the source file and search for an identifier matching\n    // the given name. In that case, it's not guaranteed that this name\n    // is unique in the given declaration scope and we just return false.\n    const nodeQueue: ts.Node[] = [sourceFile];\n    while (nodeQueue.length) {\n      const node = nodeQueue.shift()!;\n      if (ts.isIdentifier(node) && node.text === name &&\n          !ignoreIdentifierCollisions.includes(node)) {\n        return false;\n      }\n      nodeQueue.push(...node.getChildren());\n    }\n    return true;\n  }\n\n  /**\n   * Records that the given identifier is used within the specified source file. This\n   * is necessary since we do not apply changes to source files per change, but still\n   * want to avoid conflicts with newly imported symbols.\n   */\n  private _recordUsedIdentifier(sourceFile: ts.SourceFile, identifierName: string) {\n    this._usedIdentifierNames.set(\n        sourceFile, (this._usedIdentifierNames.get(sourceFile) || []).concat(identifierName));\n  }\n\n  /**\n   * Determines the full end of a given node. By default the end position of a node is\n   * before all trailing comments. This could mean that generated imports shift comments.\n   */\n  private _getEndPositionOfNode(node: ts.Node) {\n    const nodeEndPos = node.getEnd();\n    const commentRanges = ts.getTrailingCommentRanges(node.getSourceFile().text, nodeEndPos);\n    if (!commentRanges || !commentRanges.length) {\n      return nodeEndPos;\n    }\n    return commentRanges[commentRanges.length - 1]!.end;\n  }\n}\n"]} |
---|