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

com.google.errorprone.bugpatterns.inject.guice.AssistedInjectScoping Maven / Gradle / Ivy

There is a newer version: 2.28.0
Show newest version
/*
 * Copyright 2013 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.inject.guice;

import static com.google.errorprone.BugPattern.SeverityLevel.ERROR;
import static com.google.errorprone.matchers.ChildMultiMatcher.MatchType.AT_LEAST_ONE;
import static com.google.errorprone.matchers.InjectMatchers.ASSISTED_ANNOTATION;
import static com.google.errorprone.matchers.InjectMatchers.ASSISTED_INJECT_ANNOTATION;
import static com.google.errorprone.matchers.InjectMatchers.GUICE_SCOPE_ANNOTATION;
import static com.google.errorprone.matchers.InjectMatchers.JAVAX_SCOPE_ANNOTATION;
import static com.google.errorprone.matchers.InjectMatchers.hasInjectAnnotation;
import static com.google.errorprone.matchers.Matchers.annotations;
import static com.google.errorprone.matchers.Matchers.anyOf;
import static com.google.errorprone.matchers.Matchers.constructor;
import static com.google.errorprone.matchers.Matchers.hasAnnotation;
import static com.google.errorprone.matchers.Matchers.methodHasParameters;
import static com.google.errorprone.matchers.Matchers.symbolHasAnnotation;

import com.google.errorprone.BugPattern;
import com.google.errorprone.VisitorState;
import com.google.errorprone.bugpatterns.BugChecker;
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.google.errorprone.matchers.MultiMatcher;
import com.google.errorprone.matchers.MultiMatcher.MultiMatchResult;
import com.sun.source.tree.AnnotationTree;
import com.sun.source.tree.ClassTree;
import com.sun.source.tree.MethodTree;

/**
 * This checker matches iff *both* of the following conditions are true: 1) The class is assisted:
 * a) If there is a constructor that is annotated with @Inject and that constructor has at least one
 * parameter that is annotated with @Assisted. b) If there is no @Inject constructor and at least
 * one constructor is annotated with {@code @AssistedInject}. 2) There is an annotation on the
 * class, and the annotation is itself annotated with {@code @ScopeAnnotation}.
 *
 * @author [email protected] (Eddie Aftandilian)
 */
@BugPattern(
    name = "GuiceAssistedInjectScoping",
    summary = "Scope annotation on implementation class of AssistedInject factory is not allowed",
    severity = ERROR)
public class AssistedInjectScoping extends BugChecker implements ClassTreeMatcher {

  /** Matches classes that have an annotation that itself is annotated with @ScopeAnnotation. */
  private static final MultiMatcher CLASS_TO_SCOPE_ANNOTATIONS =
      annotations(
          AT_LEAST_ONE,
          anyOf(
              symbolHasAnnotation(GUICE_SCOPE_ANNOTATION),
              symbolHasAnnotation(JAVAX_SCOPE_ANNOTATION)));

  /** Matches if any constructor of a class is annotated with an @Inject annotation. */
  private static final MultiMatcher CLASS_TO_INJECTED_CONSTRUCTORS =
      constructor(AT_LEAST_ONE, hasInjectAnnotation());

  /**
   * Matches if: 1) If there is a constructor that is annotated with @Inject and that constructor
   * has at least one parameter that is annotated with @Assisted. 2) If there is no @Inject
   * constructor and at least one constructor is annotated with @AssistedInject.
   */
  private static final Matcher HAS_ASSISTED_CONSTRUCTOR =
      new Matcher() {
        @Override
        public boolean matches(ClassTree classTree, VisitorState state) {
          MultiMatchResult injectedConstructors =
              CLASS_TO_INJECTED_CONSTRUCTORS.multiMatchResult(classTree, state);
          if (injectedConstructors.matches()) {
            // Check constructor with @Inject annotation for parameter with @Assisted annotation.
            return methodHasParameters(AT_LEAST_ONE, hasAnnotation(ASSISTED_ANNOTATION))
                .matches(injectedConstructors.matchingNodes().get(0), state);
          }

          return constructor(AT_LEAST_ONE, hasAnnotation(ASSISTED_INJECT_ANNOTATION))
              .matches(classTree, state);
        }
      };

  @Override
  public final Description matchClass(ClassTree classTree, VisitorState state) {
    MultiMatchResult hasScopeAnnotations =
        CLASS_TO_SCOPE_ANNOTATIONS.multiMatchResult(classTree, state);
    if (!hasScopeAnnotations.matches() || !HAS_ASSISTED_CONSTRUCTOR.matches(classTree, state)) {
      return Description.NO_MATCH;
    }

    AnnotationTree annotationWithScopeAnnotation = hasScopeAnnotations.matchingNodes().get(0);
    if (annotationWithScopeAnnotation == null) {
      throw new IllegalStateException(
          "Expected to find an annotation that was annotated with @ScopeAnnotation");
    }

    return describeMatch(
        annotationWithScopeAnnotation, SuggestedFix.delete(annotationWithScopeAnnotation));
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy