All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.palantir.baseline.errorprone.StrictUnusedVariable Maven / Gradle / Ivy

There is a newer version: 6.11.0
Show newest version
/*
 * (c) Copyright 2018 Palantir Technologies Inc. All rights reserved.
 *
 * 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.
 */

/*
 * Copyright 2018 The Error Prone Authors.
 *
 * 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.
 */

package com.palantir.baseline.errorprone;

import static com.google.common.base.MoreObjects.firstNonNull;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Strings.nullToEmpty;
import static com.google.common.collect.ImmutableList.toImmutableList;
import static com.google.common.collect.Iterables.getLast;
import static com.google.common.collect.Iterables.getOnlyElement;
import static com.google.errorprone.BugPattern.SeverityLevel.ERROR;
import static com.google.errorprone.util.ASTHelpers.getSymbol;
import static com.google.errorprone.util.ASTHelpers.getType;
import static com.google.errorprone.util.ASTHelpers.isStatic;
import static com.google.errorprone.util.ASTHelpers.isSubtype;
import static com.google.errorprone.util.SideEffectAnalysis.hasSideEffect;
import static com.sun.source.tree.Tree.Kind.POSTFIX_DECREMENT;
import static com.sun.source.tree.Tree.Kind.POSTFIX_INCREMENT;
import static com.sun.source.tree.Tree.Kind.PREFIX_DECREMENT;
import static com.sun.source.tree.Tree.Kind.PREFIX_INCREMENT;

import com.google.auto.service.AutoService;
import com.google.common.base.Ascii;
import com.google.common.base.CaseFormat;
import com.google.common.base.Preconditions;
import com.google.common.collect.ArrayListMultimap;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableListMultimap;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.ListMultimap;
import com.google.common.collect.Multimaps;
import com.google.errorprone.BugPattern;
import com.google.errorprone.VisitorState;
import com.google.errorprone.bugpatterns.BugChecker;
import com.google.errorprone.bugpatterns.UnusedVariable;
import com.google.errorprone.fixes.SuggestedFix;
import com.google.errorprone.fixes.SuggestedFixes;
import com.google.errorprone.matchers.ChildMultiMatcher.MatchType;
import com.google.errorprone.matchers.Description;
import com.google.errorprone.matchers.Matcher;
import com.google.errorprone.matchers.Matchers;
import com.google.errorprone.suppliers.Suppliers;
import com.google.errorprone.util.ASTHelpers;
import com.sun.source.tree.AnnotationTree;
import com.sun.source.tree.ArrayAccessTree;
import com.sun.source.tree.AssignmentTree;
import com.sun.source.tree.ClassTree;
import com.sun.source.tree.CompilationUnitTree;
import com.sun.source.tree.CompoundAssignmentTree;
import com.sun.source.tree.DoWhileLoopTree;
import com.sun.source.tree.EnhancedForLoopTree;
import com.sun.source.tree.ErroneousTree;
import com.sun.source.tree.ExpressionStatementTree;
import com.sun.source.tree.ExpressionTree;
import com.sun.source.tree.ForLoopTree;
import com.sun.source.tree.IdentifierTree;
import com.sun.source.tree.IfTree;
import com.sun.source.tree.LambdaExpressionTree;
import com.sun.source.tree.MemberReferenceTree;
import com.sun.source.tree.MemberSelectTree;
import com.sun.source.tree.MethodInvocationTree;
import com.sun.source.tree.MethodTree;
import com.sun.source.tree.ReturnTree;
import com.sun.source.tree.StatementTree;
import com.sun.source.tree.Tree;
import com.sun.source.tree.Tree.Kind;
import com.sun.source.tree.TryTree;
import com.sun.source.tree.UnaryTree;
import com.sun.source.tree.VariableTree;
import com.sun.source.tree.WhileLoopTree;
import com.sun.source.util.SimpleTreeVisitor;
import com.sun.source.util.TreePath;
import com.sun.source.util.TreePathScanner;
import com.sun.source.util.TreeScanner;
import com.sun.tools.javac.code.Symbol;
import com.sun.tools.javac.code.Type;
import com.sun.tools.javac.tree.JCTree;
import com.sun.tools.javac.util.Position;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.atomic.AtomicBoolean;
import javax.annotation.Nullable;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.Name;

/**
 * Copy from {@link UnusedVariable} except we consider all parameter for unused analysis. We modified the
 * `onlyCheckForReassignments` filter to exclude abstract methods, to check loggers
 */
@AutoService(BugChecker.class)
@BugPattern(
        link = "https://github.com/palantir/gradle-baseline#baseline-error-prone-checks",
        linkType = BugPattern.LinkType.CUSTOM,
        summary = "Unused.",
        severity = ERROR,
        documentSuppression = false)
public final class StrictUnusedVariable extends BugChecker implements BugChecker.CompilationUnitTreeMatcher {
    private static final ImmutableSet EXEMPT_PREFIXES = ImmutableSet.of("_");

    /** The set of annotation full names which exempt annotated element from being reported as unused. */
    private static final ImmutableSet EXEMPTING_VARIABLE_ANNOTATIONS = ImmutableSet.of(
            "javax.persistence.Basic",
            "javax.persistence.Column",
            "javax.persistence.Id",
            "javax.persistence.Version",
            "javax.xml.bind.annotation.XmlElement",
            "org.junit.Rule",
            "org.junit.jupiter.api.extension.RegisterExtension",
            "org.mockito.Mock",
            "org.openqa.selenium.support.FindBy",
            "org.openqa.selenium.support.FindBys");

