dagger.hilt.processor.internal.aggregateddeps.ComponentDependencies Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of hilt-android-compiler Show documentation
Show all versions of hilt-android-compiler Show documentation
A fast dependency injector for Android and Java.
/*
* Copyright (C) 2019 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.hilt.processor.internal.aggregateddeps;
import static com.google.common.base.Preconditions.checkState;
import static dagger.internal.codegen.extension.DaggerStreams.toImmutableSet;
import com.google.auto.value.AutoValue;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ImmutableSetMultimap;
import com.squareup.javapoet.ClassName;
import dagger.hilt.processor.internal.ClassNames;
import dagger.hilt.processor.internal.ComponentDescriptor;
import dagger.hilt.processor.internal.earlyentrypoint.AggregatedEarlyEntryPointMetadata;
import dagger.hilt.processor.internal.uninstallmodules.AggregatedUninstallModulesMetadata;
import javax.lang.model.element.TypeElement;
import javax.lang.model.util.Elements;
/** Represents information needed to create a component (i.e. modules, entry points, etc) */
@AutoValue
public abstract class ComponentDependencies {
private static Builder builder() {
return new AutoValue_ComponentDependencies.Builder();
}
/** Returns the modules for a component, without any filtering. */
public abstract ImmutableSetMultimap modules();
/** Returns the entry points associated with the given a component. */
public abstract ImmutableSetMultimap entryPoints();
/** Returns the component entry point associated with the given a component. */
public abstract ImmutableSetMultimap componentEntryPoints();
@AutoValue.Builder
abstract static class Builder {
abstract ImmutableSetMultimap.Builder modulesBuilder();
abstract ImmutableSetMultimap.Builder entryPointsBuilder();
abstract ImmutableSetMultimap.Builder componentEntryPointsBuilder();
abstract ComponentDependencies build();
}
/** Returns the component dependencies for the given metadata. */
public static ComponentDependencies from(
ImmutableSet descriptors,
ImmutableSet aggregatedDepsMetadata,
ImmutableSet aggregatedUninstallModulesMetadata,
ImmutableSet aggregatedEarlyEntryPointMetadata,
Elements elements) {
ImmutableSet uninstalledModules =
ImmutableSet.builder()
.addAll(
aggregatedUninstallModulesMetadata.stream()
.flatMap(metadata -> metadata.uninstallModuleElements().stream())
// @AggregatedUninstallModules always references the user module, so convert to
// the generated public wrapper if needed.
// TODO(bcorso): Consider converting this to the public module in the processor.
.map(module -> PkgPrivateMetadata.publicModule(module, elements))
.collect(toImmutableSet()))
.addAll(
aggregatedDepsMetadata.stream()
.flatMap(metadata -> metadata.replacedDependencies().stream())
.collect(toImmutableSet()))
.build();
ComponentDependencies.Builder componentDependencies = ComponentDependencies.builder();
ImmutableSet componentNames =
descriptors.stream().map(ComponentDescriptor::component).collect(toImmutableSet());
for (AggregatedDepsMetadata metadata : aggregatedDepsMetadata) {
for (TypeElement componentElement : metadata.componentElements()) {
ClassName componentName = ClassName.get(componentElement);
checkState(
componentNames.contains(componentName), "%s is not a valid Component.", componentName);
switch (metadata.dependencyType()) {
case MODULE:
if (!uninstalledModules.contains(metadata.dependency())) {
componentDependencies.modulesBuilder().put(componentName, metadata.dependency());
}
break;
case ENTRY_POINT:
componentDependencies.entryPointsBuilder().put(componentName, metadata.dependency());
break;
case COMPONENT_ENTRY_POINT:
componentDependencies
.componentEntryPointsBuilder()
.put(componentName, metadata.dependency());
break;
}
}
}
componentDependencies
.entryPointsBuilder()
.putAll(
ClassNames.SINGLETON_COMPONENT,
aggregatedEarlyEntryPointMetadata.stream()
.map(AggregatedEarlyEntryPointMetadata::earlyEntryPoint)
// @AggregatedEarlyEntryPointMetadata always references the user module, so convert
// to the generated public wrapper if needed.
// TODO(bcorso): Consider converting this to the public module in the processor.
.map(entryPoint -> PkgPrivateMetadata.publicEarlyEntryPoint(entryPoint, elements))
.collect(toImmutableSet()));
return componentDependencies.build();
}
}