Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 1 | // 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. |
| 14 | package com.google.devtools.build.lib.standalone; |
| 15 | |
| 16 | import static com.google.common.truth.Truth.assertThat; |
Florian Weikert | c4975fa | 2015-12-03 10:27:10 +0000 | [diff] [blame] | 17 | import static org.junit.Assert.assertEquals; |
| 18 | import static org.junit.Assert.assertTrue; |
| 19 | import static org.junit.Assert.fail; |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 20 | |
| 21 | import com.google.common.collect.ImmutableList; |
| 22 | import com.google.common.collect.ImmutableMap; |
| 23 | import com.google.common.collect.Sets; |
| 24 | import com.google.common.eventbus.EventBus; |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 25 | import com.google.devtools.build.lib.actions.ActionExecutionContext; |
| 26 | import com.google.devtools.build.lib.actions.BaseSpawn; |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 27 | import com.google.devtools.build.lib.actions.ExecException; |
| 28 | import com.google.devtools.build.lib.actions.Executor.ActionContext; |
| 29 | import com.google.devtools.build.lib.actions.Spawn; |
| 30 | import com.google.devtools.build.lib.actions.SpawnActionContext; |
| 31 | import com.google.devtools.build.lib.actions.util.ActionsTestUtil; |
| 32 | import com.google.devtools.build.lib.analysis.BlazeDirectories; |
| 33 | import com.google.devtools.build.lib.events.PrintingEventHandler; |
| 34 | import com.google.devtools.build.lib.events.Reporter; |
Ulf Adams | dba3c83 | 2016-12-21 16:50:02 +0000 | [diff] [blame^] | 35 | import com.google.devtools.build.lib.exec.ActionContextProvider; |
| 36 | import com.google.devtools.build.lib.exec.BlazeExecutor; |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 37 | import com.google.devtools.build.lib.exec.ExecutionOptions; |
| 38 | import com.google.devtools.build.lib.exec.SingleBuildFileCache; |
Ulf Adams | 6447ad9 | 2016-11-18 09:55:47 +0000 | [diff] [blame] | 39 | import com.google.devtools.build.lib.integration.util.IntegrationMock; |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 40 | import com.google.devtools.build.lib.rules.apple.AppleConfiguration; |
Dmitry Lomov | 9bf3f6a | 2016-07-07 13:50:06 +0000 | [diff] [blame] | 41 | import com.google.devtools.build.lib.testutil.TestFileOutErr; |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 42 | import com.google.devtools.build.lib.testutil.TestUtils; |
| 43 | import com.google.devtools.build.lib.util.BlazeClock; |
Dmitry Lomov | 9d74eba | 2015-11-26 11:04:01 +0000 | [diff] [blame] | 44 | import com.google.devtools.build.lib.util.OS; |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 45 | import com.google.devtools.build.lib.vfs.FileSystem; |
| 46 | import com.google.devtools.build.lib.vfs.FileSystemUtils; |
| 47 | import com.google.devtools.build.lib.vfs.Path; |
| 48 | import com.google.devtools.build.lib.vfs.util.FileSystems; |
| 49 | import com.google.devtools.common.options.OptionsParser; |
Klaus Aehlig | d2fcd9d | 2016-08-26 08:16:25 +0000 | [diff] [blame] | 50 | import java.io.IOException; |
| 51 | import java.util.Arrays; |
Florian Weikert | c4975fa | 2015-12-03 10:27:10 +0000 | [diff] [blame] | 52 | import org.junit.Before; |
| 53 | import org.junit.Test; |
| 54 | import org.junit.runner.RunWith; |
| 55 | import org.junit.runners.JUnit4; |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 56 | |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 57 | /** |
| 58 | * Test StandaloneSpawnStrategy. |
| 59 | */ |
Florian Weikert | c4975fa | 2015-12-03 10:27:10 +0000 | [diff] [blame] | 60 | @RunWith(JUnit4.class) |
| 61 | public class StandaloneSpawnStrategyTest { |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 62 | |
| 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 Peng | 22eb332 | 2016-06-21 14:38:12 +0000 | [diff] [blame] | 68 | fileSystem = FileSystems.getNativeFileSystem(); |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 69 | 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 Weikert | c4975fa | 2015-12-03 10:27:10 +0000 | [diff] [blame] | 79 | @Before |
| 80 | public final void setUp() throws Exception { |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 81 | Path testRoot = createTestRoot(); |
Ulf Adams | 015aad9 | 2016-07-13 16:49:40 +0000 | [diff] [blame] | 82 | Path workspaceDir = testRoot.getRelative("workspace-name"); |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 83 | workspaceDir.createDirectory(); |
| 84 | |
| 85 | // setup output base & directories |
| 86 | Path outputBase = testRoot.getRelative("outputBase"); |
| 87 | outputBase.createDirectory(); |
| 88 | |
Ulf Adams | 84e1032 | 2016-07-13 17:23:33 +0000 | [diff] [blame] | 89 | BlazeDirectories directories = |
| 90 | new BlazeDirectories(outputBase, outputBase, workspaceDir, "mock-product-name"); |
Ulf Adams | 6447ad9 | 2016-11-18 09:55:47 +0000 | [diff] [blame] | 91 | // This call implicitly symlinks the integration bin tools into the exec root. |
| 92 | IntegrationMock.get().getIntegrationBinTools(directories); |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 93 | OptionsParser optionsParser = OptionsParser.newOptionsParser(ExecutionOptions.class); |
| 94 | optionsParser.parse("--verbose_failures"); |
| 95 | |
| 96 | EventBus bus = new EventBus(); |
| 97 | |
Ulf Adams | 84e1032 | 2016-07-13 17:23:33 +0000 | [diff] [blame] | 98 | this.executor = |
| 99 | new BlazeExecutor( |
| 100 | directories.getExecRoot(), |
Ulf Adams | 84e1032 | 2016-07-13 17:23:33 +0000 | [diff] [blame] | 101 | 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 Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 112 | |
| 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 Lomov | 9bf3f6a | 2016-07-07 13:50:06 +0000 | [diff] [blame] | 121 | private TestFileOutErr outErr = new TestFileOutErr(); |
| 122 | |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 123 | private String out() { |
| 124 | return outErr.outAsLatin1(); |
| 125 | } |
| 126 | private String err() { |
| 127 | return outErr.errAsLatin1(); |
| 128 | } |
| 129 | |
Florian Weikert | c4975fa | 2015-12-03 10:27:10 +0000 | [diff] [blame] | 130 | @Test |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 131 | public void testBinTrueExecutesFine() throws Exception { |
Dmitry Lomov | 9d74eba | 2015-11-26 11:04:01 +0000 | [diff] [blame] | 132 | Spawn spawn = createSpawn(getTrueCommand()); |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 133 | 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 Berki | 3bae1de | 2016-08-29 09:29:37 +0000 | [diff] [blame] | 149 | ImmutableMap.<String, String>of(), |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 150 | null); |
| 151 | } |
| 152 | |
Florian Weikert | c4975fa | 2015-12-03 10:27:10 +0000 | [diff] [blame] | 153 | @Test |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 154 | public void testBinFalseYieldsException() throws Exception { |
| 155 | try { |
Dmitry Lomov | 9d74eba | 2015-11-26 11:04:01 +0000 | [diff] [blame] | 156 | run(createSpawn(getFalseCommand())); |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 157 | fail(); |
| 158 | } catch (ExecException e) { |
| 159 | assertTrue("got: " + e.getMessage(), e |
| 160 | .getMessage().startsWith("false failed: error executing command")); |
| 161 | } |
| 162 | } |
| 163 | |
Dmitry Lomov | 9d74eba | 2015-11-26 11:04:01 +0000 | [diff] [blame] | 164 | 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 Weikert | c4975fa | 2015-12-03 10:27:10 +0000 | [diff] [blame] | 172 | @Test |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 173 | 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 Weikert | c4975fa | 2015-12-03 10:27:10 +0000 | [diff] [blame] | 180 | @Test |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 181 | public void testCommandRunsInWorkingDir() throws Exception { |
| 182 | Spawn spawn = createSpawn("/bin/pwd"); |
| 183 | run(spawn); |
| 184 | assertEquals(executor.getExecRoot() + "\n", out()); |
| 185 | } |
| 186 | |
Florian Weikert | c4975fa | 2015-12-03 10:27:10 +0000 | [diff] [blame] | 187 | @Test |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 188 | 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 Weikert | c4975fa | 2015-12-03 10:27:10 +0000 | [diff] [blame] | 196 | @Test |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 197 | 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 Duque | be10218 | 2016-05-23 14:03:55 +0000 | [diff] [blame] | 204 | // Test an action with environment variables set indicating an action running on a darwin host |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 205 | // system. Such actions should fail given the fact that these tests run on a non darwin |
| 206 | // architecture. |
Florian Weikert | c4975fa | 2015-12-03 10:27:10 +0000 | [diff] [blame] | 207 | @Test |
Dmitry Lomov | 4359ca2 | 2015-11-30 16:41:01 +0000 | [diff] [blame] | 208 | public void testIOSEnvironmentOnNonDarwin() throws Exception { |
| 209 | if (OS.getCurrent() == OS.DARWIN) { |
| 210 | return; |
| 211 | } |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 212 | 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 Lomov | 4359ca2 | 2015-11-30 16:41:01 +0000 | [diff] [blame] | 219 | fail("action should fail due to being unable to resolve SDKROOT"); |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 220 | } catch (ExecException e) { |
Dmitry Lomov | 4359ca2 | 2015-11-30 16:41:01 +0000 | [diff] [blame] | 221 | assertThat(e.getMessage()).contains("Cannot locate iOS SDK on non-darwin operating system"); |
Philipp Wollermann | 1ee9441 | 2015-11-25 13:52:17 +0000 | [diff] [blame] | 222 | } |
| 223 | } |
| 224 | } |