blob: 049ef85ae670c074020c7e02a169c8906350bca1 [file] [log] [blame]
Philipp Wollermann1ee94412015-11-25 13:52:17 +00001// Copyright 2015 The Bazel Authors. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14package com.google.devtools.build.lib.standalone;
15
16import static com.google.common.truth.Truth.assertThat;
lberkiaea56b32017-05-30 12:35:33 +020017import static com.google.common.truth.Truth.assertWithMessage;
jcater7472b372019-04-30 07:40:50 -070018import static com.google.devtools.build.lib.testutil.MoreAsserts.assertThrows;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000019
20import com.google.common.collect.ImmutableList;
21import com.google.common.collect.ImmutableMap;
22import com.google.common.collect.Sets;
23import com.google.common.eventbus.EventBus;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000024import com.google.devtools.build.lib.actions.ActionExecutionContext;
ulfjack5b99e502017-07-20 22:22:36 +020025import com.google.devtools.build.lib.actions.ActionInputPrefetcher;
tomlu3d1a1942017-11-29 14:01:21 -080026import com.google.devtools.build.lib.actions.ActionKeyContext;
ulfjackacd291a2017-06-16 15:25:42 +020027import com.google.devtools.build.lib.actions.Artifact;
28import com.google.devtools.build.lib.actions.Artifact.ArtifactExpander;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000029import com.google.devtools.build.lib.actions.ExecException;
Ulf Adamscddaaa62017-03-02 17:32:28 +000030import com.google.devtools.build.lib.actions.ResourceManager;
31import com.google.devtools.build.lib.actions.ResourceSet;
ulfjack4d7f8f72017-11-29 03:37:04 -080032import com.google.devtools.build.lib.actions.SimpleSpawn;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000033import com.google.devtools.build.lib.actions.Spawn;
34import com.google.devtools.build.lib.actions.SpawnActionContext;
ruperts4050a892017-10-07 00:46:20 +020035import com.google.devtools.build.lib.actions.SpawnResult;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000036import com.google.devtools.build.lib.actions.util.ActionsTestUtil;
37import com.google.devtools.build.lib.analysis.BlazeDirectories;
janakr3b63a4e2017-09-14 09:55:40 +020038import com.google.devtools.build.lib.analysis.ServerDirectories;
philwo3bcb9f62017-09-06 12:52:21 +020039import com.google.devtools.build.lib.clock.BlazeClock;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000040import com.google.devtools.build.lib.events.PrintingEventHandler;
41import com.google.devtools.build.lib.events.Reporter;
felly472320c2018-10-29 14:27:00 -070042import com.google.devtools.build.lib.exec.BinTools;
Ulf Adamsdba3c832016-12-21 16:50:02 +000043import com.google.devtools.build.lib.exec.BlazeExecutor;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000044import com.google.devtools.build.lib.exec.ExecutionOptions;
45import com.google.devtools.build.lib.exec.SingleBuildFileCache;
Googler3b9e1522018-03-26 11:03:30 -070046import com.google.devtools.build.lib.exec.SpawnActionContextMaps;
ulfjack19befaf2017-07-24 11:09:40 +020047import com.google.devtools.build.lib.exec.local.LocalEnvProvider;
ulfjackacd291a2017-06-16 15:25:42 +020048import com.google.devtools.build.lib.exec.local.LocalExecutionOptions;
ulfjack19befaf2017-07-24 11:09:40 +020049import com.google.devtools.build.lib.exec.local.LocalSpawnRunner;
Ulf Adams6447ad92016-11-18 09:55:47 +000050import com.google.devtools.build.lib.integration.util.IntegrationMock;
kchodorow85ae1902017-04-22 15:07:22 -040051import com.google.devtools.build.lib.testutil.TestConstants;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000052import com.google.devtools.build.lib.testutil.TestUtils;
Dmitry Lomov9d74eba2015-11-26 11:04:01 +000053import com.google.devtools.build.lib.util.OS;
ulfjackacd291a2017-06-16 15:25:42 +020054import com.google.devtools.build.lib.util.io.FileOutErr;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000055import com.google.devtools.build.lib.vfs.FileSystem;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000056import com.google.devtools.build.lib.vfs.Path;
57import com.google.devtools.build.lib.vfs.util.FileSystems;
ulfjackacd291a2017-06-16 15:25:42 +020058import com.google.devtools.common.options.Options;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000059import com.google.devtools.common.options.OptionsParser;
Klaus Aehligd2fcd9d2016-08-26 08:16:25 +000060import java.io.IOException;
ulfjackacd291a2017-06-16 15:25:42 +020061import java.util.Collection;
ruperts7967f332017-11-21 16:37:13 -080062import java.util.List;
Florian Weikertc4975fa2015-12-03 10:27:10 +000063import org.junit.Before;
64import org.junit.Test;
65import org.junit.runner.RunWith;
66import org.junit.runners.JUnit4;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000067
Philipp Wollermann1ee94412015-11-25 13:52:17 +000068/**
69 * Test StandaloneSpawnStrategy.
70 */
Florian Weikertc4975fa2015-12-03 10:27:10 +000071@RunWith(JUnit4.class)
72public class StandaloneSpawnStrategyTest {
ulfjackacd291a2017-06-16 15:25:42 +020073 private static final ArtifactExpander SIMPLE_ARTIFACT_EXPANDER =
74 new ArtifactExpander() {
75 @Override
76 public void expand(Artifact artifact, Collection<? super Artifact> output) {
77 output.add(artifact);
78 }
79 };
Philipp Wollermann1ee94412015-11-25 13:52:17 +000080
Klaus Aehlig777b30d2017-02-24 16:30:15 +000081 private Reporter reporter =
82 new Reporter(new EventBus(), PrintingEventHandler.ERRORS_AND_WARNINGS_TO_STDERR);
Philipp Wollermann1ee94412015-11-25 13:52:17 +000083 private BlazeExecutor executor;
84 private FileSystem fileSystem;
ulfjackacd291a2017-06-16 15:25:42 +020085 private FileOutErr outErr;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000086
87 private Path createTestRoot() throws IOException {
Yun Peng22eb3322016-06-21 14:38:12 +000088 fileSystem = FileSystems.getNativeFileSystem();
Philipp Wollermann1ee94412015-11-25 13:52:17 +000089 Path testRoot = fileSystem.getPath(TestUtils.tmpDir());
90 try {
jmmv5cc1f652019-03-20 09:34:08 -070091 testRoot.deleteTreesBelow();
Philipp Wollermann1ee94412015-11-25 13:52:17 +000092 } catch (IOException e) {
93 System.err.println("Failed to remove directory " + testRoot + ": " + e.getMessage());
94 throw e;
95 }
96 return testRoot;
97 }
98
Florian Weikertc4975fa2015-12-03 10:27:10 +000099 @Before
100 public final void setUp() throws Exception {
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000101 Path testRoot = createTestRoot();
Ulf Adams015aad92016-07-13 16:49:40 +0000102 Path workspaceDir = testRoot.getRelative("workspace-name");
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000103 workspaceDir.createDirectory();
ulfjackacd291a2017-06-16 15:25:42 +0200104 outErr = new FileOutErr(testRoot.getRelative("stdout"), testRoot.getRelative("stderr"));
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000105
106 // setup output base & directories
107 Path outputBase = testRoot.getRelative("outputBase");
108 outputBase.createDirectory();
109
Ulf Adams84e10322016-07-13 17:23:33 +0000110 BlazeDirectories directories =
janakr3b63a4e2017-09-14 09:55:40 +0200111 new BlazeDirectories(
Klaus Aehligc2499c42018-02-27 05:47:21 -0800112 new ServerDirectories(outputBase, outputBase, outputBase),
113 workspaceDir,
cushon849df362018-05-14 01:51:45 -0700114 /* defaultSystemJavabase= */ null,
Klaus Aehligc2499c42018-02-27 05:47:21 -0800115 "mock-product-name");
Ulf Adams6447ad92016-11-18 09:55:47 +0000116 // This call implicitly symlinks the integration bin tools into the exec root.
felly472320c2018-10-29 14:27:00 -0700117 IntegrationMock.get().getIntegrationBinTools(fileSystem, directories);
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000118 OptionsParser optionsParser = OptionsParser.newOptionsParser(ExecutionOptions.class);
119 optionsParser.parse("--verbose_failures");
ulfjackacd291a2017-06-16 15:25:42 +0200120 LocalExecutionOptions localExecutionOptions = Options.getDefaults(LocalExecutionOptions.class);
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000121
Ulf Adamscddaaa62017-03-02 17:32:28 +0000122 ResourceManager resourceManager = ResourceManager.instanceForTestingOnly();
123 resourceManager.setAvailableResources(
ulfjack4f189862018-10-26 09:59:33 -0700124 ResourceSet.create(/*memoryMb=*/1, /*cpuUsage=*/1, /*localTestCount=*/1));
kchodorow85ae1902017-04-22 15:07:22 -0400125 Path execRoot = directories.getExecRoot(TestConstants.WORKSPACE_NAME);
Ulf Adams84e10322016-07-13 17:23:33 +0000126 this.executor =
127 new BlazeExecutor(
tomluf903eb52017-10-27 12:12:11 -0400128 fileSystem,
kchodorow85ae1902017-04-22 15:07:22 -0400129 execRoot,
Ulf Adams84e10322016-07-13 17:23:33 +0000130 reporter,
Ulf Adams84e10322016-07-13 17:23:33 +0000131 BlazeClock.instance(),
132 optionsParser,
Googler3b9e1522018-03-26 11:03:30 -0700133 SpawnActionContextMaps.createStub(
134 ImmutableList.of(),
philwo207ac6e2019-02-20 12:44:06 -0800135 ImmutableMap.of(
Googler3b9e1522018-03-26 11:03:30 -0700136 "",
philwo207ac6e2019-02-20 12:44:06 -0800137 ImmutableList.of(
138 new StandaloneSpawnStrategy(
Googler3b9e1522018-03-26 11:03:30 -0700139 execRoot,
philwo207ac6e2019-02-20 12:44:06 -0800140 new LocalSpawnRunner(
141 execRoot,
142 localExecutionOptions,
143 resourceManager,
144 LocalEnvProvider.UNMODIFIED,
145 BinTools.forIntegrationTesting(
146 directories, ImmutableList.of())))))),
147 ImmutableList.of());
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000148
ulfjack9f6995a2018-02-09 04:34:48 -0800149 executor.getExecRoot().createDirectoryAndParents();
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000150 }
151
152 private Spawn createSpawn(String... arguments) {
ulfjack4d7f8f72017-11-29 03:37:04 -0800153 return new SimpleSpawn(
Ulf Adamscddaaa62017-03-02 17:32:28 +0000154 new ActionsTestUtil.NullAction(),
ulfjack4d7f8f72017-11-29 03:37:04 -0800155 ImmutableList.copyOf(arguments),
156 /*environment=*/ ImmutableMap.of(),
157 /*executionInfo=*/ ImmutableMap.of(),
158 /*inputs=*/ ImmutableList.of(),
159 /*outputs=*/ ImmutableList.of(),
Ulf Adamscddaaa62017-03-02 17:32:28 +0000160 ResourceSet.ZERO);
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000161 }
162
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000163 private String out() {
164 return outErr.outAsLatin1();
165 }
166 private String err() {
167 return outErr.errAsLatin1();
168 }
169
Florian Weikertc4975fa2015-12-03 10:27:10 +0000170 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000171 public void testBinTrueExecutesFine() throws Exception {
Dmitry Lomov9d74eba2015-11-26 11:04:01 +0000172 Spawn spawn = createSpawn(getTrueCommand());
ulfjackff559b42018-05-22 09:14:10 -0700173 executor.getContext(SpawnActionContext.class).exec(spawn, createContext());
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000174
175 assertThat(out()).isEmpty();
176 assertThat(err()).isEmpty();
177 }
178
ruperts7967f332017-11-21 16:37:13 -0800179 private List<SpawnResult> run(Spawn spawn) throws Exception {
ulfjackff559b42018-05-22 09:14:10 -0700180 return executor.getContext(SpawnActionContext.class).exec(spawn, createContext());
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000181 }
ruperts4050a892017-10-07 00:46:20 +0200182
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000183 private ActionExecutionContext createContext() {
184 Path execRoot = executor.getExecRoot();
185 return new ActionExecutionContext(
186 executor,
187 new SingleBuildFileCache(execRoot.getPathString(), execRoot.getFileSystem()),
ulfjack7599a4d2017-07-21 13:58:33 +0200188 ActionInputPrefetcher.NONE,
tomlu3d1a1942017-11-29 14:01:21 -0800189 new ActionKeyContext(),
ulfjack256be112019-04-05 05:42:01 -0700190 /*metadataHandler=*/ null,
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000191 outErr,
ulfjack256be112019-04-05 05:42:01 -0700192 reporter,
193 /*clientEnv=*/ ImmutableMap.of(),
194 /*topLevelFilesets=*/ ImmutableMap.of(),
shahanb1dd4e32018-05-09 08:23:31 -0700195 SIMPLE_ARTIFACT_EXPANDER,
felly2b3befd2018-08-10 10:37:56 -0700196 /*actionFileSystem=*/ null,
197 /*skyframeDepsResult=*/ null);
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000198 }
199
Florian Weikertc4975fa2015-12-03 10:27:10 +0000200 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000201 public void testBinFalseYieldsException() throws Exception {
jcater7472b372019-04-30 07:40:50 -0700202 ExecException e = assertThrows(ExecException.class, () -> run(createSpawn(getFalseCommand())));
203 assertWithMessage("got: " + e.getMessage())
204 .that(e.getMessage().startsWith("false failed: error executing command"))
205 .isTrue();
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000206 }
207
Dmitry Lomov9d74eba2015-11-26 11:04:01 +0000208 private static String getFalseCommand() {
209 return OS.getCurrent() == OS.DARWIN ? "/usr/bin/false" : "/bin/false";
210 }
211
212 private static String getTrueCommand() {
213 return OS.getCurrent() == OS.DARWIN ? "/usr/bin/true" : "/bin/true";
214 }
215
Florian Weikertc4975fa2015-12-03 10:27:10 +0000216 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000217 public void testBinEchoPrintsArguments() throws Exception {
218 Spawn spawn = createSpawn("/bin/echo", "Hello,", "world.");
219 run(spawn);
lberkiaea56b32017-05-30 12:35:33 +0200220 assertThat(out()).isEqualTo("Hello, world.\n");
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000221 assertThat(err()).isEmpty();
222 }
223
Florian Weikertc4975fa2015-12-03 10:27:10 +0000224 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000225 public void testCommandRunsInWorkingDir() throws Exception {
226 Spawn spawn = createSpawn("/bin/pwd");
227 run(spawn);
lberkiaea56b32017-05-30 12:35:33 +0200228 assertThat(out()).isEqualTo(executor.getExecRoot() + "\n");
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000229 }
230
Florian Weikertc4975fa2015-12-03 10:27:10 +0000231 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000232 public void testCommandHonorsEnvironment() throws Exception {
ulfjack61d39662017-09-29 09:03:44 -0400233 if (OS.getCurrent() == OS.DARWIN) {
234 // // TODO(#)3795: For some reason, we get __CF_USER_TEXT_ENCODING into the env in some
235 // configurations of MacOS machines. I have been unable to reproduce on my Mac, or to track
236 // down where that env var is coming from.
237 return;
238 }
ulfjack4d7f8f72017-11-29 03:37:04 -0800239 Spawn spawn = new SimpleSpawn(
Ulf Adamscddaaa62017-03-02 17:32:28 +0000240 new ActionsTestUtil.NullAction(),
ulfjack4d7f8f72017-11-29 03:37:04 -0800241 ImmutableList.of("/usr/bin/env"),
242 /*environment=*/ ImmutableMap.of("foo", "bar", "baz", "boo"),
243 /*executionInfo=*/ ImmutableMap.of(),
244 /*inputs=*/ ImmutableList.of(),
245 /*outputs=*/ ImmutableList.of(),
Ulf Adamscddaaa62017-03-02 17:32:28 +0000246 ResourceSet.ZERO);
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000247 run(spawn);
lberkiaea56b32017-05-30 12:35:33 +0200248 assertThat(Sets.newHashSet(out().split("\n"))).isEqualTo(Sets.newHashSet("foo=bar", "baz=boo"));
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000249 }
250
Florian Weikertc4975fa2015-12-03 10:27:10 +0000251 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000252 public void testStandardError() throws Exception {
253 Spawn spawn = createSpawn("/bin/sh", "-c", "echo Oops! >&2");
254 run(spawn);
lberkiaea56b32017-05-30 12:35:33 +0200255 assertThat(err()).isEqualTo("Oops!\n");
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000256 assertThat(out()).isEmpty();
257 }
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000258}