dagger.internal.codegen.ModuleKind.java Source code

Java tutorial

Introduction

Here is the source code for dagger.internal.codegen.ModuleKind.java

Source

/*
 * Copyright (C) 2015 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;

import static com.google.common.base.Preconditions.checkArgument;
import static dagger.internal.codegen.DaggerElements.getAnnotationMirror;
import static dagger.internal.codegen.DaggerStreams.toImmutableSet;

import com.google.auto.common.MoreElements;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Sets;
import dagger.Module;
import dagger.Provides;
import dagger.producers.ProducerModule;
import dagger.producers.Produces;
import java.lang.annotation.Annotation;
import java.util.EnumSet;
import java.util.Optional;
import java.util.Set;
import javax.lang.model.element.AnnotationMirror;
import javax.lang.model.element.TypeElement;

/** Enumeration of the kinds of modules. */
enum ModuleKind {
    /** {@code @Module} */
    MODULE(Module.class, Provides.class),

    /** {@code @ProducerModule} */
    PRODUCER_MODULE(ProducerModule.class, Produces.class);

    /** Returns the annotations for modules of the given kinds. */
    static ImmutableSet<Class<? extends Annotation>> annotationsFor(Set<ModuleKind> kinds) {
        return kinds.stream().map(ModuleKind::annotation).collect(toImmutableSet());
    }

    /**
     * Returns the kind of an annotated element if it is annotated with one of the module {@linkplain
     * #annotation() annotations}.
     *
     * @throws IllegalArgumentException if the element is annotated with more than one of the module
     *     annotations
     */
    static Optional<ModuleKind> forAnnotatedElement(TypeElement element) {
        Set<ModuleKind> kinds = EnumSet.noneOf(ModuleKind.class);
        for (ModuleKind kind : values()) {
            if (MoreElements.isAnnotationPresent(element, kind.annotation())) {
                kinds.add(kind);
            }
        }

        if (kinds.size() > 1) {
            throw new IllegalArgumentException(
                    element + " cannot be annotated with more than one of " + annotationsFor(kinds));
        }
        return kinds.stream().findAny();
    }

    private final Class<? extends Annotation> moduleAnnotation;
    private final Class<? extends Annotation> methodAnnotation;

    ModuleKind(Class<? extends Annotation> moduleAnnotation, Class<? extends Annotation> methodAnnotation) {
        this.moduleAnnotation = moduleAnnotation;
        this.methodAnnotation = methodAnnotation;
    }

    /**
     * Returns the annotation mirror for this module kind on the given type.
     *
     * @throws IllegalArgumentException if the annotation is not present on the type
     */
    AnnotationMirror getModuleAnnotation(TypeElement element) {
        Optional<AnnotationMirror> result = getAnnotationMirror(element, moduleAnnotation);
        checkArgument(result.isPresent(), "annotation %s is not present on type %s", moduleAnnotation, element);
        return result.get();
    }

    /** Returns the annotation that marks a module of this kind. */
    Class<? extends Annotation> annotation() {
        return moduleAnnotation;
    }

    /** Returns the annotation for binding methods on this type of module. */
    // TODO(cgdecker): Validate how this is used... is it really correct? Producer modules can also
    // have @Provides methods.
    Class<? extends Annotation> methodAnnotation() {
        return methodAnnotation;
    }

    /** Returns the kinds of modules that a module of this kind is allowed to include. */
    ImmutableSet<ModuleKind> legalIncludedModuleKinds() {
        switch (this) {
        case MODULE:
            return Sets.immutableEnumSet(MODULE);
        case PRODUCER_MODULE:
            return Sets.immutableEnumSet(MODULE, PRODUCER_MODULE);
        }
        throw new AssertionError(this);
    }
}