| // Copyright 2022 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.auto.value.AutoValue; |
| import com.google.auto.value.extension.memoized.Memoized; |
| import com.google.common.annotations.VisibleForTesting; |
| import com.google.common.base.MoreObjects; |
| import com.google.devtools.build.lib.actions.Artifact; |
| import com.google.devtools.build.lib.skyframe.serialization.VisibleForSerialization; |
| import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec.Instantiator; |
| import com.google.devtools.build.lib.vfs.Path; |
| import com.google.devtools.build.lib.vfs.PathFragment; |
| import com.google.devtools.build.lib.vfs.Root; |
| import com.google.devtools.build.lib.vfs.RootedPath; |
| import com.google.devtools.build.skyframe.ExecutionPhaseSkyKey; |
| import com.google.devtools.build.skyframe.SkyFunctionName; |
| import com.google.errorprone.annotations.ForOverride; |
| import java.util.Objects; |
| import javax.annotation.Nullable; |
| |
| /** A request for {@link RecursiveFilesystemTraversalFunction}. */ |
| public abstract class TraversalRequest implements ExecutionPhaseSkyKey { |
| |
| // TODO(cmita): This class is only implemented outside of tests by |
| // DirectoryArtifactTraversalRequest. These should probably be consolidated and simplified. |
| |
| /** The path to start the traversal from; may be a file, a directory or a symlink. */ |
| @VisibleForTesting |
| public abstract DirectTraversalRoot root(); |
| |
| /** |
| * Whether the path is in the output tree. |
| * |
| * <p>Such paths and all their subdirectories are assumed not to define packages, so package |
| * lookup for them is skipped. |
| */ |
| protected abstract boolean isRootGenerated(); |
| |
| /** Whether Fileset assumes that output artifacts are regular files. */ |
| protected abstract boolean strictOutputFiles(); |
| |
| /** |
| * Whether to skip checking if the root (if it's a directory) contains a BUILD file. |
| * |
| * <p>Such directories are not considered to be packages when this flag is true. This needs to be |
| * true in order to traverse directories of packages, but should be false for <i>their</i> |
| * subdirectories. |
| */ |
| protected abstract boolean skipTestingForSubpackage(); |
| |
| /** |
| * Whether to emit nodes for empty directories. |
| * |
| * <p>If this returns false, empty directories will not be represented in the result of the |
| * traversal. |
| */ |
| protected abstract boolean emitEmptyDirectoryNodes(); |
| |
| /** |
| * Returns information to be attached to any error messages that may be reported. |
| * |
| * <p>This is purely informational and is not considered in equality. |
| */ |
| protected abstract String errorInfo(); |
| |
| /** |
| * Creates a new traversal request identical to this one except with the given new values for |
| * {@link #root} and {@link #skipTestingForSubpackage}. |
| */ |
| @ForOverride |
| protected abstract TraversalRequest duplicateWithOverrides( |
| DirectTraversalRoot root, boolean skipTestingForSubpackage); |
| |
| /** Creates a new request to traverse a child element in the current directory (the root). */ |
| final TraversalRequest forChildEntry(String child) { |
| DirectTraversalRoot newTraversalRoot = |
| DirectTraversalRoot.forRootAndPath( |
| root().getRootPart(), root().getRelativePart().getRelative(child)); |
| return duplicateWithOverrides(newTraversalRoot, /*skipTestingForSubpackage=*/ false); |
| } |
| |
| /** |
| * Creates a new request for a changed root. |
| * |
| * <p>This method can be used when a package is found out to be under a different root path than |
| * originally assumed. |
| */ |
| final TraversalRequest forChangedRootPath(Root newRoot) { |
| DirectTraversalRoot newTraversalRoot = |
| DirectTraversalRoot.forRootAndPath(newRoot, root().getRelativePart()); |
| return duplicateWithOverrides(newTraversalRoot, skipTestingForSubpackage()); |
| } |
| |
| @Override |
| public final SkyFunctionName functionName() { |
| return SkyFunctions.RECURSIVE_FILESYSTEM_TRAVERSAL; |
| } |
| |
| @Override |
| public final String toString() { |
| return MoreObjects.toStringHelper(this) |
| .add("root", root()) |
| .add("isRootGenerated", isRootGenerated()) |
| .add("strictOutputFiles", strictOutputFiles()) |
| .add("skipTestingForSubpackage", skipTestingForSubpackage()) |
| .add("errorInfo", errorInfo()) |
| .toString(); |
| } |
| |
| /** The root directory of a {@link TraversalRequest}. */ |
| @AutoValue |
| abstract static class DirectTraversalRoot { |
| |
| /** |
| * Returns the output Artifact corresponding to this traversal, if present. Only present when |
| * traversing a generated output. |
| */ |
| @Nullable |
| public abstract Artifact getOutputArtifact(); |
| |
| /** |
| * Returns the root part of the full path. |
| * |
| * <p>This is typically the workspace root or some output tree's root (e.g. genfiles, binfiles). |
| */ |
| public abstract Root getRootPart(); |
| |
| /** |
| * Returns the {@link #getRootPart() root}-relative part of the path. |
| * |
| * <p>This is typically the source directory under the workspace or the output file under an |
| * output directory. |
| */ |
| public abstract PathFragment getRelativePart(); |
| |
| /** Returns a {@link Path} composed of the root and relative parts. */ |
| public final Path asPath() { |
| return getRootPart().getRelative(getRelativePart()); |
| } |
| |
| /** Returns a {@link RootedPath} composed of the root and relative parts. */ |
| public final RootedPath asRootedPath() { |
| return RootedPath.toRootedPath(getRootPart(), getRelativePart()); |
| } |
| |
| @Override |
| public final boolean equals(Object o) { |
| if (o == this) { |
| return true; |
| } |
| if (o instanceof DirectTraversalRoot that) { |
| return Objects.equals(this.getOutputArtifact(), that.getOutputArtifact()) |
| && this.getRootPart().equals(that.getRootPart()) |
| && this.getRelativePart().equals(that.getRelativePart()); |
| } |
| return false; |
| } |
| |
| @Memoized |
| @Override |
| public abstract int hashCode(); |
| |
| public static DirectTraversalRoot forFileOrDirectory(Artifact fileOrDirectory) { |
| return create( |
| fileOrDirectory.isSourceArtifact() ? null : fileOrDirectory, |
| fileOrDirectory.getRoot().getRoot(), |
| fileOrDirectory.getRootRelativePath()); |
| } |
| |
| public static DirectTraversalRoot forRootedPath(RootedPath rootedPath) { |
| return forRootAndPath(rootedPath.getRoot(), rootedPath.getRootRelativePath()); |
| } |
| |
| public static DirectTraversalRoot forRootAndPath(Root rootPart, PathFragment relativePart) { |
| return create(/* outputArtifact= */ null, rootPart, relativePart); |
| } |
| |
| @Instantiator |
| @VisibleForSerialization |
| static DirectTraversalRoot create( |
| @Nullable Artifact outputArtifact, Root rootPart, PathFragment relativePart) { |
| return new AutoValue_TraversalRequest_DirectTraversalRoot( |
| outputArtifact, rootPart, relativePart); |
| } |
| } |
| } |