From 484b305907cc4d3e9d66059a7b84fc7e5dbb2d21 Mon Sep 17 00:00:00 2001 From: Chi Wang Date: Wed, 13 Sep 2023 00:52:24 +0200 Subject: [PATCH] [6.4.0] Add output name to CacheNotFoundException (#19452) to make it easier to debug errors related to remote cache eviction. Fixes #10025. PiperOrigin-RevId: 563361764 Change-Id: I2c29cd521c7ec1293f770b28a5cdcdf356750396 --- .../lib/remote/RemoteActionInputFetcher.java | 3 +- .../build/lib/remote/RemoteCache.java | 82 +++++++++++++++++-- .../lib/remote/RemoteExecutionService.java | 7 +- .../RemoteRepositoryRemoteExecutor.java | 12 ++- .../devtools/build/lib/remote/common/BUILD | 2 + .../remote/common/CacheNotFoundException.java | 18 +++- 6 files changed, 106 insertions(+), 18 deletions(-) diff --git a/src/main/java/com/google/devtools/build/lib/remote/RemoteActionInputFetcher.java b/src/main/java/com/google/devtools/build/lib/remote/RemoteActionInputFetcher.java index 04d6fb042cadf0..b0c8faf9a0f9a5 100644 --- a/src/main/java/com/google/devtools/build/lib/remote/RemoteActionInputFetcher.java +++ b/src/main/java/com/google/devtools/build/lib/remote/RemoteActionInputFetcher.java @@ -124,7 +124,8 @@ protected ListenableFuture doDownloadFile( downloadProgressReporter = new DownloadProgressReporter(NO_ACTION, "", 0); } - return remoteCache.downloadFile(context, tempPath, digest, downloadProgressReporter); + return remoteCache.downloadFile( + context, execPath.getPathString(), tempPath, digest, downloadProgressReporter); } public static class DownloadProgress implements FetchProgress { diff --git a/src/main/java/com/google/devtools/build/lib/remote/RemoteCache.java b/src/main/java/com/google/devtools/build/lib/remote/RemoteCache.java index 2ad56b0239896c..63659815361a26 100644 --- a/src/main/java/com/google/devtools/build/lib/remote/RemoteCache.java +++ b/src/main/java/com/google/devtools/build/lib/remote/RemoteCache.java @@ -33,6 +33,7 @@ import com.google.common.util.concurrent.SettableFuture; import com.google.devtools.build.lib.concurrent.ThreadSafety; import com.google.devtools.build.lib.exec.SpawnProgressEvent; +import com.google.devtools.build.lib.remote.common.CacheNotFoundException; import com.google.devtools.build.lib.remote.common.LazyFileOutputStream; import com.google.devtools.build.lib.remote.common.OutputDigestMismatchException; import com.google.devtools.build.lib.remote.common.ProgressStatusListener; @@ -201,6 +202,11 @@ protected ListenableFuture uploadBlob( return RxFutures.toListenableFuture(upload); } + public ListenableFuture downloadBlob( + RemoteActionExecutionContext context, Digest digest) { + return downloadBlob(context, /* blobName= */ "", digest); + } + /** * Downloads a blob with content hash {@code digest} and stores its content in memory. * @@ -208,14 +214,22 @@ protected ListenableFuture uploadBlob( * the content is stored in the future's {@code byte[]}. */ public ListenableFuture downloadBlob( - RemoteActionExecutionContext context, Digest digest) { + RemoteActionExecutionContext context, String blobName, Digest digest) { if (digest.getSizeBytes() == 0) { return EMPTY_BYTES; } ByteArrayOutputStream bOut = new ByteArrayOutputStream((int) digest.getSizeBytes()); + var download = downloadBlob(context, blobName, digest, bOut); SettableFuture outerF = SettableFuture.create(); + outerF.addListener( + () -> { + if (outerF.isCancelled()) { + download.cancel(/* mayInterruptIfRunning= */ true); + } + }, + directExecutor()); Futures.addCallback( - cacheProtocol.downloadBlob(context, digest, bOut), + download, new FutureCallback() { @Override public void onSuccess(Void aVoid) { @@ -237,12 +251,37 @@ public void onFailure(Throwable t) { } private ListenableFuture downloadBlob( - RemoteActionExecutionContext context, Digest digest, OutputStream out) { + RemoteActionExecutionContext context, String blobName, Digest digest, OutputStream out) { if (digest.getSizeBytes() == 0) { return COMPLETED_SUCCESS; } + var download = cacheProtocol.downloadBlob(context, digest, out); + SettableFuture future = SettableFuture.create(); + future.addListener( + () -> { + if (future.isCancelled()) { + download.cancel(/* mayInterruptIfRunning= */ true); + } + }, + directExecutor()); + Futures.addCallback( + download, + new FutureCallback() { + @Override + public void onSuccess(Void result) { + future.set(result); + } - return cacheProtocol.downloadBlob(context, digest, out); + @Override + public void onFailure(Throwable t) { + if (t instanceof CacheNotFoundException) { + ((CacheNotFoundException) t).setFilename(blobName); + } + future.setException(t); + } + }, + directExecutor()); + return future; } /** A reporter that reports download progresses. */ @@ -315,6 +354,13 @@ public ListenableFuture downloadFile( throws IOException { SettableFuture outerF = SettableFuture.create(); ListenableFuture f = downloadFile(context, localPath, digest, reporter); + outerF.addListener( + () -> { + if (outerF.isCancelled()) { + f.cancel(/* mayInterruptIfRunning= */ true); + } + }, + directExecutor()); Futures.addCallback( f, new FutureCallback() { @@ -325,7 +371,10 @@ public void onSuccess(Void unused) { @Override public void onFailure(Throwable throwable) { - if (throwable instanceof OutputDigestMismatchException) { + if (throwable instanceof CacheNotFoundException) { + var cacheNotFoundException = (CacheNotFoundException) throwable; + cacheNotFoundException.setFilename(outputPath); + } else if (throwable instanceof OutputDigestMismatchException) { OutputDigestMismatchException e = ((OutputDigestMismatchException) throwable); e.setOutputPath(outputPath); e.setLocalPath(localPath); @@ -341,11 +390,16 @@ public void onFailure(Throwable throwable) { /** Downloads a file (that is not a directory). The content is fetched from the digest. */ public ListenableFuture downloadFile( RemoteActionExecutionContext context, Path path, Digest digest) throws IOException { - return downloadFile(context, path, digest, new DownloadProgressReporter(NO_ACTION, "", 0)); + return downloadFile( + context, + path.getPathString(), + path, + digest, + new DownloadProgressReporter(NO_ACTION, "", 0)); } /** Downloads a file (that is not a directory). The content is fetched from the digest. */ - public ListenableFuture downloadFile( + private ListenableFuture downloadFile( RemoteActionExecutionContext context, Path path, Digest digest, @@ -409,7 +463,12 @@ public final List> downloadOutErr( downloads.add(Futures.immediateFailedFuture(e)); } } else if (result.hasStdoutDigest()) { - downloads.add(downloadBlob(context, result.getStdoutDigest(), outErr.getOutputStream())); + downloads.add( + downloadBlob( + context, + /* blobName= */ "", + result.getStdoutDigest(), + outErr.getOutputStream())); } if (!result.getStderrRaw().isEmpty()) { try { @@ -419,7 +478,12 @@ public final List> downloadOutErr( downloads.add(Futures.immediateFailedFuture(e)); } } else if (result.hasStderrDigest()) { - downloads.add(downloadBlob(context, result.getStderrDigest(), outErr.getErrorStream())); + downloads.add( + downloadBlob( + context, + /* blobName= */ "", + result.getStderrDigest(), + outErr.getErrorStream())); } return downloads; } diff --git a/src/main/java/com/google/devtools/build/lib/remote/RemoteExecutionService.java b/src/main/java/com/google/devtools/build/lib/remote/RemoteExecutionService.java index ecf8f50166b880..ebe6efc6cb8659 100644 --- a/src/main/java/com/google/devtools/build/lib/remote/RemoteExecutionService.java +++ b/src/main/java/com/google/devtools/build/lib/remote/RemoteExecutionService.java @@ -1003,10 +1003,11 @@ ActionResultMetadata parseActionResultMetadata( Map> dirMetadataDownloads = Maps.newHashMapWithExpectedSize(result.getOutputDirectoriesCount()); for (OutputDirectory dir : result.getOutputDirectories()) { + var outputPath = encodeBytestringUtf8(dir.getPath()); dirMetadataDownloads.put( - remotePathResolver.outputPathToLocalPath(encodeBytestringUtf8(dir.getPath())), + remotePathResolver.outputPathToLocalPath(outputPath), Futures.transformAsync( - remoteCache.downloadBlob(context, dir.getTreeDigest()), + remoteCache.downloadBlob(context, outputPath, dir.getTreeDigest()), (treeBytes) -> immediateFuture(Tree.parseFrom(treeBytes, ExtensionRegistry.getEmptyRegistry())), directExecutor())); @@ -1176,7 +1177,7 @@ public InMemoryOutput downloadOutputs(RemoteAction action, RemoteActionResult re try (SilentCloseable c = Profiler.instance().profile("Remote.downloadInMemoryOutput")) { if (inMemoryOutput != null) { ListenableFuture inMemoryOutputDownload = - remoteCache.downloadBlob(context, inMemoryOutputDigest); + remoteCache.downloadBlob(context, inMemoryOutputPath.getPathString(), inMemoryOutputDigest); waitForBulkTransfer( ImmutableList.of(inMemoryOutputDownload), /* cancelRemainingOnInterrupt= */ true); byte[] data = getFromFuture(inMemoryOutputDownload); diff --git a/src/main/java/com/google/devtools/build/lib/remote/RemoteRepositoryRemoteExecutor.java b/src/main/java/com/google/devtools/build/lib/remote/RemoteRepositoryRemoteExecutor.java index 3a731f72709be2..8fb79e208f4cc7 100644 --- a/src/main/java/com/google/devtools/build/lib/remote/RemoteRepositoryRemoteExecutor.java +++ b/src/main/java/com/google/devtools/build/lib/remote/RemoteRepositoryRemoteExecutor.java @@ -86,14 +86,18 @@ private ExecutionResult downloadOutErr(RemoteActionExecutionContext context, Act if (!result.getStdoutRaw().isEmpty()) { stdout = result.getStdoutRaw().toByteArray(); } else if (result.hasStdoutDigest()) { - stdout = Utils.getFromFuture(remoteCache.downloadBlob(context, result.getStdoutDigest())); + stdout = + Utils.getFromFuture( + remoteCache.downloadBlob(context, "", result.getStdoutDigest())); } byte[] stderr = new byte[0]; if (!result.getStderrRaw().isEmpty()) { stderr = result.getStderrRaw().toByteArray(); } else if (result.hasStderrDigest()) { - stderr = Utils.getFromFuture(remoteCache.downloadBlob(context, result.getStderrDigest())); + stderr = + Utils.getFromFuture( + remoteCache.downloadBlob(context, "", result.getStderrDigest())); } return new ExecutionResult(result.getExitCode(), stdout, stderr); @@ -138,7 +142,7 @@ public ExecutionResult execute( platform, timeout, acceptCached, - /*salt=*/ null); + /* salt= */ null); Digest actionDigest = digestUtil.compute(action); ActionKey actionKey = new ActionKey(actionDigest); CachedActionResult cachedActionResult; @@ -158,7 +162,7 @@ public ExecutionResult execute( additionalInputs.put(actionDigest, action); additionalInputs.put(commandHash, command); - remoteCache.ensureInputsPresent(context, merkleTree, additionalInputs, /*force=*/ true); + remoteCache.ensureInputsPresent(context, merkleTree, additionalInputs, /* force= */ true); } try (SilentCloseable c = diff --git a/src/main/java/com/google/devtools/build/lib/remote/common/BUILD b/src/main/java/com/google/devtools/build/lib/remote/common/BUILD index 36e0a515df1eb5..62a6dc047ec21d 100644 --- a/src/main/java/com/google/devtools/build/lib/remote/common/BUILD +++ b/src/main/java/com/google/devtools/build/lib/remote/common/BUILD @@ -14,6 +14,8 @@ java_library( name = "cache_not_found_exception", srcs = ["CacheNotFoundException.java"], deps = [ + "//third_party:guava", + "//third_party:jsr305", "@remoteapis//:build_bazel_remote_execution_v2_remote_execution_java_proto", ], ) diff --git a/src/main/java/com/google/devtools/build/lib/remote/common/CacheNotFoundException.java b/src/main/java/com/google/devtools/build/lib/remote/common/CacheNotFoundException.java index cad87a15023875..b48ccb2b23fa9b 100644 --- a/src/main/java/com/google/devtools/build/lib/remote/common/CacheNotFoundException.java +++ b/src/main/java/com/google/devtools/build/lib/remote/common/CacheNotFoundException.java @@ -15,7 +15,9 @@ package com.google.devtools.build.lib.remote.common; import build.bazel.remote.execution.v2.Digest; +import com.google.common.base.Strings; import java.io.IOException; +import javax.annotation.Nullable; /** * An exception to indicate cache misses. TODO(olaola): have a class of checked @@ -23,13 +25,27 @@ */ public final class CacheNotFoundException extends IOException { private final Digest missingDigest; + @Nullable private String filename; public CacheNotFoundException(Digest missingDigest) { - super("Missing digest: " + missingDigest.getHash() + "/" + missingDigest.getSizeBytes()); this.missingDigest = missingDigest; } + public void setFilename(@Nullable String filename) { + this.filename = filename; + } + public Digest getMissingDigest() { return missingDigest; } + + @Override + public String getMessage() { + String message = + "Missing digest: " + missingDigest.getHash() + "/" + missingDigest.getSizeBytes(); + if (!Strings.isNullOrEmpty(filename)) { + message += " for " + filename; + } + return message; + } }