blob: e04d2af5c7a9439506999bd9c90822b303f02a4a [file] [log] [blame]
// Copyright 2018 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.Joiner;
import com.google.common.collect.ImmutableMap;
import com.google.devtools.build.lib.util.io.FileOutErr;
import com.google.devtools.build.lib.vfs.Path;
import java.util.Set;
import javax.annotation.Nullable;
/**
* An {@link ExecException} thrown when an action fails to execute because one or more of its inputs
* was lost. In some cases, Bazel may know how to fix this on its own.
*/
public class LostInputsExecException extends ExecException {
private final ImmutableMap<String, ActionInput> lostInputs;
private final InputOwners inputOwners;
public LostInputsExecException(
ImmutableMap<String, ActionInput> lostInputs, InputOwners inputOwners) {
super("lost inputs with digests: " + Joiner.on(",").join(lostInputs.keySet()));
this.lostInputs = lostInputs;
this.inputOwners = inputOwners;
}
@VisibleForTesting
public ImmutableMap<String, ActionInput> getLostInputs() {
return lostInputs;
}
@VisibleForTesting
public InputOwners getInputOwners() {
return inputOwners;
}
@Override
public ActionExecutionException toActionExecutionException(
String messagePrefix, boolean verboseFailures, Action action) {
String message = messagePrefix + " failed";
return new LostInputsActionExecutionException(message + ": " + getMessage(), this, action);
}
/** An {@link ActionExecutionException} wrapping a {@link LostInputsExecException}. */
public static class LostInputsActionExecutionException extends ActionExecutionException {
/**
* If an ActionStartedEvent was emitted, then:
*
* <ul>
* <li>if rewinding is attempted, then an ActionRewindEvent should be emitted.
* <li>if rewinding fails, then an ActionCompletionEvent should be emitted.
* </ul>
*/
private boolean actionStartedEventAlreadyEmitted;
/** Used to report the action execution failure if rewinding also fails. */
private Path primaryOutputPath;
/**
* Used to report the action execution failure if rewinding also fails. Note that this will be
* closed, so it may only be used for reporting.
*/
private FileOutErr fileOutErr;
/** Used to inform rewinding that lost inputs were found during input discovery. */
private boolean fromInputDiscovery;
private LostInputsActionExecutionException(
String message, LostInputsExecException cause, Action action) {
super(message, cause, action, /*catastrophe=*/ false);
}
public ImmutableMap<String, ActionInput> getLostInputs() {
return ((LostInputsExecException) getCause()).getLostInputs();
}
public InputOwners getInputOwners() {
return ((LostInputsExecException) getCause()).getInputOwners();
}
public Path getPrimaryOutputPath() {
return primaryOutputPath;
}
public void setPrimaryOutputPath(Path primaryOutputPath) {
this.primaryOutputPath = primaryOutputPath;
}
public FileOutErr getFileOutErr() {
return fileOutErr;
}
public void setFileOutErr(FileOutErr fileOutErr) {
this.fileOutErr = fileOutErr;
}
public boolean isActionStartedEventAlreadyEmitted() {
return actionStartedEventAlreadyEmitted;
}
public void setActionStartedEventAlreadyEmitted() {
this.actionStartedEventAlreadyEmitted = true;
}
public boolean isFromInputDiscovery() {
return fromInputDiscovery;
}
public void setFromInputDiscovery() {
this.fromInputDiscovery = true;
}
}
/**
* Specifies the owning {@link Artifact}s that were responsible for the lost inputs and whether
* the inputs came from runfiles.
*/
public interface InputOwners {
/**
* Returns the owning {@link Artifact} that was responsible for the lost {@link ActionInput} or
* {@code null} if there is no such owner. Throws if {@code input} was not lost.
*/
@Nullable
Artifact getOwner(ActionInput input);
/** Returns the lost {@link ActionInput}s that came from runfiles along with their owners. */
Set<ActionInput> getRunfilesInputsAndOwners();
}
}