Skip to content

Commit

Permalink
Add ability for a Spawn to declare a required local output
Browse files Browse the repository at this point in the history
Some native actions run both remotely and locally. For example,
the CppCompileAction runs compilation remotely but does .d file
pruning locally. This change adds the necessary infrastructure
for an action to tell a Spawn which of its outputs it needs on
the local filesytem.

Progress towards bazelbuild#6862
  • Loading branch information
buchgr committed Feb 5, 2019
1 parent 2f82b33 commit 0f9f470
Show file tree
Hide file tree
Showing 16 changed files with 87 additions and 1 deletion.
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@
import com.google.devtools.common.options.OptionsProvider;
import java.io.Closeable;
import java.io.IOException;
import java.util.Collection;
import java.util.Map;
import javax.annotation.Nullable;

Expand Down Expand Up @@ -72,6 +73,8 @@ public enum ShowSubcommands {

private final ArtifactPathResolver pathResolver;

private final ImmutableList<Artifact> requiredLocalOutputs;

private ActionExecutionContext(
Executor executor,
MetadataProvider actionInputFileCache,
Expand All @@ -82,6 +85,7 @@ private ActionExecutionContext(
ExtendedEventHandler eventHandler,
Map<String, String> clientEnv,
ImmutableMap<Artifact, ImmutableList<FilesetOutputSymlink>> topLevelFilesets,
ImmutableList<Artifact> requiredLocalOutputs,
@Nullable ArtifactExpander artifactExpander,
@Nullable Environment env,
@Nullable FileSystem actionFileSystem,
Expand All @@ -102,6 +106,7 @@ private ActionExecutionContext(
this.pathResolver = ArtifactPathResolver.createPathResolver(actionFileSystem,
// executor is only ever null in testing.
executor == null ? null : executor.getExecRoot());
this.requiredLocalOutputs = Preconditions.checkNotNull(requiredLocalOutputs);
}

public ActionExecutionContext(
Expand All @@ -114,6 +119,7 @@ public ActionExecutionContext(
ExtendedEventHandler eventHandler,
Map<String, String> clientEnv,
ImmutableMap<Artifact, ImmutableList<FilesetOutputSymlink>> topLevelFilesets,
ImmutableList<Artifact> requiredLocalOutputs,
ArtifactExpander artifactExpander,
@Nullable FileSystem actionFileSystem,
@Nullable Object skyframeDepsResult) {
Expand All @@ -127,6 +133,7 @@ public ActionExecutionContext(
eventHandler,
clientEnv,
topLevelFilesets,
requiredLocalOutputs,
artifactExpander,
/*env=*/ null,
actionFileSystem,
Expand Down Expand Up @@ -154,6 +161,7 @@ public static ActionExecutionContext forInputDiscovery(
eventHandler,
clientEnv,
ImmutableMap.of(),
/*requiredLocalOutputs=*/ ImmutableList.of(),
/*artifactExpander=*/ null,
env,
actionFileSystem,
Expand Down Expand Up @@ -307,6 +315,18 @@ public ActionKeyContext getActionKeyContext() {
return actionKeyContext;
}

/**
* Returns the collection of files that this command must write and make available via
* the local {@link com.google.devtools.build.lib.vfs.FileSystem}. The returned output
* artifacts are a subset of the action's output artifacts.
*
* <p>This is for use with remote execution, where as an optimization we don't want to
* download all output files.
*/
public Collection<Artifact> getRequiredLocalOutputs() {
return requiredLocalOutputs;
}

@Override
public void close() throws IOException {
fileOutErr.close();
Expand All @@ -327,6 +347,29 @@ public ActionExecutionContext withFileOutErr(FileOutErr fileOutErr) {
eventHandler,
clientEnv,
topLevelFilesets,
requiredLocalOutputs,
artifactExpander,
env,
actionFileSystem,
skyframeDepsResult);
}

/**
* Allows us to create a new context that provides a list of output artifacts that need to
* be staged on the local filesystem.
*/
public ActionExecutionContext withRequiredLocalOutputs(Collection<? extends Artifact> requiredLocalOutputs) {
return new ActionExecutionContext(
executor,
actionInputFileCache,
actionInputPrefetcher,
actionKeyContext,
metadataHandler,
fileOutErr,
eventHandler,
clientEnv,
topLevelFilesets,
ImmutableList.copyOf(requiredLocalOutputs),
artifactExpander,
env,
actionFileSystem,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
import com.google.devtools.build.lib.actions.ActionExecutionMetadata;
import com.google.devtools.build.lib.actions.ActionInput;
import com.google.devtools.build.lib.actions.ActionStatusMessage;
import com.google.devtools.build.lib.actions.Artifact;
import com.google.devtools.build.lib.actions.Artifact.ArtifactExpander;
import com.google.devtools.build.lib.actions.ArtifactPathResolver;
import com.google.devtools.build.lib.actions.EnvironmentalExecException;
Expand All @@ -43,6 +44,7 @@
import com.google.devtools.build.lib.vfs.PathFragment;
import java.io.IOException;
import java.time.Duration;
import java.util.Collection;
import java.util.List;
import java.util.SortedMap;
import java.util.concurrent.atomic.AtomicInteger;
Expand Down Expand Up @@ -271,5 +273,10 @@ public void report(ProgressStatus state, String name) {
break;
}
}

@Override
public Collection<Artifact> getRequiredLocalOutputs() {
return actionExecutionContext.getRequiredLocalOutputs();
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,9 @@
// limitations under the License.
package com.google.devtools.build.lib.exec;

import com.google.common.collect.ImmutableList;
import com.google.devtools.build.lib.actions.ActionInput;
import com.google.devtools.build.lib.actions.Artifact;
import com.google.devtools.build.lib.actions.Artifact.ArtifactExpander;
import com.google.devtools.build.lib.actions.ArtifactPathResolver;
import com.google.devtools.build.lib.actions.ExecException;
Expand All @@ -25,6 +27,7 @@
import com.google.devtools.build.lib.vfs.PathFragment;
import java.io.IOException;
import java.time.Duration;
import java.util.Collection;
import java.util.SortedMap;

/**
Expand Down Expand Up @@ -129,6 +132,7 @@ public enum ProgressStatus {
* by different threads, so they MUST not call any shared non-thread-safe objects.
*/
interface SpawnExecutionContext {

/**
* Returns a unique id for this spawn, to be used for logging. Note that a single spawn may be
* passed to multiple {@link SpawnRunner} implementations, so any log entries should also
Expand Down Expand Up @@ -199,6 +203,18 @@ SortedMap<PathFragment, ActionInput> getInputMapping(boolean expandTreeArtifacts

/** Reports a progress update to the Spawn strategy. */
void report(ProgressStatus state, String name);

/**
* Returns the collection of files that this command must write and make available via
* the local {@link com.google.devtools.build.lib.vfs.FileSystem}. The returned output
* artifacts are a subset of the action's output artifacts.
*
* <p>This is for use with remote execution, where as an optimization we don't want to
* download all output files.
*/
default Collection<Artifact> getRequiredLocalOutputs() {
return ImmutableList.of();
}
}

/**
Expand Down Expand Up @@ -234,4 +250,4 @@ SpawnResult exec(Spawn spawn, SpawnExecutionContext context)

/* Name of the SpawnRunner. */
String getName();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -601,6 +601,7 @@ public ActionExecutionContext getContext(
: progressSuppressingEventHandler,
clientEnv,
topLevelFilesets,
ImmutableList.of(),
new ArtifactExpanderImpl(expandedInputs, expandedFilesets),
actionFileSystem,
skyframeDepsResult);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@
import static com.google.devtools.build.lib.actions.util.ActionsTestUtil.NULL_ACTION_OWNER;
import static org.junit.Assert.fail;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.devtools.build.lib.actions.util.DummyExecutor;
import com.google.devtools.build.lib.analysis.actions.SymlinkAction;
Expand Down Expand Up @@ -66,6 +67,7 @@ private ActionExecutionContext createContext() {
executor.getEventHandler(),
ImmutableMap.<String, String>of(),
ImmutableMap.of(),
/*requiredLocalOutputs=*/ ImmutableList.of(),
null,
null,
null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -147,6 +147,7 @@ public static ActionExecutionContext createContext(
executor != null ? executor.getEventHandler() : null,
ImmutableMap.copyOf(clientEnv),
ImmutableMap.of(),
ImmutableList.of(),
actionGraph == null
? createDummyArtifactExpander()
: ActionInputHelper.actionGraphArtifactExpander(actionGraph),
Expand Down Expand Up @@ -187,6 +188,7 @@ public static ActionExecutionContext createContext(ExtendedEventHandler eventHan
eventHandler,
ImmutableMap.of(),
ImmutableMap.of(),
ImmutableList.of(),
createDummyArtifactExpander(),
/*actionFileSystem=*/ null,
/*skyframeDepsResult=*/ null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
import static com.google.common.truth.Truth.assertThat;
import static com.google.devtools.build.lib.actions.util.ActionsTestUtil.NULL_ACTION_OWNER;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Sets;
Expand Down Expand Up @@ -71,6 +72,7 @@ public final void createExecutorAndContext() throws Exception {
executor.getEventHandler(),
ImmutableMap.<String, String>of(),
ImmutableMap.of(),
ImmutableList.of(),
null,
null,
null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -203,6 +203,7 @@ public void expand(Artifact artifact, Collection<? super Artifact> output) {
executor.getEventHandler(),
ImmutableMap.<String, String>of(),
ImmutableMap.of(),
ImmutableList.of(),
artifactExpander,
/*actionFileSystem=*/ null,
/*skyframeDepsResult=*/ null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
import static com.google.common.truth.Truth.assertThat;
import static com.google.devtools.build.lib.actions.util.ActionsTestUtil.NULL_ACTION_OWNER;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.devtools.build.lib.actions.ActionExecutionContext;
import com.google.devtools.build.lib.actions.ActionInputPrefetcher;
Expand Down Expand Up @@ -89,6 +90,7 @@ public void testSymlink() throws Exception {
executor.getEventHandler(),
ImmutableMap.<String, String>of(),
ImmutableMap.of(),
ImmutableList.of(),
null,
/*actionFileSystem=*/ null,
/*skyframeDepsResult=*/ null));
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -192,6 +192,7 @@ private ActionExecutionContext createContext(Executor executor) {
executor.getEventHandler(),
ImmutableMap.of(),
ImmutableMap.of(),
ImmutableList.of(),
null,
/*actionFileSystem=*/ null,
/*skyframeDepsResult=*/ null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2231,6 +2231,7 @@ public ActionExecutionContext build() {
reporter,
clientEnv,
ImmutableMap.of(),
ImmutableList.of(),
artifactExpander,
/*actionFileSystem=*/ null,
/*skyframeDepsResult*/ null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -96,6 +96,7 @@ public FakeActionExecutionContext(
/*eventHandler=*/ null,
/* clientEnv= */ ImmutableMap.of(),
/* topLevelFilesets= */ ImmutableMap.of(),
/* requiredLocalOutputs= */ ImmutableList.of(),
/* artifactExpander= */ null,
/* actionFileSystem= */ null,
/* skyframeDepsResult= */ null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
import static com.google.common.truth.Truth.assertThat;
import static com.google.devtools.build.lib.actions.util.ActionsTestUtil.NULL_ACTION_OWNER;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.devtools.build.lib.actions.ActionExecutionContext;
import com.google.devtools.build.lib.actions.ActionKeyContext;
Expand Down Expand Up @@ -125,6 +126,7 @@ private ActionExecutionContext makeDummyContext() {
executor.getEventHandler(),
ImmutableMap.of(),
ImmutableMap.of(),
ImmutableList.of(),
null,
null,
null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@

import static com.google.common.truth.Truth.assertThat;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.devtools.build.lib.actions.AbstractAction;
Expand Down Expand Up @@ -90,6 +91,7 @@ public final void createExecutorAndContext() throws Exception {
executor.getEventHandler(),
ImmutableMap.<String, String>of(),
ImmutableMap.of(),
ImmutableList.of(),
null,
/*actionFileSystem=*/ null,
/*skyframeDepsResult=*/ null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -794,6 +794,7 @@ public void testJ2ObjCCustomModuleMap() throws Exception {
null,
ImmutableMap.<String, String>of(),
ImmutableMap.of(),
ImmutableList.of(),
DUMMY_ARTIFACT_EXPANDER,
null,
null);
Expand Down Expand Up @@ -847,6 +848,7 @@ public void testModuleMapFromGenJarTreeArtifact() throws Exception {
null,
ImmutableMap.of(),
ImmutableMap.of(),
ImmutableList.of(),
DUMMY_ARTIFACT_EXPANDER,
null,
null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -195,6 +195,7 @@ private ActionExecutionContext createContext() {
executor.getEventHandler(),
ImmutableMap.<String, String>of(),
ImmutableMap.of(),
ImmutableList.of(),
SIMPLE_ARTIFACT_EXPANDER,
/*actionFileSystem=*/ null,
/*skyframeDepsResult=*/ null);
Expand Down

0 comments on commit 0f9f470

Please sign in to comment.