dagger2.internal.codegen.ProvidesMethodValidator.java Source code

Java tutorial

Introduction

Here is the source code for dagger2.internal.codegen.ProvidesMethodValidator.java

Source

/*
 * Copyright (C) 2014 Google, Inc.
 *
 * 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 dagger2.internal.codegen;

import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import dagger2.Module;
import dagger2.Provides;
import java.util.Set;
import javax.lang.model.element.AnnotationMirror;
import javax.lang.model.element.Element;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.TypeElement;
import javax.lang.model.type.DeclaredType;
import javax.lang.model.type.TypeKind;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.util.Elements;

import static com.google.auto.common.MoreElements.isAnnotationPresent;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import static dagger2.internal.codegen.ConfigurationAnnotations.getMapKeys;
import static dagger2.internal.codegen.ErrorMessages.BINDING_METHOD_ABSTRACT;
import static dagger2.internal.codegen.ErrorMessages.BINDING_METHOD_MUST_RETURN_A_VALUE;
import static dagger2.internal.codegen.ErrorMessages.BINDING_METHOD_NOT_IN_MODULE;
import static dagger2.internal.codegen.ErrorMessages.BINDING_METHOD_NOT_MAP_HAS_MAP_KEY;
import static dagger2.internal.codegen.ErrorMessages.BINDING_METHOD_PRIVATE;
import static dagger2.internal.codegen.ErrorMessages.BINDING_METHOD_SET_VALUES_RAW_SET;
import static dagger2.internal.codegen.ErrorMessages.BINDING_METHOD_TYPE_PARAMETER;
import static dagger2.internal.codegen.ErrorMessages.BINDING_METHOD_WITH_MULTIPLE_MAP_KEY;
import static dagger2.internal.codegen.ErrorMessages.BINDING_METHOD_WITH_NO_MAP_KEY;
import static dagger2.internal.codegen.ErrorMessages.PROVIDES_METHOD_RETURN_TYPE;
import static dagger2.internal.codegen.ErrorMessages.PROVIDES_METHOD_SET_VALUES_RETURN_SET;
import static dagger2.internal.codegen.ErrorMessages.PROVIDES_OR_PRODUCES_METHOD_MULTIPLE_QUALIFIERS;
import static dagger2.internal.codegen.InjectionAnnotations.getQualifiers;
import static javax.lang.model.element.Modifier.ABSTRACT;
import static javax.lang.model.element.Modifier.PRIVATE;
import static javax.lang.model.type.TypeKind.ARRAY;
import static javax.lang.model.type.TypeKind.DECLARED;
import static javax.lang.model.type.TypeKind.VOID;

/**
 * A {@link Validator} for {@link Provides} methods.
 *
 * @author Gregory Kick
 * @since 2.0
 */
final class ProvidesMethodValidator implements Validator<ExecutableElement> {
    private final Elements elements;

    ProvidesMethodValidator(Elements elements) {
        this.elements = checkNotNull(elements);
    }

    private TypeElement getSetElement() {
        return elements.getTypeElement(Set.class.getCanonicalName());
    }

