// 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.actions; import com.google.common.annotations.VisibleForTesting; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; import com.google.common.collect.Iterables; import com.google.devtools.build.lib.actions.extra.EnvironmentVariable; import com.google.devtools.build.lib.actions.extra.SpawnInfo; import com.google.devtools.build.lib.util.CommandDescriptionForm; import com.google.devtools.build.lib.util.CommandFailureUtils; import com.google.devtools.build.lib.vfs.Path; import com.google.devtools.build.lib.vfs.PathFragment; import java.util.Collection; import java.util.List; import java.util.Map; import java.util.Set; import javax.annotation.concurrent.Immutable; /** * Base implementation of a Spawn. */ @Immutable public class BaseSpawn implements Spawn { private final ImmutableList arguments; private final ImmutableMap environment; private final ImmutableMap executionInfo; private final ImmutableMap runfilesManifests; private final RunfilesSupplier runfilesSupplier; private final ActionMetadata action; private final ResourceSet localResources; // TODO(bazel-team): When we migrate ActionSpawn to use this constructor decide on and enforce // policy on runfilesManifests and runfilesSupplier being non-empty (ie: are overlapping mappings // allowed?). @VisibleForTesting BaseSpawn(List arguments, Map environment, Map executionInfo, Map runfilesManifests, RunfilesSupplier runfilesSupplier, ActionMetadata action, ResourceSet localResources) { this.arguments = ImmutableList.copyOf(arguments); this.environment = ImmutableMap.copyOf(environment); this.executionInfo = ImmutableMap.copyOf(executionInfo); this.runfilesManifests = ImmutableMap.copyOf(runfilesManifests); this.runfilesSupplier = runfilesSupplier; this.action = action; this.localResources = localResources; } /** * Returns a new Spawn. The caller must not modify the parameters after the call; neither will * this method. */ public BaseSpawn(List arguments, Map environment, Map executionInfo, RunfilesSupplier runfilesSupplier, ActionMetadata action, ResourceSet localResources) { this(arguments, environment, executionInfo, ImmutableMap.of(), runfilesSupplier, action, localResources); } /** * Returns a new Spawn. The caller must not modify the parameters after the call; neither will * this method. */ public BaseSpawn(List arguments, Map environment, Map executionInfo, Map runfilesManifests, ActionMetadata action, ResourceSet localResources) { this(arguments, environment, executionInfo, runfilesManifests, EmptyRunfilesSupplier.INSTANCE, action, localResources); } /** * Returns a new Spawn. */ public BaseSpawn(List arguments, Map environment, Map executionInfo, ActionMetadata action, ResourceSet localResources) { this(arguments, environment, executionInfo, ImmutableMap.of(), action, localResources); } public static PathFragment runfilesForFragment(PathFragment pathFragment) { return pathFragment.getParentDirectory().getChild(pathFragment.getBaseName() + ".runfiles"); } @Override public boolean isRemotable() { return !executionInfo.containsKey("local"); } @Override public final ImmutableMap getExecutionInfo() { return executionInfo; } @Override public String asShellCommand(Path workingDir) { return asShellCommand(getArguments(), workingDir, getEnvironment()); } @Override public ImmutableMap getRunfilesManifests() { return runfilesManifests; } @Override public RunfilesSupplier getRunfilesSupplier() { return runfilesSupplier; } @Override public ImmutableList getFilesetManifests() { return ImmutableList.of(); } @Override public SpawnInfo getExtraActionInfo() { SpawnInfo.Builder info = SpawnInfo.newBuilder(); info.addAllArgument(getArguments()); for (Map.Entry variable : getEnvironment().entrySet()) { info.addVariable(EnvironmentVariable.newBuilder() .setName(variable.getKey()) .setValue(variable.getValue()).build()); } for (ActionInput input : getInputFiles()) { // Explicitly ignore middleman artifacts here. if (!(input instanceof Artifact) || !((Artifact) input).isMiddlemanArtifact()) { info.addInputFile(input.getExecPathString()); } } info.addAllOutputFile(ActionInputHelper.toExecPaths(getOutputFiles())); return info.build(); } @Override public ImmutableList getArguments() { // TODO(bazel-team): this method should be final, as the correct value of the args can be // injected in the ctor. return arguments; } @Override public ImmutableMap getEnvironment() { PathFragment runfilesRoot = getRunfilesRoot(); if (runfilesRoot == null) { return environment; } else { ImmutableMap.Builder env = ImmutableMap.builder(); env.putAll(environment); // TODO(bazel-team): Unify these into a single env variable. String runfilesRootString = runfilesRoot.getPathString(); env.put("JAVA_RUNFILES", runfilesRootString); env.put("PYTHON_RUNFILES", runfilesRootString); return env.build(); } } /** @return the runfiles directory if there is only one, otherwise null */ private PathFragment getRunfilesRoot() { Set runfilesSupplierRoots = runfilesSupplier.getRunfilesDirs(); if (runfilesSupplierRoots.size() == 1 && runfilesManifests.isEmpty()) { return Iterables.getOnlyElement(runfilesSupplierRoots); } else if (runfilesManifests.size() == 1 && runfilesSupplierRoots.isEmpty()) { return Iterables.getOnlyElement(runfilesManifests.keySet()); } else { return null; } } @Override public Iterable getInputFiles() { return action.getInputs(); } @Override public Collection getOutputFiles() { return action.getOutputs(); } @Override public ActionMetadata getResourceOwner() { return action; } @Override public ResourceSet getLocalResources() { return localResources; } @Override public ActionOwner getOwner() { return action.getOwner(); } @Override public String getMnemonic() { return action.getMnemonic(); } /** * Convert a working dir + environment map + arg list into a Bourne shell * command. */ public static String asShellCommand(Collection arguments, Path workingDirectory, Map environment) { // We print this command out in such a way that it can safely be // copied+pasted as a Bourne shell command. This is extremely valuable for // debugging. return CommandFailureUtils.describeCommand(CommandDescriptionForm.COMPLETE, arguments, environment, workingDirectory.getPathString()); } /** * A local spawn requiring zero resources. */ public static class Local extends BaseSpawn { public Local(List arguments, Map environment, ActionMetadata action) { super(arguments, environment, ImmutableMap.of("local", ""), action, ResourceSet.ZERO); } } }