|  | // Copyright 2014 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.collect.ImmutableSet; | 
|  | import com.google.devtools.build.lib.cmdline.RepositoryName; | 
|  | import com.google.devtools.build.lib.collect.nestedset.NestedSet; | 
|  | import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder; | 
|  | import com.google.devtools.build.lib.collect.nestedset.Order; | 
|  | import com.google.devtools.build.lib.concurrent.ThreadSafety.Immutable; | 
|  | import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadSafe; | 
|  | import com.google.devtools.build.lib.util.Preconditions; | 
|  | import com.google.devtools.build.lib.vfs.PathFragment; | 
|  | import com.google.devtools.build.lib.vfs.RootedPath; | 
|  | import com.google.devtools.build.skyframe.LegacySkyKey; | 
|  | import com.google.devtools.build.skyframe.SkyKey; | 
|  | import com.google.devtools.build.skyframe.SkyValue; | 
|  | import java.io.Serializable; | 
|  | import java.util.Objects; | 
|  |  | 
|  | /** | 
|  | * This value represents the result of looking up all the packages under a given package path root, | 
|  | * starting at a given directory. | 
|  | */ | 
|  | @Immutable | 
|  | @ThreadSafe | 
|  | public class RecursivePkgValue implements SkyValue { | 
|  | static final RecursivePkgValue EMPTY = | 
|  | new RecursivePkgValue(NestedSetBuilder.<String>emptySet(Order.STABLE_ORDER)); | 
|  |  | 
|  | private final NestedSet<String> packages; | 
|  |  | 
|  | private RecursivePkgValue(NestedSet<String> packages) { | 
|  | this.packages = packages; | 
|  | } | 
|  |  | 
|  | static RecursivePkgValue create(NestedSetBuilder<String> packages) { | 
|  | if (packages.isEmpty()) { | 
|  | return EMPTY; | 
|  | } | 
|  | return new RecursivePkgValue(packages.build()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Create a transitive package lookup request. | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static SkyKey key(RepositoryName repositoryName, RootedPath rootedPath, | 
|  | ImmutableSet<PathFragment> excludedPaths) { | 
|  | return LegacySkyKey.create( | 
|  | SkyFunctions.RECURSIVE_PKG, new RecursivePkgKey(repositoryName, rootedPath, excludedPaths)); | 
|  | } | 
|  |  | 
|  | public NestedSet<String> getPackages() { | 
|  | return packages; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * A RecursivePkgKey is a tuple of a {@link RootedPath}, {@code rootedPath}, defining the | 
|  | * directory to recurse beneath in search of packages, and an {@link ImmutableSet} of {@link | 
|  | * PathFragment}s, {@code excludedPaths}, relative to {@code rootedPath.getRoot}, defining the | 
|  | * set of subdirectories beneath {@code rootedPath} to skip. | 
|  | * | 
|  | * <p>Throws {@link IllegalArgumentException} if {@code excludedPaths} contains any paths that | 
|  | * are equal to {@code rootedPath} or that are not beneath {@code rootedPath}. | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static final class RecursivePkgKey implements Serializable { | 
|  | private final RepositoryName repositoryName; | 
|  | private final RootedPath rootedPath; | 
|  | private final ImmutableSet<PathFragment> excludedPaths; | 
|  |  | 
|  | public RecursivePkgKey(RepositoryName repositoryName, RootedPath rootedPath, | 
|  | ImmutableSet<PathFragment> excludedPaths) { | 
|  | PathFragment.checkAllPathsAreUnder(excludedPaths, | 
|  | rootedPath.getRelativePath()); | 
|  | Preconditions.checkState(!repositoryName.isDefault()); | 
|  | this.repositoryName = repositoryName; | 
|  | this.rootedPath = Preconditions.checkNotNull(rootedPath); | 
|  | this.excludedPaths = Preconditions.checkNotNull(excludedPaths); | 
|  | } | 
|  |  | 
|  | public RepositoryName getRepository() { | 
|  | return repositoryName; | 
|  | } | 
|  |  | 
|  | public RootedPath getRootedPath() { | 
|  | return rootedPath; | 
|  | } | 
|  |  | 
|  | public ImmutableSet<PathFragment> getExcludedPaths() { | 
|  | return excludedPaths; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public String toString() { | 
|  | return "rootedPath=" + rootedPath + ", excludedPaths=<omitted>)"; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public boolean equals(Object o) { | 
|  | if (this == o) { | 
|  | return true; | 
|  | } | 
|  | if (!(o instanceof RecursivePkgKey)) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | RecursivePkgKey that = (RecursivePkgKey) o; | 
|  | return excludedPaths.equals(that.excludedPaths) && rootedPath.equals(that.rootedPath); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public int hashCode() { | 
|  | return Objects.hash(rootedPath, excludedPaths); | 
|  | } | 
|  | } | 
|  | } |