-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathanalyzer.mjs
executable file
·114 lines (114 loc) · 17.3 KB
/
analyzer.mjs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
/**
* @license
* Copyright Google LLC
*
* 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
*/
import { readFileSync } from 'fs';
import { dirname, join, resolve } from 'path';
import ts from 'typescript';
import { getFileStatus } from './file_system.js';
import { getModuleReferences } from './parser.js';
/** Default extensions that the analyzer uses for resolving imports. */
const DEFAULT_EXTENSIONS = ['ts', 'js', 'd.ts'];
/**
* Analyzer that can be used to detect import cycles within source files. It supports
* custom module resolution, source file caching and collects unresolved specifiers.
*/
export class Analyzer {
constructor(resolveModuleFn, ignoreTypeOnlyChecks = false, extensions = DEFAULT_EXTENSIONS) {
this.resolveModuleFn = resolveModuleFn;
this.extensions = extensions;
this._sourceFileCache = new Map();
this.unresolvedModules = new Set();
this.unresolvedFiles = new Map();
this._ignoreTypeOnlyChecks = !!ignoreTypeOnlyChecks;
}
/** Finds all cycles in the specified source file. */
findCycles(sf, visited = new WeakSet(), path = []) {
const previousIndex = path.indexOf(sf);
// If the given node is already part of the current path, then a cycle has
// been found. Add the reference chain which represents the cycle to the results.
if (previousIndex !== -1) {
return [path.slice(previousIndex)];
}
// If the node has already been visited, then it's not necessary to go check its edges
// again. Cycles would have been already detected and collected in the first check.
if (visited.has(sf)) {
return [];
}
path.push(sf);
visited.add(sf);
// Go through all edges, which are determined through import/exports, and collect cycles.
const result = [];
for (const ref of getModuleReferences(sf, this._ignoreTypeOnlyChecks)) {
const targetFile = this._resolveImport(ref, sf.fileName);
if (targetFile !== null) {
result.push(...this.findCycles(this.getSourceFile(targetFile), visited, path.slice()));
}
}
return result;
}
/** Gets the TypeScript source file of the specified path. */
getSourceFile(filePath) {
const resolvedPath = resolve(filePath);
if (this._sourceFileCache.has(resolvedPath)) {
return this._sourceFileCache.get(resolvedPath);
}
const fileContent = readFileSync(resolvedPath, 'utf8');
const sourceFile = ts.createSourceFile(resolvedPath, fileContent, ts.ScriptTarget.Latest, false);
this._sourceFileCache.set(resolvedPath, sourceFile);
return sourceFile;
}
/** Resolves the given import specifier with respect to the specified containing file path. */
_resolveImport(specifier, containingFilePath) {
if (specifier.charAt(0) === '.') {
const resolvedPath = this._resolveFileSpecifier(specifier, containingFilePath);
if (resolvedPath === null) {
this._trackUnresolvedFileImport(specifier, containingFilePath);
}
return resolvedPath;
}
if (this.resolveModuleFn) {
const targetFile = this.resolveModuleFn(specifier);
if (targetFile !== null) {
const resolvedPath = this._resolveFileSpecifier(targetFile);
if (resolvedPath !== null) {
return resolvedPath;
}
}
}
this.unresolvedModules.add(specifier);
return null;
}
/** Tracks the given file import as unresolved. */
_trackUnresolvedFileImport(specifier, originFilePath) {
if (!this.unresolvedFiles.has(originFilePath)) {
this.unresolvedFiles.set(originFilePath, [specifier]);
}
this.unresolvedFiles.get(originFilePath).push(specifier);
}
/** Resolves the given import specifier to the corresponding source file. */
_resolveFileSpecifier(specifier, containingFilePath) {
const importFullPath = containingFilePath !== undefined ? join(dirname(containingFilePath), specifier) : specifier;
const stat = getFileStatus(importFullPath);
if (stat && stat.isFile()) {
return importFullPath;
}
for (const extension of this.extensions) {
const pathWithExtension = `${importFullPath}.${extension}`;
const withExtensionStat = getFileStatus(pathWithExtension);
if (withExtensionStat?.isFile()) {
return pathWithExtension;
}
}
// Directories should be considered last. TypeScript first looks for source files, then
// falls back to directories if no file with appropriate extension could be found.
if (stat && stat.isDirectory()) {
return this._resolveFileSpecifier(join(importFullPath, 'index'));
}
return null;
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"analyzer.js","sourceRoot":"","sources":["../../../../../ng-dev/ts-circular-dependencies/analyzer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC\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 {readFileSync} from 'fs';\nimport {dirname, join, resolve} from 'path';\nimport ts from 'typescript';\nimport {CircularDependenciesParserOptions} from './config.js';\n\nimport {getFileStatus} from './file_system.js';\nimport {getModuleReferences} from './parser.js';\n\nexport type ModuleResolver = (specifier: string) => string | null;\n\n/**\n * Reference chains describe a sequence of source files which are connected through imports.\n * e.g. `file_a.ts` imports `file_b.ts`, whereas `file_b.ts` imports `file_c.ts`. The reference\n * chain data structure could be used to represent this import sequence.\n */\nexport type ReferenceChain<T = ts.SourceFile> = T[];\n\n/** Default extensions that the analyzer uses for resolving imports. */\nconst DEFAULT_EXTENSIONS = ['ts', 'js', 'd.ts'];\n\n/**\n * Analyzer that can be used to detect import cycles within source files. It supports\n * custom module resolution, source file caching and collects unresolved specifiers.\n */\nexport class Analyzer {\n  private _sourceFileCache = new Map<string, ts.SourceFile>();\n\n  private _ignoreTypeOnlyChecks: boolean;\n\n  unresolvedModules = new Set<string>();\n  unresolvedFiles = new Map<string, string[]>();\n\n  constructor(\n    public resolveModuleFn?: ModuleResolver,\n    ignoreTypeOnlyChecks: boolean = false,\n    public extensions: string[] = DEFAULT_EXTENSIONS,\n  ) {\n    this._ignoreTypeOnlyChecks = !!ignoreTypeOnlyChecks;\n  }\n\n  /** Finds all cycles in the specified source file. */\n  findCycles(\n    sf: ts.SourceFile,\n    visited = new WeakSet<ts.SourceFile>(),\n    path: ReferenceChain = [],\n  ): ReferenceChain[] {\n    const previousIndex = path.indexOf(sf);\n    // If the given node is already part of the current path, then a cycle has\n    // been found. Add the reference chain which represents the cycle to the results.\n    if (previousIndex !== -1) {\n      return [path.slice(previousIndex)];\n    }\n    // If the node has already been visited, then it's not necessary to go check its edges\n    // again. Cycles would have been already detected and collected in the first check.\n    if (visited.has(sf)) {\n      return [];\n    }\n    path.push(sf);\n    visited.add(sf);\n    // Go through all edges, which are determined through import/exports, and collect cycles.\n    const result: ReferenceChain[] = [];\n    for (const ref of getModuleReferences(sf, this._ignoreTypeOnlyChecks)) {\n      const targetFile = this._resolveImport(ref, sf.fileName);\n      if (targetFile !== null) {\n        result.push(...this.findCycles(this.getSourceFile(targetFile), visited, path.slice()));\n      }\n    }\n    return result;\n  }\n\n  /** Gets the TypeScript source file of the specified path. */\n  getSourceFile(filePath: string): ts.SourceFile {\n    const resolvedPath = resolve(filePath);\n    if (this._sourceFileCache.has(resolvedPath)) {\n      return this._sourceFileCache.get(resolvedPath)!;\n    }\n    const fileContent = readFileSync(resolvedPath, 'utf8');\n    const sourceFile = ts.createSourceFile(\n      resolvedPath,\n      fileContent,\n      ts.ScriptTarget.Latest,\n      false,\n    );\n    this._sourceFileCache.set(resolvedPath, sourceFile);\n    return sourceFile;\n  }\n\n  /** Resolves the given import specifier with respect to the specified containing file path. */\n  private _resolveImport(specifier: string, containingFilePath: string): string | null {\n    if (specifier.charAt(0) === '.') {\n      const resolvedPath = this._resolveFileSpecifier(specifier, containingFilePath);\n      if (resolvedPath === null) {\n        this._trackUnresolvedFileImport(specifier, containingFilePath);\n      }\n      return resolvedPath;\n    }\n    if (this.resolveModuleFn) {\n      const targetFile = this.resolveModuleFn(specifier);\n      if (targetFile !== null) {\n        const resolvedPath = this._resolveFileSpecifier(targetFile);\n        if (resolvedPath !== null) {\n          return resolvedPath;\n        }\n      }\n    }\n    this.unresolvedModules.add(specifier);\n    return null;\n  }\n\n  /** Tracks the given file import as unresolved. */\n  private _trackUnresolvedFileImport(specifier: string, originFilePath: string) {\n    if (!this.unresolvedFiles.has(originFilePath)) {\n      this.unresolvedFiles.set(originFilePath, [specifier]);\n    }\n    this.unresolvedFiles.get(originFilePath)!.push(specifier);\n  }\n\n  /** Resolves the given import specifier to the corresponding source file. */\n  private _resolveFileSpecifier(specifier: string, containingFilePath?: string): string | null {\n    const importFullPath =\n      containingFilePath !== undefined ? join(dirname(containingFilePath), specifier) : specifier;\n    const stat = getFileStatus(importFullPath);\n    if (stat && stat.isFile()) {\n      return importFullPath;\n    }\n    for (const extension of this.extensions) {\n      const pathWithExtension = `${importFullPath}.${extension}`;\n      const withExtensionStat = getFileStatus(pathWithExtension);\n      if (withExtensionStat?.isFile()) {\n        return pathWithExtension;\n      }\n    }\n    // Directories should be considered last. TypeScript first looks for source files, then\n    // falls back to directories if no file with appropriate extension could be found.\n    if (stat && stat.isDirectory()) {\n      return this._resolveFileSpecifier(join(importFullPath, 'index'));\n    }\n    return null;\n  }\n}\n"]}