    /** The set of types exempting a type that is extending or implementing them. */
    private static final ImmutableSet EXEMPTING_SUPER_TYPES = ImmutableSet.of(
            // Don't check record fields
            "java.lang.Record");

    /** The set of types exempting a field of type extending them. */
    private static final ImmutableSet EXEMPTING_FIELD_SUPER_TYPES =
            ImmutableSet.of("org.junit.rules.TestRule", "org.slf4j.Logger", "com.palantir.logsafe.logger.SafeLogger");

    private static final ImmutableList SPECIAL_FIELDS = ImmutableList.of(
            "serialVersionUID",
            // TAG fields are used by convention in Android apps.
            "TAG");
    private static final String UNUSED = "unused";

    private static final Matcher ALTERNATIVE_SUPPRESSED = Matchers.allOf(
            // Matchers.annotations will throw given a tree that doesn't support annotations, so we must enumerate
            // supported types.
            Matchers.kindAnyOf(Set.of(
                    Kind.CLASS, Kind.VARIABLE, Kind.METHOD, Kind.ANNOTATED_TYPE, Kind.PACKAGE, Kind.COMPILATION_UNIT)),
            Matchers.annotations(
                    MatchType.AT_LEAST_ONE,
                    Matchers.allOf(
                            Matchers.isType("java.lang.SuppressWarnings"),
                            Matchers.hasArgumentWithValue(
                                    "value",
                                    Matchers.anyOf(
                                            Matchers.stringLiteral("UnusedVariable"),
                                            Matchers.stringLiteral(UNUSED))))));

    @Override
    public Description matchCompilationUnit(CompilationUnitTree tree, VisitorState state) {
        // We will skip reporting on the whole compilation if there are any native methods found.
        // Use a TreeScanner to find all local variables and fields.
        if (hasNativeMethods(tree)) {
            return Description.NO_MATCH;
        }

        VariableFinder variableFinder = new VariableFinder(state);
        variableFinder.scan(state.getPath(), null);

        checkUsedVariables(state, variableFinder);

        // Map of symbols to variable declarations. Initially this is a map of all of the local variable
        // and fields. As we go we remove those variables which are used.
        Map unusedElements = variableFinder.unusedElements;

        // Whether a symbol should only be checked for reassignments (e.g. public methods' parameters).
        Set onlyCheckForReassignments = variableFinder.onlyCheckForReassignments;

        // Map of symbols to their usage sites. In this map we also include the definition site in
        // addition to all the trees where symbol is used. This map is designed to keep the usage sites
        // of variables (parameters, fields, locals).
        //
        // We populate this map when analyzing the unused variables and then use it to generate
        // appropriate fixes for them.
        ListMultimap usageSites = variableFinder.usageSites;

        FilterUsedVariables filterUsedVariables = new FilterUsedVariables(unusedElements, usageSites);
        filterUsedVariables.scan(state.getPath(), null);

        // Keeps track of whether a symbol was _ever_ used (between reassignments).
        Set isEverUsed = filterUsedVariables.isEverUsed;
        List unusedSpecs = filterUsedVariables.unusedSpecs;

        // Add the left-over unused variables...
        for (Map.Entry entry : unusedElements.entrySet()) {
            unusedSpecs.add(UnusedSpec.of(entry.getKey(), entry.getValue(), usageSites.get(entry.getKey()), null));
        }

        ImmutableListMultimap unusedSpecsBySymbol =
                Multimaps.index(unusedSpecs, UnusedSpec::symbol);

        for (Map.Entry> entry :
                unusedSpecsBySymbol.asMap().entrySet()) {
            Symbol unusedSymbol = entry.getKey();
            Collection specs = entry.getValue();

            ImmutableList allUsageSites =
                    specs.stream().flatMap(u -> u.usageSites().stream()).collect(toImmutableList());
            if (!unusedElements.containsKey(unusedSymbol)) {
                isEverUsed.add(unusedSymbol);
            }
            SuggestedFix makeFirstAssignmentDeclaration =
                    makeAssignmentDeclaration(unusedSymbol, specs, allUsageSites, state);
            // Don't complain if this is a public method and we only overwrote it once.
            if (onlyCheckForReassignments.contains(unusedSymbol) && specs.size() <= 1) {
                continue;
            }
            Tree unused = specs.iterator().next().variableTree().getLeaf();
            Symbol.VarSymbol symbol = (Symbol.VarSymbol) unusedSymbol;
            ImmutableList fixes;
            if (symbol.getKind() == ElementKind.PARAMETER && !isEverUsed.contains(unusedSymbol)) {
                Symbol.MethodSymbol methodSymbol = (Symbol.MethodSymbol) symbol.owner;
                int index;
                if (methodSymbol.params == null) {
                    // if the parameter is for a lambda is defined in a static initializer, params is null
                    index = -1;
                } else {
                    index = methodSymbol.params.indexOf(symbol);
                }
                // If we can not find the parameter in the owning method, then it must be a parameter to a lambda
                // defined within the method
                if (index == -1) {
                    fixes = buildUnusedLambdaParameterFix(symbol, entry.getValue(), state);
                } else {
                    fixes = buildUnusedParameterFixes(symbol, methodSymbol, allUsageSites, state);
                }
            } else {
                fixes = buildUnusedVarFixes(symbol, allUsageSites, state);
            }
            state.reportMatch(buildDescription(unused)
                    .setMessage(String.format(
                            "%s %s '%s' is never read. Intentional occurrences are acknowledged by renaming "
                                    + "the unused variable with a leading underscore. '_%s', for example.",
                            unused instanceof VariableTree ? "The" : "The assignment to this",
                            describeVariable(symbol),
                            symbol.name,
                            symbol.name))
                    .addAllFixes(fixes.stream()
                            .map(f -> SuggestedFix.builder()
                                    .merge(makeFirstAssignmentDeclaration)
                                    .merge(f)
                                    .build())
                            .collect(toImmutableList()))
                    .build());
        }
        return Description.NO_MATCH;
    }

