blob: a456e066652df22f7a9ef7f24c21e77f07efe55f [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;
Florian Weikertc4975fa2015-12-03 10:27:10 +000017import static org.junit.Assert.assertEquals;
18import static org.junit.Assert.assertTrue;
19import static org.junit.Assert.fail;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000020
21import com.google.common.collect.ImmutableList;
22import com.google.common.collect.ImmutableMap;
23import com.google.common.collect.Sets;
24import com.google.common.eventbus.EventBus;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000025import com.google.devtools.build.lib.actions.ActionExecutionContext;
26import com.google.devtools.build.lib.actions.BaseSpawn;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000027import com.google.devtools.build.lib.actions.ExecException;
28import com.google.devtools.build.lib.actions.Executor.ActionContext;
29import com.google.devtools.build.lib.actions.Spawn;
30import com.google.devtools.build.lib.actions.SpawnActionContext;
31import com.google.devtools.build.lib.actions.util.ActionsTestUtil;
32import com.google.devtools.build.lib.analysis.BlazeDirectories;
33import com.google.devtools.build.lib.events.PrintingEventHandler;
34import com.google.devtools.build.lib.events.Reporter;
Ulf Adamsdba3c832016-12-21 16:50:02 +000035import com.google.devtools.build.lib.exec.ActionContextProvider;
36import com.google.devtools.build.lib.exec.BlazeExecutor;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000037import com.google.devtools.build.lib.exec.ExecutionOptions;
38import com.google.devtools.build.lib.exec.SingleBuildFileCache;
Ulf Adams6447ad92016-11-18 09:55:47 +000039import com.google.devtools.build.lib.integration.util.IntegrationMock;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000040import com.google.devtools.build.lib.rules.apple.AppleConfiguration;
Dmitry Lomov9bf3f6a2016-07-07 13:50:06 +000041import com.google.devtools.build.lib.testutil.TestFileOutErr;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000042import com.google.devtools.build.lib.testutil.TestUtils;
43import com.google.devtools.build.lib.util.BlazeClock;
Dmitry Lomov9d74eba2015-11-26 11:04:01 +000044import com.google.devtools.build.lib.util.OS;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000045import com.google.devtools.build.lib.vfs.FileSystem;
46import com.google.devtools.build.lib.vfs.FileSystemUtils;
47import com.google.devtools.build.lib.vfs.Path;
48import com.google.devtools.build.lib.vfs.util.FileSystems;
49import com.google.devtools.common.options.OptionsParser;
Klaus Aehligd2fcd9d2016-08-26 08:16:25 +000050import java.io.IOException;
51import java.util.Arrays;
Florian Weikertc4975fa2015-12-03 10:27:10 +000052import org.junit.Before;
53import org.junit.Test;
54import org.junit.runner.RunWith;
55import org.junit.runners.JUnit4;
Philipp Wollermann1ee94412015-11-25 13:52:17 +000056
Philipp Wollermann1ee94412015-11-25 13:52:17 +000057/**
58 * Test StandaloneSpawnStrategy.
59 */
Florian Weikertc4975fa2015-12-03 10:27:10 +000060@RunWith(JUnit4.class)
61public class StandaloneSpawnStrategyTest {
Philipp Wollermann1ee94412015-11-25 13:52:17 +000062
63 private Reporter reporter = new Reporter(PrintingEventHandler.ERRORS_AND_WARNINGS_TO_STDERR);
64 private BlazeExecutor executor;
65 private FileSystem fileSystem;
66
67 private Path createTestRoot() throws IOException {
Yun Peng22eb3322016-06-21 14:38:12 +000068 fileSystem = FileSystems.getNativeFileSystem();
Philipp Wollermann1ee94412015-11-25 13:52:17 +000069 Path testRoot = fileSystem.getPath(TestUtils.tmpDir());
70 try {
71 FileSystemUtils.deleteTreesBelow(testRoot);
72 } catch (IOException e) {
73 System.err.println("Failed to remove directory " + testRoot + ": " + e.getMessage());
74 throw e;
75 }
76 return testRoot;
77 }
78
Florian Weikertc4975fa2015-12-03 10:27:10 +000079 @Before
80 public final void setUp() throws Exception {
Philipp Wollermann1ee94412015-11-25 13:52:17 +000081 Path testRoot = createTestRoot();
Ulf Adams015aad92016-07-13 16:49:40 +000082 Path workspaceDir = testRoot.getRelative("workspace-name");
Philipp Wollermann1ee94412015-11-25 13:52:17 +000083 workspaceDir.createDirectory();
84
85 // setup output base & directories
86 Path outputBase = testRoot.getRelative("outputBase");
87 outputBase.createDirectory();
88
Ulf Adams84e10322016-07-13 17:23:33 +000089 BlazeDirectories directories =
90 new BlazeDirectories(outputBase, outputBase, workspaceDir, "mock-product-name");
Ulf Adams6447ad92016-11-18 09:55:47 +000091 // This call implicitly symlinks the integration bin tools into the exec root.
92 IntegrationMock.get().getIntegrationBinTools(directories);
Philipp Wollermann1ee94412015-11-25 13:52:17 +000093 OptionsParser optionsParser = OptionsParser.newOptionsParser(ExecutionOptions.class);
94 optionsParser.parse("--verbose_failures");
95
96 EventBus bus = new EventBus();
97
Ulf Adams84e10322016-07-13 17:23:33 +000098 this.executor =
99 new BlazeExecutor(
100 directories.getExecRoot(),
Ulf Adams84e10322016-07-13 17:23:33 +0000101 reporter,
102 bus,
103 BlazeClock.instance(),
104 optionsParser,
105 /* verboseFailures */ false,
106 /* showSubcommands */ false,
107 ImmutableList.<ActionContext>of(),
108 ImmutableMap.<String, SpawnActionContext>of(
109 "",
110 new StandaloneSpawnStrategy(directories.getExecRoot(), false, "mock-product-name")),
111 ImmutableList.<ActionContextProvider>of());
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000112
113 executor.getExecRoot().createDirectory();
114 }
115
116 private Spawn createSpawn(String... arguments) {
117 return new BaseSpawn.Local(Arrays.asList(arguments), ImmutableMap.<String, String>of(),
118 new ActionsTestUtil.NullAction());
119 }
120
Dmitry Lomov9bf3f6a2016-07-07 13:50:06 +0000121 private TestFileOutErr outErr = new TestFileOutErr();
122
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000123 private String out() {
124 return outErr.outAsLatin1();
125 }
126 private String err() {
127 return outErr.errAsLatin1();
128 }
129
Florian Weikertc4975fa2015-12-03 10:27:10 +0000130 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000131 public void testBinTrueExecutesFine() throws Exception {
Dmitry Lomov9d74eba2015-11-26 11:04:01 +0000132 Spawn spawn = createSpawn(getTrueCommand());
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000133 executor.getSpawnActionContext(spawn.getMnemonic()).exec(spawn, createContext());
134
135 assertThat(out()).isEmpty();
136 assertThat(err()).isEmpty();
137 }
138
139 private void run(Spawn spawn) throws Exception {
140 executor.getSpawnActionContext(spawn.getMnemonic()).exec(spawn, createContext());
141 }
142 private ActionExecutionContext createContext() {
143 Path execRoot = executor.getExecRoot();
144 return new ActionExecutionContext(
145 executor,
146 new SingleBuildFileCache(execRoot.getPathString(), execRoot.getFileSystem()),
147 null,
148 outErr,
Lukacs Berki3bae1de2016-08-29 09:29:37 +0000149 ImmutableMap.<String, String>of(),
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000150 null);
151 }
152
Florian Weikertc4975fa2015-12-03 10:27:10 +0000153 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000154 public void testBinFalseYieldsException() throws Exception {
155 try {
Dmitry Lomov9d74eba2015-11-26 11:04:01 +0000156 run(createSpawn(getFalseCommand()));
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000157 fail();
158 } catch (ExecException e) {
159 assertTrue("got: " + e.getMessage(), e
160 .getMessage().startsWith("false failed: error executing command"));
161 }
162 }
163
Dmitry Lomov9d74eba2015-11-26 11:04:01 +0000164 private static String getFalseCommand() {
165 return OS.getCurrent() == OS.DARWIN ? "/usr/bin/false" : "/bin/false";
166 }
167
168 private static String getTrueCommand() {
169 return OS.getCurrent() == OS.DARWIN ? "/usr/bin/true" : "/bin/true";
170 }
171
Florian Weikertc4975fa2015-12-03 10:27:10 +0000172 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000173 public void testBinEchoPrintsArguments() throws Exception {
174 Spawn spawn = createSpawn("/bin/echo", "Hello,", "world.");
175 run(spawn);
176 assertEquals("Hello, world.\n", out());
177 assertThat(err()).isEmpty();
178 }
179
Florian Weikertc4975fa2015-12-03 10:27:10 +0000180 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000181 public void testCommandRunsInWorkingDir() throws Exception {
182 Spawn spawn = createSpawn("/bin/pwd");
183 run(spawn);
184 assertEquals(executor.getExecRoot() + "\n", out());
185 }
186
Florian Weikertc4975fa2015-12-03 10:27:10 +0000187 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000188 public void testCommandHonorsEnvironment() throws Exception {
189 Spawn spawn = new BaseSpawn.Local(Arrays.asList("/usr/bin/env"),
190 ImmutableMap.of("foo", "bar", "baz", "boo"),
191 new ActionsTestUtil.NullAction());
192 run(spawn);
193 assertEquals(Sets.newHashSet("foo=bar", "baz=boo"), Sets.newHashSet(out().split("\n")));
194 }
195
Florian Weikertc4975fa2015-12-03 10:27:10 +0000196 @Test
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000197 public void testStandardError() throws Exception {
198 Spawn spawn = createSpawn("/bin/sh", "-c", "echo Oops! >&2");
199 run(spawn);
200 assertEquals("Oops!\n", err());
201 assertThat(out()).isEmpty();
202 }
203
Luis Fernando Pino Duquebe102182016-05-23 14:03:55 +0000204 // Test an action with environment variables set indicating an action running on a darwin host
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000205 // system. Such actions should fail given the fact that these tests run on a non darwin
206 // architecture.
Florian Weikertc4975fa2015-12-03 10:27:10 +0000207 @Test
Dmitry Lomov4359ca22015-11-30 16:41:01 +0000208 public void testIOSEnvironmentOnNonDarwin() throws Exception {
209 if (OS.getCurrent() == OS.DARWIN) {
210 return;
211 }
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000212 Spawn spawn = new BaseSpawn.Local(Arrays.asList("/bin/sh", "-c", "echo $SDKROOT"),
213 ImmutableMap.<String, String>of(AppleConfiguration.APPLE_SDK_VERSION_ENV_NAME, "8.4",
214 AppleConfiguration.APPLE_SDK_PLATFORM_ENV_NAME, "iPhoneSimulator"),
215 new ActionsTestUtil.NullAction());
216
217 try {
218 run(spawn);
Dmitry Lomov4359ca22015-11-30 16:41:01 +0000219 fail("action should fail due to being unable to resolve SDKROOT");
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000220 } catch (ExecException e) {
Dmitry Lomov4359ca22015-11-30 16:41:01 +0000221 assertThat(e.getMessage()).contains("Cannot locate iOS SDK on non-darwin operating system");
Philipp Wollermann1ee94412015-11-25 13:52:17 +0000222 }
223 }
224}