140 lines
5.9 KiB
TypeScript
140 lines
5.9 KiB
TypeScript
/**
|
|
* @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 {Replacement, RuleFailure, Rules} from 'tslint';
|
|
import * as ts from 'typescript';
|
|
|
|
import {getHelper, HelperFunction} from '../renderer-to-renderer2/helpers';
|
|
import {migrateExpression, replaceImport} from '../renderer-to-renderer2/migration';
|
|
import {findCoreImport, findRendererReferences} from '../renderer-to-renderer2/util';
|
|
|
|
/**
|
|
* TSLint rule that migrates from `Renderer` to `Renderer2`. More information on how it works:
|
|
* https://hackmd.angular.io/UTzUZTnPRA-cSa_4mHyfYw
|
|
*/
|
|
export class Rule extends Rules.TypedRule {
|
|
applyWithProgram(sourceFile: ts.SourceFile, program: ts.Program): RuleFailure[] {
|
|
const typeChecker = program.getTypeChecker();
|
|
const printer = ts.createPrinter();
|
|
const failures: RuleFailure[] = [];
|
|
const rendererImport = findCoreImport(sourceFile, 'Renderer');
|
|
|
|
// If there are no imports for the `Renderer`, we can exit early.
|
|
if (!rendererImport) {
|
|
return failures;
|
|
}
|
|
|
|
const {typedNodes, methodCalls, forwardRefs} =
|
|
findRendererReferences(sourceFile, typeChecker, rendererImport);
|
|
const helpersToAdd = new Set<HelperFunction>();
|
|
|
|
failures.push(this._getNamedImportsFailure(rendererImport, sourceFile, printer));
|
|
typedNodes.forEach(node => failures.push(this._getTypedNodeFailure(node, sourceFile)));
|
|
forwardRefs.forEach(node => failures.push(this._getIdentifierNodeFailure(node, sourceFile)));
|
|
|
|
methodCalls.forEach(call => {
|
|
const {failure, requiredHelpers} =
|
|
this._getMethodCallFailure(call, sourceFile, typeChecker, printer);
|
|
|
|
failures.push(failure);
|
|
|
|
if (requiredHelpers) {
|
|
requiredHelpers.forEach(helperName => helpersToAdd.add(helperName));
|
|
}
|
|
});
|
|
|
|
// Some of the methods can't be mapped directly to `Renderer2` and need extra logic around them.
|
|
// The safest way to do so is to declare helper functions similar to the ones emitted by TS
|
|
// which encapsulate the extra "glue" logic. We should only emit these functions once per
|
|
// file and only if they're needed.
|
|
if (helpersToAdd.size) {
|
|
failures.push(this._getHelpersFailure(helpersToAdd, sourceFile, printer));
|
|
}
|
|
|
|
return failures;
|
|
}
|
|
|
|
/** Gets a failure for an import of the Renderer. */
|
|
private _getNamedImportsFailure(
|
|
node: ts.NamedImports, sourceFile: ts.SourceFile, printer: ts.Printer): RuleFailure {
|
|
const replacementText = printer.printNode(
|
|
ts.EmitHint.Unspecified, replaceImport(node, 'Renderer', 'Renderer2'), sourceFile);
|
|
|
|
return new RuleFailure(
|
|
sourceFile, node.getStart(), node.getEnd(),
|
|
'Imports of deprecated Renderer are not allowed. Please use Renderer2 instead.',
|
|
this.ruleName, new Replacement(node.getStart(), node.getWidth(), replacementText));
|
|
}
|
|
|
|
/** Gets a failure for a typed node (e.g. function parameter or property). */
|
|
private _getTypedNodeFailure(
|
|
node: ts.ParameterDeclaration|ts.PropertyDeclaration|ts.AsExpression,
|
|
sourceFile: ts.SourceFile): RuleFailure {
|
|
const type = node.type!;
|
|
|
|
return new RuleFailure(
|
|
sourceFile, type.getStart(), type.getEnd(),
|
|
'References to deprecated Renderer are not allowed. Please use Renderer2 instead.',
|
|
this.ruleName, new Replacement(type.getStart(), type.getWidth(), 'Renderer2'));
|
|
}
|
|
|
|
/** Gets a failure for an identifier node. */
|
|
private _getIdentifierNodeFailure(node: ts.Identifier, sourceFile: ts.SourceFile): RuleFailure {
|
|
return new RuleFailure(
|
|
sourceFile, node.getStart(), node.getEnd(),
|
|
'References to deprecated Renderer are not allowed. Please use Renderer2 instead.',
|
|
this.ruleName, new Replacement(node.getStart(), node.getWidth(), 'Renderer2'));
|
|
}
|
|
|
|
/** Gets a failure for a Renderer method call. */
|
|
private _getMethodCallFailure(
|
|
call: ts.CallExpression, sourceFile: ts.SourceFile, typeChecker: ts.TypeChecker,
|
|
printer: ts.Printer): {failure: RuleFailure, requiredHelpers?: HelperFunction[]} {
|
|
const {node, requiredHelpers} = migrateExpression(call, typeChecker);
|
|
let fix: Replacement|undefined;
|
|
|
|
if (node) {
|
|
// If we migrated the node to a new expression, replace only the call expression.
|
|
fix = new Replacement(
|
|
call.getStart(), call.getWidth(),
|
|
printer.printNode(ts.EmitHint.Unspecified, node, sourceFile));
|
|
} else if (call.parent && ts.isExpressionStatement(call.parent)) {
|
|
// Otherwise if the call is inside an expression statement, drop the entire statement.
|
|
// This takes care of any trailing semicolons. We only need to drop nodes for cases like
|
|
// `setBindingDebugInfo` which have been noop for a while so they can be removed safely.
|
|
fix = new Replacement(call.parent.getStart(), call.parent.getWidth(), '');
|
|
}
|
|
|
|
return {
|
|
failure: new RuleFailure(
|
|
sourceFile, call.getStart(), call.getEnd(), 'Calls to Renderer methods are not allowed',
|
|
this.ruleName, fix),
|
|
requiredHelpers
|
|
};
|
|
}
|
|
|
|
/** Gets a failure that inserts the required helper functions at the bottom of the file. */
|
|
private _getHelpersFailure(
|
|
helpersToAdd: Set<HelperFunction>, sourceFile: ts.SourceFile,
|
|
printer: ts.Printer): RuleFailure {
|
|
const helpers: Replacement[] = [];
|
|
const endOfFile = sourceFile.endOfFileToken;
|
|
|
|
helpersToAdd.forEach(helperName => {
|
|
helpers.push(new Replacement(
|
|
endOfFile.getStart(), endOfFile.getWidth(), getHelper(helperName, sourceFile, printer)));
|
|
});
|
|
|
|
// Add a failure at the end of the file which we can use as an anchor to insert the helpers.
|
|
return new RuleFailure(
|
|
sourceFile, endOfFile.getStart(), endOfFile.getStart() + 1,
|
|
'File should contain Renderer helper functions. Run tslint with --fix to generate them.',
|
|
this.ruleName, helpers);
|
|
}
|
|
}
|