    private void checkUsedVariables(VisitorState state, VariableFinder variableFinder) {
        VariableUsage variableUsage = new VariableUsage();
        variableUsage.scan(state.getPath(), null);
        variableFinder.exemptedVariables.forEach((key, value) -> {
            List usageSites = variableUsage.usageSites.get(key);
            if (usageSites.size() <= 1) {
                return;
            }
            state.reportMatch(buildDescription(value)
                    .setMessage(String.format(
                            "The %s '%s' is read but has 'StrictUnusedVariable' suppressed because of its name.",
                            describeVariable((Symbol.VarSymbol) key), key.name))
                    .addFix(constructUsedVariableSuggestedFix(usageSites, state))
                    .build());
        });
    }

    @Override
    public boolean isSuppressed(Tree tree, VisitorState state) {
        return super.isSuppressed(tree, state) || ALTERNATIVE_SUPPRESSED.matches(tree, state);
    }

    private static SuggestedFix constructUsedVariableSuggestedFix(List usagePaths, VisitorState state) {
        SuggestedFix.Builder fix = SuggestedFix.builder();
        for (TreePath usagePath : usagePaths) {
            if (usagePath.getLeaf() instanceof VariableTree) {
                VariableTree variableTree = (VariableTree) usagePath.getLeaf();
                String variableName = variableTree.getName().toString();

                // see buildUnusedLambdaParameterFix for a similar pattern
                if (state.getEndPosition(variableTree.getType()) == -1) {
                    renameVariable(variableTree, variableName, fix);
                    continue;
                }
                int startPos = state.getEndPosition(variableTree.getType()) + 1;
                int endPos = state.getEndPosition(variableTree);

                // Ignore the initializer if there is one
                if (variableTree.getInitializer() != null) {
                    endPos = startPos + variableName.length();
                }
                if (startPos == Position.NOPOS || endPos == Position.NOPOS) {
                    // TODO(b/118437729): handle bogus source positions in enum declarations
                    continue;
                }

                renameVariable(startPos, endPos, variableName, fix);
            } else if (usagePath.getLeaf() instanceof IdentifierTree) {
                JCTree.JCIdent identifierTree = (JCTree.JCIdent) usagePath.getLeaf();
                int startPos = identifierTree.getStartPosition();
                int endPos = state.getEndPosition(identifierTree);
                if (startPos == Position.NOPOS || endPos == Position.NOPOS) {
                    // TODO(b/118437729): handle bogus source positions in enum declarations
                    continue;
                }

                renameVariable(startPos, endPos, identifierTree.getName().toString(), fix);
            }
        }
        return fix.build();
    }

    private static Optional stripUnusedPrefix(String variableName) {
        return EXEMPT_PREFIXES.stream()
                .filter(variableName::startsWith)
                .findFirst()
                .map(prefix -> prefix.length() == variableName.length()
                        // Fall back to a generic variable name if the prefix is the entire variable name
                        ? "value"
                        : CaseFormat.UPPER_CAMEL.to(CaseFormat.LOWER_CAMEL, variableName.substring(prefix.length())));
    }

    private static void renameVariable(int startPos, int endPos, String name, SuggestedFix.Builder fix) {
        stripUnusedPrefix(name).ifPresent(newName -> fix.replace(startPos, endPos, newName));
    }

    private static void renameVariable(Tree node, String name, SuggestedFix.Builder fix) {
        stripUnusedPrefix(name).ifPresent(newName -> fix.replace(node, newName));
    }

    private static SuggestedFix makeAssignmentDeclaration(
            Symbol unusedSymbol,
            Collection specs,
            ImmutableList allUsageSites,
            VisitorState state) {
        if (unusedSymbol.getKind() != ElementKind.LOCAL_VARIABLE) {
            return SuggestedFix.builder().build();
        }
        Optional removedVariableTree = allUsageSites.stream()
                .filter(tp -> tp.getLeaf() instanceof VariableTree)
                .findFirst()
                .map(tp -> (VariableTree) tp.getLeaf());
        Optional reassignment = specs.stream()
                .map(UnusedSpec::terminatingAssignment)
                .filter(Optional::isPresent)
                .map(Optional::get)
                .filter(a -> allUsageSites.stream().noneMatch(tp -> tp.getLeaf().equals(a)))
                .findFirst();
        if (!removedVariableTree.isPresent() || !reassignment.isPresent()) {
            return SuggestedFix.builder().build();
        }
        return SuggestedFix.prefixWith(
                reassignment.get(),
                state.getSourceForNode(removedVariableTree.get().getType()) + " ");
    }

    private static String describeVariable(Symbol.VarSymbol symbol) {
        switch (symbol.getKind()) {
            case FIELD:
                return "field";
            case LOCAL_VARIABLE:
                return "local variable";
            case PARAMETER:
                return "parameter";
            default:
                return "variable";
        }
    }

