Package com.google.errorprone.bugpatterns

Source Code of com.google.errorprone.bugpatterns.SelfEquals

/*
* Copyright 2012 Google 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.
*/

package com.google.errorprone.bugpatterns;

import static com.google.errorprone.BugPattern.Category.GUAVA;
import static com.google.errorprone.BugPattern.MaturityLevel.MATURE;
import static com.google.errorprone.BugPattern.SeverityLevel.ERROR;
import static com.google.errorprone.matchers.Matchers.allOf;
import static com.google.errorprone.matchers.Matchers.methodSelect;
import static com.google.errorprone.matchers.Matchers.receiverSameAsArgument;
import static com.google.errorprone.matchers.Matchers.sameArgument;
import static com.google.errorprone.matchers.Matchers.staticMethod;

import com.google.errorprone.BugPattern;
import com.google.errorprone.VisitorState;
import com.google.errorprone.bugpatterns.BugChecker.MethodInvocationTreeMatcher;
import com.google.errorprone.fixes.Fix;
import com.google.errorprone.fixes.SuggestedFix;
import com.google.errorprone.matchers.Description;
import com.google.errorprone.matchers.Matcher;
import com.google.errorprone.matchers.Matchers;
import com.google.errorprone.util.ASTHelpers;

import com.sun.source.tree.ExpressionTree;
import com.sun.source.tree.MethodInvocationTree;
import com.sun.source.tree.Tree.Kind;
import com.sun.source.util.TreePath;
import com.sun.tools.javac.code.Symbol.TypeSymbol;
import com.sun.tools.javac.tree.JCTree.JCBlock;
import com.sun.tools.javac.tree.JCTree.JCFieldAccess;
import com.sun.tools.javac.tree.JCTree.JCStatement;
import com.sun.tools.javac.tree.JCTree.JCVariableDecl;

import java.util.List;

/**
* @author alexeagle@google.com (Alex Eagle)
*/
@BugPattern(name = "SelfEquals",
    summary = "An object is tested for equality to itself",
    explanation =
        "The arguments to this equal method are the same object, so it always returns " +
        "true.  Either change the arguments to point to different objects or substitute true.",
    category = GUAVA, severity = ERROR, maturity = MATURE)
public class SelfEquals extends BugChecker implements MethodInvocationTreeMatcher {

  /**
   * Matches calls to the Guava method Objects.equal() in which the two arguments are
   * the same reference.
   *
   * Example: Objects.equal(foo, foo)
   */
  @SuppressWarnings({"unchecked"})
  private static final Matcher<MethodInvocationTree> guavaMatcher = allOf(
      methodSelect(staticMethod("com.google.common.base.Objects", "equal")),
      sameArgument(0, 1));

  /**
   * Matches calls to any instance method called "equals" with exactly one argument in which the
   * receiver is the same reference as the argument.
   *
   * Example: foo.equals(foo)
   *
   * TODO(user): This may match too many things, if people are calling methods "equals" that
   * don't really mean equals.
   */
  private static final Matcher<MethodInvocationTree> equalsMatcher = allOf(
      methodSelect(Matchers.instanceMethod(Matchers.<ExpressionTree>anything(), "equals")),
      receiverSameAsArgument(0));

  /**
   * The state of the matcher.  Caches the result of matches() for use in describe().
   */
  private MatchState matchState = MatchState.NONE;

  private enum MatchState {
    NONE,
    OBJECTS_EQUAL,
    EQUALS
  }

  /**
   * Should this matcher check for Objects.equal(foo, foo)?
   */
  private boolean checkGuava = true;

  /**
   * Should this matcher check for foo.equals(foo)?
   */
  private boolean checkEquals = false;

  public SelfEquals() {
  }

  /**
   * Construct a new SelfEquals matcher.
   *
   * @param checkGuava Check for Guava Objects.equal(foo, foo) pattern?
   * @param checkEquals Check for foo.equals(foo) pattern?
   */
  // TODO(user): looks like this wants to be two different checks
  public SelfEquals(boolean checkGuava, boolean checkEquals) {
    if (!checkGuava && !checkEquals) {
      throw new IllegalArgumentException("SelfEquals should check something");
    }
    this.checkGuava = checkGuava;
    this.checkEquals = checkEquals;
  }

  @Override
  public Description matchMethodInvocation(MethodInvocationTree methodInvocationTree, VisitorState state) {
    if (checkGuava && guavaMatcher.matches(methodInvocationTree, state)) {
      matchState = MatchState.OBJECTS_EQUAL;
      return describe(methodInvocationTree, state);
    } else if (checkEquals && equalsMatcher.matches(methodInvocationTree, state)) {
      matchState = MatchState.EQUALS;
      return describe(methodInvocationTree, state);
    } else {
      return Description.NO_MATCH;
    }
  }

  public Description describe(MethodInvocationTree methodInvocationTree, VisitorState state) {
    if (matchState == MatchState.NONE) {
      throw new IllegalStateException("describe() called without a match");
    }

    // If we don't find a good field to use, then just replace with "true"
    Fix fix = SuggestedFix.replace(methodInvocationTree, "true");

    if (matchState == MatchState.OBJECTS_EQUAL) {
      /**
       * Cases:
       *    1) Objects.equal(foo, foo) ==> Objects.equal(foo, other.foo)
       *    2) Objects.equal(foo, this.foo) ==> Objects.equal(other.foo, this.foo)
       *    3) Objects.equal(this.foo, foo) ==> Objects.equal(this.foo, other.foo)
       *    4) Objects.equal(this.foo, this.foo) ==> Objects.equal(this.foo, other.foo)
       */
      // Assumption: Both arguments are either identifiers or field accesses.
      List<? extends ExpressionTree> args = methodInvocationTree.getArguments();
      for (ExpressionTree arg : args) {
        switch (arg.getKind()) {
          case IDENTIFIER: case MEMBER_SELECT:
            break;
          default:
            throw new IllegalStateException("Expected arg " + arg + " to be a field access or "
                + "identifier");
        }
      }

      // Choose argument to replace.
      ExpressionTree toReplace;
      if (args.get(1).getKind() == Kind.IDENTIFIER) {
        toReplace = args.get(1);
      } else if (args.get(0).getKind() == Kind.IDENTIFIER) {
        toReplace = args.get(0);
      } else {
        // If we don't have a good reason to replace one or the other, replace the second.
        toReplace = args.get(1);
      }
      // Find containing block
      TreePath path = state.getPath();
      while(path.getLeaf().getKind() != Kind.BLOCK) {
        path = path.getParentPath();
      }
      JCBlock block = (JCBlock)path.getLeaf();
      for (JCStatement jcStatement : block.getStatements()) {
        if (jcStatement.getKind() == Kind.VARIABLE) {
          JCVariableDecl declaration = (JCVariableDecl) jcStatement;
          TypeSymbol variableTypeSymbol = declaration.getType().type.tsym;

          if (ASTHelpers.getSymbol(toReplace).isMemberOf(variableTypeSymbol, state.getTypes())) {
            if (toReplace.getKind() == Kind.IDENTIFIER) {
              fix = SuggestedFix.prefixWith(toReplace, declaration.getName() + ".");
            } else {
              fix = SuggestedFix.replace(((JCFieldAccess) toReplace).getExpression(),
                  declaration.getName().toString());
            }
          }
        }
      }
    }

    return describeMatch(methodInvocationTree, fix);
  }
}
TOP

Related Classes of com.google.errorprone.bugpatterns.SelfEquals

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.