/* * 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 androidx.room.compiler.processing.XProcessingEnv; import androidx.room.compiler.processing.XTypeElement; 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; /** 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, XProcessingEnv env) { 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)) .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 (XTypeElement componentElement : metadata.componentElements()) { ClassName componentName = componentElement.getClassName(); 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(PkgPrivateMetadata::publicEarlyEntryPoint) .collect(toImmutableSet())); return componentDependencies.build(); } }