Add a flag for disabling local fallback for actions that failed remotely.

--
PiperOrigin-RevId: 141817345
MOS_MIGRATED_REVID=141817345
diff --git a/src/main/java/com/google/devtools/build/lib/remote/RemoteOptions.java b/src/main/java/com/google/devtools/build/lib/remote/RemoteOptions.java
index 538c554..53fb4a9 100644
--- a/src/main/java/com/google/devtools/build/lib/remote/RemoteOptions.java
+++ b/src/main/java/com/google/devtools/build/lib/remote/RemoteOptions.java
@@ -114,4 +114,12 @@
       help = "Whether to accept remotely cached action results."
     )
   public boolean remoteAcceptCached;
+
+  @Option(
+      name = "remote_allow_local_fallback",
+      defaultValue = "true",
+      category = "remote",
+      help = "Whether to fall back to standalone strategy if remote fails."
+    )
+  public boolean remoteAllowLocalFallback;
 }
diff --git a/src/main/java/com/google/devtools/build/lib/remote/RemoteSpawnStrategy.java b/src/main/java/com/google/devtools/build/lib/remote/RemoteSpawnStrategy.java
index 4dc998e..14a275e 100644
--- a/src/main/java/com/google/devtools/build/lib/remote/RemoteSpawnStrategy.java
+++ b/src/main/java/com/google/devtools/build/lib/remote/RemoteSpawnStrategy.java
@@ -67,6 +67,7 @@
   private final RemoteWorkExecutor remoteWorkExecutor;
   private final boolean verboseFailures;
   private final boolean remoteAcceptCached;
+  private final boolean remoteAllowLocalFallback;
 
   RemoteSpawnStrategy(
       Map<String, String> clientEnv,
@@ -82,6 +83,7 @@
     this.remoteActionCache = actionCache;
     this.remoteWorkExecutor = workExecutor;
     this.remoteAcceptCached = options.remoteAcceptCached;
+    this.remoteAllowLocalFallback = options.remoteAllowLocalFallback;
   }
 
   private Action buildAction(
@@ -239,7 +241,7 @@
         remoteActionCache.downloadAllResults(result, execRoot);
         return;
       }
-      if (status.getError() == ExecutionStatus.ErrorCode.EXEC_FAILED) {
+      if (status.getError() == ExecutionStatus.ErrorCode.EXEC_FAILED || !remoteAllowLocalFallback) {
         passRemoteOutErr(result, actionExecutionContext.getFileOutErr());
         throw new UserExecException(status.getErrorDetail());
       }
@@ -258,10 +260,18 @@
         stackTrace = "\n" + Throwables.getStackTraceAsString(e);
       }
       eventHandler.handle(Event.warn(mnemonic + " remote work failed (" + e + ")" + stackTrace));
-      execLocally(spawn, actionExecutionContext, actionKey);
+      if (remoteAllowLocalFallback) {
+        execLocally(spawn, actionExecutionContext, actionKey);
+      } else {
+        throw new UserExecException(e);
+      }
     } catch (CacheNotFoundException e) {
       eventHandler.handle(Event.warn(mnemonic + " remote work results cache miss (" + e + ")"));
-      execLocally(spawn, actionExecutionContext, actionKey);
+      if (remoteAllowLocalFallback) {
+        execLocally(spawn, actionExecutionContext, actionKey);
+      } else {
+        throw new UserExecException(e);
+      }
     } catch (UnsupportedOperationException e) {
       eventHandler.handle(
           Event.warn(mnemonic + " unsupported operation for action cache (" + e + ")"));