// 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.analysis.actions; import static java.nio.charset.StandardCharsets.ISO_8859_1; import com.google.common.annotations.VisibleForTesting; import com.google.common.base.CharMatcher; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; import com.google.common.collect.ImmutableSet; import com.google.common.collect.Iterables; import com.google.common.collect.Lists; import com.google.devtools.build.lib.actions.AbstractAction; import com.google.devtools.build.lib.actions.Action; import com.google.devtools.build.lib.actions.ActionExecutionContext; import com.google.devtools.build.lib.actions.ActionExecutionException; import com.google.devtools.build.lib.actions.ActionInput; import com.google.devtools.build.lib.actions.ActionInputHelper; import com.google.devtools.build.lib.actions.ActionOwner; import com.google.devtools.build.lib.actions.Artifact; import com.google.devtools.build.lib.actions.BaseSpawn; import com.google.devtools.build.lib.actions.CommandAction; import com.google.devtools.build.lib.actions.CompositeRunfilesSupplier; import com.google.devtools.build.lib.actions.EmptyRunfilesSupplier; import com.google.devtools.build.lib.actions.ExecException; import com.google.devtools.build.lib.actions.ExecutionInfoSpecifier; import com.google.devtools.build.lib.actions.Executor; import com.google.devtools.build.lib.actions.ParameterFile.ParameterFileType; import com.google.devtools.build.lib.actions.ResourceSet; import com.google.devtools.build.lib.actions.RunfilesSupplier; import com.google.devtools.build.lib.actions.Spawn; import com.google.devtools.build.lib.actions.SpawnActionContext; import com.google.devtools.build.lib.actions.extra.EnvironmentVariable; import com.google.devtools.build.lib.actions.extra.ExtraActionInfo; import com.google.devtools.build.lib.actions.extra.SpawnInfo; import com.google.devtools.build.lib.analysis.AnalysisEnvironment; import com.google.devtools.build.lib.analysis.FilesToRunProvider; import com.google.devtools.build.lib.analysis.TransitiveInfoCollection; import com.google.devtools.build.lib.analysis.config.BuildConfiguration; import com.google.devtools.build.lib.collect.CollectionUtils; import com.google.devtools.build.lib.collect.IterablesChain; import com.google.devtools.build.lib.collect.nestedset.NestedSet; import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder; import com.google.devtools.build.lib.syntax.SkylarkList; import com.google.devtools.build.lib.util.Fingerprint; import com.google.devtools.build.lib.util.Preconditions; import com.google.devtools.build.lib.util.ShellEscaper; import com.google.devtools.build.lib.vfs.PathFragment; import com.google.protobuf.GeneratedMessage.GeneratedExtension; import java.nio.charset.Charset; import java.util.ArrayList; import java.util.Collections; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; import java.util.Set; import javax.annotation.CheckReturnValue; import javax.annotation.Nullable; /** An Action representing an arbitrary subprocess to be forked and exec'd. */ public class SpawnAction extends AbstractAction implements ExecutionInfoSpecifier, CommandAction { /** Sets extensions on ExtraActionInfo **/ protected static class ExtraActionInfoSupplier<T> { private final GeneratedExtension<ExtraActionInfo, T> extension; private final T value; protected ExtraActionInfoSupplier(GeneratedExtension<ExtraActionInfo, T> extension, T value) { this.extension = extension; this.value = value; } void extend(ExtraActionInfo.Builder builder) { builder.setExtension(extension, value); } } private static final String GUID = "ebd6fce3-093e-45ee-adb6-bf513b602f0d"; private final CommandLine argv; private final boolean executeUnconditionally; private final boolean isShellCommand; private final String progressMessage; private final String mnemonic; private final ResourceSet resourceSet; private final ImmutableMap<String, String> environment; private final ImmutableSet<String> clientEnvironmentVariables; private final ImmutableMap<String, String> executionInfo; private final ExtraActionInfoSupplier<?> extraActionInfoSupplier; /** * Constructs a SpawnAction using direct initialization arguments. * * <p>All collections provided must not be subsequently modified. * * @param owner the owner of the Action. * @param tools the set of files comprising the tool that does the work (e.g. compiler). * @param inputs the set of all files potentially read by this action; must not be subsequently * modified. * @param outputs the set of all files written by this action; must not be subsequently modified. * @param resourceSet the resources consumed by executing this Action * @param environment the map of environment variables. * @param clientEnvironmentVariables the set of variables to be inherited from the client * environment. * @param argv the command line to execute. This is merely a list of options to the executable, * and is uninterpreted by the build tool for the purposes of dependency checking; typically * it may include the names of input and output files, but this is not necessary. * @param isShellCommand Whether the command line represents a shell command with the given shell * executable. This is used to give better error messages. * @param progressMessage the message printed during the progression of the build * @param mnemonic the mnemonic that is reported in the master log. */ public SpawnAction( ActionOwner owner, Iterable<Artifact> tools, Iterable<Artifact> inputs, Iterable<Artifact> outputs, ResourceSet resourceSet, CommandLine argv, boolean isShellCommand, Map<String, String> environment, Set<String> clientEnvironmentVariables, String progressMessage, String mnemonic) { this( owner, tools, inputs, outputs, resourceSet, argv, isShellCommand, ImmutableMap.copyOf(environment), ImmutableSet.copyOf(clientEnvironmentVariables), ImmutableMap.<String, String>of(), progressMessage, EmptyRunfilesSupplier.INSTANCE, mnemonic, false, null); } /** * Constructs a SpawnAction using direct initialization arguments. * * <p>All collections provided must not be subsequently modified. * * @param owner the owner of the Action. * @param tools the set of files comprising the tool that does the work (e.g. compiler). This is a * subset of "inputs" and is only used by the WorkerSpawnStrategy. * @param inputs the set of all files potentially read by this action; must not be subsequently * modified. * @param outputs the set of all files written by this action; must not be subsequently modified. * @param resourceSet the resources consumed by executing this Action * @param environment the map of environment variables. * @param clientEnvironmentVariables the set of variables to be inherited from the client * environment. * @param executionInfo out-of-band information for scheduling the spawn. * @param argv the argv array (including argv[0]) of arguments to pass. This is merely a list of * options to the executable, and is uninterpreted by the build tool for the purposes of * dependency checking; typically it may include the names of input and output files, but this * is not necessary. * @param isShellCommand Whether the command line represents a shell command with the given shell * executable. This is used to give better error messages. * @param progressMessage the message printed during the progression of the build * @param runfilesSupplier {@link RunfilesSupplier}s describing the runfiles for the action * @param mnemonic the mnemonic that is reported in the master log. */ public SpawnAction( ActionOwner owner, Iterable<Artifact> tools, Iterable<Artifact> inputs, Iterable<Artifact> outputs, ResourceSet resourceSet, CommandLine argv, boolean isShellCommand, ImmutableMap<String, String> environment, ImmutableSet<String> clientEnvironmentVariables, ImmutableMap<String, String> executionInfo, String progressMessage, RunfilesSupplier runfilesSupplier, String mnemonic, boolean executeUnconditionally, ExtraActionInfoSupplier<?> extraActionInfoSupplier) { super(owner, tools, inputs, runfilesSupplier, outputs); this.resourceSet = resourceSet; this.executionInfo = executionInfo; this.environment = environment; this.clientEnvironmentVariables = clientEnvironmentVariables; this.argv = argv; this.isShellCommand = isShellCommand; this.progressMessage = progressMessage; this.mnemonic = mnemonic; this.executeUnconditionally = executeUnconditionally; this.extraActionInfoSupplier = extraActionInfoSupplier; } @Override @VisibleForTesting public List<String> getArguments() { return ImmutableList.copyOf(argv.arguments()); } @Override public SkylarkList<String> getSkylarkArgv() { return SkylarkList.createImmutable(getArguments()); } /** Returns command argument, argv[0]. */ @VisibleForTesting public String getCommandFilename() { return Iterables.getFirst(argv.arguments(), null); } /** * Returns the (immutable) list of arguments, excluding the command name, * argv[0]. */ @VisibleForTesting public List<String> getRemainingArguments() { return ImmutableList.copyOf(Iterables.skip(argv.arguments(), 1)); } @VisibleForTesting public boolean isShellCommand() { return isShellCommand; } @Override public boolean isVolatile() { return executeUnconditionally; } @Override public boolean executeUnconditionally() { return executeUnconditionally; } /** * Executes the action without handling ExecException errors. * * <p>Called by {@link #execute}. */ protected void internalExecute( ActionExecutionContext actionExecutionContext) throws ExecException, InterruptedException { getContext(actionExecutionContext.getExecutor()) .exec(getSpawn(actionExecutionContext.getClientEnv()), actionExecutionContext); } @Override public void execute(ActionExecutionContext actionExecutionContext) throws ActionExecutionException, InterruptedException { Executor executor = actionExecutionContext.getExecutor(); try { internalExecute(actionExecutionContext); } catch (ExecException e) { String failMessage = progressMessage; if (isShellCommand()) { // The possible reasons it could fail are: shell executable not found, shell // exited non-zero, or shell died from signal. The first is impossible // and the second two aren't very interesting, so in the interests of // keeping the noise-level down, we don't print a reason why, just the // command that failed. // // 0=shell executable, 1=shell command switch, 2=command failMessage = "error executing shell command: " + "'" + truncate(Iterables.get(argv.arguments(), 2), 200) + "'"; } throw e.toActionExecutionException(failMessage, executor.getVerboseFailures(), this); } } /** * Returns s, truncated to no more than maxLen characters, appending an * ellipsis if truncation occurred. */ private static String truncate(String s, int maxLen) { return s.length() > maxLen ? s.substring(0, maxLen - "...".length()) + "..." : s; } /** * Returns a Spawn that is representative of the command that this Action * will execute. This function must not modify any state. * * This method is final, as it is merely a shorthand use of the generic way to obtain a spawn, * which also depends on the client environment. Subclasses that which to override the way to get * a spawn should override {@link #getSpawn(Map)} instead. */ public final Spawn getSpawn() { return getSpawn(null); } /** * Return a spawn that is representative of the command that this Action will execute in the given * client environment. */ public Spawn getSpawn(Map<String, String> clientEnv) { return new ActionSpawn(clientEnv); } @Override protected String computeKey() { Fingerprint f = new Fingerprint(); f.addString(GUID); f.addStrings(argv.arguments()); f.addString(getMnemonic()); // We don't need the toolManifests here, because they are a subset of the inputManifests by // definition and the output of an action shouldn't change whether something is considered a // tool or not. f.addPaths(getRunfilesSupplier().getRunfilesDirs()); ImmutableList<Artifact> runfilesManifests = getRunfilesSupplier().getManifests(); f.addInt(runfilesManifests.size()); for (Artifact runfilesManifest : runfilesManifests) { f.addPath(runfilesManifest.getExecPath()); } f.addStringMap(getEnvironment()); f.addStrings(getClientEnvironmentVariables()); f.addStringMap(getExecutionInfo()); return f.hexDigestAndReset(); } @Override public String describeKey() { StringBuilder message = new StringBuilder(); message.append(getProgressMessage()); message.append('\n'); for (Map.Entry<String, String> entry : getEnvironment().entrySet()) { message.append(" Environment variable: "); message.append(ShellEscaper.escapeString(entry.getKey())); message.append('='); message.append(ShellEscaper.escapeString(entry.getValue())); message.append('\n'); } for (String var : getClientEnvironmentVariables()) { message.append(" Environment variables taken from the client environment: "); message.append(ShellEscaper.escapeString(var)); message.append('\n'); } for (String argument : ShellEscaper.escapeAll(argv.arguments())) { message.append(" Argument: "); message.append(argument); message.append('\n'); } return message.toString(); } @Override public final String getMnemonic() { return mnemonic; } @Override protected String getRawProgressMessage() { if (progressMessage != null) { return progressMessage; } return super.getRawProgressMessage(); } @Override public ExtraActionInfo.Builder getExtraActionInfo() { ExtraActionInfo.Builder builder = super.getExtraActionInfo(); if (extraActionInfoSupplier == null) { SpawnInfo spawnInfo = getExtraActionSpawnInfo(); return builder .setExtension(SpawnInfo.spawnInfo, spawnInfo); } else { extraActionInfoSupplier.extend(builder); return builder; } } /** * Returns information about this spawn action for use by the extra action mechanism. * * <p>Subclasses of SpawnAction may override this in order to provide action-specific behaviour. * This can be necessary, for example, when the action discovers inputs. */ protected SpawnInfo getExtraActionSpawnInfo() { SpawnInfo.Builder info = SpawnInfo.newBuilder(); Spawn spawn = getSpawn(); info.addAllArgument(spawn.getArguments()); for (Map.Entry<String, String> variable : spawn.getEnvironment().entrySet()) { info.addVariable( EnvironmentVariable.newBuilder() .setName(variable.getKey()) .setValue(variable.getValue()) .build()); } for (ActionInput input : spawn.getInputFiles()) { // Explicitly ignore middleman artifacts here. if (!(input instanceof Artifact) || !((Artifact) input).isMiddlemanArtifact()) { info.addInputFile(input.getExecPathString()); } } info.addAllOutputFile(ActionInputHelper.toExecPaths(spawn.getOutputFiles())); return info.build(); } @Override public ImmutableMap<String, String> getEnvironment() { return environment; } @Override public Iterable<String> getClientEnvironmentVariables() { return clientEnvironmentVariables; } /** * Returns the out-of-band execution data for this action. */ @Override public Map<String, String> getExecutionInfo() { return executionInfo; } protected SpawnActionContext getContext(Executor executor) { return executor.getSpawnActionContext(getMnemonic()); } /** * A spawn instance that is tied to a specific SpawnAction. */ public class ActionSpawn extends BaseSpawn { private final List<Artifact> filesets = new ArrayList<>(); private final ImmutableMap<String, String> effectiveEnvironment; /** * Creates an ActionSpawn with the given environment variables. * * <p>Subclasses of ActionSpawn may subclass in order to provide action-specific values for * environment variables or action inputs. */ protected ActionSpawn(Map<String, String> clientEnv) { super(ImmutableList.copyOf(argv.arguments()), ImmutableMap.<String, String>of(), executionInfo, SpawnAction.this.getRunfilesSupplier(), SpawnAction.this, resourceSet); for (Artifact input : getInputs()) { if (input.isFileset()) { filesets.add(input); } } LinkedHashMap<String, String> env = new LinkedHashMap<>(SpawnAction.this.getEnvironment()); if (clientEnv != null) { for (String var : SpawnAction.this.getClientEnvironmentVariables()) { String value = clientEnv.get(var); if (value == null) { env.remove(var); } else { env.put(var, value); } } } effectiveEnvironment = ImmutableMap.copyOf(env); } @Override public ImmutableMap<String, String> getEnvironment() { return effectiveEnvironment; } @Override public ImmutableList<Artifact> getFilesetManifests() { return ImmutableList.copyOf(filesets); } @Override public Iterable<? extends ActionInput> getInputFiles() { // Remove Fileset directories in inputs list. Instead, these are // included as manifests in getEnvironment(). List<Artifact> inputs = Lists.newArrayList(getInputs()); inputs.removeAll(filesets); inputs.removeAll(this.getRunfilesSupplier().getManifests()); return inputs; } } /** * Builder class to construct {@link SpawnAction} instances. */ public static class Builder { private final NestedSetBuilder<Artifact> toolsBuilder = NestedSetBuilder.stableOrder(); private final NestedSetBuilder<Artifact> inputsBuilder = NestedSetBuilder.stableOrder(); private final List<Artifact> outputs = new ArrayList<>(); private final List<RunfilesSupplier> inputRunfilesSuppliers = new ArrayList<>(); private final List<RunfilesSupplier> toolRunfilesSuppliers = new ArrayList<>(); private ResourceSet resourceSet = AbstractAction.DEFAULT_RESOURCE_SET; private ImmutableMap<String, String> environment = ImmutableMap.of(); private ImmutableSet<String> clientEnvironmentVariables = ImmutableSet.of(); private ImmutableMap<String, String> executionInfo = ImmutableMap.of(); private boolean isShellCommand = false; private boolean useDefaultShellEnvironment = false; protected boolean executeUnconditionally; private PathFragment executable; // executableArgs does not include the executable itself. private List<String> executableArgs; private final IterablesChain.Builder<String> argumentsBuilder = IterablesChain.builder(); private CommandLine commandLine; private String progressMessage; private ParamFileInfo paramFileInfo = null; private String mnemonic = "Unknown"; protected ExtraActionInfoSupplier<?> extraActionInfoSupplier = null; private boolean disableSandboxing = false; /** * Creates a SpawnAction builder. */ public Builder() {} /** * Creates a builder that is a copy of another builder. */ public Builder(Builder other) { this.toolsBuilder.addTransitive(other.toolsBuilder.build()); this.inputsBuilder.addTransitive(other.inputsBuilder.build()); this.outputs.addAll(other.outputs); this.inputRunfilesSuppliers.addAll(other.inputRunfilesSuppliers); this.toolRunfilesSuppliers.addAll(other.toolRunfilesSuppliers); this.resourceSet = other.resourceSet; this.environment = other.environment; this.clientEnvironmentVariables = other.clientEnvironmentVariables; this.executionInfo = other.executionInfo; this.isShellCommand = other.isShellCommand; this.useDefaultShellEnvironment = other.useDefaultShellEnvironment; this.executable = other.executable; this.executableArgs = (other.executableArgs != null) ? Lists.newArrayList(other.executableArgs) : null; this.argumentsBuilder.add(other.argumentsBuilder.build()); this.commandLine = other.commandLine; this.progressMessage = other.progressMessage; this.paramFileInfo = other.paramFileInfo; this.mnemonic = other.mnemonic; } /** * Builds the SpawnAction and ParameterFileWriteAction (if param file is used) using the passed- * in action configuration. The first item of the returned array is always the SpawnAction * itself. * * <p>This method makes a copy of all the collections, so it is safe to reuse the builder after * this method returns. * * <p>This is annotated with @CheckReturnValue, which causes a compiler error when you call this * method and ignore its return value. This is because some time ago, calling .build() had the * side-effect of registering it with the RuleContext that was passed in to the constructor. * This logic was removed, but if people don't notice and still rely on the side-effect, things * may break. * * @return the SpawnAction and any actions required by it, with the first item always being the * SpawnAction itself. */ @CheckReturnValue public Action[] build(ActionConstructionContext context) { return build(context.getActionOwner(), context.getAnalysisEnvironment(), context.getConfiguration()); } @VisibleForTesting @CheckReturnValue public Action[] build(ActionOwner owner, AnalysisEnvironment analysisEnvironment, BuildConfiguration configuration) { Iterable<String> arguments = argumentsBuilder.build(); // Check to see if we need to use param file. Artifact paramsFile = ParamFileHelper.getParamsFileMaybe( buildExecutableArgs(configuration.getShellExecutable()), arguments, commandLine, paramFileInfo, configuration, analysisEnvironment, outputs); // If param file is to be used, set up the param file write action as well. ParameterFileWriteAction paramFileWriteAction = null; if (paramsFile != null) { paramFileWriteAction = ParamFileHelper.createParameterFileWriteAction( arguments, commandLine, owner, paramsFile, paramFileInfo); } List<Action> actions = new ArrayList<>(2); actions.add( buildSpawnAction( owner, configuration.getLocalShellEnvironment(), configuration.getVariableShellEnvironment(), configuration.getShellExecutable(), paramsFile)); if (paramFileWriteAction != null) { actions.add(paramFileWriteAction); } return actions.toArray(new Action[actions.size()]); } /** * Builds the SpawnAction using the passed-in action configuration. * * <p>This method makes a copy of all the collections, so it is safe to reuse the builder after * this method returns. * * <p>This method is invoked by {@link SpawnActionTemplate} in the execution phase. It is * important that analysis-phase objects (RuleContext, Configuration, etc.) are not directly * referenced in this function to prevent them from bleeding into the execution phase. * * @param owner the {@link ActionOwner} for the SpawnAction * @param defaultShellEnvironment the default shell environment to use. May be null if not used. * @param defaultShellExecutable the default shell executable path. May be null if not used. * @param paramsFile the parameter file for the SpawnAction. May be null if not used. * @param paramFileWriteAction the action generating the parameter file. May be null if not * used. * @return the SpawnAction and any actions required by it, with the first item always being the * SpawnAction itself. */ SpawnAction buildSpawnAction( ActionOwner owner, @Nullable Map<String, String> defaultShellEnvironment, @Nullable Set<String> variableShellEnvironment, @Nullable PathFragment defaultShellExecutable, @Nullable Artifact paramsFile) { List<String> argv = buildExecutableArgs(defaultShellExecutable); Iterable<String> arguments = argumentsBuilder.build(); CommandLine actualCommandLine; if (paramsFile != null) { inputsBuilder.add(paramsFile); actualCommandLine = ParamFileHelper.createWithParamsFile(argv, paramFileInfo, paramsFile); } else { actualCommandLine = ParamFileHelper.createWithoutParamsFile(argv, arguments, commandLine); } NestedSet<Artifact> tools = toolsBuilder.build(); // Tools are by definition a subset of the inputs, so make sure they're present there, too. NestedSet<Artifact> inputsAndTools = NestedSetBuilder.<Artifact>stableOrder() .addTransitive(inputsBuilder.build()) .addTransitive(tools) .build(); Map<String, String> env; Set<String> clientEnv; if (useDefaultShellEnvironment) { env = Preconditions.checkNotNull(defaultShellEnvironment); clientEnv = Preconditions.checkNotNull(variableShellEnvironment); } else { env = this.environment; clientEnv = this.clientEnvironmentVariables; } if (disableSandboxing) { ImmutableMap.Builder<String, String> builder = ImmutableMap.builder(); builder.putAll(executionInfo); builder.put("nosandbox", "1"); executionInfo = builder.build(); } return createSpawnAction( owner, tools, inputsAndTools, ImmutableList.copyOf(outputs), resourceSet, actualCommandLine, isShellCommand, ImmutableMap.copyOf(env), ImmutableSet.copyOf(clientEnv), ImmutableMap.copyOf(executionInfo), progressMessage, new CompositeRunfilesSupplier( Iterables.concat(this.inputRunfilesSuppliers, this.toolRunfilesSuppliers)), mnemonic); } /** Creates a SpawnAction. */ protected SpawnAction createSpawnAction( ActionOwner owner, NestedSet<Artifact> tools, NestedSet<Artifact> inputsAndTools, ImmutableList<Artifact> outputs, ResourceSet resourceSet, CommandLine actualCommandLine, boolean isShellCommand, ImmutableMap<String, String> env, ImmutableSet<String> clientEnvironmentVariables, ImmutableMap<String, String> executionInfo, String progressMessage, RunfilesSupplier runfilesSupplier, String mnemonic) { return new SpawnAction( owner, tools, inputsAndTools, outputs, resourceSet, actualCommandLine, isShellCommand, env, clientEnvironmentVariables, executionInfo, progressMessage, runfilesSupplier, mnemonic, executeUnconditionally, extraActionInfoSupplier); } private List<String> buildExecutableArgs(@Nullable PathFragment defaultShellExecutable) { if (isShellCommand && executable == null) { Preconditions.checkNotNull(defaultShellExecutable); executable = defaultShellExecutable; } Preconditions.checkNotNull(executable); Preconditions.checkNotNull(executableArgs); return ImmutableList.<String>builder() .add(executable.getPathString()) .addAll(executableArgs) .build(); } /** * Adds an artifact that is necessary for executing the spawn itself (e.g. a compiler), in * contrast to an artifact that is necessary for the spawn to do its work (e.g. source code). * * <p>The artifact is implicitly added to the inputs of the action as well. */ public Builder addTool(Artifact tool) { toolsBuilder.add(tool); return this; } /** * Adds an input to this action. */ public Builder addInput(Artifact artifact) { inputsBuilder.add(artifact); return this; } /** * Adds tools to this action. */ public Builder addTools(Iterable<Artifact> artifacts) { toolsBuilder.addAll(artifacts); return this; } /** * Adds inputs to this action. */ public Builder addInputs(Iterable<Artifact> artifacts) { inputsBuilder.addAll(artifacts); return this; } /** @deprecated Use {@link #addTransitiveInputs} to avoid excessive memory use. */ @Deprecated public Builder addInputs(NestedSet<Artifact> artifacts) { // Do not delete this method, or else addInputs(Iterable) calls with a NestedSet argument // will not be flagged. inputsBuilder.addAll((Iterable<Artifact>) artifacts); return this; } /** * Adds transitive inputs to this action. */ public Builder addTransitiveInputs(NestedSet<Artifact> artifacts) { inputsBuilder.addTransitive(artifacts); return this; } public Builder addRunfilesSupplier(RunfilesSupplier supplier) { inputRunfilesSuppliers.add(supplier); return this; } public Builder addOutput(Artifact artifact) { outputs.add(artifact); return this; } public Builder addOutputs(Iterable<Artifact> artifacts) { Iterables.addAll(outputs, artifacts); return this; } /** * Checks whether the action produces any outputs */ public boolean hasOutputs() { return !outputs.isEmpty(); } public Builder setResources(ResourceSet resourceSet) { this.resourceSet = resourceSet; return this; } /** * Sets the map of environment variables. */ public Builder setEnvironment(Map<String, String> environment) { this.environment = ImmutableMap.copyOf(environment); this.useDefaultShellEnvironment = false; return this; } /** Sets the environment variables to be inherited from the client environment. */ public Builder setClientEnvironmentVariables(Set<String> clientEnvironmentVariables) { this.clientEnvironmentVariables = ImmutableSet.copyOf(clientEnvironmentVariables); this.useDefaultShellEnvironment = false; return this; } /** * Sets the map of execution info. */ public Builder setExecutionInfo(Map<String, String> info) { this.executionInfo = ImmutableMap.copyOf(info); return this; } /** * Sets the environment to the configurations default shell environment, * see {@link BuildConfiguration#getLocalShellEnvironment}. */ public Builder useDefaultShellEnvironment() { this.environment = null; this.clientEnvironmentVariables = null; this.useDefaultShellEnvironment = true; return this; } /** * Makes the action always execute, even if none of its inputs have changed. * * <p>Only use this when absolutely necessary, since this is a performance hit and we'd like to * get rid of this mechanism eventually. You'll eventually be able to declare a Skyframe * dependency on the build ID, which would accomplish the same thing. */ public Builder executeUnconditionally() { // This should really be implemented by declaring a Skyframe dependency on the build ID // instead, however, we can't just do that yet from within actions, so we need to go through // Action.executeUnconditionally() which in turn is called by ActionCacheChecker. this.executeUnconditionally = true; return this; } /** * Sets the executable path; the path is interpreted relative to the * execution root. * * <p>Calling this method overrides any previous values set via calls to * {@link #setExecutable(Artifact)}, {@link #setJavaExecutable}, or * {@link #setShellCommand(String)}. */ public Builder setExecutable(PathFragment executable) { this.executable = executable; this.executableArgs = Lists.newArrayList(); this.isShellCommand = false; return this; } /** * Sets the executable as an artifact. * * <p>Calling this method overrides any previous values set via calls to * {@link #setExecutable(Artifact)}, {@link #setJavaExecutable}, or * {@link #setShellCommand(String)}. */ public Builder setExecutable(Artifact executable) { addTool(executable); return setExecutable(executable.getExecPath()); } /** * Sets the executable as a configured target. Automatically adds the files to run to the tools * and inputs and uses the executable of the target as the executable. * * <p>Calling this method overrides any previous values set via calls to * {@link #setExecutable(Artifact)}, {@link #setJavaExecutable}, or * {@link #setShellCommand(String)}. */ public Builder setExecutable(TransitiveInfoCollection executable) { FilesToRunProvider provider = executable.getProvider(FilesToRunProvider.class); Preconditions.checkArgument(provider != null); return setExecutable(provider); } /** * Sets the executable as a configured target. Automatically adds the files to run to the tools * and inputs and uses the executable of the target as the executable. * * <p>Calling this method overrides any previous values set via calls to {@link #setExecutable}, * {@link #setJavaExecutable}, or {@link #setShellCommand(String)}. */ public Builder setExecutable(FilesToRunProvider executableProvider) { Preconditions.checkArgument(executableProvider.getExecutable() != null, "The target does not have an executable"); setExecutable(executableProvider.getExecutable().getExecPath()); return addTool(executableProvider); } private Builder setJavaExecutable(PathFragment javaExecutable, Artifact deployJar, List<String> jvmArgs, String... launchArgs) { this.executable = javaExecutable; this.executableArgs = Lists.newArrayList(); executableArgs.add("-Xverify:none"); executableArgs.addAll(jvmArgs); Collections.addAll(executableArgs, launchArgs); toolsBuilder.add(deployJar); this.isShellCommand = false; return this; } /** * Sets the executable to be a java class executed from the given deploy * jar. The deploy jar is automatically added to the action inputs. * * <p>Calling this method overrides any previous values set via calls to * {@link #setExecutable}, {@link #setJavaExecutable}, or * {@link #setShellCommand(String)}. */ public Builder setJavaExecutable(PathFragment javaExecutable, Artifact deployJar, String javaMainClass, List<String> jvmArgs) { return setJavaExecutable(javaExecutable, deployJar, jvmArgs, "-cp", deployJar.getExecPathString(), javaMainClass); } /** * Sets the executable to be a jar executed from the given deploy jar. The deploy jar is * automatically added to the action inputs. * * <p>This method is similar to {@link #setJavaExecutable} but it assumes that the Jar artifact * declares a main class. * * <p>Calling this method overrides any previous values set via calls to {@link #setExecutable}, * {@link #setJavaExecutable}, or {@link #setShellCommand(String)}. */ public Builder setJarExecutable(PathFragment javaExecutable, Artifact deployJar, List<String> jvmArgs) { return setJavaExecutable(javaExecutable, deployJar, jvmArgs, "-jar", deployJar.getExecPathString()); } /** * Sets the executable to be the shell and adds the given command as the * command to be executed. * * <p>Note that this will not clear the arguments, so any arguments will * be passed in addition to the command given here. * * <p>Calling this method overrides any previous values set via calls to * {@link #setExecutable(Artifact)}, {@link #setJavaExecutable}, or * {@link #setShellCommand(String)}. */ public Builder setShellCommand(String command) { this.executable = null; // 0=shell command switch, 1=command this.executableArgs = Lists.newArrayList("-c", command); this.isShellCommand = true; return this; } /** * Sets the executable to be the shell and adds the given interned commands as the * commands to be executed. */ public Builder setShellCommand(Iterable<String> command) { this.executable = PathFragment.create(Iterables.getFirst(command, null)); // The first item of the commands is the shell executable that should be used. this.executableArgs = ImmutableList.copyOf(Iterables.skip(command, 1)); this.isShellCommand = true; return this; } /** * Adds an executable and its runfiles, which is necessary for executing the spawn itself (e.g. * a compiler), in contrast to artifacts that are necessary for the spawn to do its work (e.g. * source code). */ public Builder addTool(FilesToRunProvider tool) { addTools(tool.getFilesToRun()); toolRunfilesSuppliers.add(tool.getRunfilesSupplier()); return this; } /** * Appends the arguments to the list of executable arguments. */ public Builder addExecutableArguments(String... arguments) { Preconditions.checkState(executableArgs != null); Collections.addAll(executableArgs, arguments); return this; } /** * Add multiple arguments in the order they are returned by the collection * to the list of executable arguments. */ public Builder addExecutableArguments(Iterable<String> arguments) { Preconditions.checkState(executableArgs != null); Iterables.addAll(executableArgs, arguments); return this; } /** * Appends the argument to the list of command-line arguments. */ public Builder addArgument(String argument) { Preconditions.checkState(commandLine == null); argumentsBuilder.addElement(argument); return this; } /** * Appends the arguments to the list of command-line arguments. */ public Builder addArguments(String... arguments) { Preconditions.checkState(commandLine == null); argumentsBuilder.add(ImmutableList.copyOf(arguments)); return this; } /** * Add multiple arguments in the order they are returned by the collection. */ public Builder addArguments(Iterable<String> arguments) { Preconditions.checkState(commandLine == null); argumentsBuilder.add(CollectionUtils.makeImmutable(arguments)); return this; } /** * Appends the argument both to the inputs and to the list of command-line * arguments. */ public Builder addInputArgument(Artifact argument) { Preconditions.checkState(commandLine == null); addInput(argument); addArgument(argument.getExecPathString()); return this; } /** * Appends the arguments both to the inputs and to the list of command-line * arguments. */ public Builder addInputArguments(Iterable<Artifact> arguments) { for (Artifact argument : arguments) { addInputArgument(argument); } return this; } /** * Appends the argument both to the outputs and to the list of command-line * arguments. */ public Builder addOutputArgument(Artifact argument) { Preconditions.checkState(commandLine == null); outputs.add(argument); argumentsBuilder.addElement(argument.getExecPathString()); return this; } /** * Sets a delegate to compute the command line at a later time. This method * cannot be used in conjunction with the {@link #addArgument} or {@link * #addArguments} methods. * * <p>The main intention of this method is to save memory by allowing * client-controlled sharing between actions and configured targets. * Objects passed to this method MUST be immutable. */ public Builder setCommandLine(CommandLine commandLine) { Preconditions.checkState(argumentsBuilder.isEmpty()); this.commandLine = commandLine; return this; } public Builder setProgressMessage(String progressMessage) { this.progressMessage = progressMessage; return this; } public Builder setMnemonic(String mnemonic) { Preconditions.checkArgument( !mnemonic.isEmpty() && CharMatcher.javaLetterOrDigit().matchesAllOf(mnemonic), "mnemonic must only contain letters and/or digits, and have non-zero length, was: \"%s\"", mnemonic); this.mnemonic = mnemonic; return this; } public <T> Builder setExtraActionInfo( GeneratedExtension<ExtraActionInfo, T> extension, T value) { this.extraActionInfoSupplier = new ExtraActionInfoSupplier<>(extension, value); return this; } /** * Enable use of a parameter file and set the encoding to ISO-8859-1 (latin1). * * <p>In order to use parameter files, at least one output artifact must be specified. */ public Builder useParameterFile(ParameterFileType parameterFileType) { return useParameterFile(parameterFileType, ISO_8859_1, "@"); } /** * Force the use of a parameter file and set the encoding to ISO-8859-1 (latin1). * * <p>In order to use parameter files, at least one output artifact must be specified. */ public Builder alwaysUseParameterFile(ParameterFileType parameterFileType) { return useParameterFile(parameterFileType, ISO_8859_1, "@", /*always=*/ true); } /** * Enable or disable the use of a parameter file, set the encoding to the given value, and * specify the argument prefix to use in passing the parameter file name to the tool. * * <p>The default argument prefix is "@". In order to use parameter files, at least one output * artifact must be specified. */ public Builder useParameterFile( ParameterFileType parameterFileType, Charset charset, String flagPrefix) { return useParameterFile(parameterFileType, charset, flagPrefix, /*always=*/ false); } private Builder useParameterFile( ParameterFileType parameterFileType, Charset charset, String flagPrefix, boolean always) { paramFileInfo = new ParamFileInfo(parameterFileType, charset, flagPrefix, always); return this; } public Builder disableSandboxing() { this.disableSandboxing = true; return this; } } }