blob: ec9d208e748e6b4ed3d4ffd995a1ca62e9f3742b [file] [log] [blame]
/*
* Copyright (C) 2016 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.auto.common.MoreElements.asExecutable;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkState;
import static dagger.internal.codegen.Accessibility.isTypeAccessibleFrom;
import static dagger.internal.codegen.CodeBlocks.toParametersCodeBlock;
import static dagger.internal.codegen.ContributionBinding.Kind.INJECTION;
import static dagger.internal.codegen.FactoryGenerator.checkNotNullProvidesMethod;
import static dagger.internal.codegen.InjectionMethods.ProvisionMethod.requiresInjectionMethod;
import static dagger.internal.codegen.TypeNames.rawTypeName;
import static javax.lang.model.element.Modifier.STATIC;
import com.google.auto.common.MoreTypes;
import com.squareup.javapoet.ClassName;
import com.squareup.javapoet.CodeBlock;
import com.squareup.javapoet.TypeName;
import dagger.internal.codegen.InjectionMethods.ProvisionMethod;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.type.DeclaredType;
/**
* A binding expression that invokes methods or constructors directly for a provision binding when
* possible.
*/
final class SimpleMethodBindingExpression extends SimpleInvocationBindingExpression {
private final CompilerOptions compilerOptions;
private final ProvisionBinding provisionBinding;
private final ComponentBindingExpressions componentBindingExpressions;
private final GeneratedComponentModel generatedComponentModel;
SimpleMethodBindingExpression(
CompilerOptions compilerOptions,
ProvisionBinding provisionBinding,
BindingExpression delegate,
ComponentBindingExpressions componentBindingExpressions,
GeneratedComponentModel generatedComponentModel) {
super(delegate);
checkArgument(
provisionBinding.implicitDependencies().isEmpty(),
"framework deps are not currently supported");
checkArgument(!provisionBinding.scope().isPresent());
checkArgument(!provisionBinding.requiresModuleInstance());
checkArgument(provisionBinding.bindingElement().isPresent());
this.compilerOptions = compilerOptions;
this.provisionBinding = provisionBinding;
this.componentBindingExpressions = componentBindingExpressions;
this.generatedComponentModel = generatedComponentModel;
}
@Override
CodeBlock getInstanceDependencyExpression(
DependencyRequest.Kind requestKind, ClassName requestingClass) {
return requiresInjectionMethod(provisionBinding, requestingClass.packageName())
? invokeInjectionMethod(requestingClass)
: invokeMethod(requestingClass);
}
private CodeBlock invokeMethod(ClassName requestingClass) {
// TODO(dpb): align this with the contents of InlineMethods.create
CodeBlock arguments =
provisionBinding
.dependencies()
.stream()
.map(request -> dependencyArgument(request, requestingClass))
.collect(toParametersCodeBlock());
ExecutableElement method = asExecutable(provisionBinding.bindingElement().get());
switch (method.getKind()) {
case CONSTRUCTOR:
return CodeBlock.of("new $T($L)", constructorTypeName(requestingClass), arguments);
case METHOD:
checkState(method.getModifiers().contains(STATIC));
return maybeCheckForNulls(
CodeBlock.of(
"$T.$L($L)",
provisionBinding.bindingTypeElement().get(),
method.getSimpleName(),
arguments));
default:
throw new IllegalStateException();
}
}
private TypeName constructorTypeName(ClassName requestingClass) {
DeclaredType type = MoreTypes.asDeclared(provisionBinding.key().type());
TypeName typeName = TypeName.get(type);
if (type.getTypeArguments()
.stream()
.allMatch(t -> isTypeAccessibleFrom(t, requestingClass.packageName()))) {
return typeName;
}
return rawTypeName(typeName);
}
private CodeBlock invokeInjectionMethod(ClassName requestingClass) {
return injectMembers(
maybeCheckForNulls(
ProvisionMethod.invoke(
provisionBinding,
request -> dependencyArgument(request, requestingClass),
requestingClass)));
}
private CodeBlock dependencyArgument(DependencyRequest dependency, ClassName requestingClass) {
return componentBindingExpressions.getDependencyArgumentExpression(dependency, requestingClass);
}
private CodeBlock maybeCheckForNulls(CodeBlock methodCall) {
return !provisionBinding.bindingKind().equals(INJECTION)
&& provisionBinding.shouldCheckForNull(compilerOptions)
? checkNotNullProvidesMethod(methodCall)
: methodCall;
}
private CodeBlock injectMembers(CodeBlock instance) {
if (provisionBinding.injectionSites().isEmpty()) {
return instance;
}
// Java 7 type inference can't figure out that instance in
// injectParameterized(Parameterized_Factory.newParameterized()) is Parameterized<T> and not
// Parameterized<Object>
if (!MoreTypes.asDeclared(provisionBinding.key().type()).getTypeArguments().isEmpty()) {
TypeName keyType = TypeName.get(provisionBinding.key().type());
instance = CodeBlock.of("($T) ($T) $L", keyType, rawTypeName(keyType), instance);
}
return CodeBlock.of(
"$N($L)",
generatedComponentModel.getMembersInjectionMethod(provisionBinding.key()),
instance);
}
}