    private static boolean hasNativeMethods(CompilationUnitTree tree) {
        AtomicBoolean hasAnyNativeMethods = new AtomicBoolean(false);
        new TreeScanner() {
            @Override
            public Void visitMethod(MethodTree tree, Void unused) {
                if (tree.getModifiers().getFlags().contains(Modifier.NATIVE)) {
                    hasAnyNativeMethods.set(true);
                }
                return null;
            }
        }.scan(tree, null);
        return hasAnyNativeMethods.get();
    }

    // https://docs.oracle.com/javase/specs/jls/se11/html/jls-14.html#jls-ExpressionStatement
    private static final ImmutableSet TOP_LEVEL_EXPRESSIONS = ImmutableSet.of(
            Tree.Kind.ASSIGNMENT,
            Tree.Kind.PREFIX_INCREMENT,
            Tree.Kind.PREFIX_DECREMENT,
            Tree.Kind.POSTFIX_INCREMENT,
            Tree.Kind.POSTFIX_DECREMENT,
            Tree.Kind.METHOD_INVOCATION,
            Tree.Kind.NEW_CLASS);

    private static boolean needsBlock(TreePath path) {
        Tree leaf = path.getLeaf();
        class Visitor extends SimpleTreeVisitor {

            @Override
            public Boolean visitIf(IfTree tree, Void unused) {
                return tree.getThenStatement() == leaf || tree.getElseStatement() == leaf;
            }

            @Override
            public Boolean visitDoWhileLoop(DoWhileLoopTree tree, Void unused) {
                return tree.getStatement() == leaf;
            }

            @Override
            public Boolean visitWhileLoop(WhileLoopTree tree, Void unused) {
                return tree.getStatement() == leaf;
            }

            @Override
            public Boolean visitForLoop(ForLoopTree tree, Void unused) {
                return tree.getStatement() == leaf;
            }

            @Override
            public Boolean visitEnhancedForLoop(EnhancedForLoopTree tree, Void unused) {
                return tree.getStatement() == leaf;
            }
        }
        return firstNonNull(path.getParentPath().getLeaf().accept(new Visitor(), null), false);
    }

    private static ImmutableList buildUnusedVarFixes(
            Symbol varSymbol, List usagePaths, VisitorState state) {
        // Don't suggest a fix for fields annotated @Inject: we can warn on them, but they *could* be
        // used outside the class.
        if (ASTHelpers.hasDirectAnnotationWithSimpleName(varSymbol, "Inject")) {
            return ImmutableList.of();
        }
        ElementKind varKind = varSymbol.getKind();
        SuggestedFix.Builder fix = SuggestedFix.builder().setShortDescription("remove unused variable");
        for (TreePath usagePath : usagePaths) {
            StatementTree statement = (StatementTree) usagePath.getLeaf();
            if (statement.getKind() == Tree.Kind.VARIABLE) {
                if (getSymbol(statement).getKind() == ElementKind.PARAMETER) {
                    continue;
                }
                VariableTree variableTree = (VariableTree) statement;
                ExpressionTree initializer = variableTree.getInitializer();
                if (hasSideEffect(initializer) && TOP_LEVEL_EXPRESSIONS.contains(initializer.getKind())) {
                    if (varKind == ElementKind.FIELD) {
                        String newContent = String.format(
                                "%s{ %s; }", isStatic(varSymbol) ? "static " : "", state.getSourceForNode(initializer));
                        fix.merge(SuggestedFixes.replaceIncludingComments(usagePath, newContent, state));
                    } else {
                        fix.replace(statement, String.format("%s;", state.getSourceForNode(initializer)));
                    }
                } else if (isEnhancedForLoopVar(usagePath)) {
                    String modifiers = nullToEmpty(
                            variableTree.getModifiers() == null
                                    ? null
                                    : state.getSourceForNode(variableTree.getModifiers()));
                    String newContent = String.format(
                            "%s%s unused",
                            modifiers.isEmpty() ? "" : (modifiers + " "),
                            state.getSourceForNode(variableTree.getType()));
                    // The new content for the second fix should be identical to the content for the first
                    // fix in this case because we can't just remove the enhanced for loop variable.
                    fix.replace(variableTree, newContent);
                } else {
                    String replacement = needsBlock(usagePath) ? "{}" : "";
                    fix.merge(SuggestedFixes.replaceIncludingComments(usagePath, replacement, state));
                }
                continue;
            } else if (statement.getKind() == Tree.Kind.EXPRESSION_STATEMENT) {
                JCTree tree = (JCTree) ((ExpressionStatementTree) statement).getExpression();

                if (tree instanceof CompoundAssignmentTree) {
                    if (hasSideEffect(((CompoundAssignmentTree) tree).getExpression())) {
                        // If it's a compound assignment, there's no reason we'd want to remove the expression,
                        // so don't set `encounteredSideEffects` based on this usage.
                        SuggestedFix replacement = SuggestedFix.replace(
                                tree.getStartPosition(),
                                ((JCTree.JCAssignOp) tree).getExpression().getStartPosition(),
                                "");
                        fix.merge(replacement);
                        continue;
                    }
                } else if (tree instanceof AssignmentTree) {
                    if (hasSideEffect(((AssignmentTree) tree).getExpression())) {
                        fix.replace(
                                tree.getStartPosition(),
                                ((JCTree.JCAssign) tree).getExpression().getStartPosition(),
                                "");
                        continue;
                    }
                }
            }
            String replacement = needsBlock(usagePath) ? "{}" : "";
            fix.replace(statement, replacement);
        }
        return ImmutableList.of(fix.build());
    }

