2019-06-10 13:08:54 -04:00
|
|
|
/**
|
|
|
|
* @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 {RuleFailure, Rules} from 'tslint';
|
|
|
|
import * as ts from 'typescript';
|
2019-06-18 05:28:23 -04:00
|
|
|
|
2019-10-05 17:41:37 -04:00
|
|
|
import {NgDefinitionCollector} from '../missing-injectable/definition_collector';
|
2019-06-18 05:28:23 -04:00
|
|
|
import {TslintUpdateRecorder} from '../missing-injectable/google3/tslint_update_recorder';
|
|
|
|
import {MissingInjectableTransform} from '../missing-injectable/transform';
|
|
|
|
|
2019-06-10 13:08:54 -04:00
|
|
|
|
2019-10-05 17:41:37 -04:00
|
|
|
|
2019-06-10 13:08:54 -04:00
|
|
|
/**
|
2019-10-05 17:41:37 -04:00
|
|
|
* TSLint rule that flags classes which are declared as providers in "NgModule",
|
|
|
|
* "Directive" or "Component" definitions while not being decorated with any
|
|
|
|
* Angular decorator (e.g. "@Injectable").
|
2019-06-10 13:08:54 -04:00
|
|
|
*/
|
|
|
|
export class Rule extends Rules.TypedRule {
|
|
|
|
applyWithProgram(sourceFile: ts.SourceFile, program: ts.Program): RuleFailure[] {
|
|
|
|
const ruleName = this.ruleName;
|
|
|
|
const typeChecker = program.getTypeChecker();
|
|
|
|
const sourceFiles = program.getSourceFiles().filter(
|
|
|
|
s => !s.isDeclarationFile && !program.isSourceFileFromExternalLibrary(s));
|
2019-10-05 17:41:37 -04:00
|
|
|
const definitionCollector = new NgDefinitionCollector(typeChecker);
|
2019-06-10 13:08:54 -04:00
|
|
|
const failures: RuleFailure[] = [];
|
|
|
|
|
2019-10-05 17:41:37 -04:00
|
|
|
// Analyze source files by detecting all "NgModule", "Directive" or
|
|
|
|
// "Component" definitions.
|
|
|
|
sourceFiles.forEach(sourceFile => definitionCollector.visitNode(sourceFile));
|
2019-06-10 13:08:54 -04:00
|
|
|
|
2019-10-05 17:41:37 -04:00
|
|
|
const {resolvedModules, resolvedDirectives} = definitionCollector;
|
2019-06-10 13:08:54 -04:00
|
|
|
const transformer = new MissingInjectableTransform(typeChecker, getUpdateRecorder);
|
|
|
|
const updateRecorders = new Map<ts.SourceFile, TslintUpdateRecorder>();
|
|
|
|
|
2019-10-05 17:41:37 -04:00
|
|
|
[...transformer.migrateModules(resolvedModules),
|
|
|
|
...transformer.migrateDirectives(resolvedDirectives),
|
|
|
|
].forEach(({message, node}) => {
|
|
|
|
// Only report failures for the current source file that is visited.
|
|
|
|
if (node.getSourceFile() === sourceFile) {
|
|
|
|
failures.push(new RuleFailure(node.getSourceFile(), node.getStart(), 0, message, ruleName));
|
|
|
|
}
|
2019-06-10 13:08:54 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
// Record the changes collected in the import manager and NgModule manager.
|
|
|
|
transformer.recordChanges();
|
|
|
|
|
|
|
|
if (updateRecorders.has(sourceFile)) {
|
|
|
|
failures.push(...updateRecorders.get(sourceFile) !.failures);
|
|
|
|
}
|
|
|
|
|
|
|
|
return failures;
|
|
|
|
|
|
|
|
/** Gets the update recorder for the specified source file. */
|
|
|
|
function getUpdateRecorder(sourceFile: ts.SourceFile): TslintUpdateRecorder {
|
|
|
|
if (updateRecorders.has(sourceFile)) {
|
|
|
|
return updateRecorders.get(sourceFile) !;
|
|
|
|
}
|
|
|
|
const recorder = new TslintUpdateRecorder(ruleName, sourceFile);
|
|
|
|
updateRecorders.set(sourceFile, recorder);
|
|
|
|
return recorder;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|