| // Copyright 2015 The Bazel Authors. All rights reserved. |
| // |
| // 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 com.google.devtools.build.lib.skyframe; |
| |
| import com.google.common.base.Function; |
| import com.google.common.base.Objects; |
| import com.google.common.base.Throwables; |
| import com.google.common.collect.ImmutableList; |
| import com.google.common.collect.ImmutableMap; |
| import com.google.common.collect.ImmutableSet; |
| import com.google.common.collect.Iterables; |
| import com.google.common.collect.Maps; |
| import com.google.common.collect.Sets; |
| import com.google.common.collect.Sets.SetView; |
| import com.google.devtools.build.lib.cmdline.Label; |
| import com.google.devtools.build.lib.cmdline.PackageIdentifier; |
| import com.google.devtools.build.lib.cmdline.RepositoryName; |
| import com.google.devtools.build.lib.cmdline.TargetPattern; |
| import com.google.devtools.build.lib.cmdline.TargetPattern.Type; |
| import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadSafe; |
| import com.google.devtools.build.lib.events.Event; |
| import com.google.devtools.build.lib.events.ExtendedEventHandler; |
| import com.google.devtools.build.lib.packages.BuildFileNotFoundException; |
| import com.google.devtools.build.lib.packages.NoSuchPackageException; |
| import com.google.devtools.build.lib.packages.NoSuchTargetException; |
| import com.google.devtools.build.lib.packages.Package; |
| import com.google.devtools.build.lib.packages.Target; |
| import com.google.devtools.build.lib.pkgcache.PathPackageLocator; |
| import com.google.devtools.build.lib.pkgcache.RecursivePackageProvider; |
| import com.google.devtools.build.lib.rules.repository.RepositoryDirectoryValue; |
| import com.google.devtools.build.lib.skyframe.TargetPatternValue.TargetPatternKey; |
| import com.google.devtools.build.lib.util.Preconditions; |
| import com.google.devtools.build.lib.vfs.Path; |
| import com.google.devtools.build.lib.vfs.PathFragment; |
| import com.google.devtools.build.lib.vfs.RootedPath; |
| import com.google.devtools.build.skyframe.SkyKey; |
| import com.google.devtools.build.skyframe.SkyValue; |
| import com.google.devtools.build.skyframe.WalkableGraph; |
| import java.util.ArrayList; |
| import java.util.List; |
| import java.util.Map; |
| import java.util.Set; |
| import java.util.logging.Logger; |
| |
| /** |
| * A {@link RecursivePackageProvider} backed by a {@link WalkableGraph}, used by {@code |
| * SkyQueryEnvironment} to look up the packages and targets matching the universe that's been |
| * preloaded in {@code graph}. |
| */ |
| @ThreadSafe |
| public final class GraphBackedRecursivePackageProvider implements RecursivePackageProvider { |
| |
| private final WalkableGraph graph; |
| private final PathPackageLocator pkgPath; |
| private final ImmutableList<TargetPatternKey> universeTargetPatternKeys; |
| |
| private static final Logger LOGGER = Logger |
| .getLogger(GraphBackedRecursivePackageProvider.class.getName()); |
| |
| public GraphBackedRecursivePackageProvider(WalkableGraph graph, |
| ImmutableList<TargetPatternKey> universeTargetPatternKeys, |
| PathPackageLocator pkgPath) { |
| this.pkgPath = pkgPath; |
| this.graph = Preconditions.checkNotNull(graph); |
| this.universeTargetPatternKeys = Preconditions.checkNotNull(universeTargetPatternKeys); |
| } |
| |
| @Override |
| public Package getPackage(ExtendedEventHandler eventHandler, PackageIdentifier packageName) |
| throws NoSuchPackageException, InterruptedException { |
| SkyKey pkgKey = PackageValue.key(packageName); |
| |
| PackageValue pkgValue = (PackageValue) graph.getValue(pkgKey); |
| if (pkgValue != null) { |
| return pkgValue.getPackage(); |
| } |
| NoSuchPackageException nspe = (NoSuchPackageException) graph.getException(pkgKey); |
| if (nspe != null) { |
| throw nspe; |
| } |
| if (graph.isCycle(pkgKey)) { |
| throw new NoSuchPackageException(packageName, "Package depends on a cycle"); |
| } else { |
| // If the package key does not exist in the graph, then it must not correspond to any package, |
| // because the SkyQuery environment has already loaded the universe. |
| throw new BuildFileNotFoundException(packageName, "BUILD file not found on package path"); |
| } |
| } |
| |
| @Override |
| public Map<PackageIdentifier, Package> bulkGetPackages(Iterable<PackageIdentifier> pkgIds) |
| throws NoSuchPackageException, InterruptedException { |
| Set<SkyKey> pkgKeys = ImmutableSet.copyOf(PackageValue.keys(pkgIds)); |
| |
| ImmutableMap.Builder<PackageIdentifier, Package> pkgResults = ImmutableMap.builder(); |
| Map<SkyKey, SkyValue> packages = graph.getSuccessfulValues(pkgKeys); |
| for (Map.Entry<SkyKey, SkyValue> pkgEntry : packages.entrySet()) { |
| PackageIdentifier pkgId = (PackageIdentifier) pkgEntry.getKey().argument(); |
| PackageValue pkgValue = (PackageValue) pkgEntry.getValue(); |
| pkgResults.put(pkgId, Preconditions.checkNotNull(pkgValue.getPackage(), pkgId)); |
| } |
| |
| SetView<SkyKey> unknownKeys = Sets.difference(pkgKeys, packages.keySet()); |
| if (!Iterables.isEmpty(unknownKeys)) { |
| LOGGER.warning("Unable to find " + unknownKeys + " in the batch lookup of " + pkgKeys |
| + ". Successfully looked up " + packages.keySet()); |
| } |
| for (Map.Entry<SkyKey, Exception> missingOrExceptionEntry : |
| graph.getMissingAndExceptions(unknownKeys).entrySet()) { |
| PackageIdentifier pkgIdentifier = |
| (PackageIdentifier) missingOrExceptionEntry.getKey().argument(); |
| Exception exception = missingOrExceptionEntry.getValue(); |
| if (exception == null) { |
| // If the package key does not exist in the graph, then it must not correspond to any |
| // package, because the SkyQuery environment has already loaded the universe. |
| throw new BuildFileNotFoundException(pkgIdentifier, "Package not found"); |
| } |
| Throwables.propagateIfInstanceOf(exception, NoSuchPackageException.class); |
| Throwables.propagate(exception); |
| } |
| return pkgResults.build(); |
| } |
| |
| |
| @Override |
| public boolean isPackage(ExtendedEventHandler eventHandler, PackageIdentifier packageName) |
| throws InterruptedException { |
| SkyKey packageLookupKey = PackageLookupValue.key(packageName); |
| PackageLookupValue packageLookupValue = (PackageLookupValue) graph.getValue(packageLookupKey); |
| if (packageLookupValue == null) { |
| // Package lookups can't depend on Skyframe cycles. |
| Preconditions.checkState(!graph.isCycle(packageLookupKey), packageLookupKey); |
| Exception exception = graph.getException(packageLookupKey); |
| if (exception == null) { |
| // If the package lookup key does not exist in the graph, then it must not correspond to any |
| // package, because the SkyQuery environment has already loaded the universe. |
| return false; |
| } else { |
| if (exception instanceof NoSuchPackageException |
| || exception instanceof InconsistentFilesystemException) { |
| eventHandler.handle(Event.error(exception.getMessage())); |
| return false; |
| } else { |
| throw new IllegalStateException( |
| "During package lookup for '" + packageName + "', got unexpected exception type", |
| exception); |
| } |
| } |
| } |
| return packageLookupValue.packageExists(); |
| } |
| |
| @Override |
| public Iterable<PathFragment> getPackagesUnderDirectory( |
| ExtendedEventHandler eventHandler, |
| RepositoryName repository, |
| PathFragment directory, |
| ImmutableSet<PathFragment> blacklistedSubdirectories, |
| ImmutableSet<PathFragment> excludedSubdirectories) |
| throws InterruptedException { |
| PathFragment.checkAllPathsAreUnder(blacklistedSubdirectories, directory); |
| PathFragment.checkAllPathsAreUnder(excludedSubdirectories, directory); |
| |
| if (excludedSubdirectories.contains(directory)) { |
| return ImmutableList.of(); |
| } |
| |
| // Check that this package is covered by at least one of our universe patterns. |
| boolean inUniverse = false; |
| for (TargetPatternKey patternKey : universeTargetPatternKeys) { |
| TargetPattern pattern = patternKey.getParsedPattern(); |
| boolean isTBD = pattern.getType().equals(Type.TARGETS_BELOW_DIRECTORY); |
| PackageIdentifier packageIdentifier = PackageIdentifier.create(repository, directory); |
| if (isTBD && pattern.containsAllTransitiveSubdirectoriesForTBD(packageIdentifier)) { |
| inUniverse = true; |
| break; |
| } |
| } |
| |
| if (!inUniverse) { |
| return ImmutableList.of(); |
| } |
| |
| List<Path> roots = new ArrayList<>(); |
| if (repository.isMain()) { |
| roots.addAll(pkgPath.getPathEntries()); |
| } else { |
| RepositoryDirectoryValue repositoryValue = |
| (RepositoryDirectoryValue) graph.getValue(RepositoryDirectoryValue.key(repository)); |
| if (repositoryValue == null || !repositoryValue.repositoryExists()) { |
| // If this key doesn't exist, the repository is outside the universe, so we return |
| // "nothing". |
| return ImmutableList.of(); |
| } |
| roots.add(repositoryValue.getPath()); |
| } |
| |
| // If we found a TargetsBelowDirectory pattern in the universe that contains this directory, |
| // then we can look for packages in and under it in the graph. If we didn't find one, then the |
| // directory wasn't in the universe, so return an empty list. |
| ImmutableList.Builder<PathFragment> builder = ImmutableList.builder(); |
| for (Path root : roots) { |
| RootedPath rootedDir = RootedPath.toRootedPath(root, directory); |
| TraversalInfo info = |
| new TraversalInfo(rootedDir, blacklistedSubdirectories, excludedSubdirectories); |
| collectPackagesUnder(eventHandler, repository, ImmutableSet.of(info), builder); |
| } |
| return builder.build(); |
| } |
| |
| private void collectPackagesUnder( |
| ExtendedEventHandler eventHandler, |
| final RepositoryName repository, |
| Set<TraversalInfo> traversals, |
| ImmutableList.Builder<PathFragment> builder) |
| throws InterruptedException { |
| Map<TraversalInfo, SkyKey> traversalToKeyMap = |
| Maps.asMap( |
| traversals, |
| new Function<TraversalInfo, SkyKey>() { |
| @Override |
| public SkyKey apply(TraversalInfo traversalInfo) { |
| return CollectPackagesUnderDirectoryValue.key( |
| repository, traversalInfo.rootedDir, traversalInfo.blacklistedSubdirectories); |
| } |
| }); |
| Map<SkyKey, SkyValue> values = graph.getSuccessfulValues(traversalToKeyMap.values()); |
| |
| ImmutableSet.Builder<TraversalInfo> subdirTraversalBuilder = ImmutableSet.builder(); |
| for (Map.Entry<TraversalInfo, SkyKey> entry : traversalToKeyMap.entrySet()) { |
| TraversalInfo info = entry.getKey(); |
| SkyKey key = entry.getValue(); |
| SkyValue val = values.get(key); |
| CollectPackagesUnderDirectoryValue collectPackagesValue = |
| (CollectPackagesUnderDirectoryValue) val; |
| if (collectPackagesValue != null) { |
| if (collectPackagesValue.isDirectoryPackage()) { |
| builder.add(info.rootedDir.getRelativePath()); |
| } |
| |
| if (collectPackagesValue.getErrorMessage() != null) { |
| eventHandler.handle(Event.error(collectPackagesValue.getErrorMessage())); |
| } |
| |
| ImmutableMap<RootedPath, Boolean> subdirectoryTransitivelyContainsPackages = |
| collectPackagesValue.getSubdirectoryTransitivelyContainsPackagesOrErrors(); |
| for (RootedPath subdirectory : subdirectoryTransitivelyContainsPackages.keySet()) { |
| if (subdirectoryTransitivelyContainsPackages.get(subdirectory)) { |
| PathFragment subdirectoryRelativePath = subdirectory.getRelativePath(); |
| ImmutableSet<PathFragment> blacklistedSubdirectoriesBeneathThisSubdirectory = |
| PathFragment.filterPathsStartingWith( |
| info.blacklistedSubdirectories, subdirectoryRelativePath); |
| ImmutableSet<PathFragment> excludedSubdirectoriesBeneathThisSubdirectory = |
| PathFragment.filterPathsStartingWith( |
| info.excludedSubdirectories, subdirectoryRelativePath); |
| if (!excludedSubdirectoriesBeneathThisSubdirectory.contains(subdirectoryRelativePath)) { |
| subdirTraversalBuilder.add( |
| new TraversalInfo( |
| subdirectory, |
| blacklistedSubdirectoriesBeneathThisSubdirectory, |
| excludedSubdirectoriesBeneathThisSubdirectory)); |
| } |
| } |
| } |
| } |
| } |
| |
| ImmutableSet<TraversalInfo> subdirTraversals = subdirTraversalBuilder.build(); |
| if (!subdirTraversals.isEmpty()) { |
| collectPackagesUnder(eventHandler, repository, subdirTraversals, builder); |
| } |
| } |
| |
| @Override |
| public Target getTarget(ExtendedEventHandler eventHandler, Label label) |
| throws NoSuchPackageException, NoSuchTargetException, InterruptedException { |
| return getPackage(eventHandler, label.getPackageIdentifier()).getTarget(label.getName()); |
| } |
| |
| private static final class TraversalInfo { |
| private final RootedPath rootedDir; |
| // Set of blacklisted directories. The graph is assumed to be prepopulated with |
| // CollectPackagesUnderDirectoryValue nodes whose keys have blacklisted packages embedded in |
| // them. Therefore, we need to be careful to request and use the same sort of keys here in our |
| // traversal. |
| private final ImmutableSet<PathFragment> blacklistedSubdirectories; |
| // Set of directories, targets under which should be excluded from the traversal results. |
| // Excluded directory information isn't part of the graph keys in the prepopulated graph, so we |
| // need to perform the filtering ourselves. |
| private final ImmutableSet<PathFragment> excludedSubdirectories; |
| |
| private TraversalInfo( |
| RootedPath rootedDir, |
| ImmutableSet<PathFragment> blacklistedSubdirectories, |
| ImmutableSet<PathFragment> excludedSubdirectories) { |
| this.rootedDir = rootedDir; |
| this.blacklistedSubdirectories = blacklistedSubdirectories; |
| this.excludedSubdirectories = excludedSubdirectories; |
| } |
| |
| @Override |
| public int hashCode() { |
| return Objects.hashCode(rootedDir, blacklistedSubdirectories, excludedSubdirectories); |
| } |
| |
| @Override |
| public boolean equals(Object obj) { |
| if (this == obj) { |
| return true; |
| } |
| if (obj instanceof TraversalInfo) { |
| TraversalInfo otherTraversal = (TraversalInfo) obj; |
| return Objects.equal(rootedDir, otherTraversal.rootedDir) |
| && Objects.equal(blacklistedSubdirectories, otherTraversal.blacklistedSubdirectories) |
| && Objects.equal(excludedSubdirectories, otherTraversal.excludedSubdirectories); |
| } |
| return false; |
| } |
| } |
| } |