    private static ImmutableList buildUnusedLambdaParameterFix(
            Symbol.VarSymbol _symbol, Collection values, VisitorState state) {
        SuggestedFix.Builder fix = SuggestedFix.builder();

        for (UnusedSpec unusedSpec : values) {
            Tree leaf = unusedSpec.variableTree().getLeaf();
            if (!(leaf instanceof VariableTree)) {
                continue;
            }

            VariableTree tree = (VariableTree) leaf;
            if (state.getEndPosition(tree.getType()) == -1) {
                fix.replace(tree, "_" + tree.getName());
            } else {
                int startPos = state.getEndPosition(tree.getType()) + 1;
                int endPos = state.getEndPosition(tree);
                fix.replace(startPos, endPos, "_" + tree.getName());
            }
        }

        return ImmutableList.of(fix.build());
    }

    private static ImmutableList buildUnusedParameterFixes(
            Symbol varSymbol, Symbol.MethodSymbol methodSymbol, List usagePaths, VisitorState state) {
        boolean isPrivateMethod = methodSymbol.getModifiers().contains(Modifier.PRIVATE);
        int index = methodSymbol.params.indexOf(varSymbol);
        Preconditions.checkState(index != -1, "symbol %s must be a parameter to the owning method", varSymbol);
        SuggestedFix.Builder fix = SuggestedFix.builder();
        for (TreePath path : usagePaths) {
            fix.delete(path.getLeaf());
        }

        // Remove parameter if the method is private since we can automatically fix all invocation sites
        // Otherwise add `_` prefix to the variable name
        if (isPrivateMethod) {
            new TreePathScanner() {
                @Override
                public Void visitMethodInvocation(MethodInvocationTree tree, Void unused) {
                    if (getSymbol(tree).equals(methodSymbol)) {
                        removeByIndex(tree.getArguments());
                    }
                    return super.visitMethodInvocation(tree, null);
                }

                @Override
                public Void visitMethod(MethodTree tree, Void unused) {
                    if (getSymbol(tree).equals(methodSymbol)) {
                        removeByIndex(tree.getParameters());
                    }
                    return super.visitMethod(tree, null);
                }

                private void removeByIndex(List trees) {
                    if (index >= trees.size()) {
                        // possible when removing a varargs parameter with no corresponding formal parameters
                        return;
                    }
                    if (trees.size() == 1) {
                        Tree tree = getOnlyElement(trees);
                        if (((JCTree) tree).getStartPosition() == -1 || state.getEndPosition(tree) == -1) {
                            // TODO(b/118437729): handle bogus source positions in enum declarations
                            return;
                        }
                        fix.delete(tree);
                        return;
                    }
                    int startPos;
                    int endPos;
                    if (index >= 1) {
                        startPos = state.getEndPosition(trees.get(index - 1));
                        endPos = state.getEndPosition(trees.get(index));
                    } else {
                        startPos = ((JCTree) trees.get(index)).getStartPosition();
                        endPos = ((JCTree) trees.get(index + 1)).getStartPosition();
                    }
                    if (index == methodSymbol.params().size() - 1 && methodSymbol.isVarArgs()) {
                        endPos = state.getEndPosition(getLast(trees));
                    }
                    if (startPos == Position.NOPOS || endPos == Position.NOPOS) {
                        // TODO(b/118437729): handle bogus source positions in enum declarations
                        return;
                    }
                    fix.replace(startPos, endPos, "");
                }
            }.scan(state.getPath().getCompilationUnit(), null);
        } else {
            new TreePathScanner() {
                @Override
                public Void visitMethod(MethodTree methodTree, Void unused) {
                    if (getSymbol(methodTree).equals(methodSymbol)) {
                        renameByIndex(methodTree.getParameters());
                    }
                    return super.visitMethod(methodTree, null);
                }

                private void renameByIndex(List trees) {
                    if (index >= trees.size()) {
                        // possible when removing a varargs parameter with no corresponding formal parameters
                        return;
                    }

                    VariableTree tree = trees.get(index);
                    int startPos = state.getEndPosition(tree.getType()) + 1;
                    int endPos = state.getEndPosition(trees.get(index));
                    if (index == methodSymbol.params().size() - 1 && methodSymbol.isVarArgs()) {
                        endPos = state.getEndPosition(getLast(trees));
                    }
                    if (startPos == Position.NOPOS || endPos == Position.NOPOS) {
                        // TODO(b/118437729): handle bogus source positions in enum declarations
                        return;
                    }
                    String name = tree.getName().toString();
                    if (name.startsWith(UNUSED)) {
                        fix.replace(
                                startPos,
                                endPos,
                                "_"
                                        + (name.equals(UNUSED)
                                                ? "value"
                                                : CaseFormat.UPPER_CAMEL.to(
                                                        CaseFormat.LOWER_CAMEL, name.substring(UNUSED.length()))));
                    } else {
                        fix.replace(startPos, endPos, "_" + tree.getName());
                    }
                }
            }.scan(state.getPath().getCompilationUnit(), null);
        }
        return ImmutableList.of(fix.build());
    }

