/* * Copyright (C) 2017 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.writing; import static com.google.common.base.Preconditions.checkNotNull; import static com.squareup.javapoet.MethodSpec.methodBuilder; import static dagger.internal.codegen.base.Util.reentrantComputeIfAbsent; import static dagger.internal.codegen.langmodel.Accessibility.isTypeAccessibleFrom; import static dagger.internal.codegen.writing.ComponentImplementation.MethodSpecKind.MEMBERS_INJECTION_METHOD; import static javax.lang.model.element.Modifier.PRIVATE; import com.google.common.collect.ImmutableSet; import com.squareup.javapoet.ClassName; import com.squareup.javapoet.CodeBlock; import com.squareup.javapoet.MethodSpec; import com.squareup.javapoet.ParameterSpec; import com.squareup.javapoet.TypeName; import dagger.internal.codegen.binding.Binding; import dagger.internal.codegen.binding.BindingGraph; import dagger.internal.codegen.binding.MembersInjectionBinding; import dagger.internal.codegen.binding.MembersInjectionBinding.InjectionSite; import dagger.internal.codegen.binding.ProvisionBinding; import dagger.internal.codegen.kotlin.KotlinMetadataUtil; import dagger.internal.codegen.langmodel.DaggerElements; import dagger.internal.codegen.langmodel.DaggerTypes; import dagger.internal.codegen.writing.InjectionMethods.InjectionSiteMethod; import dagger.model.Key; import java.util.LinkedHashMap; import java.util.Map; import javax.lang.model.element.Name; import javax.lang.model.element.TypeElement; import javax.lang.model.type.TypeMirror; /** Manages the member injection methods for a component. */ final class MembersInjectionMethods { private final Map membersInjectionMethods = new LinkedHashMap<>(); private final ComponentImplementation componentImplementation; private final ComponentBindingExpressions bindingExpressions; private final BindingGraph graph; private final DaggerElements elements; private final DaggerTypes types; private final KotlinMetadataUtil metadataUtil; MembersInjectionMethods( ComponentImplementation componentImplementation, ComponentBindingExpressions bindingExpressions, BindingGraph graph, DaggerElements elements, DaggerTypes types, KotlinMetadataUtil metadataUtil) { this.componentImplementation = checkNotNull(componentImplementation); this.bindingExpressions = checkNotNull(bindingExpressions); this.graph = checkNotNull(graph); this.elements = checkNotNull(elements); this.types = checkNotNull(types); this.metadataUtil = metadataUtil; } /** * Returns the members injection {@link MethodSpec} for the given {@link Key}, creating it if * necessary. */ MethodSpec getOrCreate(Key key) { return reentrantComputeIfAbsent(membersInjectionMethods, key, this::membersInjectionMethod); } private MethodSpec membersInjectionMethod(Key key) { Binding binding = graph.membersInjectionBinding(key).isPresent() ? graph.membersInjectionBinding(key).get() : graph.contributionBinding(key); TypeMirror keyType = binding.key().type(); TypeMirror membersInjectedType = isTypeAccessibleFrom(keyType, componentImplementation.name().packageName()) ? keyType : elements.getTypeElement(Object.class).asType(); TypeName membersInjectedTypeName = TypeName.get(membersInjectedType); Name bindingTypeName = binding.bindingTypeElement().get().getSimpleName(); // TODO(ronshapiro): include type parameters in this name e.g. injectFooOfT, and outer class // simple names Foo.Builder -> injectFooBuilder String methodName = componentImplementation.getUniqueMethodName("inject" + bindingTypeName); ParameterSpec parameter = ParameterSpec.builder(membersInjectedTypeName, "instance").build(); MethodSpec.Builder methodBuilder = methodBuilder(methodName) .addModifiers(PRIVATE) .returns(membersInjectedTypeName) .addParameter(parameter); TypeElement canIgnoreReturnValue = elements.getTypeElement("com.google.errorprone.annotations.CanIgnoreReturnValue"); if (canIgnoreReturnValue != null) { methodBuilder.addAnnotation(ClassName.get(canIgnoreReturnValue)); } CodeBlock instance = CodeBlock.of("$N", parameter); methodBuilder.addCode( InjectionSiteMethod.invokeAll( injectionSites(binding), componentImplementation.name(), instance, membersInjectedType, request -> bindingExpressions .getDependencyArgumentExpression(request, componentImplementation.name()) .codeBlock(), types, metadataUtil)); methodBuilder.addStatement("return $L", instance); MethodSpec method = methodBuilder.build(); componentImplementation.addMethod(MEMBERS_INJECTION_METHOD, method); return method; } private static ImmutableSet injectionSites(Binding binding) { if (binding instanceof ProvisionBinding) { return ((ProvisionBinding) binding).injectionSites(); } else if (binding instanceof MembersInjectionBinding) { return ((MembersInjectionBinding) binding).injectionSites(); } throw new IllegalArgumentException(binding.key().toString()); } }