Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Unused identifiers compiler code #9200

Merged
merged 44 commits into from
Jun 24, 2016
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
44 commits
Select commit Hold shift + click to select a range
617b819
Code changes to update references of the Identifiers
sarangan12 May 24, 2016
a8bc8b4
Added code for handling function, method and coonstructor level local…
sarangan12 May 26, 2016
01966ba
Rebased with origin master
sarangan12 May 26, 2016
1e5ca92
Code changes to handle unused private variables, private methods and …
sarangan12 May 27, 2016
890d178
Code changes to handle namespace level elements
sarangan12 Jun 1, 2016
69dbeea
Code changes to handle unimplemented interfaces
sarangan12 Jun 1, 2016
c78d0e2
Code to optimize the d.ts check
sarangan12 Jun 1, 2016
107b369
Correct Code change to handle the parameters for methods inside inter…
sarangan12 Jun 1, 2016
481baa3
Fix for lint error
sarangan12 Jun 2, 2016
cd1ce0f
Remove Trailing whitespace
sarangan12 Jun 2, 2016
a38149d
Code changes to handle interface implementations
sarangan12 Jun 3, 2016
0571c19
Changes to display the error position correctly
sarangan12 Jun 3, 2016
e17ed58
Compiler Test Cases
sarangan12 Jun 3, 2016
2b7f3a7
Merge remote-tracking branch 'origin' into UpdateReferences
sarangan12 Jun 6, 2016
5a34352
Adding condition to ignore constructor parameters
sarangan12 Jun 15, 2016
93b7490
Removing unnecessary tests
sarangan12 Jun 15, 2016
7aba626
Additional changes for compiler code
sarangan12 Jun 15, 2016
ed5052d
Additional changes to handle constructor scenario
sarangan12 Jun 15, 2016
f5cdc9c
Fixing the consolidated case
sarangan12 Jun 15, 2016
8c3c7b1
Changed logic to search for private instead of public
sarangan12 Jun 15, 2016
dfad7cc
Response to PR Comments
sarangan12 Jun 15, 2016
c325625
Merge remote-tracking branch 'origin' into UnusedIdentifiersCompilerCode
sarangan12 Jun 15, 2016
6a711bc
Changed the error code in test cases as result of merge with master
sarangan12 Jun 15, 2016
d62a43f
Adding the missing file
sarangan12 Jun 16, 2016
043a625
Adding the missing file II
sarangan12 Jun 16, 2016
8f9d4ae
Response to PR comments
sarangan12 Jun 20, 2016
f15448a
Merge remote-tracking branch 'origin' into UnusedIdentifiersCompilerCode
sarangan12 Jun 20, 2016
c82453f
Code changes for checking unused imports
sarangan12 Jun 20, 2016
bcd6fc4
Test Cases for Unused Imports
sarangan12 Jun 20, 2016
49385f4
Response to PR comments
sarangan12 Jun 21, 2016
5993015
Code change specific to position of Import Declaration
sarangan12 Jun 21, 2016
f464f92
Code change for handling the position for unused import
sarangan12 Jun 21, 2016
3b5f8d2
New scenarios for handling parameters in lambda function, type parame…
sarangan12 Jun 22, 2016
ed282d7
Additional scenarios based on PR comments
sarangan12 Jun 22, 2016
e502ba0
Removing a redundant check
sarangan12 Jun 22, 2016
0e2e43d
Added ambient check to imports and typeparatmeter reporting
sarangan12 Jun 23, 2016
d6c2bcd
Merge remote-tracking branch 'origin' into UnusedIdentifiersCompilerCode
sarangan12 Jun 23, 2016
f93c6c8
Added one more scenario to handle type parameters
sarangan12 Jun 24, 2016
4521058
Added new scenario for TypeParameter on Interface
sarangan12 Jun 24, 2016
5eb7153
Refactoring the code
sarangan12 Jun 24, 2016
1401506
Merge remote-tracking branch 'origin' into UnusedIdentifiersCompilerCode
sarangan12 Jun 24, 2016
5361e5f
Added scenario to handle private class elements declared in constructor.
sarangan12 Jun 24, 2016
7fc4616
Minor change to erro reporting
sarangan12 Jun 24, 2016
9753d09
Merge remote-tracking branch 'origin' into UnusedIdentifiersCompilerCode
sarangan12 Jun 24, 2016
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
35 changes: 16 additions & 19 deletions src/compiler/checker.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14447,10 +14447,8 @@ namespace ts {
error(local.valueDeclaration, Diagnostics._0_is_declared_but_never_used, key);
}
else if (local.valueDeclaration.kind === SyntaxKind.Parameter && compilerOptions.noUnusedParameters) {
if (local.valueDeclaration.modifiers) {
if (getCombinedNodeFlags(local.valueDeclaration) & NodeFlags.Private) {
error(local.valueDeclaration, Diagnostics._0_is_declared_but_never_used, key);
}
if (getCombinedNodeFlags(local.valueDeclaration) & NodeFlags.Private) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

do not think you need this condition, the getCombinedNodeFlags should be sufficient.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed. Removed the condition

error(local.valueDeclaration, Diagnostics._0_is_declared_but_never_used, key);
}
else {
error(local.valueDeclaration, Diagnostics._0_is_declared_but_never_used, key);
Expand All @@ -14462,7 +14460,7 @@ namespace ts {
}
}

function checkUnusedModuleLocals(node: ModuleDeclaration): void {
function checkUnusedModuleLocals(node: ModuleDeclaration | SourceFile): void {
if (compilerOptions.noUnusedLocals && !isInAmbientContext(node)) {
for (const key in node.locals) {
if (hasProperty(node.locals, key)) {
Expand All @@ -14471,13 +14469,12 @@ namespace ts {
if ((local.valueDeclaration && local.valueDeclaration.kind)) {
error(local.valueDeclaration, Diagnostics._0_is_declared_but_never_used, key);
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

no need to check kind in local.valueDeclaration.kind. all nodes have kinds.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Removed it

else if (local.declarations &&
(local.declarations[0].kind === SyntaxKind.InterfaceDeclaration || local.declarations[0].kind === SyntaxKind.ModuleDeclaration)) {
error(local.declarations[0], Diagnostics._0_is_declared_but_never_used, key);
}
else if (local.declarations && local.declarations[0].kind === SyntaxKind.ImportEqualsDeclaration) {
error(local.declarations[0].name, Diagnostics._0_is_declared_but_never_used, key);
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why do we need these checks? why not just

forEach(local.declarations, d => error(d, Diagnostics._0_is_declared_but_never_used, key));

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have already tried this one. But I feel this might not work. The reason is that ImportEqualsDeclaration is a special case in which we need to use local.declarations[0].name instead of local.declarations[0].

Scenario:
module A {
export class Calculator {
public handelChar() {
}
}
}

module B {
import a = A;
}

Here 'a' must be indicated as error instead of "import a = A". But, I have optimized it a little so the condition does not look long.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i would say all errors should be on the declaration.name if it exists.

else if (local.declarations) {
error(local.declarations[0], Diagnostics._0_is_declared_but_never_used, key);
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit, can we rename this to checkUnusuedModuleLocals

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

if (local.declarations) {
    const declaration = local.declarations[0];
    error(declaration.name? declaration.name : declaation, Diagnostics._0_is_declared_but_never_used, key);
}

}
}
}
Expand All @@ -14498,7 +14495,7 @@ namespace ts {
}
}

function checkUnusedTypeParameters(node: ClassDeclaration | FunctionDeclaration | MethodDeclaration | FunctionExpression | ArrowFunction | ConstructorDeclaration | SignatureDeclaration) {
function checkUnusedTypeParameters(node: ClassDeclaration | FunctionDeclaration | MethodDeclaration | FunctionExpression | ArrowFunction | ConstructorDeclaration | SignatureDeclaration | InterfaceDeclaration) {
if (compilerOptions.noUnusedLocals && !isInAmbientContext(node)) {
if (node.typeParameters) {
for (const typeParameter of node.typeParameters) {
Expand All @@ -14520,15 +14517,13 @@ namespace ts {
if (hasProperty(node.locals, local)) {
const localValue = node.locals[local];
if (localValue.declarations && !localValue.exportSymbol) {
for (const declaration of localValue.declarations) {
const symbol = declaration.symbol;
if (!symbol.hasReference) {
if (declaration.kind === SyntaxKind.ImportSpecifier || declaration.kind === SyntaxKind.ImportClause || declaration.kind === SyntaxKind.NamespaceImport) {
error(declaration, Diagnostics._0_is_declared_but_never_used, symbol.name);
}
else if (declaration.kind === SyntaxKind.ImportEqualsDeclaration) {
error(declaration.name, Diagnostics._0_is_declared_but_never_used, symbol.name);
}
const declaration = localValue.declarations[0];
if (!localValue.hasReference) {
if (declaration.kind === SyntaxKind.ImportSpecifier || declaration.kind === SyntaxKind.ImportClause || declaration.kind === SyntaxKind.NamespaceImport) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

if (declaration.kind === SyntaxKind.ImportSpecifier || declaration.kind === SyntaxKind.ImportClause || declaration.kind === SyntaxKind.NamespaceImport || declaration.kind === SyntaxKind.ImportEqualsDeclaration) {
     error(declaration.name, Diagnostics._0_is_declared_but_never_used, localValue.name);
}

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i am still not sure i understand why checkUnusedImports is not part of the checkUnusedModuleLocals?

error(declaration, Diagnostics._0_is_declared_but_never_used, localValue.name);
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i do not think this is correct. you already have the symbol, and the symbol is not exported, so localValue === localValue.declarations[0].symbol.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Changed the code.

else if (declaration.kind === SyntaxKind.ImportEqualsDeclaration) {
error(declaration.name, Diagnostics._0_is_declared_but_never_used, localValue.name);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

no need to check the isInAmbientContext here. similar to type parameters, there is no reason to have an import that is never used even in ambient context.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can we merge this function with the checkUnusedModulePrivates ?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Removed the ambient context check. I prefer to keep them separate for sake of clarity,

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i am not sure i see the point here. just merge them togather, and get rid of all the checks on the kinds.

}
}
}
Expand Down Expand Up @@ -16010,6 +16005,7 @@ namespace ts {

if (produceDiagnostics) {
checkTypeForDuplicateIndexSignatures(node);
checkUnusedTypeParameters(node);
}
}

Expand Down Expand Up @@ -16934,6 +16930,7 @@ namespace ts {
forEach(node.statements, checkSourceElement);
if (isExternalModule(node)) {
checkUnusedImports(node);
checkUnusedModuleLocals(node);
}
checkDeferredNodes();
deferredNodes = undefined;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

checkUnusedModuleLocals(node); here as well. if this is a module, all declarations are local unless exported so for instance:

export {};  // makes the file a module

var x; // should be reported as unused

export var y; // should not

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added the code and test code

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
tests/cases/compiler/unusedTypeParameterInInterface1.ts(2,15): error TS6133: 'T' is declared but never used.


==== tests/cases/compiler/unusedTypeParameterInInterface1.ts (1 errors) ====

interface int<T> {
~
!!! error TS6133: 'T' is declared but never used.

}
7 changes: 7 additions & 0 deletions tests/baselines/reference/unusedTypeParameterInInterface1.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
//// [unusedTypeParameterInInterface1.ts]

interface int<T> {

}

//// [unusedTypeParameterInInterface1.js]
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
tests/cases/compiler/unusedTypeParameterInInterface2.ts(2,18): error TS6133: 'U' is declared but never used.


==== tests/cases/compiler/unusedTypeParameterInInterface2.ts (1 errors) ====

interface int<T, U, V> {
~
!!! error TS6133: 'U' is declared but never used.
f1(a: T): string;
c: V;
}
8 changes: 8 additions & 0 deletions tests/baselines/reference/unusedTypeParameterInInterface2.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
//// [unusedTypeParameterInInterface2.ts]

interface int<T, U, V> {
f1(a: T): string;
c: V;
}

//// [unusedTypeParameterInInterface2.js]
12 changes: 12 additions & 0 deletions tests/baselines/reference/unusedVariablesinModules1.errors.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
tests/cases/compiler/unusedVariablesinModules1.ts(4,5): error TS6133: 'x' is declared but never used.


==== tests/cases/compiler/unusedVariablesinModules1.ts (1 errors) ====

export {};

var x: string;
~
!!! error TS6133: 'x' is declared but never used.

export var y: string;
11 changes: 11 additions & 0 deletions tests/baselines/reference/unusedVariablesinModules1.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
//// [unusedVariablesinModules1.ts]

export {};

var x: string;

export var y: string;

//// [unusedVariablesinModules1.js]
"use strict";
var x;
6 changes: 6 additions & 0 deletions tests/cases/compiler/unusedTypeParameterInInterface1.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
//@noUnusedLocals:true
//@noUnusedParameters:true

interface int<T> {

}
7 changes: 7 additions & 0 deletions tests/cases/compiler/unusedTypeParameterInInterface2.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
//@noUnusedLocals:true
//@noUnusedParameters:true

interface int<T, U, V> {
f1(a: T): string;
c: V;
}
8 changes: 8 additions & 0 deletions tests/cases/compiler/unusedVariablesinModules1.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
//@noUnusedLocals:true
//@noUnusedParameters:true

export {};

var x: string;

export var y: string;