    private static boolean isEnhancedForLoopVar(TreePath variablePath) {
        Tree tree = variablePath.getLeaf();
        Tree parent = variablePath.getParentPath().getLeaf();
        return parent instanceof EnhancedForLoopTree && ((EnhancedForLoopTree) parent).getVariable() == tree;
    }

    /**
     * Looks at the list of {@code annotations} and see if there is any annotation which exists
     * {@code exemptingAnnotations}.
     */
    private static boolean exemptedByAnnotation(List annotations, VisitorState unused) {
        for (AnnotationTree annotation : annotations) {
            if (((JCTree.JCAnnotation) annotation).type != null) {
                Symbol.TypeSymbol tsym = ((JCTree.JCAnnotation) annotation).type.tsym;
                if (EXEMPTING_VARIABLE_ANNOTATIONS.contains(
                        tsym.getQualifiedName().toString())) {
                    return true;
                }
            }
        }
        return false;
    }

    private static boolean exemptedByName(Name name) {
        return EXEMPT_PREFIXES.stream()
                .anyMatch(prefix -> Ascii.toLowerCase(name.toString()).startsWith(prefix));
    }

    private final class VariableFinder extends TreePathScanner {
        private final Map unusedElements = new HashMap<>();

        private final Set onlyCheckForReassignments = new HashSet<>();

        private final ListMultimap usageSites = ArrayListMultimap.create();

        private final Map exemptedVariables = new HashMap<>();

        private final VisitorState state;

        private VariableFinder(VisitorState state) {
            this.state = state;
        }

        @Override
        @SuppressWarnings("SwitchStatementDefaultCase")
        public Void visitVariable(VariableTree variableTree, Void unused) {
            if (isSuppressed(variableTree, state)) {
                return null;
            }
            Symbol.VarSymbol symbol = getSymbol(variableTree);
            if (symbol == null) {
                return null;
            }
            if (exemptedByName(variableTree.getName())) {
                exemptedVariables.put(symbol, variableTree);
                return null;
            }
            if (symbol.getKind() == ElementKind.FIELD && exemptedFieldBySuperType(getType(variableTree), state)) {
                return null;
            }
            super.visitVariable(variableTree, null);
            // Return if the element is exempted by an annotation.
            if (exemptedByAnnotation(variableTree.getModifiers().getAnnotations(), state)) {
                return null;
            }
            switch (symbol.getKind()) {
                case FIELD:
                    // We are only interested in private fields and those which are not special.
                    if (isFieldEligibleForChecking(variableTree, symbol)) {
                        unusedElements.put(symbol, getCurrentPath());
                        usageSites.put(symbol, getCurrentPath());
                    }
                    break;
                case LOCAL_VARIABLE:
                    unusedElements.put(symbol, getCurrentPath());
                    usageSites.put(symbol, getCurrentPath());
                    break;
                case PARAMETER:
                    // ignore the receiver parameter
                    if (variableTree.getName().contentEquals("this")) {
                        return null;
                    }
                    unusedElements.put(symbol, getCurrentPath());
                    if (!isParameterSubjectToAnalysis(symbol)) {
                        onlyCheckForReassignments.add(symbol);
                    }
                    break;
                default:
                    break;
            }
            return null;
        }

        private boolean exemptedFieldBySuperType(Type type, VisitorState state) {
            return EXEMPTING_FIELD_SUPER_TYPES.stream()
                    .anyMatch(t -> isSubtype(type, state.getTypeFromString(t), state));
        }

        private boolean isFieldEligibleForChecking(VariableTree variableTree, Symbol.VarSymbol symbol) {
            return variableTree.getModifiers().getFlags().contains(Modifier.PRIVATE)
                    && !SPECIAL_FIELDS.contains(symbol.getSimpleName().toString());
        }

        /** Returns whether {@code sym} can be removed without updating call sites in other files. */
        @SuppressWarnings("PreferSafeLoggingPreconditions")
        private boolean isParameterSubjectToAnalysis(Symbol sym) {
            checkArgument(sym.getKind() == ElementKind.PARAMETER);
            Symbol enclosingMethod = sym.owner;

            return !enclosingMethod.getModifiers().contains(Modifier.ABSTRACT)
                    && !enclosingMethod.getModifiers().contains(Modifier.DEFAULT);
        }

        @Override
        public Void visitTry(TryTree node, Void unused) {
            // Skip resources, as while these may not be referenced, they are used.
            scan(node.getBlock(), null);
            scan(node.getCatches(), null);
            scan(node.getFinallyBlock(), null);
            return null;
        }

        @Override
        public Void visitClass(ClassTree tree, Void unused) {
            if (isSuppressed(tree, state)) {
                return null;
            }
            if (EXEMPTING_SUPER_TYPES.stream()
                    .anyMatch(t ->
                            isSubtype(getType(tree), Suppliers.typeFromString(t).get(state), state))) {
                return null;
            }
            return super.visitClass(tree, null);
        }

        @Override
        public Void visitLambdaExpression(LambdaExpressionTree node, Void unused) {
            scan(node.getBody(), null);
            scan(node.getParameters(), null);
            return null;
        }

        @Override
        public Void visitMethod(MethodTree tree, Void unused) {
            return isSuppressed(tree, state) ? null : super.visitMethod(tree, null);
        }
    }

