1*f585d8a3SJacky Wang /* 2*f585d8a3SJacky Wang * Copyright (C) 2015 The Dagger Authors. 3*f585d8a3SJacky Wang * 4*f585d8a3SJacky Wang * Licensed under the Apache License, Version 2.0 (the "License"); 5*f585d8a3SJacky Wang * you may not use this file except in compliance with the License. 6*f585d8a3SJacky Wang * You may obtain a copy of the License at 7*f585d8a3SJacky Wang * 8*f585d8a3SJacky Wang * http://www.apache.org/licenses/LICENSE-2.0 9*f585d8a3SJacky Wang * 10*f585d8a3SJacky Wang * Unless required by applicable law or agreed to in writing, software 11*f585d8a3SJacky Wang * distributed under the License is distributed on an "AS IS" BASIS, 12*f585d8a3SJacky Wang * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*f585d8a3SJacky Wang * See the License for the specific language governing permissions and 14*f585d8a3SJacky Wang * limitations under the License. 15*f585d8a3SJacky Wang */ 16*f585d8a3SJacky Wang 17*f585d8a3SJacky Wang package dagger.model; 18*f585d8a3SJacky Wang 19*f585d8a3SJacky Wang import static com.google.auto.common.MoreElements.isAnnotationPresent; 20*f585d8a3SJacky Wang import static com.google.common.base.Preconditions.checkArgument; 21*f585d8a3SJacky Wang 22*f585d8a3SJacky Wang import com.google.auto.common.AnnotationMirrors; 23*f585d8a3SJacky Wang import com.google.auto.common.MoreElements; 24*f585d8a3SJacky Wang import com.google.auto.common.MoreTypes; 25*f585d8a3SJacky Wang import com.google.auto.value.AutoValue; 26*f585d8a3SJacky Wang import com.google.common.base.Equivalence; 27*f585d8a3SJacky Wang import com.squareup.javapoet.ClassName; 28*f585d8a3SJacky Wang import dagger.Reusable; 29*f585d8a3SJacky Wang import javax.inject.Singleton; 30*f585d8a3SJacky Wang import javax.lang.model.element.AnnotationMirror; 31*f585d8a3SJacky Wang import javax.lang.model.element.TypeElement; 32*f585d8a3SJacky Wang 33*f585d8a3SJacky Wang /** A representation of a {@link javax.inject.Scope}. */ 34*f585d8a3SJacky Wang @AutoValue 35*f585d8a3SJacky Wang // TODO(ronshapiro): point to SimpleAnnotationMirror 36*f585d8a3SJacky Wang public abstract class Scope { wrappedScopeAnnotation()37*f585d8a3SJacky Wang abstract Equivalence.Wrapper<AnnotationMirror> wrappedScopeAnnotation(); 38*f585d8a3SJacky Wang 39*f585d8a3SJacky Wang /** The {@link AnnotationMirror} that represents the scope annotation. */ scopeAnnotation()40*f585d8a3SJacky Wang public final AnnotationMirror scopeAnnotation() { 41*f585d8a3SJacky Wang return wrappedScopeAnnotation().get(); 42*f585d8a3SJacky Wang } 43*f585d8a3SJacky Wang 44*f585d8a3SJacky Wang /** The scope annotation element. */ scopeAnnotationElement()45*f585d8a3SJacky Wang public final TypeElement scopeAnnotationElement() { 46*f585d8a3SJacky Wang return MoreTypes.asTypeElement(scopeAnnotation().getAnnotationType()); 47*f585d8a3SJacky Wang } 48*f585d8a3SJacky Wang 49*f585d8a3SJacky Wang /** 50*f585d8a3SJacky Wang * Creates a {@link Scope} object from the {@link javax.inject.Scope}-annotated annotation type. 51*f585d8a3SJacky Wang */ scope(AnnotationMirror scopeAnnotation)52*f585d8a3SJacky Wang public static Scope scope(AnnotationMirror scopeAnnotation) { 53*f585d8a3SJacky Wang checkArgument(isScope(scopeAnnotation)); 54*f585d8a3SJacky Wang return new AutoValue_Scope(AnnotationMirrors.equivalence().wrap(scopeAnnotation)); 55*f585d8a3SJacky Wang } 56*f585d8a3SJacky Wang 57*f585d8a3SJacky Wang /** 58*f585d8a3SJacky Wang * Returns {@code true} if {@link #scopeAnnotation()} is a {@link javax.inject.Scope} annotation. 59*f585d8a3SJacky Wang */ isScope(AnnotationMirror scopeAnnotation)60*f585d8a3SJacky Wang public static boolean isScope(AnnotationMirror scopeAnnotation) { 61*f585d8a3SJacky Wang return isScope(MoreElements.asType(scopeAnnotation.getAnnotationType().asElement())); 62*f585d8a3SJacky Wang } 63*f585d8a3SJacky Wang 64*f585d8a3SJacky Wang /** 65*f585d8a3SJacky Wang * Returns {@code true} if {@code scopeAnnotationType} is a {@link javax.inject.Scope} annotation. 66*f585d8a3SJacky Wang */ isScope(TypeElement scopeAnnotationType)67*f585d8a3SJacky Wang public static boolean isScope(TypeElement scopeAnnotationType) { 68*f585d8a3SJacky Wang return isAnnotationPresent(scopeAnnotationType, SCOPE.canonicalName()) 69*f585d8a3SJacky Wang || isAnnotationPresent(scopeAnnotationType, SCOPE_JAVAX.canonicalName()); 70*f585d8a3SJacky Wang } 71*f585d8a3SJacky Wang 72*f585d8a3SJacky Wang private static final ClassName PRODUCTION_SCOPE = 73*f585d8a3SJacky Wang ClassName.get("dagger.producers", "ProductionScope"); 74*f585d8a3SJacky Wang private static final ClassName SINGLETON = ClassName.get("jakarta.inject", "Singleton"); 75*f585d8a3SJacky Wang private static final ClassName SINGLETON_JAVAX = ClassName.get("javax.inject", "Singleton"); 76*f585d8a3SJacky Wang private static final ClassName REUSABLE = ClassName.get("dagger", "Reusable"); 77*f585d8a3SJacky Wang private static final ClassName SCOPE = ClassName.get("jakarta.inject", "Scope"); 78*f585d8a3SJacky Wang private static final ClassName SCOPE_JAVAX = ClassName.get("javax.inject", "Scope"); 79*f585d8a3SJacky Wang 80*f585d8a3SJacky Wang /** Returns {@code true} if this scope is the {@link Singleton @Singleton} scope. */ isSingleton()81*f585d8a3SJacky Wang public final boolean isSingleton() { 82*f585d8a3SJacky Wang return isScope(SINGLETON) || isScope(SINGLETON_JAVAX); 83*f585d8a3SJacky Wang } 84*f585d8a3SJacky Wang 85*f585d8a3SJacky Wang /** Returns {@code true} if this scope is the {@link Reusable @Reusable} scope. */ isReusable()86*f585d8a3SJacky Wang public final boolean isReusable() { 87*f585d8a3SJacky Wang return isScope(REUSABLE); 88*f585d8a3SJacky Wang } 89*f585d8a3SJacky Wang 90*f585d8a3SJacky Wang /** Returns {@code true} if this scope is the {@code @ProductionScope} scope. */ isProductionScope()91*f585d8a3SJacky Wang public final boolean isProductionScope() { 92*f585d8a3SJacky Wang return isScope(PRODUCTION_SCOPE); 93*f585d8a3SJacky Wang } 94*f585d8a3SJacky Wang isScope(ClassName className)95*f585d8a3SJacky Wang private boolean isScope(ClassName className) { 96*f585d8a3SJacky Wang return scopeAnnotationElement().getQualifiedName().contentEquals(className.canonicalName()); 97*f585d8a3SJacky Wang } 98*f585d8a3SJacky Wang 99*f585d8a3SJacky Wang /** Returns a debug representation of the scope. */ 100*f585d8a3SJacky Wang @Override toString()101*f585d8a3SJacky Wang public final String toString() { 102*f585d8a3SJacky Wang return scopeAnnotation().toString(); 103*f585d8a3SJacky Wang } 104*f585d8a3SJacky Wang } 105