681137cad2
Reviewed-by: jjg
324 lines
13 KiB
Java
324 lines
13 KiB
Java
/*
|
|
* Copyright (c) 2023, 2024, 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 8304400 8332226
|
|
* @summary Test source launcher running Java programs contained in one module
|
|
* @modules jdk.compiler/com.sun.tools.javac.launcher
|
|
* @run junit ModuleSourceLauncherTests
|
|
*/
|
|
|
|
import static org.junit.jupiter.api.Assertions.*;
|
|
|
|
import org.junit.jupiter.api.Test;
|
|
import org.junit.jupiter.api.io.TempDir;
|
|
|
|
import java.nio.file.Files;
|
|
import java.nio.file.Path;
|
|
import java.util.List;
|
|
import java.util.Set;
|
|
import java.util.spi.ToolProvider;
|
|
|
|
class ModuleSourceLauncherTests {
|
|
@Test
|
|
void testHelloModularWorld(@TempDir Path base) throws Exception {
|
|
var packageFolder = Files.createDirectories(base.resolve("com/greetings"));
|
|
var mainFile = Files.writeString(packageFolder.resolve("Main.java"),
|
|
"""
|
|
package com.greetings;
|
|
public class Main {
|
|
public static void main(String... args) {
|
|
System.out.println("Greetings!");
|
|
System.out.println(" module -> " + Main.class.getModule().getName());
|
|
System.out.println(" package -> " + Main.class.getPackageName());
|
|
System.out.println(" class -> " + Main.class.getSimpleName());
|
|
}
|
|
}
|
|
""");
|
|
Files.writeString(base.resolve("module-info.java"),
|
|
"""
|
|
module com.greetings {}
|
|
""");
|
|
|
|
var run = Run.of(mainFile);
|
|
assertAll("Run -> " + run,
|
|
() -> assertLinesMatch(
|
|
"""
|
|
Greetings!
|
|
module -> com.greetings
|
|
package -> com.greetings
|
|
class -> Main
|
|
""".lines(),
|
|
run.stdOut().lines()),
|
|
() -> assertTrue(run.stdErr().isEmpty()),
|
|
() -> assertNull(run.exception())
|
|
);
|
|
|
|
var module = run.result().programClass().getModule();
|
|
assertEquals("com.greetings", module.getName());
|
|
var reference = module.getLayer().configuration().findModule(module.getName()).orElseThrow().reference();
|
|
try (var reader = reference.open()) {
|
|
assertLinesMatch(
|
|
"""
|
|
com/
|
|
com/greetings/
|
|
com/greetings/Main.class
|
|
com/greetings/Main.java
|
|
module-info.class
|
|
module-info.java
|
|
""".lines(),
|
|
reader.list());
|
|
}
|
|
}
|
|
|
|
@Test
|
|
void testTwoAndHalfPackages(@TempDir Path base) throws Exception {
|
|
var fooFolder = Files.createDirectories(base.resolve("foo"));
|
|
var program = Files.writeString(fooFolder.resolve("Main.java"),
|
|
"""
|
|
package foo;
|
|
public class Main {
|
|
public static void main(String... args) throws Exception {
|
|
var module = Main.class.getModule();
|
|
System.out.println("To the " + bar.Bar.class + " from " + module);
|
|
try (var stream = module.getResourceAsStream("baz/baz.txt")) {
|
|
System.out.println(new String(stream.readAllBytes()));
|
|
}
|
|
}
|
|
}
|
|
""");
|
|
var barFolder = Files.createDirectories(base.resolve("bar"));
|
|
Files.writeString(barFolder.resolve("Bar.java"), "package bar; public record Bar() {}");
|
|
var bazFolder = Files.createDirectories(base.resolve("baz"));
|
|
Files.writeString(bazFolder.resolve("baz.txt"), "baz");
|
|
var badFolder = Files.createDirectories(base.resolve(".bad"));
|
|
Files.writeString(badFolder.resolve("bad.txt"), "bad");
|
|
|
|
Files.writeString(base.resolve("module-info.java"),
|
|
"""
|
|
module m {
|
|
exports foo;
|
|
exports bar;
|
|
opens baz;
|
|
}
|
|
""");
|
|
|
|
var run = Run.of(program);
|
|
var result = run.result();
|
|
assertAll("Run -> " + run,
|
|
() -> assertLinesMatch(
|
|
"""
|
|
To the class bar.Bar from module m
|
|
baz
|
|
""".lines(),
|
|
run.stdOut().lines()),
|
|
() -> assertTrue(run.stdErr().isEmpty()),
|
|
() -> assertNull(run.exception()),
|
|
() -> assertEquals(Set.of("foo", "bar", "baz"), result.programClass().getModule().getPackages())
|
|
);
|
|
|
|
var module = run.result().programClass().getModule();
|
|
assertEquals("m", module.getName());
|
|
var reference = module.getLayer().configuration().findModule(module.getName()).orElseThrow().reference();
|
|
try (var reader = reference.open()) {
|
|
var actual = reader.list().toList();
|
|
assertLinesMatch(
|
|
"""
|
|
.bad/
|
|
.bad/bad.txt
|
|
bar/
|
|
bar/Bar.class
|
|
bar/Bar.java
|
|
baz/
|
|
baz/baz.txt
|
|
foo/
|
|
foo/Main.class
|
|
foo/Main.java
|
|
module-info.class
|
|
module-info.java
|
|
""".lines().toList(),
|
|
actual, "Actual lines -> " + actual);
|
|
}
|
|
}
|
|
|
|
@Test
|
|
void testUserModuleOnModulePath(@TempDir Path base) throws Exception {
|
|
Files.createDirectories(base.resolve("foo", "foo"));
|
|
Files.writeString(base.resolve("foo", "module-info.java"),
|
|
"""
|
|
module foo {
|
|
exports foo;
|
|
}
|
|
""");
|
|
Files.writeString(base.resolve("foo", "foo", "Foo.java"),
|
|
"""
|
|
package foo;
|
|
public record Foo() {}
|
|
""");
|
|
var javac = ToolProvider.findFirst("javac").orElseThrow();
|
|
javac.run(System.out, System.err, "--module-source-path", base.toString(), "--module", "foo", "-d", base.toString());
|
|
|
|
Files.createDirectories(base.resolve("bar", "bar"));
|
|
Files.writeString(base.resolve("bar", "module-info.java"),
|
|
"""
|
|
module bar {
|
|
requires foo;
|
|
}
|
|
""");
|
|
Files.writeString(base.resolve("bar", "bar","Prog1.java"),
|
|
"""
|
|
package bar;
|
|
class Prog1 {
|
|
public static void main(String... args) {
|
|
System.out.println(new foo.Foo());
|
|
}
|
|
}
|
|
""");
|
|
|
|
var command = List.of(
|
|
Path.of(System.getProperty("java.home"), "bin", "java").toString(),
|
|
"-p", ".",
|
|
"bar/bar/Prog1.java");
|
|
var redirectedOut = base.resolve("out.redirected");
|
|
var redirectedErr = base.resolve("err.redirected");
|
|
var process = new ProcessBuilder(command)
|
|
.directory(base.toFile())
|
|
.redirectOutput(redirectedOut.toFile())
|
|
.redirectError(redirectedErr.toFile())
|
|
.start();
|
|
var code = process.waitFor();
|
|
var out = Files.readAllLines(redirectedOut);
|
|
var err = Files.readAllLines(redirectedErr);
|
|
|
|
assertAll(
|
|
() -> assertEquals(0, code),
|
|
() -> assertLinesMatch(
|
|
"""
|
|
Foo[]
|
|
""".lines(), out.stream()),
|
|
() -> assertTrue(err.isEmpty())
|
|
);
|
|
}
|
|
|
|
@Test
|
|
void testServiceLoading(@TempDir Path base) throws Exception {
|
|
var packageFolder = Files.createDirectories(base.resolve("p"));
|
|
var mainFile = Files.writeString(packageFolder.resolve("Main.java"),
|
|
"""
|
|
package p;
|
|
|
|
import java.util.ServiceLoader;
|
|
import java.util.spi.ToolProvider;
|
|
|
|
class Main {
|
|
public static void main(String... args) throws Exception {
|
|
System.out.println(Main.class + " in " + Main.class.getModule());
|
|
System.out.println("1");
|
|
System.out.println(Main.class.getResource("/p/Main.java"));
|
|
System.out.println(Main.class.getResource("/p/Main.class"));
|
|
System.out.println("2");
|
|
System.out.println(Main.class.getResource("/p/Tool.java"));
|
|
System.out.println(Main.class.getResource("/p/Tool.class"));
|
|
System.out.println("3");
|
|
System.out.println(ToolProvider.findFirst("p.Tool")); // empty due to SCL being used
|
|
System.out.println("4");
|
|
listToolProvidersIn(Main.class.getModule().getLayer());
|
|
System.out.println("5");
|
|
Class.forName("p.Tool"); // trigger compilation of "p/Tool.java"
|
|
System.out.println(Main.class.getResource("/p/Tool.class"));
|
|
System.out.println("6");
|
|
listToolProvidersIn(Main.class.getModule().getLayer());
|
|
}
|
|
|
|
static void listToolProvidersIn(ModuleLayer layer) {
|
|
try {
|
|
ServiceLoader.load(layer, ToolProvider.class).stream()
|
|
.filter(service -> service.type().getModule().getLayer() == layer)
|
|
.map(ServiceLoader.Provider::get)
|
|
.forEach(System.out::println);
|
|
} catch (java.util.ServiceConfigurationError error) {
|
|
error.printStackTrace(System.err);
|
|
}
|
|
}
|
|
}
|
|
""");
|
|
Files.writeString(packageFolder.resolve("Tool.java"),
|
|
"""
|
|
package p;
|
|
|
|
import java.io.PrintWriter;
|
|
import java.util.spi.ToolProvider;
|
|
|
|
public record Tool(String name) implements ToolProvider {
|
|
public static void main(String... args) {
|
|
System.exit(new Tool().run(System.out, System.err, args));
|
|
}
|
|
|
|
public Tool() {
|
|
this(Tool.class.getName());
|
|
}
|
|
|
|
@Override
|
|
public int run(PrintWriter out, PrintWriter err, String... args) {
|
|
out.println(name + "/out");
|
|
err.println(name + "/err");
|
|
return 0;
|
|
}
|
|
}
|
|
""");
|
|
Files.writeString(base.resolve("module-info.java"),
|
|
"""
|
|
module m {
|
|
uses java.util.spi.ToolProvider;
|
|
provides java.util.spi.ToolProvider with p.Tool;
|
|
}
|
|
""");
|
|
|
|
var run = Run.of(mainFile);
|
|
assertAll("Run -> " + run,
|
|
() -> assertLinesMatch(
|
|
"""
|
|
class p.Main in module m
|
|
1
|
|
.*/p/Main.java
|
|
.*:p/Main.class
|
|
2
|
|
.*/p/Tool.java
|
|
null
|
|
3
|
|
Optional.empty
|
|
4
|
|
Tool[name=p.Tool]
|
|
5
|
|
.*:p/Tool.class
|
|
6
|
|
Tool[name=p.Tool]
|
|
""".lines(),
|
|
run.stdOut().lines()),
|
|
() -> assertTrue(run.stdErr().isEmpty()),
|
|
() -> assertNull(run.exception())
|
|
);
|
|
}
|
|
}
|