blob: 38c0f1eb71d9efe5e7101fa72c4b67e3266296a3 [file] [log] [blame]
// 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);
}
}
}