| // 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.annotations.VisibleForTesting; |
| import com.google.common.base.Objects; |
| import com.google.common.base.Optional; |
| import com.google.devtools.build.lib.actions.FilesetTraversalParams.PackageBoundaryMode; |
| 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.skyframe.RecursiveFilesystemTraversalFunction.DanglingSymlinkException; |
| import com.google.devtools.build.lib.skyframe.RecursiveFilesystemTraversalFunction.FileType; |
| 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.LegacySkyKey; |
| import com.google.devtools.build.skyframe.SkyKey; |
| import com.google.devtools.build.skyframe.SkyValue; |
| import java.util.regex.Pattern; |
| import javax.annotation.Nullable; |
| |
| /** |
| * Collection of files found while recursively traversing a path. |
| * |
| * <p>The path may refer to files, symlinks or directories that may or may not exist. |
| * |
| * <p>Traversing a file or a symlink results in a single {@link ResolvedFile} corresponding to the |
| * file or symlink. |
| * |
| * <p>Traversing a directory results in a collection of {@link ResolvedFile}s for all files and |
| * symlinks under it, and in all of its subdirectories. The {@link TraversalRequest} can specify |
| * whether to traverse source subdirectories that are packages (have BUILD files in them). |
| * |
| * <p>Traversing a symlink that points to a directory is the same as traversing a normal directory. |
| * The paths in the result will not be resolved; the files will be listed under the symlink, as if |
| * it was the actual directory they reside in. |
| * |
| * <p>Editing a file that is part of this traversal, or adding or removing a file in a directory |
| * that is part of this traversal, will invalidate this {@link SkyValue}. This also applies to |
| * directories that are symlinked to. |
| */ |
| public final class RecursiveFilesystemTraversalValue implements SkyValue { |
| static final RecursiveFilesystemTraversalValue EMPTY = new RecursiveFilesystemTraversalValue( |
| Optional.<ResolvedFile>absent(), |
| NestedSetBuilder.<ResolvedFile>emptySet(Order.STABLE_ORDER)); |
| |
| /** The root of the traversal. May only be absent for the {@link #EMPTY} instance. */ |
| private final Optional<ResolvedFile> resolvedRoot; |
| |
| /** The transitive closure of {@link ResolvedFile}s. */ |
| private final NestedSet<ResolvedFile> resolvedPaths; |
| |
| private RecursiveFilesystemTraversalValue(Optional<ResolvedFile> resolvedRoot, |
| NestedSet<ResolvedFile> resolvedPaths) { |
| this.resolvedRoot = Preconditions.checkNotNull(resolvedRoot); |
| this.resolvedPaths = Preconditions.checkNotNull(resolvedPaths); |
| } |
| |
| static RecursiveFilesystemTraversalValue of(ResolvedFile resolvedRoot, |
| NestedSet<ResolvedFile> resolvedPaths) { |
| if (resolvedPaths.isEmpty()) { |
| return EMPTY; |
| } else { |
| return new RecursiveFilesystemTraversalValue(Optional.of(resolvedRoot), resolvedPaths); |
| } |
| } |
| |
| static RecursiveFilesystemTraversalValue of(ResolvedFile singleMember) { |
| return new RecursiveFilesystemTraversalValue(Optional.of(singleMember), |
| NestedSetBuilder.<ResolvedFile>create(Order.STABLE_ORDER, singleMember)); |
| } |
| |
| /** Returns the root of the traversal; absent only for the {@link #EMPTY} instance. */ |
| public Optional<ResolvedFile> getResolvedRoot() { |
| return resolvedRoot; |
| } |
| |
| /** |
| * Retrieves the set of {@link ResolvedFile}s that were found by this traversal. |
| * |
| * <p>The returned set may be empty if no files were found, or the ones found were to be |
| * considered non-existent. Unless it's empty, the returned set always includes the |
| * {@link #getResolvedRoot() resolved root}. |
| * |
| * <p>The returned set also includes symlinks. If a symlink points to a directory, its contents |
| * are also included in this set, and their path will start with the symlink's path, just like on |
| * a usual Unix file system. |
| */ |
| public NestedSet<ResolvedFile> getTransitiveFiles() { |
| return resolvedPaths; |
| } |
| |
| public static SkyKey key(TraversalRequest traversal) { |
| return LegacySkyKey.create(SkyFunctions.RECURSIVE_FILESYSTEM_TRAVERSAL, traversal); |
| } |
| |
| /** The parameters of a file or directory traversal. */ |
| public static final class TraversalRequest { |
| |
| /** The path to start the traversal from; may be a file, a directory or a symlink. */ |
| final RootedPath path; |
| |
| /** |
| * 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. |
| */ |
| final boolean isGenerated; |
| |
| /** Whether traversal should descend into directories that are roots of subpackages. */ |
| final PackageBoundaryMode crossPkgBoundaries; |
| |
| /** |
| * 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. |
| */ |
| final boolean skipTestingForSubpackage; |
| |
| /** A pattern that files must match to be included in this traversal (may be null.) */ |
| @Nullable |
| final Pattern pattern; |
| |
| /** Information to be attached to any error messages that may be reported. */ |
| @Nullable final String errorInfo; |
| |
| public TraversalRequest(RootedPath path, boolean isRootGenerated, |
| PackageBoundaryMode crossPkgBoundaries, boolean skipTestingForSubpackage, |
| @Nullable String errorInfo, @Nullable Pattern pattern) { |
| this.path = path; |
| this.isGenerated = isRootGenerated; |
| this.crossPkgBoundaries = crossPkgBoundaries; |
| this.skipTestingForSubpackage = skipTestingForSubpackage; |
| this.errorInfo = errorInfo; |
| this.pattern = pattern; |
| } |
| |
| private TraversalRequest duplicate(RootedPath newRoot, boolean newSkipTestingForSubpackage) { |
| return new TraversalRequest(newRoot, isGenerated, crossPkgBoundaries, |
| newSkipTestingForSubpackage, errorInfo, pattern); |
| } |
| |
| /** Creates a new request to traverse a child element in the current directory (the root). */ |
| TraversalRequest forChildEntry(RootedPath newPath) { |
| return duplicate(newPath, 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. |
| */ |
| TraversalRequest forChangedRootPath(Path newRoot) { |
| return duplicate(RootedPath.toRootedPath(newRoot, path.getRelativePath()), |
| skipTestingForSubpackage); |
| } |
| |
| @Override |
| public boolean equals(Object obj) { |
| if (this == obj) { |
| return true; |
| } |
| if (!(obj instanceof TraversalRequest)) { |
| return false; |
| } |
| TraversalRequest o = (TraversalRequest) obj; |
| return path.equals(o.path) && isGenerated == o.isGenerated |
| && crossPkgBoundaries == o.crossPkgBoundaries |
| && skipTestingForSubpackage == o.skipTestingForSubpackage |
| && Objects.equal(pattern, o.pattern); |
| } |
| |
| @Override |
| public int hashCode() { |
| return Objects.hashCode(path, isGenerated, crossPkgBoundaries, skipTestingForSubpackage, |
| pattern); |
| } |
| |
| @Override |
| public String toString() { |
| return String.format( |
| "TraversalParams(root=%s, is_generated=%d, skip_testing_for_subpkg=%d," |
| + " pkg_boundaries=%s, pattern=%s)", path, isGenerated ? 1 : 0, |
| skipTestingForSubpackage ? 1 : 0, crossPkgBoundaries, |
| pattern == null ? "null" : pattern.pattern()); |
| } |
| } |
| |
| private static final class Symlink { |
| private final RootedPath linkName; |
| private final PathFragment unresolvedLinkTarget; |
| // The resolved link target is returned by ResolvedFile.getPath() |
| |
| private Symlink(RootedPath linkName, PathFragment unresolvedLinkTarget) { |
| this.linkName = Preconditions.checkNotNull(linkName); |
| this.unresolvedLinkTarget = Preconditions.checkNotNull(unresolvedLinkTarget); |
| } |
| |
| PathFragment getNameInSymlinkTree() { |
| return linkName.getRelativePath(); |
| } |
| |
| @Override |
| public boolean equals(Object obj) { |
| if (this == obj) { |
| return true; |
| } |
| if (!(obj instanceof Symlink)) { |
| return false; |
| } |
| Symlink o = (Symlink) obj; |
| return linkName.equals(o.linkName) && unresolvedLinkTarget.equals(o.unresolvedLinkTarget); |
| } |
| |
| @Override |
| public int hashCode() { |
| return Objects.hashCode(linkName, unresolvedLinkTarget); |
| } |
| |
| @Override |
| public String toString() { |
| return String.format("Symlink(link_name=%s, unresolved_target=%s)", |
| linkName, unresolvedLinkTarget); |
| } |
| } |
| |
| private static final class RegularFile implements ResolvedFile { |
| private final RootedPath path; |
| @Nullable private final FileStateValue metadata; |
| |
| /** C'tor for {@link #stripMetadataForTesting()}. */ |
| private RegularFile(RootedPath path) { |
| this.path = Preconditions.checkNotNull(path); |
| this.metadata = null; |
| } |
| |
| RegularFile(RootedPath path, FileStateValue metadata) { |
| this.path = Preconditions.checkNotNull(path); |
| this.metadata = Preconditions.checkNotNull(metadata); |
| } |
| |
| @Override |
| public FileType getType() { |
| return FileType.FILE; |
| } |
| |
| @Override |
| public RootedPath getPath() { |
| return path; |
| } |
| |
| @Override |
| @Nullable |
| public int getMetadataHash() { |
| return metadata == null ? 0 : metadata.hashCode(); |
| } |
| |
| @Override |
| public boolean equals(Object obj) { |
| if (this == obj) { |
| return true; |
| } |
| if (!(obj instanceof RegularFile)) { |
| return false; |
| } |
| return this.path.equals(((RegularFile) obj).path) |
| && Objects.equal(this.metadata, ((RegularFile) obj).metadata); |
| } |
| |
| @Override |
| public int hashCode() { |
| return Objects.hashCode(path, metadata); |
| } |
| |
| @Override |
| public String toString() { |
| return String.format("RegularFile(path=%s)", path); |
| } |
| |
| @Override |
| public ResolvedFile stripMetadataForTesting() { |
| return new RegularFile(path); |
| } |
| |
| @Override |
| public PathFragment getNameInSymlinkTree() { |
| return path.getRelativePath(); |
| } |
| |
| @Override |
| public PathFragment getTargetInSymlinkTree(boolean followSymlinks) { |
| return path.asPath().asFragment(); |
| } |
| } |
| |
| private static final class Directory implements ResolvedFile { |
| private final RootedPath path; |
| |
| Directory(RootedPath path) { |
| this.path = Preconditions.checkNotNull(path); |
| } |
| |
| @Override |
| public FileType getType() { |
| return FileType.DIRECTORY; |
| } |
| |
| @Override |
| public RootedPath getPath() { |
| return path; |
| } |
| |
| @Override |
| public int getMetadataHash() { |
| return FileStateValue.DIRECTORY_FILE_STATE_NODE.hashCode(); |
| } |
| |
| @Override |
| public boolean equals(Object obj) { |
| if (this == obj) { |
| return true; |
| } |
| if (!(obj instanceof Directory)) { |
| return false; |
| } |
| return this.path.equals(((Directory) obj).path); |
| } |
| |
| @Override |
| public int hashCode() { |
| return path.hashCode(); |
| } |
| |
| @Override |
| public String toString() { |
| return String.format("Directory(path=%s)", path); |
| } |
| |
| @Override |
| public ResolvedFile stripMetadataForTesting() { |
| return this; |
| } |
| |
| @Override |
| public PathFragment getNameInSymlinkTree() { |
| return path.getRelativePath(); |
| } |
| |
| @Override |
| public PathFragment getTargetInSymlinkTree(boolean followSymlinks) { |
| return path.asPath().asFragment(); |
| } |
| } |
| |
| private static final class DanglingSymlink implements ResolvedFile { |
| private final Symlink symlink; |
| @Nullable private final FileStateValue metadata; |
| |
| private DanglingSymlink(Symlink symlink) { |
| this.symlink = symlink; |
| this.metadata = null; |
| } |
| |
| private DanglingSymlink(RootedPath linkNamePath, PathFragment linkTargetPath) { |
| this.symlink = new Symlink(linkNamePath, linkTargetPath); |
| this.metadata = null; |
| } |
| |
| DanglingSymlink(RootedPath linkNamePath, PathFragment linkTargetPath, |
| FileStateValue metadata) { |
| this.symlink = new Symlink(linkNamePath, linkTargetPath); |
| this.metadata = Preconditions.checkNotNull(metadata); |
| } |
| |
| @Override |
| public FileType getType() { |
| return FileType.DANGLING_SYMLINK; |
| } |
| |
| @Override |
| @Nullable |
| public RootedPath getPath() { |
| return null; |
| } |
| |
| @Override |
| @Nullable |
| public int getMetadataHash() { |
| return metadata == null ? 0 : metadata.hashCode(); |
| } |
| |
| @Override |
| public boolean equals(Object obj) { |
| if (this == obj) { |
| return true; |
| } |
| if (!(obj instanceof DanglingSymlink)) { |
| return false; |
| } |
| return Objects.equal(this.metadata, ((DanglingSymlink) obj).metadata) |
| && this.symlink.equals(((DanglingSymlink) obj).symlink); |
| } |
| |
| @Override |
| public int hashCode() { |
| return Objects.hashCode(metadata, symlink); |
| } |
| |
| @Override |
| public String toString() { |
| return String.format("DanglingSymlink(%s)", symlink); |
| } |
| |
| @Override |
| public ResolvedFile stripMetadataForTesting() { |
| return new DanglingSymlink(symlink); |
| } |
| |
| @Override |
| public PathFragment getNameInSymlinkTree() { |
| return symlink.getNameInSymlinkTree(); |
| } |
| |
| @Override |
| public PathFragment getTargetInSymlinkTree(boolean followSymlinks) |
| throws DanglingSymlinkException { |
| if (followSymlinks) { |
| throw new DanglingSymlinkException(symlink.linkName.asPath().getPathString(), |
| symlink.unresolvedLinkTarget.getPathString()); |
| } else { |
| return symlink.unresolvedLinkTarget; |
| } |
| } |
| } |
| |
| private static final class SymlinkToFile implements ResolvedFile { |
| private final RootedPath path; |
| @Nullable private final FileStateValue metadata; |
| private final Symlink symlink; |
| |
| /** C'tor for {@link #stripMetadataForTesting()}. */ |
| private SymlinkToFile(RootedPath targetPath, Symlink symlink) { |
| this.path = Preconditions.checkNotNull(targetPath); |
| this.metadata = null; |
| this.symlink = Preconditions.checkNotNull(symlink); |
| } |
| |
| private SymlinkToFile( |
| RootedPath targetPath, RootedPath linkNamePath, PathFragment linkTargetPath) { |
| this.path = Preconditions.checkNotNull(targetPath); |
| this.metadata = null; |
| this.symlink = new Symlink(linkNamePath, linkTargetPath); |
| } |
| |
| SymlinkToFile(RootedPath targetPath, RootedPath linkNamePath, |
| PathFragment linkTargetPath, FileStateValue metadata) { |
| this.path = Preconditions.checkNotNull(targetPath); |
| this.metadata = Preconditions.checkNotNull(metadata); |
| this.symlink = new Symlink(linkNamePath, linkTargetPath); |
| } |
| |
| @Override |
| public FileType getType() { |
| return FileType.SYMLINK_TO_FILE; |
| } |
| |
| @Override |
| public RootedPath getPath() { |
| return path; |
| } |
| |
| @Override |
| @Nullable |
| public int getMetadataHash() { |
| return metadata == null ? 0 : metadata.hashCode(); |
| } |
| |
| @Override |
| public boolean equals(Object obj) { |
| if (this == obj) { |
| return true; |
| } |
| if (!(obj instanceof SymlinkToFile)) { |
| return false; |
| } |
| return this.path.equals(((SymlinkToFile) obj).path) |
| && Objects.equal(this.metadata, ((SymlinkToFile) obj).metadata) |
| && this.symlink.equals(((SymlinkToFile) obj).symlink); |
| } |
| |
| @Override |
| public int hashCode() { |
| return Objects.hashCode(path, metadata, symlink); |
| } |
| |
| @Override |
| public String toString() { |
| return String.format("SymlinkToFile(target=%s, %s)", path, symlink); |
| } |
| |
| @Override |
| public ResolvedFile stripMetadataForTesting() { |
| return new SymlinkToFile(path, symlink); |
| } |
| |
| @Override |
| public PathFragment getNameInSymlinkTree() { |
| return symlink.getNameInSymlinkTree(); |
| } |
| |
| @Override |
| public PathFragment getTargetInSymlinkTree(boolean followSymlinks) { |
| return followSymlinks ? path.asPath().asFragment() : symlink.unresolvedLinkTarget; |
| } |
| } |
| |
| private static final class SymlinkToDirectory implements ResolvedFile { |
| private final RootedPath path; |
| @Nullable private final int metadataHash; |
| private final Symlink symlink; |
| |
| /** C'tor for {@link #stripMetadataForTesting()}. */ |
| private SymlinkToDirectory(RootedPath targetPath, Symlink symlink) { |
| this.path = Preconditions.checkNotNull(targetPath); |
| this.metadataHash = 0; |
| this.symlink = symlink; |
| } |
| |
| private SymlinkToDirectory( |
| RootedPath targetPath, RootedPath linkNamePath, PathFragment linkValue) { |
| this.path = Preconditions.checkNotNull(targetPath); |
| this.metadataHash = 0; |
| this.symlink = new Symlink(linkNamePath, linkValue); |
| } |
| |
| SymlinkToDirectory(RootedPath targetPath, RootedPath linkNamePath, |
| PathFragment linkValue, int metadataHash) { |
| this.path = Preconditions.checkNotNull(targetPath); |
| this.metadataHash = metadataHash; |
| this.symlink = new Symlink(linkNamePath, linkValue); |
| } |
| |
| @Override |
| public FileType getType() { |
| return FileType.SYMLINK_TO_DIRECTORY; |
| } |
| |
| @Override |
| public RootedPath getPath() { |
| return path; |
| } |
| |
| @Override |
| @Nullable |
| public int getMetadataHash() { |
| return metadataHash; |
| } |
| |
| @Override |
| public boolean equals(Object obj) { |
| if (this == obj) { |
| return true; |
| } |
| if (!(obj instanceof SymlinkToDirectory)) { |
| return false; |
| } |
| return this.path.equals(((SymlinkToDirectory) obj).path) |
| && Objects.equal(this.metadataHash, ((SymlinkToDirectory) obj).metadataHash) |
| && this.symlink.equals(((SymlinkToDirectory) obj).symlink); |
| } |
| |
| @Override |
| public int hashCode() { |
| return Objects.hashCode(path, metadataHash, symlink); |
| } |
| |
| @Override |
| public String toString() { |
| return String.format("SymlinkToDirectory(target=%s, %s)", path, symlink); |
| } |
| |
| @Override |
| public ResolvedFile stripMetadataForTesting() { |
| return new SymlinkToDirectory(path, symlink); |
| } |
| |
| @Override |
| public PathFragment getNameInSymlinkTree() { |
| return symlink.getNameInSymlinkTree(); |
| } |
| |
| @Override |
| public PathFragment getTargetInSymlinkTree(boolean followSymlinks) { |
| return followSymlinks ? path.asPath().asFragment() : symlink.unresolvedLinkTarget; |
| } |
| } |
| |
| static final class ResolvedFileFactory { |
| private ResolvedFileFactory() {} |
| |
| public static ResolvedFile regularFile(RootedPath path, FileStateValue metadata) { |
| return new RegularFile(path, metadata); |
| } |
| |
| public static ResolvedFile directory(RootedPath path) { |
| return new Directory(path); |
| } |
| |
| public static ResolvedFile symlinkToFile(RootedPath targetPath, RootedPath linkNamePath, |
| PathFragment linkTargetPath, FileStateValue metadata) { |
| return new SymlinkToFile(targetPath, linkNamePath, linkTargetPath, metadata); |
| } |
| |
| public static ResolvedFile symlinkToDirectory(RootedPath targetPath, |
| RootedPath linkNamePath, PathFragment linkValue, int metadataHash) { |
| return new SymlinkToDirectory(targetPath, linkNamePath, linkValue, metadataHash); |
| } |
| |
| public static ResolvedFile danglingSymlink(RootedPath linkNamePath, PathFragment linkValue, |
| FileStateValue metadata) { |
| return new DanglingSymlink(linkNamePath, linkValue, metadata); |
| } |
| } |
| |
| @VisibleForTesting |
| static final class ResolvedFileFactoryForTesting { |
| private ResolvedFileFactoryForTesting() {} |
| |
| static ResolvedFile regularFileForTesting(RootedPath path) { |
| return new RegularFile(path); |
| } |
| |
| static ResolvedFile symlinkToFileForTesting( |
| RootedPath targetPath, RootedPath linkNamePath, PathFragment linkTargetPath) { |
| return new SymlinkToFile(targetPath, linkNamePath, linkTargetPath); |
| } |
| |
| static ResolvedFile symlinkToDirectoryForTesting( |
| RootedPath targetPath, RootedPath linkNamePath, PathFragment linkValue) { |
| return new SymlinkToDirectory(targetPath, linkNamePath, linkValue); |
| } |
| |
| public static ResolvedFile danglingSymlinkForTesting( |
| RootedPath linkNamePath, PathFragment linkValue) { |
| return new DanglingSymlink(linkNamePath, linkValue); |
| } |
| } |
| |
| /** |
| * Path and type information about a single file or symlink. |
| * |
| * <p>The object stores things such as the absolute path of the file or symlink, its exact type |
| * and, if it's a symlink, the resolved and unresolved link target paths. |
| */ |
| public static interface ResolvedFile { |
| /** Type of the entity under {@link #getPath()}. */ |
| FileType getType(); |
| |
| /** |
| * Path of the file, directory or resolved target of the symlink. |
| * |
| * <p>May only return null for dangling symlinks. |
| */ |
| @Nullable |
| RootedPath getPath(); |
| |
| /** |
| * Hash code of associated metadata. |
| * |
| * <p>This is usually some hash of the {@link FileStateValue} of the underlying filesystem |
| * entity. |
| * |
| * <p>If tests stripped the metadata or the {@link ResolvedFile} was created by the |
| * {@link ResolvedFileFactoryForTesting}, this method returns 0. |
| */ |
| int getMetadataHash(); |
| |
| /** |
| * Returns the path of the Fileset-output symlink relative to the output directory. |
| * |
| * <p>The path should contain the FilesetEntry-specific destination directory (if any) and |
| * should have necessary prefixes stripped (if any). |
| */ |
| PathFragment getNameInSymlinkTree(); |
| |
| /** |
| * Returns the path of the symlink target. |
| * |
| * @throws DanglingSymlinkException if the target cannot be resolved because the symlink is |
| * dangling |
| */ |
| PathFragment getTargetInSymlinkTree(boolean followSymlinks) throws DanglingSymlinkException; |
| |
| /** |
| * Returns a copy of this object with the metadata stripped away. |
| * |
| * <p>This method should only be used by tests that wish to assert that this |
| * {@link ResolvedFile} refers to the expected absolute path and has the expected type, without |
| * asserting its actual contents (which the metadata is a function of). |
| */ |
| @VisibleForTesting |
| ResolvedFile stripMetadataForTesting(); |
| } |
| |
| @Override |
| public boolean equals(Object obj) { |
| if (this == obj) { |
| return true; |
| } |
| if (!(obj instanceof RecursiveFilesystemTraversalValue)) { |
| return false; |
| } |
| RecursiveFilesystemTraversalValue o = (RecursiveFilesystemTraversalValue) obj; |
| return resolvedRoot.equals(o.resolvedRoot) && resolvedPaths.equals(o.resolvedPaths); |
| } |
| |
| @Override |
| public int hashCode() { |
| return Objects.hashCode(resolvedRoot, resolvedPaths); |
| } |
| } |