blob: fe46c07d99276fecdb7df11e3dba80cf14700f9b [file] [log] [blame]
// Copyright 2015 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.shell;
import static com.google.common.truth.Truth.assertThat;
import static com.google.devtools.build.lib.shell.ShellUtils.prettyPrintArgv;
import static com.google.devtools.build.lib.shell.ShellUtils.tokenize;
import static org.junit.Assert.fail;
import com.google.common.collect.Lists;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
/** Tests for ShellUtils that call out to Bash. */
@RunWith(JUnit4.class)
public class ShellUtilsWithBashTest {
private void assertTokenizeIsDualToPrettyPrint(String... args) throws Exception {
List<String> in = Arrays.asList(args);
String shellCommand = prettyPrintArgv(in);
// Assert that pretty-print is correct, i.e. dual to the actual /bin/sh
// tokenization. This test assumes no newlines in the input:
String[] execArgs = {
"/bin/sh",
"-c",
"for i in " + shellCommand + "; do echo \"$i\"; done" // tokenize, one word per line
};
String stdout = null;
try {
stdout = new String(new Command(execArgs).execute().getStdout());
} catch (Exception e) {
fail("/bin/sh failed:\n" + in + "\n" + shellCommand + "\n" + e.getMessage());
}
// We can't use stdout.split("\n") here,
// because String.split() ignores trailing empty strings.
ArrayList<String> words = Lists.newArrayList();
int index;
while ((index = stdout.indexOf('\n')) >= 0) {
words.add(stdout.substring(0, index));
stdout = stdout.substring(index + 1);
}
assertThat(words).isEqualTo(in);
// Assert that tokenize is dual to pretty-print:
List<String> out = new ArrayList<>();
try {
tokenize(out, shellCommand);
} finally {
if (out.isEmpty()) { // i.e. an exception
System.err.println(in);
}
}
assertThat(out).isEqualTo(in);
}
@Test
public void testTokenizeIsDualToPrettyPrint() throws Exception {
// tokenize() is the inverse of prettyPrintArgv(). (However, the reverse
// is not true, since there are many ways to escape the same string,
// e.g. "foo" and 'foo'.)
assertTokenizeIsDualToPrettyPrint("foo");
assertTokenizeIsDualToPrettyPrint("foo bar");
assertTokenizeIsDualToPrettyPrint("foo bar", "wiz");
assertTokenizeIsDualToPrettyPrint("'foo'");
assertTokenizeIsDualToPrettyPrint("\\'foo\\'");
assertTokenizeIsDualToPrettyPrint("${filename%.c}.o");
assertTokenizeIsDualToPrettyPrint("<html!>");
assertTokenizeIsDualToPrettyPrint("");
assertTokenizeIsDualToPrettyPrint("!@#$%^&*()");
assertTokenizeIsDualToPrettyPrint("x'y\" z");
}
}