    private static final class FilterUsedVariables extends TreePathScanner {
        private boolean leftHandSideAssignment = false;
        // When this greater than zero, the usage of identifiers are real.
        private int inArrayAccess = 0;
        // This is true when we are processing a `return` statement. Elements used in return statement
        // must not be considered unused.
        private boolean inReturnStatement = false;
        // When this greater than zero, the usage of identifiers are real because they are in a method
        // call.
        private int inMethodCall = 0;

        private final Set hasBeenAssigned = new HashSet<>();

        private TreePath currentExpressionStatement = null;

        private final Map unusedElements;

        private final ListMultimap usageSites;

        // Keeps track of whether a symbol was _ever_ used (between reassignments).
        private final Set isEverUsed = new HashSet<>();

        private final List unusedSpecs = new ArrayList<>();

        private final ImmutableMap declarationSites;

        private FilterUsedVariables(Map unusedElements, ListMultimap usageSites) {
            this.unusedElements = unusedElements;
            this.usageSites = usageSites;
            this.declarationSites = ImmutableMap.copyOf(unusedElements);
        }

        private boolean isInExpressionStatementTree() {
            Tree parent = getCurrentPath().getParentPath().getLeaf();
            return parent != null && parent.getKind() == Tree.Kind.EXPRESSION_STATEMENT;
        }

        private boolean isUsed(@Nullable Symbol symbol) {
            return symbol != null
                    && (!leftHandSideAssignment || inReturnStatement || inArrayAccess > 0 || inMethodCall > 0)
                    && unusedElements.containsKey(symbol);
        }

        @Override
        public Void visitVariable(VariableTree tree, Void unused) {
            Symbol.VarSymbol symbol = getSymbol(tree);
            if (hasBeenAssigned(tree, symbol)) {
                hasBeenAssigned.add(symbol);
            }
            return super.visitVariable(tree, null);
        }

        private boolean hasBeenAssigned(VariableTree tree, Symbol.VarSymbol symbol) {
            if (symbol == null) {
                return false;
            }
            // Parameters and enhanced for loop variables are always considered assigned.
            if (symbol.getKind() == ElementKind.PARAMETER) {
                return true;
            }
            if (getCurrentPath().getParentPath().getLeaf() instanceof EnhancedForLoopTree) {
                return true;
            }
            // Otherwise it's assigned if the VariableTree has an initializer.
            if (unusedElements.containsKey(symbol) && tree.getInitializer() != null) {
                return true;
            }
            return false;
        }

        @Override
        public Void visitExpressionStatement(ExpressionStatementTree tree, Void unused) {
            currentExpressionStatement = getCurrentPath();
            super.visitExpressionStatement(tree, null);
            currentExpressionStatement = null;
            return null;
        }

        @Override
        public Void visitIdentifier(IdentifierTree tree, Void unused) {
            Symbol symbol = getSymbol(tree);
            // Filtering out identifier symbol from vars map. These are real usages of identifiers.
            if (isUsed(symbol)) {
                unusedElements.remove(symbol);
            }
            if (currentExpressionStatement != null && unusedElements.containsKey(symbol)) {
                usageSites.put(symbol, currentExpressionStatement);
            }
            return null;
        }

        @Override
        public Void visitAssignment(AssignmentTree tree, Void unused) {
            scan(tree.getExpression(), null);
            // If a variable is used in the left hand side of an assignment that does not count as a
            // usage.
            if (isInExpressionStatementTree()) {
                handleReassignment(tree);
                leftHandSideAssignment = true;
                scan(tree.getVariable(), null);
                leftHandSideAssignment = false;
            } else {
                super.visitAssignment(tree, null);
            }
            return null;
        }

        /**
         * Deals with assignment trees; works out if the assignment definitely overwrites the variable in all ways that
         * could be observed as we scan forwards.
         */
        private void handleReassignment(AssignmentTree tree) {
            Tree parent = getCurrentPath().getParentPath().getLeaf();
            if (!(parent instanceof StatementTree)) {
                return;
            }
            if (tree.getVariable().getKind() != Tree.Kind.IDENTIFIER) {
                return;
            }
            if (ASTHelpers.findEnclosingNode(getCurrentPath(), ForLoopTree.class) != null) {
                return;
            }
            Symbol symbol = getSymbol(tree.getVariable());
            // Check if it was actually assigned to at this depth (or is a parameter).
            if (!((hasBeenAssigned.contains(symbol) && symbol.getKind() == ElementKind.LOCAL_VARIABLE)
                    || symbol.getKind() == ElementKind.PARAMETER)) {
                return;
            }
            if (!declarationSites.containsKey(symbol)) {
                return;
            }
            hasBeenAssigned.add(symbol);
            TreePath assignmentSite = declarationSites.get(symbol);
            if (scopeDepth(assignmentSite) != Iterables.size(getCurrentPath().getParentPath())) {
                return;
            }
            if (unusedElements.containsKey(symbol)) {
                unusedSpecs.add(UnusedSpec.of(symbol, assignmentSite, usageSites.get(symbol), tree));
            } else {
                isEverUsed.add(symbol);
            }
            unusedElements.put(symbol, getCurrentPath());
            usageSites.removeAll(symbol);
            usageSites.put(symbol, getCurrentPath().getParentPath());
        }

        // This is a crude proxy for when a variable is unconditionally overwritten. It doesn't match
        // all cases, but it catches a reassignment at the same depth.
        private static int scopeDepth(TreePath assignmentSite) {
            if (assignmentSite.getParentPath().getLeaf() instanceof EnhancedForLoopTree) {
                return Iterables.size(assignmentSite) + 1;
            }
            if (assignmentSite.getLeaf() instanceof VariableTree) {
                Symbol.VarSymbol symbol = getSymbol((VariableTree) assignmentSite.getLeaf());
                if (symbol.getKind() == ElementKind.PARAMETER) {
                    return Iterables.size(assignmentSite) + 1;
                }
            }
            return Iterables.size(assignmentSite);
        }

