dagger.internal.codegen.bindinggraphvalidation.SetMultibindingValidator Maven / Gradle / Ivy
Show all versions of dagger-compiler Show documentation
/*
* Copyright (C) 2021 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.bindinggraphvalidation;
import static dagger.internal.codegen.model.BindingKind.DELEGATE;
import static dagger.internal.codegen.model.BindingKind.MULTIBOUND_SET;
import static javax.tools.Diagnostic.Kind.ERROR;
import com.google.common.base.Joiner;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.Multimap;
import dagger.internal.codegen.model.Binding;
import dagger.internal.codegen.model.BindingGraph;
import dagger.internal.codegen.model.DiagnosticReporter;
import dagger.internal.codegen.model.Key;
import dagger.internal.codegen.validation.ValidationBindingGraphPlugin;
import java.util.Optional;
import javax.inject.Inject;
/** Validates that there are not multiple set binding contributions to the same binding. */
final class SetMultibindingValidator extends ValidationBindingGraphPlugin {
@Inject
SetMultibindingValidator() {
}
@Override
public String pluginName() {
return "Dagger/SetMultibinding";
}
@Override
public void visitGraph(BindingGraph bindingGraph, DiagnosticReporter diagnosticReporter) {
bindingGraph.bindings().stream()
.filter(binding -> binding.kind().equals(MULTIBOUND_SET))
.forEach(
binding ->
checkForDuplicateSetContributions(binding, bindingGraph, diagnosticReporter));
}
private void checkForDuplicateSetContributions(
Binding binding, BindingGraph bindingGraph, DiagnosticReporter diagnosticReporter) {
// Map of delegate target key to the original contribution binding
Multimap dereferencedBindsTargets = HashMultimap.create();
for (Binding dep : bindingGraph.requestedBindings(binding)) {
if (dep.kind().equals(DELEGATE)) {
dereferenceDelegateBinding(dep, bindingGraph)
.ifPresent(dereferencedKey -> dereferencedBindsTargets.put(dereferencedKey, dep));
}
}
dereferencedBindsTargets
.asMap()
.forEach(
(targetKey, contributions) -> {
if (contributions.size() > 1) {
diagnosticReporter.reportComponent(
ERROR,
bindingGraph.componentNode(binding.componentPath()).get(),
"Multiple set contributions into %s for the same contribution key: %s.\n\n"
+ " %s\n",
binding.key(),
targetKey,
Joiner.on("\n ").join(contributions));
}
});
}
/**
* Returns the dereferenced key of a delegate binding (going through other delegates as well).
*
* If the binding cannot be dereferenced (because it leads to a missing binding or duplicate
* bindings) then {@link Optional#empty()} is returned.
*/
private Optional dereferenceDelegateBinding(Binding binding, BindingGraph bindingGraph) {
ImmutableSet delegateSet = bindingGraph.requestedBindings(binding);
if (delegateSet.size() != 1) {
// If there isn't exactly 1 delegate then it means either a MissingBinding or DuplicateBinding
// error will be reported. Just return nothing rather than trying to dereference further, as
// anything we report here will just be noise on top of the other error anyway.
return Optional.empty();
}
// If there is a binding, first we check if that is a delegate binding so we can dereference
// that binding if needed.
Binding delegate = Iterables.getOnlyElement(delegateSet);
if (delegate.kind().equals(DELEGATE)) {
return dereferenceDelegateBinding(delegate, bindingGraph);
}
return Optional.of(delegate.key());
}
}