/* * Copyright (C) 2014 The Dagger 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 dagger.internal.codegen.binding; import static com.google.common.base.Preconditions.checkArgument; import static com.google.common.base.Preconditions.checkNotNull; import static com.google.common.collect.Iterables.consumingIterable; import static dagger.internal.codegen.base.ComponentAnnotation.subcomponentAnnotation; import static dagger.internal.codegen.base.ModuleAnnotation.moduleAnnotation; import static dagger.internal.codegen.base.MoreAnnotationMirrors.getTypeListValue; import static dagger.internal.codegen.binding.ComponentCreatorAnnotation.subcomponentCreatorAnnotations; import static dagger.internal.codegen.langmodel.DaggerElements.isAnyAnnotationPresent; import static javax.lang.model.util.ElementFilter.typesIn; import com.google.auto.common.MoreElements; import com.google.auto.common.MoreTypes; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableSet; import com.google.common.collect.Iterables; import com.google.common.collect.Sets; import dagger.Component; import dagger.Module; import dagger.internal.codegen.langmodel.DaggerElements; import dagger.internal.codegen.langmodel.DaggerTypes; import java.lang.annotation.Annotation; import java.util.ArrayDeque; import java.util.List; import java.util.Optional; import java.util.Queue; import java.util.Set; import javax.lang.model.element.AnnotationMirror; import javax.lang.model.element.Element; import javax.lang.model.element.TypeElement; import javax.lang.model.type.DeclaredType; import javax.lang.model.type.TypeKind; import javax.lang.model.type.TypeMirror; /** * Utility methods related to dagger configuration annotations (e.g.: {@link Component} and {@link * Module}). */ public final class ConfigurationAnnotations { public static Optional getSubcomponentCreator(TypeElement subcomponent) { checkArgument(subcomponentAnnotation(subcomponent).isPresent()); for (TypeElement nestedType : typesIn(subcomponent.getEnclosedElements())) { if (isSubcomponentCreator(nestedType)) { return Optional.of(nestedType); } } return Optional.empty(); } static boolean isSubcomponentCreator(Element element) { return isAnyAnnotationPresent(element, subcomponentCreatorAnnotations()); } // Dagger 1 support. public static ImmutableList getModuleInjects(AnnotationMirror moduleAnnotation) { checkNotNull(moduleAnnotation); return getTypeListValue(moduleAnnotation, "injects"); } /** Returns the first type that specifies this' nullability, or empty if none. */ public static Optional getNullableType(Element element) { List mirrors = element.getAnnotationMirrors(); for (AnnotationMirror mirror : mirrors) { if (mirror.getAnnotationType().asElement().getSimpleName().contentEquals("Nullable")) { return Optional.of(mirror.getAnnotationType()); } } return Optional.empty(); } /** * Returns the full set of modules transitively {@linkplain Module#includes included} from the * given seed modules. If a module is malformed and a type listed in {@link Module#includes} is * not annotated with {@link Module}, it is ignored. * * @deprecated Use {@link ComponentDescriptor#modules()}. */ @Deprecated public static ImmutableSet getTransitiveModules( DaggerTypes types, DaggerElements elements, Iterable seedModules) { TypeMirror objectType = elements.getTypeElement(Object.class).asType(); Queue moduleQueue = new ArrayDeque<>(); Iterables.addAll(moduleQueue, seedModules); Set moduleElements = Sets.newLinkedHashSet(); for (TypeElement moduleElement : consumingIterable(moduleQueue)) { moduleAnnotation(moduleElement) .ifPresent( moduleAnnotation -> { ImmutableSet.Builder moduleDependenciesBuilder = ImmutableSet.builder(); moduleDependenciesBuilder.addAll(moduleAnnotation.includes()); // We don't recur on the parent class because we don't want the parent class as a // root that the component depends on, and also because we want the dependencies // rooted against this element, not the parent. addIncludesFromSuperclasses( types, moduleElement, moduleDependenciesBuilder, objectType); ImmutableSet moduleDependencies = moduleDependenciesBuilder.build(); moduleElements.add(moduleElement); for (TypeElement dependencyType : moduleDependencies) { if (!moduleElements.contains(dependencyType)) { moduleQueue.add(dependencyType); } } }); } return ImmutableSet.copyOf(moduleElements); } /** Returns the enclosed types annotated with the given annotation. */ public static ImmutableList enclosedAnnotatedTypes( TypeElement typeElement, Class annotation) { final ImmutableList.Builder builders = ImmutableList.builder(); for (TypeElement element : typesIn(typeElement.getEnclosedElements())) { if (MoreElements.isAnnotationPresent(element, annotation)) { builders.add(MoreTypes.asDeclared(element.asType())); } } return builders.build(); } /** Traverses includes from superclasses and adds them into the builder. */ private static void addIncludesFromSuperclasses( DaggerTypes types, TypeElement element, ImmutableSet.Builder builder, TypeMirror objectType) { // Also add the superclass to the queue, in case any @Module definitions were on that. TypeMirror superclass = element.getSuperclass(); while (!types.isSameType(objectType, superclass) && superclass.getKind().equals(TypeKind.DECLARED)) { element = MoreElements.asType(types.asElement(superclass)); moduleAnnotation(element) .ifPresent(moduleAnnotation -> builder.addAll(moduleAnnotation.includes())); superclass = element.getSuperclass(); } } private ConfigurationAnnotations() {} }