jdk-24/test/jdk/tools/launcher/SourceMode.java

494 lines
19 KiB
Java

/*
* Copyright (c) 2017, 2021, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/**
* @test
* @bug 8192920 8204588 8210275 8286571
* @summary Test source mode
* @modules jdk.compiler jdk.jlink
* @run main SourceMode
*/
import java.io.IOException;
import java.io.PrintStream;
import java.io.PrintWriter;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.attribute.PosixFilePermission;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.spi.ToolProvider;
public class SourceMode extends TestHelper {
public static void main(String... args) throws Exception {
new SourceMode().run(args);
}
// To reduce the chance of creating shebang lines that are too long,
// use a shorter path for a java command if the standard path is too long.
private final Path shebangJavaCmd;
// Whether or not to automatically skip the shebang tests
private final boolean skipShebangTest;
private final PrintStream log;
private static final String thisVersion = System.getProperty("java.specification.version");
SourceMode() throws Exception {
log = System.err;
if (isWindows) {
// Skip shebang tests on Windows, because that requires Cygwin.
skipShebangTest = true;
shebangJavaCmd = null;
} else {
// Try to ensure the path to the Java launcher is reasonably short,
// to work around the mostly undocumented limit of 120 characters
// for a shebang line.
// The value of 120 is the typical kernel compile-time buffer limit.
// The following limit of 80 allows room for arguments to be placed
// after the path to the launcher on the shebang line.
Path cmd = Paths.get(javaCmd);
if (cmd.toString().length() < 80) {
shebangJavaCmd = cmd;
} else {
// Create a small image in the current directory, such that
// the path for the launcher is just "tmpJDK/bin/java".
Path tmpJDK = Paths.get("tmpJDK");
ToolProvider jlink = ToolProvider.findFirst("jlink")
.orElseThrow(() -> new Exception("cannot find jlink"));
jlink.run(System.out, System.err,
"--add-modules", "jdk.compiler,jdk.zipfs", "--output", tmpJDK.toString());
shebangJavaCmd = tmpJDK.resolve("bin").resolve("java");
}
log.println("Using java command: " + shebangJavaCmd);
skipShebangTest = false;
}
}
// java Simple.java 1 2 3
@Test
void testSimpleJava() throws IOException {
starting("testSimpleJava");
Path file = getSimpleFile("Simple.java", false);
TestResult tr = doExec(javaCmd, file.toString(), "1", "2", "3");
if (!tr.isOK())
error(tr, "Bad exit code: " + tr.exitValue);
if (!tr.contains("[1, 2, 3]"))
error(tr, "Expected output not found");
show(tr);
}
// java --source N --enable-preview Simple.java hello
// on minimal jdk image containing jdk.compiler
@Test
void test8286571() throws IOException {
starting("test8286571");
var pw = new PrintWriter(System.out);
int rc = ToolProvider.findFirst("jlink").orElseThrow().run(
pw, pw,
"--add-modules",
"jdk.compiler",
"--output",
"comp_only");
if (rc != 0)
throw new AssertionError("Jlink failed: rc = " + rc);
Path file = getSimpleFile("Simple.java", false);
TestResult tr = doExec(
Path.of("comp_only", "bin", isWindows ? "java.exe" : "java").toString(),
"--source", thisVersion,
"--enable-preview",
file.toString(), "hello");
if (!tr.isOK())
error(tr, "Bad exit code: " + tr.exitValue);
if (!tr.contains("[hello]"))
error(tr, "Expected output not found");
show(tr);
}
// java --source N simple 1 2 3
@Test
void testSimple() throws IOException {
starting("testSimple");
Path file = getSimpleFile("simple", false);
TestResult tr = doExec(javaCmd, "--source", thisVersion, file.toString(), "1", "2", "3");
if (!tr.isOK())
error(tr, "Bad exit code: " + tr.exitValue);
if (!tr.contains("[1, 2, 3]"))
error(tr, "Expected output not found");
show(tr);
}
// execSimple 1 2 3
@Test
void testExecSimple() throws IOException {
starting("testExecSimple");
if (skipShebangTest) {
log.println("SKIPPED");
return;
}
Path file = setExecutable(getSimpleFile("execSimple", true));
TestResult tr = doExec(file.toAbsolutePath().toString(), "1", "2", "3");
if (!tr.isOK())
error(tr, "Bad exit code: " + tr.exitValue);
if (!tr.contains("[1, 2, 3]"))
error(tr, "Expected output not found");
show(tr);
}
// java @simpleJava.at (contains Simple.java 1 2 3)
@Test
void testSimpleJavaAtFile() throws IOException {
starting("testSimpleJavaAtFile");
Path file = getSimpleFile("Simple.java", false);
Path atFile = Paths.get("simpleJava.at");
createFile(atFile, List.of(file + " 1 2 3"));
TestResult tr = doExec(javaCmd, "@" + atFile);
if (!tr.isOK())
error(tr, "Bad exit code: " + tr.exitValue);
if (!tr.contains("[1, 2, 3]"))
error(tr, "Expected output not found");
show(tr);
}
// java @simple.at (contains --source N simple 1 2 3)
@Test
void testSimpleAtFile() throws IOException {
starting("testSimpleAtFile");
Path file = getSimpleFile("simple", false);
Path atFile = Paths.get("simple.at");
createFile(atFile, List.of("--source " + thisVersion + " " + file + " 1 2 3"));
TestResult tr = doExec(javaCmd, "@" + atFile);
if (!tr.isOK())
error(tr, "Bad exit code: " + tr.exitValue);
if (!tr.contains("[1, 2, 3]"))
error(tr, "Expected output not found");
show(tr);
}
// java -cp classes Main.java 1 2 3
@Test
void testClasspath() throws IOException {
starting("testClasspath");
Path base = Files.createDirectories(Paths.get("testClasspath"));
Path otherJava = base.resolve("Other.java");
createFile(otherJava, List.of(
"public class Other {",
" public static String join(String[] args) {",
" return String.join(\"-\", args);",
" }",
"}"
));
Path classes = Files.createDirectories(base.resolve("classes"));
Path mainJava = base.resolve("Main.java");
createFile(mainJava, List.of(
"class Main {",
" public static void main(String[] args) {",
" System.out.println(Other.join(args));",
" }}"
));
compile("-d", classes.toString(), otherJava.toString());
TestResult tr = doExec(javaCmd, "-cp", classes.toString(),
mainJava.toString(), "1", "2", "3");
if (!tr.isOK())
error(tr, "Bad exit code: " + tr.exitValue);
if (!tr.contains("1-2-3"))
error(tr, "Expected output not found");
show(tr);
}
// java --add-exports=... Export.java --help
@Test
void testAddExports() throws IOException {
if (!isEnglishLocale()) {
return;
}
starting("testAddExports");
Path exportJava = Paths.get("Export.java");
createFile(exportJava, List.of(
"public class Export {",
" public static void main(String[] args) {",
" new com.sun.tools.javac.main.Main(\"demo\").compile(args);",
" }",
"}"
));
// verify access fails without --add-exports
TestResult tr1 = doExec(javaCmd, exportJava.toString(), "--help");
if (tr1.isOK())
error(tr1, "Compilation succeeded unexpectedly");
show(tr1);
// verify access succeeds with --add-exports
TestResult tr2 = doExec(javaCmd,
"--add-exports", "jdk.compiler/com.sun.tools.javac.main=ALL-UNNAMED",
exportJava.toString(), "--help");
if (!tr2.isOK())
error(tr2, "Bad exit code: " + tr2.exitValue);
if (!(tr2.contains("demo") && tr2.contains("Usage")))
error(tr2, "Expected output not found");
show(tr2);
}
// java -cp ... HelloWorld.java (for a class "java" in package "HelloWorld")
@Test
void testClassNamedJava() throws IOException {
starting("testClassNamedJava");
Path base = Files.createDirectories(Paths.get("testClassNamedJava"));
Path src = Files.createDirectories(base.resolve("src"));
Path srcfile = src.resolve("java.java");
createFile(srcfile, List.of(
"package HelloWorld;",
"class java {",
" public static void main(String... args) {",
" System.out.println(HelloWorld.java.class.getName());",
" }",
"}"
));
Path classes = base.resolve("classes");
compile("-d", classes.toString(), srcfile.toString());
TestResult tr =
doExec(javaCmd, "-cp", classes.toString(), "HelloWorld.java");
if (!tr.isOK())
error(tr, "Command failed");
if (!tr.contains("HelloWorld.java"))
error(tr, "Expected output not found");
show(tr);
}
// java --source N -cp ... HelloWorld
@Test
void testSourceClasspath() throws IOException {
if (!isEnglishLocale()) {
return;
}
starting("testSourceClasspath");
Path base = Files.createDirectories(Paths.get("testSourceClasspath"));
Path src = Files.createDirectories(base.resolve("src"));
Path srcfile = src.resolve("java.java");
createFile(srcfile, List.of(
"class HelloWorld {",
" public static void main(String... args) {",
" System.out.println(\"Hello World\");",
" }",
"}"
));
Path classes = base.resolve("classes");
compile("-d", classes.toString(), srcfile.toString());
TestResult tr =
doExec(javaCmd, "--source", thisVersion, "-cp", classes.toString(), "HelloWorld");
if (tr.isOK())
error(tr, "Command succeeded unexpectedly");
if (!tr.contains("file not found: HelloWorld"))
error(tr, "Expected output not found");
show(tr);
}
// java --source
@Test
void testSourceNoArg() throws IOException {
starting("testSourceNoArg");
TestResult tr = doExec(javaCmd, "--source");
if (tr.isOK())
error(tr, "Command succeeded unexpectedly");
if (!tr.contains("--source requires source version"))
error(tr, "Expected output not found");
show(tr);
}
// java --source N -jar simple.jar
@Test
void testSourceJarConflict() throws IOException {
starting("testSourceJarConflict");
Path base = Files.createDirectories(Paths.get("testSourceJarConflict"));
Path file = getSimpleFile("Simple.java", false);
Path classes = Files.createDirectories(base.resolve("classes"));
compile("-d", classes.toString(), file.toString());
Path simpleJar = base.resolve("simple.jar");
createJar("cf", simpleJar.toString(), "-C", classes.toString(), ".");
TestResult tr =
doExec(javaCmd, "--source", thisVersion, "-jar", simpleJar.toString());
if (tr.isOK())
error(tr, "Command succeeded unexpectedly");
if (!tr.contains("Option -jar is not allowed with --source"))
error(tr, "Expected output not found");
show(tr);
}
// java --source N -m jdk.compiler
@Test
void testSourceModuleConflict() throws IOException {
starting("testSourceModuleConflict");
TestResult tr = doExec(javaCmd, "--source", thisVersion, "-m", "jdk.compiler");
if (tr.isOK())
error(tr, "Command succeeded unexpectedly");
if (!tr.contains("Option -m is not allowed with --source"))
error(tr, "Expected output not found");
show(tr);
}
// #!.../java --source N -version
@Test
void testTerminalOptionInShebang() throws IOException {
starting("testTerminalOptionInShebang");
if (skipShebangTest || isAIX || isMacOSX) {
// On MacOSX, we cannot distinguish between terminal options on the
// shebang line and those on the command line.
// On Solaris, all options after the first on the shebang line are
// ignored. Similar on AIX.
log.println("SKIPPED");
return;
}
Path base = Files.createDirectories(
Paths.get("testTerminalOptionInShebang"));
Path bad = base.resolve("bad");
createFile(bad, List.of(
"#!" + shebangJavaCmd + " --source " + thisVersion + " -version"));
setExecutable(bad);
TestResult tr = doExec(bad.toString());
if (!tr.contains("Option -version is not allowed in this context"))
error(tr, "Expected output not found");
show(tr);
}
// #!.../java --source N @bad.at (contains -version)
@Test
void testTerminalOptionInShebangAtFile() throws IOException {
starting("testTerminalOptionInShebangAtFile");
if (skipShebangTest || isAIX || isMacOSX) {
// On MacOSX, we cannot distinguish between terminal options in a
// shebang @-file and those on the command line.
// On Solaris, all options after the first on the shebang line are
// ignored. Similar on AIX.
log.println("SKIPPED");
return;
}
// Use a short directory name, to avoid line length limitations
Path base = Files.createDirectories(Paths.get("testBadAtFile"));
Path bad_at = base.resolve("bad.at");
createFile(bad_at, List.of("-version"));
Path bad = base.resolve("bad");
createFile(bad, List.of(
"#!" + shebangJavaCmd + " --source " + thisVersion + " @" + bad_at));
setExecutable(bad);
TestResult tr = doExec(bad.toString());
if (!tr.contains("Option -version in @testBadAtFile/bad.at is "
+ "not allowed in this context"))
error(tr, "Expected output not found");
show(tr);
}
// #!.../java --source N HelloWorld
@Test
void testMainClassInShebang() throws IOException {
starting("testMainClassInShebang");
if (skipShebangTest || isAIX || isMacOSX) {
// On MacOSX, we cannot distinguish between a main class on the
// shebang line and one on the command line.
// On Solaris, all options after the first on the shebang line are
// ignored. Similar on AIX.
log.println("SKIPPED");
return;
}
Path base = Files.createDirectories(Paths.get("testMainClassInShebang"));
Path bad = base.resolve("bad");
createFile(bad, List.of(
"#!" + shebangJavaCmd + " --source " + thisVersion + " HelloWorld"));
setExecutable(bad);
TestResult tr = doExec(bad.toString());
if (!tr.contains("Cannot specify main class in this context"))
error(tr, "Expected output not found");
show(tr);
}
//--------------------------------------------------------------------------
private void starting(String label) {
System.out.println();
System.out.println("*** Starting: " + label + " (stdout)");
System.err.println();
System.err.println("*** Starting: " + label + " (stderr)");
}
private void show(TestResult tr) {
log.println("*** Test Output:");
for (String line: tr.testOutput) {
log.println(line);
}
log.println("*** End Of Test Output:");
}
private Map<String,String> getLauncherDebugEnv() {
return Map.of("_JAVA_LAUNCHER_DEBUG", "1");
}
private Path getSimpleFile(String name, boolean shebang) throws IOException {
Path file = Paths.get(name);
if (!Files.exists(file)) {
createFile(file, List.of(
(shebang ? "#!" + shebangJavaCmd + " --source=" + thisVersion: ""),
"public class Simple {",
" public static void main(String[] args) {",
" System.out.println(java.util.Arrays.toString(args));",
" }}"));
}
return file;
}
private void createFile(Path file, List<String> lines) throws IOException {
lines.stream()
.filter(line -> line.length() > 128)
.forEach(line -> {
log.println("*** Warning: long line ("
+ line.length()
+ " chars) in file " + file);
log.println("*** " + line);
});
log.println("*** File: " + file);
lines.stream().forEach(log::println);
log.println("*** End Of File");
createFile(file.toFile(), lines);
}
private Path setExecutable(Path file) throws IOException {
Set<PosixFilePermission> perms = Files.getPosixFilePermissions(file);
perms.add(PosixFilePermission.OWNER_EXECUTE);
Files.setPosixFilePermissions(file, perms);
return file;
}
private void error(TestResult tr, String message) {
show(tr);
throw new RuntimeException(message);
}
}