build: allow custom module resolution for ts-circular-deps tests (#36226)

Currently the `ts-circular-deps` tool uses a hard-coded module resolver
that only works in the `angular/angular` repository.

If the tool is consumed in other repositories through the shared
dev-infra package, the module resolution won't work, and a few
resolvable imports (usually cross-entry-points) are accidentally
skipped. For each test, the resolution might differ, so tests can
now configure their module resolution in a configuration file.

Note that we intentionally don't rely on tsconfig's for module
resolution as parsing their mappings rather complicates the
circular dependency tool. Additionally, not every test has a
corresponding tsconfig file.

Also, hard-coding mappings to `@angular/*` while accepting a
path to the packages folder would work, but it would mean
that the circular deps tool is no longer self-contained. Rather,
and also for better flexibility, a custom resolver should be
specified.

PR Close #36226
This commit is contained in:
Paul Gschwendtner 2020-03-24 15:43:54 +01:00 committed by Alex Rickabaugh
parent b07b6edc2a
commit 44acf6734b
4 changed files with 117 additions and 58 deletions

View File

@ -0,0 +1,59 @@
/**
* @license
* Copyright Google Inc. 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
*/
import {dirname, isAbsolute, resolve} from 'path';
import {ModuleResolver} from './analyzer';
/** Configuration for a circular dependencies test. */
export interface CircularDependenciesTestConfig {
/** Base directory used for shortening paths in the golden file. */
baseDir: string;
/** Path to the golden file that is used for checking and approving. */
goldenFile: string;
/** Glob that resolves source files which should be checked. */
glob: string
/**
* Optional module resolver function that can be used to resolve modules
* to absolute file paths.
*/
resolveModule?: ModuleResolver;
/**
* Optional command that will be displayed if the golden check failed. This can be used
* to consistently use script aliases for checking/approving the golden.
*/
approveCommand?: string;
}
/**
* Loads the configuration for the circular dependencies test. If the config cannot be
* loaded, an error will be printed and the process exists with a non-zero exit code.
*/
export function loadTestConfig(configPath: string): CircularDependenciesTestConfig {
const configBaseDir = dirname(configPath);
const resolveRelativePath = (relativePath: string) => resolve(configBaseDir, relativePath);
try {
const config = require(configPath) as CircularDependenciesTestConfig;
if (!isAbsolute(config.baseDir)) {
config.baseDir = resolveRelativePath(config.baseDir);
}
if (!isAbsolute(config.goldenFile)) {
config.goldenFile = resolveRelativePath(config.goldenFile);
}
if (!isAbsolute(config.glob)) {
config.glob = resolveRelativePath(config.glob);
}
return config;
} catch (e) {
console.error('Could not load test configuration file at: ' + configPath);
console.error(`Failed with: ${e.message}`);
process.exit(1);
}
}

View File

@ -9,7 +9,7 @@
import {existsSync, readFileSync, writeFileSync} from 'fs'; import {existsSync, readFileSync, writeFileSync} from 'fs';
import {sync as globSync} from 'glob'; import {sync as globSync} from 'glob';
import {join, relative, resolve} from 'path'; import {isAbsolute, relative, resolve} from 'path';
import * as ts from 'typescript'; import * as ts from 'typescript';
import * as yargs from 'yargs'; import * as yargs from 'yargs';
import chalk from 'chalk'; import chalk from 'chalk';
@ -17,56 +17,36 @@ import chalk from 'chalk';
import {Analyzer, ReferenceChain} from './analyzer'; import {Analyzer, ReferenceChain} from './analyzer';
import {compareGoldens, convertReferenceChainToGolden, Golden} from './golden'; import {compareGoldens, convertReferenceChainToGolden, Golden} from './golden';
import {convertPathToForwardSlash} from './file_system'; import {convertPathToForwardSlash} from './file_system';
import {loadTestConfig, CircularDependenciesTestConfig} from './config';
import {getRepoBaseDir} from '../utils/config';
const projectDir = getRepoBaseDir();
const packagesDir = join(projectDir, 'packages/');
// The default glob does not capture deprecated packages such as http, or the webworker platform.
const defaultGlob =
join(packagesDir, '!(http|platform-webworker|platform-webworker-dynamic)/**/*.ts');
if (require.main === module) { if (require.main === module) {
const {_: command, goldenFile, glob, baseDir, warnings} = const {_: command, config: configArg, warnings} =
yargs.help() yargs.help()
.strict() .strict()
.command('check <goldenFile>', 'Checks if the circular dependencies have changed.') .command('check', 'Checks if the circular dependencies have changed.')
.command('approve <goldenFile>', 'Approves the current circular dependencies.') .command('approve', 'Approves the current circular dependencies.')
.demandCommand() .demandCommand()
.option( .option(
'approve', 'config',
{type: 'boolean', description: 'Approves the current circular dependencies.'}) {type: 'string', demandOption: true, description: 'Path to the configuration file.'})
.option('warnings', {type: 'boolean', description: 'Prints all warnings.'}) .option('warnings', {type: 'boolean', description: 'Prints all warnings.'})
.option('base-dir', {
type: 'string',
description: 'Base directory used for shortening paths in the golden file.',
default: projectDir,
defaultDescription: 'Project directory'
})
.option('glob', {
type: 'string',
description: 'Glob that matches source files which should be checked.',
default: defaultGlob,
defaultDescription: 'All release packages'
})
.argv; .argv;
const configPath = isAbsolute(configArg) ? configArg : resolve(configArg);
const config = loadTestConfig(configPath);
const isApprove = command.includes('approve'); const isApprove = command.includes('approve');
process.exit(main(baseDir, isApprove, goldenFile, glob, warnings)); process.exit(main(isApprove, config, warnings));
} }
/** /**
* Runs the ts-circular-dependencies tool. * Runs the ts-circular-dependencies tool.
* @param baseDir Base directory which is used to build up relative file paths in goldens.
* @param approve Whether the detected circular dependencies should be approved. * @param approve Whether the detected circular dependencies should be approved.
* @param goldenFile Path to the golden file. * @param config Configuration for the current circular dependencies test.
* @param glob Glob that is used to collect all source files which should be checked/approved. * @param printWarnings Whether warnings should be printed out.
* @param printWarnings Whether warnings should be printed. Warnings for unresolved modules/files
* are not printed by default.
* @returns Status code. * @returns Status code.
*/ */
export function main( export function main(
baseDir: string, approve: boolean, goldenFile: string, glob: string, approve: boolean, config: CircularDependenciesTestConfig, printWarnings: boolean): number {
printWarnings: boolean): number { const {baseDir, goldenFile, glob, resolveModule, approveCommand} = config;
const analyzer = new Analyzer(resolveModule); const analyzer = new Analyzer(resolveModule);
const cycles: ReferenceChain[] = []; const cycles: ReferenceChain[] = [];
const checkedNodes = new WeakSet<ts.SourceFile>(); const checkedNodes = new WeakSet<ts.SourceFile>();
@ -90,17 +70,21 @@ export function main(
return 1; return 1;
} }
const warningsCount = analyzer.unresolvedFiles.size + analyzer.unresolvedModules.size;
// By default, warnings for unresolved files or modules are not printed. This is because // By default, warnings for unresolved files or modules are not printed. This is because
// it's common that third-party modules are not resolved/visited. Also generated files // it's common that third-party modules are not resolved/visited. Also generated files
// from the View Engine compiler (i.e. factories, summaries) cannot be resolved. // from the View Engine compiler (i.e. factories, summaries) cannot be resolved.
if (printWarnings && if (printWarnings && warningsCount !== 0) {
(analyzer.unresolvedFiles.size !== 0 || analyzer.unresolvedModules.size !== 0)) { console.info(chalk.yellow('⚠ The following imports could not be resolved:'));
console.info(chalk.yellow('The following imports could not be resolved:'));
analyzer.unresolvedModules.forEach(specifier => console.info(`${specifier}`)); analyzer.unresolvedModules.forEach(specifier => console.info(`${specifier}`));
analyzer.unresolvedFiles.forEach((value, key) => { analyzer.unresolvedFiles.forEach((value, key) => {
console.info(`${getRelativePath(baseDir, key)}`); console.info(`${getRelativePath(baseDir, key)}`);
value.forEach(specifier => console.info(` ${specifier}`)); value.forEach(specifier => console.info(` ${specifier}`));
}); });
} else {
console.info(chalk.yellow(`${warningsCount} imports could not be resolved.`));
console.info(chalk.yellow(` Please rerun with "--warnings" to inspect unresolved imports.`));
} }
const expected: Golden = JSON.parse(readFileSync(goldenFile, 'utf8')); const expected: Golden = JSON.parse(readFileSync(goldenFile, 'utf8'));
@ -122,16 +106,12 @@ export function main(
chalk.yellow(` Fixed circular dependencies that need to be removed from the golden:`)); chalk.yellow(` Fixed circular dependencies that need to be removed from the golden:`));
fixedCircularDeps.forEach(c => console.error(`${convertReferenceChainToString(c)}`)); fixedCircularDeps.forEach(c => console.error(`${convertReferenceChainToString(c)}`));
console.info(); console.info();
// Print the command for updating the golden. Note that we hard-code the script name for if (approveCommand) {
// approving default packages golden in `goldens/`. We cannot infer the script name passed to console.info(chalk.yellow(` Please approve the new golden with: ${approveCommand}`));
// Yarn automatically since script are launched in a child process where `argv0` is different.
if (resolve(goldenFile) === resolve(projectDir, 'goldens/packages-circular-deps.json')) {
console.info(
chalk.yellow(` Please approve the new golden with: yarn ts-circular-deps:approve`));
} else { } else {
console.info(chalk.yellow( console.info(chalk.yellow(
` Please update the golden. The following command can be ` + ` Please update the golden. The following command can be ` +
`run: yarn ts-circular-deps approve ${getRelativePath(baseDir, goldenFile)}.`)); `run: yarn ts-circular-deps approve ${getRelativePath(process.cwd(), goldenFile)}.`));
} }
} }
return 1; return 1;
@ -146,14 +126,3 @@ function getRelativePath(baseDir: string, path: string) {
function convertReferenceChainToString(chain: ReferenceChain<string>) { function convertReferenceChainToString(chain: ReferenceChain<string>) {
return chain.join(' → '); return chain.join(' → ');
} }
/**
* Custom module resolver that maps specifiers starting with `@angular/` to the
* local packages folder.
*/
function resolveModule(specifier: string) {
if (specifier.startsWith('@angular/')) {
return packagesDir + specifier.substr('@angular/'.length);
}
return null;
}

View File

@ -35,9 +35,9 @@
"tslint": "tsc -p tools/tsconfig.json && tslint -c tslint.json \"+(packages|modules|scripts|tools)/**/*.+(js|ts)\"", "tslint": "tsc -p tools/tsconfig.json && tslint -c tslint.json \"+(packages|modules|scripts|tools)/**/*.+(js|ts)\"",
"public-api:check": "node goldens/public-api/manage.js test", "public-api:check": "node goldens/public-api/manage.js test",
"public-api:update": "node goldens/public-api/manage.js accept", "public-api:update": "node goldens/public-api/manage.js accept",
"ts-circular-deps": "ts-node dev-infra/ts-circular-dependencies/index.ts", "ts-circular-deps": "ts-node dev-infra/ts-circular-dependencies/index.ts --config ./packages/circular-deps-test.conf.js",
"ts-circular-deps:check": "yarn -s ts-circular-deps check ./goldens/packages-circular-deps.json", "ts-circular-deps:check": "yarn -s ts-circular-deps check",
"ts-circular-deps:approve": "yarn -s ts-circular-deps approve ./goldens/packages-circular-deps.json" "ts-circular-deps:approve": "yarn -s ts-circular-deps approve"
}, },
"// 1": "dependencies are used locally and by bazel", "// 1": "dependencies are used locally and by bazel",
"dependencies": { "dependencies": {

View File

@ -0,0 +1,31 @@
/**
* @license
* Copyright Google Inc. 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
*/
const path = require('path');
module.exports = {
baseDir: '../',
goldenFile: '../goldens/packages-circular-deps.json',
// The test should not capture deprecated packages such as `http`, or the `webworker` platform.
glob: `./!(http|platform-webworker|platform-webworker-dynamic)/**/*.ts`,
// Command that will be displayed if the golden needs to be updated.
approveCommand: 'yarn ts-circular-deps:approve',
resolveModule: resolveModule
};
/**
* Custom module resolver that maps specifiers starting with `@angular/` to the
* local packages folder. This ensures that cross package/entry-point dependencies
* can be detected.
*/
function resolveModule(specifier) {
if (specifier.startsWith('@angular/')) {
return path.join(__dirname, specifier.substr('@angular/'.length));
}
return null;
}