    @Override
    public ValidationReport<ExecutableElement> validate(ExecutableElement providesMethodElement) {
        ValidationReport.Builder<ExecutableElement> builder = ValidationReport.Builder.about(providesMethodElement);

        Provides providesAnnotation = providesMethodElement.getAnnotation(Provides.class);
        checkArgument(providesAnnotation != null);

        Element enclosingElement = providesMethodElement.getEnclosingElement();
        if (!isAnnotationPresent(enclosingElement, Module.class)) {
            builder.addItem(formatModuleErrorMessage(BINDING_METHOD_NOT_IN_MODULE), providesMethodElement);
        }

        if (!providesMethodElement.getTypeParameters().isEmpty()) {
            builder.addItem(formatErrorMessage(BINDING_METHOD_TYPE_PARAMETER), providesMethodElement);
        }

        Set<Modifier> modifiers = providesMethodElement.getModifiers();
        if (modifiers.contains(PRIVATE)) {
            builder.addItem(formatErrorMessage(BINDING_METHOD_PRIVATE), providesMethodElement);
        }
        if (modifiers.contains(ABSTRACT)) {
            builder.addItem(formatErrorMessage(BINDING_METHOD_ABSTRACT), providesMethodElement);
        }

        TypeMirror returnType = providesMethodElement.getReturnType();
        TypeKind returnTypeKind = returnType.getKind();
        if (returnTypeKind.equals(VOID)) {
            builder.addItem(formatErrorMessage(BINDING_METHOD_MUST_RETURN_A_VALUE), providesMethodElement);
        }

        // check mapkey is right
        if (!providesAnnotation.type().equals(Provides.Type.MAP)
                && (getMapKeys(providesMethodElement) != null && getMapKeys(providesMethodElement).size() > 0)) {
            builder.addItem(formatErrorMessage(BINDING_METHOD_NOT_MAP_HAS_MAP_KEY), providesMethodElement);
        }

        validateMethodQualifiers(builder, providesMethodElement);

        switch (providesAnnotation.type()) {
        case UNIQUE: // fall through
        case SET:
            validateKeyType(builder, returnType);
            break;
        case MAP:
            validateKeyType(builder, returnType);
            ImmutableSet<? extends AnnotationMirror> annotationMirrors = getMapKeys(providesMethodElement);
            switch (annotationMirrors.size()) {
            case 0:
                builder.addItem(formatErrorMessage(BINDING_METHOD_WITH_NO_MAP_KEY), providesMethodElement);
                break;
            case 1:
                break;
            default:
                builder.addItem(formatErrorMessage(BINDING_METHOD_WITH_MULTIPLE_MAP_KEY), providesMethodElement);
                break;
            }
            break;
        case SET_VALUES:
            if (!returnTypeKind.equals(DECLARED)) {
                builder.addItem(PROVIDES_METHOD_SET_VALUES_RETURN_SET, providesMethodElement);
            } else {
                DeclaredType declaredReturnType = (DeclaredType) returnType;
                // TODO(gak): should we allow "covariant return" for set values?
                if (!declaredReturnType.asElement().equals(getSetElement())) {
                    builder.addItem(PROVIDES_METHOD_SET_VALUES_RETURN_SET, providesMethodElement);
                } else if (declaredReturnType.getTypeArguments().isEmpty()) {
                    builder.addItem(formatErrorMessage(BINDING_METHOD_SET_VALUES_RAW_SET), providesMethodElement);
                } else {
                    validateKeyType(builder, Iterables.getOnlyElement(declaredReturnType.getTypeArguments()));
                }
            }
            break;
        default:
            throw new AssertionError();
        }

        return builder.build();
    }

    /** Validates that a Provides or Produces method doesn't have multiple qualifiers. */
    static void validateMethodQualifiers(ValidationReport.Builder<ExecutableElement> builder,
            ExecutableElement methodElement) {
        ImmutableSet<? extends AnnotationMirror> qualifiers = getQualifiers(methodElement);
        if (qualifiers.size() > 1) {
            for (AnnotationMirror qualifier : qualifiers) {
                builder.addItem(PROVIDES_OR_PRODUCES_METHOD_MULTIPLE_QUALIFIERS, methodElement, qualifier);
            }
        }
    }

    private String formatErrorMessage(String msg) {
        return String.format(msg, Provides.class.getSimpleName());
    }

    private String formatModuleErrorMessage(String msg) {
        return String.format(msg, Provides.class.getSimpleName(), Module.class.getSimpleName());
    }

    private void validateKeyType(ValidationReport.Builder<? extends Element> reportBuilder, TypeMirror type) {
        TypeKind kind = type.getKind();
        if (!(kind.isPrimitive() || kind.equals(DECLARED) || kind.equals(ARRAY))) {
            reportBuilder.addItem(PROVIDES_METHOD_RETURN_TYPE, reportBuilder.getSubject());
        }
    }
}