| // 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.actions; |
| |
| import com.google.common.annotations.VisibleForTesting; |
| import com.google.common.base.MoreObjects; |
| import com.google.common.base.Preconditions; |
| import com.google.common.io.BaseEncoding; |
| import com.google.devtools.build.lib.actions.cache.DigestUtils; |
| import com.google.devtools.build.lib.concurrent.ThreadSafety.Immutable; |
| import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadSafe; |
| import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec; |
| import com.google.devtools.build.lib.util.BigIntegerFingerprint; |
| import com.google.devtools.build.lib.vfs.DigestHashFunction; |
| import com.google.devtools.build.lib.vfs.FileStatus; |
| import com.google.devtools.build.lib.vfs.Path; |
| import com.google.devtools.build.lib.vfs.PathFragment; |
| import com.google.devtools.build.lib.vfs.Symlinks; |
| import com.google.devtools.build.skyframe.SkyValue; |
| import java.io.ByteArrayInputStream; |
| import java.io.IOException; |
| import java.math.BigInteger; |
| import java.util.Arrays; |
| import java.util.Objects; |
| import javax.annotation.Nullable; |
| |
| /** |
| * State of a file system object for the execution phase. |
| * |
| * <p>This is not used by Skyframe for invalidation, it is primarily used by the action cache and |
| * the various {@link com.google.devtools.build.lib.exec.SpawnRunner} implementations. |
| * |
| * <p>We have the following cases: |
| * |
| * <ul> |
| * <li>an ordinary file, in which case we would expect to see a digest and size; |
| * <li>a directory, in which case we would expect to see an mtime; |
| * <li>an intentionally omitted file which the build system is aware of but doesn't actually |
| * exist, where all access methods are unsupported; |
| * <li>a "middleman marker" object, which has a null digest, 0 size, and mtime of 0. |
| * <li>The "self data" of a TreeArtifact, where we would expect to see a digest representing the |
| * artifact's contents, and a size of 0. |
| * <li>a file that's only stored by a remote caching/execution system, in which case we would |
| * expect to see a digest and size. |
| * </ul> |
| */ |
| @Immutable |
| @ThreadSafe |
| public abstract class FileArtifactValue implements SkyValue { |
| @AutoCodec public static final FileArtifactValue DEFAULT_MIDDLEMAN = new SingletonMarkerValue(); |
| /** Data that marks that a file is not present on the filesystem. */ |
| @AutoCodec public static final FileArtifactValue MISSING_FILE_MARKER = new SingletonMarkerValue(); |
| |
| /** |
| * Represents an omitted file -- we are aware of it but it doesn't exist. All access methods are |
| * unsupported. |
| */ |
| @AutoCodec public static final FileArtifactValue OMITTED_FILE_MARKER = new OmittedFileValue(); |
| |
| /** |
| * Marker interface for singleton implementations of this class. |
| * |
| * <p>Needed for a correct implementation of {@code equals}. |
| */ |
| public interface Singleton {} |
| |
| /** |
| * The type of the underlying file system object. If it is a regular file, then it is guaranteed |
| * to have a digest. Otherwise it does not have a digest. |
| */ |
| public abstract FileStateType getType(); |
| |
| /** |
| * Returns a digest of the content of the underlying file system object; must always return a |
| * non-null value for instances of type {@link FileStateType#REGULAR_FILE}. Otherwise may return |
| * null. |
| * |
| * <p>All instances of this interface must either have a digest or return a last-modified time. |
| * Clients should prefer using the digest for content identification (e.g., for caching), and only |
| * fall back to the last-modified time if no digest is available. |
| * |
| * <p>The return value is owned by this object and must not be modified. |
| */ |
| @Nullable |
| public abstract byte[] getDigest(); |
| |
| /** Returns the file's size, or 0 if the underlying file system object is not a file. */ |
| // TODO(ulfjack): Throw an exception if it's not a file. |
| public abstract long getSize(); |
| |
| /** |
| * Returns the last modified time; see the documentation of {@link #getDigest} for when this can |
| * and should be called. |
| */ |
| public abstract long getModifiedTime(); |
| |
| @Nullable |
| @Override |
| public BigInteger getValueFingerprint() { |
| byte[] digest = getDigest(); |
| if (digest != null) { |
| return new BigIntegerFingerprint().addDigestedBytes(digest).getFingerprint(); |
| } |
| // TODO(janakr): return fingerprint in other cases: symlink, directory. |
| return null; |
| } |
| |
| /** |
| * Index used to resolve remote files. |
| * |
| * <p>0 indicates that no such information is available which can mean that it's either a local |
| * file, empty, or an omitted output. |
| */ |
| public int getLocationIndex() { |
| return 0; |
| } |
| |
| /** Returns {@code true} if this is a special marker as opposed to a representing a real file. */ |
| public final boolean isMarkerValue() { |
| return this instanceof Singleton; |
| } |
| |
| /** |
| * Provides a best-effort determination whether the file was changed since the digest was |
| * computed. This method performs file system I/O, so may be expensive. It's primarily intended to |
| * avoid storing bad cache entries in an action cache. It should return true if there is a chance |
| * that the file was modified since the digest was computed. Better not upload if we are not sure |
| * that the cache entry is reliable. |
| */ |
| public abstract boolean wasModifiedSinceDigest(Path path) throws IOException; |
| |
| /** Returns {@code true} if the file only exists remotely. */ |
| public boolean isRemote() { |
| return false; |
| } |
| |
| @Override |
| public boolean equals(Object o) { |
| if (this == o) { |
| return true; |
| } |
| if (!(o instanceof FileArtifactValue)) { |
| return false; |
| } |
| if ((this instanceof Singleton) || (o instanceof Singleton)) { |
| return false; |
| } |
| FileArtifactValue m = (FileArtifactValue) o; |
| if (getType() != m.getType()) { |
| return false; |
| } |
| if (getDigest() != null) { |
| return Arrays.equals(getDigest(), m.getDigest()) && getSize() == m.getSize(); |
| } else { |
| return getModifiedTime() == m.getModifiedTime(); |
| } |
| } |
| |
| @Override |
| public int hashCode() { |
| if (this instanceof Singleton) { |
| return System.identityHashCode(this); |
| } |
| // Hash digest by content, not reference. |
| if (getDigest() != null) { |
| return 37 * Long.hashCode(getSize()) + Arrays.hashCode(getDigest()); |
| } else { |
| return Long.hashCode(getModifiedTime()); |
| } |
| } |
| |
| public static FileArtifactValue create(Artifact artifact, FileValue fileValue) |
| throws IOException { |
| boolean isFile = fileValue.isFile(); |
| FileContentsProxy proxy = getProxyFromFileStateValue(fileValue.realFileStateValue()); |
| return create( |
| artifact.getPath(), |
| isFile, |
| isFile ? fileValue.getSize() : 0, |
| proxy, |
| isFile ? fileValue.getDigest() : null, |
| !artifact.isConstantMetadata()); |
| } |
| |
| public static FileArtifactValue create(Artifact artifact, ArtifactFileMetadata metadata) |
| throws IOException { |
| boolean isFile = metadata.isFile(); |
| FileContentsProxy proxy = getProxyFromFileStateValue(metadata.realFileStateValue()); |
| return create( |
| artifact.getPath(), |
| isFile, |
| isFile ? metadata.getSize() : 0, |
| proxy, |
| isFile ? metadata.getDigest() : null, |
| !artifact.isConstantMetadata()); |
| } |
| |
| public static FileArtifactValue create( |
| Artifact artifact, |
| ArtifactPathResolver resolver, |
| ArtifactFileMetadata fileValue, |
| @Nullable byte[] injectedDigest) |
| throws IOException { |
| boolean isFile = fileValue.isFile(); |
| FileContentsProxy proxy = getProxyFromFileStateValue(fileValue.realFileStateValue()); |
| return create( |
| resolver.toPath(artifact), |
| isFile, |
| isFile ? fileValue.getSize() : 0, |
| proxy, |
| injectedDigest, |
| !artifact.isConstantMetadata()); |
| } |
| |
| @VisibleForTesting |
| public static FileArtifactValue create(Artifact artifact) throws IOException { |
| return create(artifact.getPath(), !artifact.isConstantMetadata()); |
| } |
| |
| public static FileArtifactValue createShareable(Path path) throws IOException { |
| return create(path, /*isShareable=*/ true); |
| } |
| |
| public static FileArtifactValue create(Path path, boolean isShareable) throws IOException { |
| // Caution: there's a race condition between stating the file and computing the |
| // digest. We need to stat first, since we're using the stat to detect changes. |
| // We follow symlinks here to be consistent with getDigest. |
| return create(path, path.stat(Symlinks.FOLLOW), isShareable); |
| } |
| |
| public static FileArtifactValue create(Path path, FileStatus stat, boolean isShareable) |
| throws IOException { |
| return create( |
| path, stat.isFile(), stat.getSize(), FileContentsProxy.create(stat), null, isShareable); |
| } |
| |
| private static FileArtifactValue create( |
| Path path, |
| boolean isFile, |
| long size, |
| FileContentsProxy proxy, |
| @Nullable byte[] digest, |
| boolean isShareable) |
| throws IOException { |
| if (!isFile) { |
| // In this case, we need to store the mtime because the action cache uses mtime for |
| // directories to determine if this artifact has changed. We want this code path to go away |
| // somehow. |
| return new DirectoryArtifactValue(path.getLastModifiedTime()); |
| } |
| if (digest == null) { |
| digest = DigestUtils.getDigestOrFail(path, size); |
| } |
| Preconditions.checkState(digest != null, path); |
| return createNormalFile(digest, proxy, size, isShareable); |
| } |
| |
| public static FileArtifactValue createForVirtualActionInput(byte[] digest, long size) { |
| return new RegularFileArtifactValue(digest, /*proxy=*/ null, size); |
| } |
| |
| @VisibleForTesting |
| public static FileArtifactValue createNormalFile( |
| byte[] digest, @Nullable FileContentsProxy proxy, long size, boolean isShareable) { |
| return isShareable |
| ? new RegularFileArtifactValue(digest, proxy, size) |
| : new UnshareableRegularFileArtifactValue(digest, proxy, size); |
| } |
| |
| public static FileArtifactValue createNormalFile( |
| ArtifactFileMetadata artifactMetadata, boolean isShareable) { |
| FileContentsProxy proxy = getProxyFromFileStateValue(artifactMetadata.realFileStateValue()); |
| return createNormalFile( |
| artifactMetadata.getDigest(), proxy, artifactMetadata.getSize(), isShareable); |
| } |
| |
| public static FileArtifactValue createDirectoryWithHash(byte[] digest) { |
| return new HashedDirectoryArtifactValue(digest); |
| } |
| |
| public static FileArtifactValue createDirectory(long mtime) { |
| return new DirectoryArtifactValue(mtime); |
| } |
| |
| /** |
| * Creates a FileArtifactValue used as a 'proxy' input for other ArtifactValues. These are used in |
| * {@link com.google.devtools.build.lib.actions.ActionCacheChecker}. |
| */ |
| public static FileArtifactValue createProxy(byte[] digest) { |
| Preconditions.checkNotNull(digest); |
| return createNormalFile(digest, /*proxy=*/ null, /*size=*/ 0, /*isShareable=*/ true); |
| } |
| |
| private static final class DirectoryArtifactValue extends FileArtifactValue { |
| private final long mtime; |
| |
| private DirectoryArtifactValue(long mtime) { |
| this.mtime = mtime; |
| } |
| |
| @Override |
| public FileStateType getType() { |
| return FileStateType.DIRECTORY; |
| } |
| |
| @Nullable |
| @Override |
| public byte[] getDigest() { |
| return null; |
| } |
| |
| @Override |
| public BigInteger getValueFingerprint() { |
| BigIntegerFingerprint fp = new BigIntegerFingerprint(); |
| fp.addString(getClass().getCanonicalName()); |
| fp.addLong(mtime); |
| return fp.getFingerprint(); |
| } |
| |
| @Override |
| public long getModifiedTime() { |
| return mtime; |
| } |
| |
| @Override |
| public long getSize() { |
| return 0; |
| } |
| |
| @Override |
| public boolean wasModifiedSinceDigest(Path path) throws IOException { |
| return false; |
| } |
| |
| @Override |
| public String toString() { |
| return MoreObjects.toStringHelper(this).add("mtime", mtime).toString(); |
| } |
| } |
| |
| private static final class HashedDirectoryArtifactValue extends FileArtifactValue { |
| private final byte[] digest; |
| |
| private HashedDirectoryArtifactValue(byte[] digest) { |
| this.digest = digest; |
| } |
| |
| @Override |
| public FileStateType getType() { |
| return FileStateType.DIRECTORY; |
| } |
| |
| @Nullable |
| @Override |
| public byte[] getDigest() { |
| return digest; |
| } |
| |
| @Override |
| public long getModifiedTime() { |
| return 0; |
| } |
| |
| @Override |
| public long getSize() { |
| return 0; |
| } |
| |
| @Override |
| public boolean wasModifiedSinceDigest(Path path) throws IOException { |
| // TODO(ulfjack): Ideally, we'd attempt to detect intra-build modifications here. I'm |
| // consciously deferring work here as this code will most likely change again, and we're |
| // already doing better than before by detecting inter-build modifications. |
| return false; |
| } |
| |
| @Override |
| public String toString() { |
| return MoreObjects.toStringHelper(this).add("digest", digest).toString(); |
| } |
| |
| @Override |
| public boolean equals(Object o) { |
| if (this == o) { |
| return true; |
| } |
| if (!(o instanceof HashedDirectoryArtifactValue)) { |
| return false; |
| } |
| HashedDirectoryArtifactValue r = (HashedDirectoryArtifactValue) o; |
| return Arrays.equals(digest, r.digest); |
| } |
| |
| @Override |
| public int hashCode() { |
| return Arrays.hashCode(digest); |
| } |
| } |
| |
| private static class RegularFileArtifactValue extends FileArtifactValue { |
| private final byte[] digest; |
| @Nullable private final FileContentsProxy proxy; |
| private final long size; |
| |
| private RegularFileArtifactValue(byte[] digest, @Nullable FileContentsProxy proxy, long size) { |
| this.digest = Preconditions.checkNotNull(digest); |
| this.proxy = proxy; |
| this.size = size; |
| } |
| |
| @Override |
| public FileStateType getType() { |
| return FileStateType.REGULAR_FILE; |
| } |
| |
| @Override |
| public byte[] getDigest() { |
| return digest; |
| } |
| |
| @Override |
| public long getSize() { |
| return size; |
| } |
| |
| @Override |
| public boolean wasModifiedSinceDigest(Path path) throws IOException { |
| if (proxy == null) { |
| return false; |
| } |
| FileStatus stat = path.statIfFound(Symlinks.FOLLOW); |
| return stat == null || !stat.isFile() || !proxy.equals(FileContentsProxy.create(stat)); |
| } |
| |
| @Override |
| public long getModifiedTime() { |
| throw new UnsupportedOperationException( |
| "regular file's mtime should never be called. (" + this + ")"); |
| } |
| |
| @Override |
| public String toString() { |
| return MoreObjects.toStringHelper(this) |
| .add("digest", BaseEncoding.base16().lowerCase().encode(digest)) |
| .add("size", size) |
| .add("proxy", proxy).toString(); |
| } |
| |
| @Override |
| public boolean equals(Object o) { |
| if (this == o) { |
| return true; |
| } |
| if (!(o instanceof RegularFileArtifactValue)) { |
| return false; |
| } |
| RegularFileArtifactValue r = (RegularFileArtifactValue) o; |
| return Arrays.equals(digest, r.digest) |
| && Objects.equals(proxy, r.proxy) |
| && size == r.size |
| && dataIsShareable() == r.dataIsShareable(); |
| } |
| |
| @Override |
| public int hashCode() { |
| return (proxy != null ? 127 * proxy.hashCode() : 0) |
| + 37 * Long.hashCode(getSize()) + Arrays.hashCode(getDigest()); |
| } |
| } |
| |
| private static final class UnshareableRegularFileArtifactValue extends RegularFileArtifactValue { |
| private UnshareableRegularFileArtifactValue( |
| byte[] digest, @Nullable FileContentsProxy proxy, long size) { |
| super(digest, proxy, size); |
| } |
| |
| @Override |
| public boolean dataIsShareable() { |
| return false; |
| } |
| } |
| |
| /** Metadata for remotely stored files. */ |
| public static final class RemoteFileArtifactValue extends FileArtifactValue { |
| private final byte[] digest; |
| private final long size; |
| private final int locationIndex; |
| |
| public RemoteFileArtifactValue(byte[] digest, long size, int locationIndex) { |
| this.digest = digest; |
| this.size = size; |
| this.locationIndex = locationIndex; |
| } |
| |
| @Override |
| public FileStateType getType() { |
| return FileStateType.REGULAR_FILE; |
| } |
| |
| @Override |
| public byte[] getDigest() { |
| return digest; |
| } |
| |
| @Override |
| public long getSize() { |
| return size; |
| } |
| |
| @Override |
| public long getModifiedTime() { |
| throw new UnsupportedOperationException( |
| "RemoteFileArifactValue doesn't support getModifiedTime"); |
| } |
| |
| @Override |
| public int getLocationIndex() { |
| return locationIndex; |
| } |
| |
| @Override |
| public boolean wasModifiedSinceDigest(Path path) { |
| return false; |
| } |
| |
| @Override |
| public boolean isRemote() { |
| return true; |
| } |
| |
| @Override |
| public String toString() { |
| return MoreObjects.toStringHelper(this) |
| .add("digest", bytesToString(digest)) |
| .add("size", size) |
| .add("locationIndex", locationIndex) |
| .toString(); |
| } |
| } |
| |
| private static String bytesToString(byte[] bytes) { |
| return "0x" + BaseEncoding.base16().omitPadding().encode(bytes); |
| } |
| |
| /** File stored inline in metadata. */ |
| public static class InlineFileArtifactValue extends FileArtifactValue { |
| private final byte[] data; |
| private final byte[] digest; |
| |
| private InlineFileArtifactValue(byte[] data, byte[] digest) { |
| this.data = Preconditions.checkNotNull(data); |
| this.digest = Preconditions.checkNotNull(digest); |
| } |
| |
| private InlineFileArtifactValue(byte[] bytes) { |
| this(bytes, |
| DigestHashFunction.getDefaultUnchecked().getHashFunction().hashBytes(bytes).asBytes()); |
| } |
| |
| public static InlineFileArtifactValue create(byte[] bytes, boolean shareable) { |
| return shareable |
| ? new InlineFileArtifactValue(bytes) |
| : new UnshareableInlineFileArtifactValue(bytes); |
| } |
| |
| public ByteArrayInputStream getInputStream() { |
| return new ByteArrayInputStream(data); |
| } |
| |
| @Override |
| public FileStateType getType() { |
| return FileStateType.REGULAR_FILE; |
| } |
| |
| @Override |
| public byte[] getDigest() { |
| return digest; |
| } |
| |
| @Override |
| public long getSize() { |
| return data.length; |
| } |
| |
| @Override |
| public long getModifiedTime() { |
| throw new UnsupportedOperationException(); |
| } |
| |
| @Override |
| public boolean wasModifiedSinceDigest(Path path) { |
| throw new UnsupportedOperationException(); |
| } |
| } |
| |
| private static final class UnshareableInlineFileArtifactValue extends InlineFileArtifactValue { |
| UnshareableInlineFileArtifactValue(byte[] bytes) { |
| super(bytes); |
| } |
| |
| @Override |
| public boolean dataIsShareable() { |
| return false; |
| } |
| } |
| |
| /** |
| * Used to resolve source symlinks when diskless. |
| * |
| * <p>When {@link com.google.devtools.build.lib.skyframe.ActionFileSystem} creates symlinks, it |
| * relies on metadata ({@link FileArtifactValue}) to resolve the actual underlying data. In the |
| * case of remote or inline files, this information is self-contained. However, in the case of |
| * source files, the path is required to resolve the content. |
| */ |
| public static final class SourceFileArtifactValue extends FileArtifactValue { |
| private final PathFragment execPath; |
| private final byte[] digest; |
| private final long size; |
| |
| public SourceFileArtifactValue( |
| PathFragment execPath, byte[] digest, long size) { |
| this.execPath = Preconditions.checkNotNull(execPath); |
| this.digest = Preconditions.checkNotNull(digest); |
| this.size = size; |
| } |
| |
| public PathFragment getExecPath() { |
| return execPath; |
| } |
| |
| @Override |
| public FileStateType getType() { |
| return FileStateType.REGULAR_FILE; |
| } |
| |
| @Override |
| public byte[] getDigest() { |
| return digest; |
| } |
| |
| @Override |
| public long getSize() { |
| return size; |
| } |
| |
| @Override |
| public long getModifiedTime() { |
| throw new UnsupportedOperationException(); |
| } |
| |
| @Override |
| public boolean wasModifiedSinceDigest(Path path) { |
| throw new UnsupportedOperationException(); |
| } |
| } |
| |
| private static FileContentsProxy getProxyFromFileStateValue(FileStateValue value) { |
| if (value instanceof FileStateValue.RegularFileStateValue) { |
| return ((FileStateValue.RegularFileStateValue) value).getContentsProxy(); |
| } else if (value instanceof FileStateValue.SpecialFileStateValue) { |
| return ((FileStateValue.SpecialFileStateValue) value).getContentsProxy(); |
| } |
| return null; |
| } |
| |
| private static final class SingletonMarkerValue extends FileArtifactValue implements Singleton { |
| @Override |
| public FileStateType getType() { |
| return FileStateType.NONEXISTENT; |
| } |
| |
| @Nullable |
| @Override |
| public byte[] getDigest() { |
| return null; |
| } |
| |
| @Override |
| public long getSize() { |
| return 0; |
| } |
| |
| @Override |
| public long getModifiedTime() { |
| return 0; |
| } |
| |
| @Override |
| public boolean wasModifiedSinceDigest(Path path) throws IOException { |
| return false; |
| } |
| |
| @Nullable |
| @Override |
| public BigInteger getValueFingerprint() { |
| return BigInteger.TEN; |
| } |
| |
| @Override |
| public String toString() { |
| return "singleton marker artifact value (" + hashCode() + ")"; |
| } |
| } |
| |
| private static final class OmittedFileValue extends FileArtifactValue implements Singleton { |
| @Override |
| public FileStateType getType() { |
| return FileStateType.NONEXISTENT; |
| } |
| |
| @Override |
| public byte[] getDigest() { |
| throw new UnsupportedOperationException(); |
| } |
| |
| @Override |
| public long getSize() { |
| throw new UnsupportedOperationException(); |
| } |
| |
| @Override |
| public long getModifiedTime() { |
| throw new UnsupportedOperationException(); |
| } |
| |
| @Override |
| public boolean wasModifiedSinceDigest(Path path) throws IOException { |
| return false; |
| } |
| |
| @Override |
| public String toString() { |
| return "OMITTED_FILE_MARKER"; |
| } |
| } |
| } |