This repository has been archived by the owner on Mar 25, 2021. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 889
/
noInferrableTypesRule.ts
129 lines (118 loc) · 5 KB
/
noInferrableTypesRule.ts
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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
/**
* @license
* Copyright 2015 Palantir Technologies, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { hasModifier } from "tsutils";
import * as ts from "typescript";
import * as Lint from "../index";
const OPTION_IGNORE_PARMS = "ignore-params";
const OPTION_IGNORE_PROPERTIES = "ignore-properties";
interface Options {
ignoreParameters: boolean;
ignoreProperties: boolean;
}
export class Rule extends Lint.Rules.AbstractRule {
/* tslint:disable:object-literal-sort-keys */
public static metadata: Lint.IRuleMetadata = {
ruleName: "no-inferrable-types",
description: "Disallows explicit type declarations for variables or parameters initialized to a number, string, or boolean.",
rationale: "Explicit types where they can be easily inferred by the compiler make code more verbose.",
optionsDescription: Lint.Utils.dedent`
Two arguments may be optionally provided:
* \`${OPTION_IGNORE_PARMS}\` allows specifying an inferrable type annotation for function params.
This can be useful when combining with the \`typedef\` rule.
* \`${OPTION_IGNORE_PROPERTIES}\` allows specifying an inferrable type annotation for class properties.`,
options: {
type: "array",
items: {
type: "string",
enum: [OPTION_IGNORE_PARMS, OPTION_IGNORE_PROPERTIES],
},
minLength: 0,
maxLength: 2,
},
hasFix: true,
optionExamples: [
true,
[true, OPTION_IGNORE_PARMS],
[true, OPTION_IGNORE_PARMS, OPTION_IGNORE_PROPERTIES],
],
type: "typescript",
typescriptOnly: true,
};
/* tslint:enable:object-literal-sort-keys */
public static FAILURE_STRING_FACTORY(type: string) {
return `Type ${type} trivially inferred from a ${type} literal, remove type annotation`;
}
public apply(sourceFile: ts.SourceFile): Lint.RuleFailure[] {
return this.applyWithWalker(new NoInferrableTypesWalker(sourceFile, this.ruleName, {
ignoreParameters: this.ruleArguments.indexOf(OPTION_IGNORE_PARMS) !== -1,
ignoreProperties: this.ruleArguments.indexOf(OPTION_IGNORE_PROPERTIES) !== -1,
}));
}
}
class NoInferrableTypesWalker extends Lint.AbstractWalker<Options> {
public walk(sourceFile: ts.SourceFile) {
const cb = (node: ts.Node): void => {
if (shouldCheck(node, this.options)) {
const { name, type, initializer } = node;
if (type !== undefined && initializer !== undefined
&& typeIsInferrable(type.kind, initializer)) {
const fix = Lint.Replacement.deleteFromTo(name.end, type.end);
this.addFailureAtNode(type, Rule.FAILURE_STRING_FACTORY(ts.tokenToString(type.kind)!), fix);
}
}
return ts.forEachChild(node, cb);
};
return ts.forEachChild(sourceFile, cb);
}
}
function shouldCheck(
node: ts.Node,
{ ignoreParameters, ignoreProperties }: Options,
): node is ts.ParameterDeclaration | ts.PropertyDeclaration | ts.VariableDeclaration {
switch (node.kind) {
case ts.SyntaxKind.Parameter:
return !ignoreParameters &&
!hasModifier(node.modifiers, ts.SyntaxKind.ReadonlyKeyword) &&
// "ignore-properties" also works for parameter properties
!(ignoreProperties && node.modifiers !== undefined);
case ts.SyntaxKind.PropertyDeclaration:
return !ignoreProperties && !hasModifier(node.modifiers, ts.SyntaxKind.ReadonlyKeyword);
case ts.SyntaxKind.VariableDeclaration:
return true;
default:
return false;
}
}
function typeIsInferrable(type: ts.SyntaxKind, initializer: ts.Expression): boolean {
switch (type) {
case ts.SyntaxKind.BooleanKeyword:
return initializer.kind === ts.SyntaxKind.TrueKeyword || initializer.kind === ts.SyntaxKind.FalseKeyword;
case ts.SyntaxKind.NumberKeyword:
return Lint.isNumeric(initializer);
case ts.SyntaxKind.StringKeyword:
switch (initializer.kind) {
case ts.SyntaxKind.StringLiteral:
case ts.SyntaxKind.NoSubstitutionTemplateLiteral:
case ts.SyntaxKind.TemplateExpression:
return true;
default:
return false;
}
default:
return false;
}
}