        @Override
        public Void visitMemberSelect(MemberSelectTree memberSelectTree, Void unused) {
            Symbol symbol = getSymbol(memberSelectTree);
            if (isUsed(symbol)) {
                unusedElements.remove(symbol);
            } else if (currentExpressionStatement != null && unusedElements.containsKey(symbol)) {
                usageSites.put(symbol, currentExpressionStatement);
            }
            // Clear leftHandSideAssignment and descend down the tree to catch any variables in the
            // receiver of this member select, which _are_ considered used.
            boolean wasLeftHandAssignment = leftHandSideAssignment;
            leftHandSideAssignment = false;
            super.visitMemberSelect(memberSelectTree, null);
            leftHandSideAssignment = wasLeftHandAssignment;
            return null;
        }

        @Override
        public Void visitMemberReference(MemberReferenceTree tree, Void unused) {
            super.visitMemberReference(tree, null);
            Symbol.MethodSymbol symbol = getSymbol(tree);
            if (symbol != null) {
                symbol.getParameters().forEach(unusedElements::remove);
            }
            return null;
        }

        @Override
        public Void visitCompoundAssignment(CompoundAssignmentTree tree, Void unused) {
            if (isInExpressionStatementTree()) {
                leftHandSideAssignment = true;
                scan(tree.getVariable(), null);
                leftHandSideAssignment = false;
                scan(tree.getExpression(), null);
            } else {
                super.visitCompoundAssignment(tree, null);
            }
            return null;
        }

        @Override
        public Void visitArrayAccess(ArrayAccessTree node, Void unused) {
            inArrayAccess++;
            super.visitArrayAccess(node, null);
            inArrayAccess--;
            return null;
        }

        @Override
        public Void visitReturn(ReturnTree node, Void unused) {
            inReturnStatement = true;
            scan(node.getExpression(), null);
            inReturnStatement = false;
            return null;
        }

        @Override
        public Void visitUnary(UnaryTree tree, Void unused) {
            // If unary expression is inside another expression, then this is a real usage of unary
            // operand.
            // Example:
            //   array[i++] = 0; // 'i' has a real usage here. 'array' might not have.
            //   list.get(i++);
            // But if it is like this:
            //   i++;
            // Then it is possible that this is not a real usage of 'i'.
            if (isInExpressionStatementTree()
                    && (tree.getKind() == POSTFIX_DECREMENT
                            || tree.getKind() == POSTFIX_INCREMENT
                            || tree.getKind() == PREFIX_DECREMENT
                            || tree.getKind() == PREFIX_INCREMENT)) {
                leftHandSideAssignment = true;
                scan(tree.getExpression(), null);
                leftHandSideAssignment = false;
            } else {
                super.visitUnary(tree, null);
            }
            return null;
        }

        @Override
        public Void visitErroneous(ErroneousTree tree, Void unused) {
            return scan(tree.getErrorTrees(), null);
        }

        /** Looks at method invocations and removes the invoked private methods from {@code #unusedElements}. */
        @Override
        public Void visitMethodInvocation(MethodInvocationTree tree, Void unused) {
            inMethodCall++;
            super.visitMethodInvocation(tree, null);
            inMethodCall--;
            return null;
        }
    }

    static class VariableUsage extends TreePathScanner {
        public final ListMultimap usageSites = ArrayListMultimap.create();

        @Override
        public Void visitVariable(VariableTree tree, Void unused) {
            usageSites.put(getSymbol(tree), getCurrentPath());
            return super.visitVariable(tree, null);
        }

        @Override
        public Void visitIdentifier(IdentifierTree tree, Void unused) {
            usageSites.put(getSymbol(tree), getCurrentPath());
            return super.visitIdentifier(tree, null);
        }

        @Override
        public Void visitMemberSelect(MemberSelectTree memberSelectTree, Void unused) {
            usageSites.put(getSymbol(memberSelectTree), getCurrentPath());
            return super.visitMemberSelect(memberSelectTree, null);
        }
    }

    interface UnusedSpec {
        /** {@link Symbol} of the unsued element. */
        Symbol symbol();

        /** {@link VariableTree} for the original declaration site. */
        TreePath variableTree();

        /**
         * All the usage sites of this variable that we claim are unused (including the initial declaration/assignment).
         */
        ImmutableList usageSites();

        /**
         * If this usage chain was terminated by an unconditional reassignment, the corresponding
         * {@link AssignmentTree}.
         */
        Optional terminatingAssignment();

        static UnusedSpec of(
                Symbol symbol,
                TreePath variableTree,
                Iterable treePaths,
                @Nullable AssignmentTree assignmentTree) {
            final ImmutableList treePaths1 = ImmutableList.copyOf(treePaths);
            return new UnusedSpec() {
                @Override
                public Symbol symbol() {
                    return symbol;
                }

                @Override
                public TreePath variableTree() {
                    return variableTree;
                }

                @Override
                public ImmutableList usageSites() {
                    return treePaths1;
                }

                @Override
                public Optional terminatingAssignment() {
                    return Optional.ofNullable(assignmentTree);
                }
            };
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy