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

com.google.errorprone.bugpatterns.ImmutableMemberCollection Maven / Gradle / Ivy

There is a newer version: 2.27.1
Show newest version
/*
 * Copyright 2016 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.google.errorprone.bugpatterns;

import static com.google.common.collect.ImmutableMap.toImmutableMap;
import static com.google.common.collect.Streams.stream;
import static com.google.errorprone.BugPattern.SeverityLevel.SUGGESTION;
import static com.google.errorprone.matchers.Matchers.allOf;
import static com.google.errorprone.matchers.Matchers.anyOf;
import static com.google.errorprone.matchers.Matchers.hasAnnotationWithSimpleName;
import static com.google.errorprone.matchers.Matchers.hasModifier;
import static com.google.errorprone.matchers.Matchers.isSameType;
import static com.google.errorprone.matchers.Matchers.kindIs;
import static com.google.errorprone.util.ASTHelpers.getReceiver;
import static com.google.errorprone.util.ASTHelpers.getSymbol;
import static com.google.errorprone.util.ASTHelpers.shouldKeep;

import com.google.auto.value.AutoValue;
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.ImmutableSetMultimap;
import com.google.common.collect.ImmutableSortedMap;
import com.google.common.collect.ImmutableSortedSet;
import com.google.common.collect.ListMultimap;
import com.google.common.collect.SetMultimap;
import com.google.errorprone.BugPattern;
import com.google.errorprone.VisitorState;
import com.google.errorprone.bugpatterns.BugChecker.ClassTreeMatcher;
import com.google.errorprone.fixes.SuggestedFix;
import com.google.errorprone.matchers.Description;
import com.google.errorprone.matchers.Matcher;
import com.sun.source.tree.AssignmentTree;
import com.sun.source.tree.ClassTree;
import com.sun.source.tree.CompilationUnitTree;
import com.sun.source.tree.ExpressionTree;
import com.sun.source.tree.IdentifierTree;
import com.sun.source.tree.MemberSelectTree;
import com.sun.source.tree.MethodInvocationTree;
import com.sun.source.tree.ParameterizedTypeTree;
import com.sun.source.tree.Tree;
import com.sun.source.tree.Tree.Kind;
import com.sun.source.tree.VariableTree;
import com.sun.source.util.TreePathScanner;
import com.sun.tools.javac.code.Symbol;
import com.sun.tools.javac.code.Symbol.VarSymbol;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.NavigableSet;
import java.util.Optional;
import java.util.Set;
import java.util.SortedMap;
import javax.lang.model.element.Modifier;

/** Refactoring to suggest Immutable types for member collection that are not mutated. */
@BugPattern(
    summary = "If you don't intend to mutate a member collection prefer using Immutable types.",
    severity = SUGGESTION)
public final class ImmutableMemberCollection extends BugChecker implements ClassTreeMatcher {

  private static final ImmutableSet MUTATING_METHODS =
      ImmutableSet.of(
          "add",
          "addAll",
          "clear",
          "compute",
          "computeIfAbsent",
          "computeIfPresent",
          "forcePut",
          "merge",
          "pollFirst",
          "pollFirstEntry",
          "pollLast",
          "pollLastEntry",
          "put",
          "putAll",
          "putIfAbsent",
          "remove",
          "removeAll",
          "removeIf",
          "replace",
          "replaceAll",
          "replaceValues",
          "retainAll",
          "set",
          "sort");

  private static final ImmutableSet> REPLACEABLE_TYPES =
      ImmutableSet.of(
          ReplaceableType.create(NavigableSet.class, ImmutableSortedSet.class),
          ReplaceableType.create(Set.class, ImmutableSet.class),
          ReplaceableType.create(List.class, ImmutableList.class),
          ReplaceableType.create(ListMultimap.class, ImmutableListMultimap.class),
          ReplaceableType.create(SetMultimap.class, ImmutableSetMultimap.class),
          ReplaceableType.create(SortedMap.class, ImmutableSortedMap.class),
          ReplaceableType.create(Map.class, ImmutableMap.class));

  private static final Matcher PRIVATE_FINAL_VAR_MATCHER =
      allOf(kindIs(Kind.VARIABLE), hasModifier(Modifier.PRIVATE), hasModifier(Modifier.FINAL));

  // TODO(ashishkedia) : Share this with ImmutableSetForContains.
  private static final Matcher EXCLUSIONS =
      anyOf(
          (t, s) -> shouldKeep(t),
          hasAnnotationWithSimpleName("Bind"),
          hasAnnotationWithSimpleName("Inject"));

  @Override
  public Description matchClass(ClassTree classTree, VisitorState state) {
    ImmutableMap replaceableVars =
        classTree.getMembers().stream()
            .filter(member -> PRIVATE_FINAL_VAR_MATCHER.matches(member, state))
            .filter(member -> !EXCLUSIONS.matches(member, state))
            .filter(member -> !isSuppressed(member, state))
            .map(VariableTree.class::cast)
            .flatMap(varTree -> stream(isReplaceable(varTree, state)))
            .collect(toImmutableMap(ReplaceableVar::symbol, var -> var));
    if (replaceableVars.isEmpty()) {
      return Description.NO_MATCH;
    }
    HashSet isPotentiallyMutated = new HashSet<>();
    ImmutableSetMultimap.Builder initTreesBuilder = ImmutableSetMultimap.builder();
    new TreePathScanner() {
      @Override
      public Void visitAssignment(AssignmentTree assignmentTree, VisitorState visitorState) {
        Symbol varSymbol = getSymbol(assignmentTree.getVariable());
        if (replaceableVars.containsKey(varSymbol) && assignmentTree.getExpression() != null) {
          initTreesBuilder.put(varSymbol, assignmentTree.getExpression());
        }
        return scan(assignmentTree.getExpression(), visitorState);
      }

      @Override
      public Void visitVariable(VariableTree variableTree, VisitorState visitorState) {
        VarSymbol varSym = getSymbol(variableTree);
        if (replaceableVars.containsKey(varSym) && variableTree.getInitializer() != null) {
          initTreesBuilder.put(varSym, variableTree.getInitializer());
        }
        return super.visitVariable(variableTree, visitorState);
      }

      @Override
      public Void visitIdentifier(IdentifierTree identifierTree, VisitorState visitorState) {
        recordVarMutation(getSymbol(identifierTree));
        return super.visitIdentifier(identifierTree, visitorState);
      }

      @Override
      public Void visitMemberSelect(MemberSelectTree memberSelectTree, VisitorState visitorState) {
        recordVarMutation(getSymbol(memberSelectTree));
        return super.visitMemberSelect(memberSelectTree, visitorState);
      }

      @Override
      public Void visitMethodInvocation(
          MethodInvocationTree methodInvocationTree, VisitorState visitorState) {
        ExpressionTree receiver = getReceiver(methodInvocationTree);
        if (replaceableVars.containsKey(getSymbol(receiver))) {
          MemberSelectTree selectTree = (MemberSelectTree) methodInvocationTree.getMethodSelect();
          if (!MUTATING_METHODS.contains(selectTree.getIdentifier().toString())) {
            // This is a safe read only method invoked on a replaceable collection member.
            methodInvocationTree.getTypeArguments().forEach(type -> scan(type, visitorState));
            methodInvocationTree.getArguments().forEach(arg -> scan(arg, visitorState));
            return null;
          }
        }
        return super.visitMethodInvocation(methodInvocationTree, visitorState);
      }

      private void recordVarMutation(Symbol sym) {
        if (replaceableVars.containsKey(sym)) {
          isPotentiallyMutated.add(sym);
        }
      }
    }.scan(state.findPathToEnclosing(CompilationUnitTree.class), state);
    ImmutableSetMultimap initTrees = initTreesBuilder.build();
    SuggestedFix.Builder suggestedFix = SuggestedFix.builder();
    replaceableVars.values().stream()
        .filter(
            var ->
                var.areAllInitImmutable(initTrees.get(var.symbol()), state)
                    || !isPotentiallyMutated.contains(var.symbol()))
        .forEach(
            replaceableVar ->
                suggestedFix.merge(
                    replaceableVar.getFix(initTrees.get(replaceableVar.symbol()), state)));
    if (suggestedFix.isEmpty()) {
      return Description.NO_MATCH;
    }
    return describeMatch(classTree, suggestedFix.build());
  }

  private static Optional isReplaceable(VariableTree tree, VisitorState state) {
    return REPLACEABLE_TYPES.stream()
        .filter(type -> isSameType(type.interfaceType()).matches(tree, state))
        .findFirst()
        .map(type -> ReplaceableVar.create(tree, type));
  }

  @AutoValue
  abstract static class ReplaceableType {
    abstract Class interfaceType();

    abstract Class immutableType();

    static  ReplaceableType create(Class interfaceType, Class immutableType) {
      return new AutoValue_ImmutableMemberCollection_ReplaceableType<>(
          interfaceType, immutableType);
    }
  }

  @AutoValue
  abstract static class ReplaceableVar {
    abstract Symbol symbol();

    abstract ReplaceableType type();

    abstract Tree declaredType();

    static ReplaceableVar create(VariableTree variableTree, ReplaceableType type) {
      return new AutoValue_ImmutableMemberCollection_ReplaceableVar(
          getSymbol(variableTree), type, variableTree.getType());
    }

    private SuggestedFix getFix(ImmutableSet initTrees, VisitorState state) {
      SuggestedFix.Builder fixBuilder =
          SuggestedFix.builder()
              .replace(stripTypeParameters(declaredType()), type().immutableType().getSimpleName())
              .addImport(type().immutableType().getName());
      initTrees.stream()
          .filter(initTree -> !isSameType(type().immutableType()).matches(initTree, state))
          .forEach(init -> fixBuilder.replace(init, wrapWithImmutableCopy(init, state)));
      return fixBuilder.build();
    }

    private String wrapWithImmutableCopy(Tree tree, VisitorState state) {
      String type = type().immutableType().getSimpleName();
      return type + ".copyOf(" + state.getSourceForNode(tree) + ")";
    }

    private boolean areAllInitImmutable(ImmutableSet initTrees, VisitorState state) {
      return initTrees.stream()
          .allMatch(initTree -> isSameType(type().immutableType()).matches(initTree, state));
    }

    private static Tree stripTypeParameters(Tree tree) {
      return tree.getKind().equals(Kind.PARAMETERIZED_TYPE)
          ? ((ParameterizedTypeTree) tree).getType()
          : tree;
    }
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy