2fc00fbdb1
Reviewed-by: psandoz, redestad
277 lines
12 KiB
Java
277 lines
12 KiB
Java
/*
|
|
* Copyright (c) 2016, 2017, 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
|
|
* @summary Test Multi-Release jar usage in runtime
|
|
* @library /test/lib
|
|
* @modules jdk.compiler
|
|
* @run testng RuntimeTest
|
|
*/
|
|
|
|
import static org.testng.Assert.*;
|
|
|
|
import java.io.BufferedReader;
|
|
import java.io.File;
|
|
import java.io.IOException;
|
|
import java.io.InputStream;
|
|
import java.io.InputStreamReader;
|
|
import java.io.UncheckedIOException;
|
|
import java.lang.reflect.InvocationTargetException;
|
|
import java.lang.reflect.Method;
|
|
import java.net.URL;
|
|
import java.net.URLClassLoader;
|
|
import java.nio.file.Files;
|
|
import java.nio.file.Path;
|
|
import java.nio.file.Paths;
|
|
import java.nio.file.StandardCopyOption;
|
|
import java.util.ArrayList;
|
|
import java.util.Arrays;
|
|
import java.util.HashMap;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.stream.Collectors;
|
|
import java.util.stream.Stream;
|
|
|
|
import org.testng.annotations.BeforeClass;
|
|
import org.testng.annotations.DataProvider;
|
|
import org.testng.annotations.Test;
|
|
|
|
import jdk.test.lib.JDKToolFinder;
|
|
import jdk.test.lib.JDKToolLauncher;
|
|
import jdk.test.lib.compiler.CompilerUtils;
|
|
import jdk.test.lib.process.OutputAnalyzer;
|
|
import jdk.test.lib.process.ProcessTools;
|
|
|
|
public class RuntimeTest {
|
|
public static final int SUCCESS = 0;
|
|
private static final String src = System.getProperty("test.src", ".");
|
|
private static final String usr = System.getProperty("user.dir", ".");
|
|
|
|
private static final Path srcFileRoot = Paths.get(src, "data", "runtimetest");
|
|
private static final Path genFileRoot = Paths.get(usr, "data", "runtimetest");
|
|
|
|
private static final int OLD_RELEASE = 8;
|
|
private static final int CURRENT_RELEASE = Runtime.version().major();
|
|
private static final int FUTURE_RELEASE = CURRENT_RELEASE + 1;
|
|
private static final String MRJAR_BOTH_RELEASES = "MV_BOTH.jar";
|
|
private static final String MRJAR_CURRENT_RELEASE = "MV_ONLY_" + CURRENT_RELEASE + ".jar";
|
|
private static final String NON_MRJAR_OLD_RELEASE = "NON_MV.jar";
|
|
|
|
private static final int[] versions = { OLD_RELEASE, CURRENT_RELEASE, FUTURE_RELEASE };
|
|
|
|
@DataProvider(name = "jarFiles")
|
|
Object[][] jarFiles() {
|
|
return new Object[][]{
|
|
{ MRJAR_BOTH_RELEASES, CURRENT_RELEASE, CURRENT_RELEASE, CURRENT_RELEASE },
|
|
{ MRJAR_CURRENT_RELEASE, CURRENT_RELEASE, CURRENT_RELEASE, CURRENT_RELEASE },
|
|
{ NON_MRJAR_OLD_RELEASE, OLD_RELEASE, OLD_RELEASE, OLD_RELEASE }
|
|
};
|
|
}
|
|
|
|
@BeforeClass
|
|
protected void setUpTest() throws Throwable {
|
|
createJarSourceFiles();
|
|
compile();
|
|
Path classes = Paths.get("classes");
|
|
jar("cfm", MRJAR_BOTH_RELEASES, "manifest.txt",
|
|
"-C", classes.resolve("v" + OLD_RELEASE).toString(), ".",
|
|
"--release", "" + CURRENT_RELEASE, "-C", classes.resolve("v" + CURRENT_RELEASE).toString(), ".",
|
|
"--release", "" + FUTURE_RELEASE, "-C", classes.resolve("v" + FUTURE_RELEASE).toString(), ".")
|
|
.shouldHaveExitValue(0);
|
|
|
|
jar("cfm", MRJAR_CURRENT_RELEASE, "manifest.txt",
|
|
"-C", classes.resolve("v" + OLD_RELEASE).toString(), ".",
|
|
"--release", "" + CURRENT_RELEASE, "-C", classes.resolve("v" + CURRENT_RELEASE).toString(), ".")
|
|
.shouldHaveExitValue(0);
|
|
jar("cfm", NON_MRJAR_OLD_RELEASE, "manifest.txt",
|
|
"-C", classes.resolve("v" + OLD_RELEASE).toString(), ".")
|
|
.shouldHaveExitValue(0);
|
|
}
|
|
|
|
@Test(dataProvider = "jarFiles")
|
|
public void testClasspath(String jar, int mainVer, int helperVer,
|
|
int resVer) throws Throwable {
|
|
String[] command = { "-cp", jar, "testpackage.Main" };
|
|
System.out.println("Command arguments:" + Arrays.asList(command));
|
|
System.out.println();
|
|
java(command).shouldHaveExitValue(SUCCESS)
|
|
.shouldContain("Main version: " + mainVer)
|
|
.shouldContain("Helpers version: " + helperVer)
|
|
.shouldContain("Resource version: " + resVer);
|
|
}
|
|
|
|
@Test(dataProvider = "jarFiles")
|
|
void testMVJarAsLib(String jar, int mainVer, int helperVer, int resVer)
|
|
throws Throwable {
|
|
String[] apps = { "UseByImport", "UseByReflection" };
|
|
for (String app : apps) {
|
|
String[] command = {"-cp",
|
|
jar + File.pathSeparatorChar + "classes/test/", app };
|
|
System.out.println("Command arguments:" + Arrays.asList(command));
|
|
System.out.println();
|
|
java(command).shouldHaveExitValue(SUCCESS)
|
|
.shouldContain("Main version: " + mainVer)
|
|
.shouldContain("Helpers version: " + helperVer)
|
|
.shouldContain("Resource version: " + resVer);
|
|
}
|
|
}
|
|
|
|
@Test(dataProvider = "jarFiles")
|
|
void testJavaJar(String jar, int mainVer, int helperVer, int resVer)
|
|
throws Throwable {
|
|
String[] command = { "-jar", jar };
|
|
System.out.println("Command arguments:" + Arrays.asList(command));
|
|
System.out.println();
|
|
java(command).shouldHaveExitValue(SUCCESS)
|
|
.shouldContain("Main version: " + mainVer)
|
|
.shouldContain("Helpers version: " + helperVer)
|
|
.shouldContain("Resource version: " + resVer);
|
|
}
|
|
|
|
@Test(dataProvider = "jarFiles")
|
|
void testURLClassLoader(String jarName, int mainVer, int helperVer,
|
|
int resVer) throws ClassNotFoundException, NoSuchMethodException,
|
|
IllegalAccessException, IllegalArgumentException,
|
|
InvocationTargetException, IOException {
|
|
Path pathToJAR = Paths.get(jarName).toAbsolutePath();
|
|
URL jarURL1 = new URL("jar:file:" + pathToJAR + "!/");
|
|
URL jarURL2 = new URL("file:///" + pathToJAR);
|
|
testURLClassLoaderURL(jarURL1, mainVer, helperVer, resVer);
|
|
testURLClassLoaderURL(jarURL2, mainVer, helperVer, resVer);
|
|
}
|
|
|
|
private static void testURLClassLoaderURL(URL jarURL,
|
|
int mainVersionExpected, int helperVersionExpected,
|
|
int resourceVersionExpected) throws ClassNotFoundException,
|
|
NoSuchMethodException, IllegalAccessException,
|
|
IllegalArgumentException, InvocationTargetException, IOException {
|
|
System.out.println(
|
|
"Testing URLClassLoader MV JAR support for URL: " + jarURL);
|
|
URL[] urls = { jarURL };
|
|
int mainVersionActual;
|
|
int helperVersionActual;
|
|
int resourceVersionActual;
|
|
try (URLClassLoader cl = URLClassLoader.newInstance(urls)) {
|
|
Class c = cl.loadClass("testpackage.Main");
|
|
Method getMainVersion = c.getMethod("getMainVersion");
|
|
mainVersionActual = (int) getMainVersion.invoke(null);
|
|
Method getHelperVersion = c.getMethod("getHelperVersion");
|
|
helperVersionActual = (int) getHelperVersion.invoke(null);
|
|
try (InputStream ris = cl.getResourceAsStream("versionResource");
|
|
BufferedReader br = new BufferedReader(
|
|
new InputStreamReader(ris))) {
|
|
resourceVersionActual = Integer.parseInt(br.readLine());
|
|
}
|
|
}
|
|
|
|
assertEquals(mainVersionActual, mainVersionExpected,
|
|
"Test failed: Expected Main class version: "
|
|
+ mainVersionExpected + " Actual version: "
|
|
+ mainVersionActual);
|
|
assertEquals(helperVersionActual, helperVersionExpected,
|
|
"Test failed: Expected Helper class version: "
|
|
+ helperVersionExpected + " Actual version: "
|
|
+ helperVersionActual);
|
|
assertEquals(resourceVersionActual, resourceVersionExpected,
|
|
"Test failed: Expected resource version: "
|
|
+ resourceVersionExpected + " Actual version: "
|
|
+ resourceVersionActual);
|
|
}
|
|
|
|
@Test(dataProvider = "jarFiles")
|
|
void testJjs(String jar, int mainVer, int helperVer, int resVer)
|
|
throws Throwable {
|
|
JDKToolLauncher launcher = JDKToolLauncher.createUsingTestJDK("jjs");
|
|
launcher.addToolArg("-cp").addToolArg(jar)
|
|
.addToolArg(src + "/data/runtimetest/MVJarJJSTestScript.js");
|
|
ProcessTools.executeCommand(launcher.getCommand())
|
|
.shouldHaveExitValue(SUCCESS)
|
|
.shouldContain("Main version: " + mainVer)
|
|
.shouldContain("Helpers version: " + helperVer)
|
|
.shouldContain("Resource version: " + resVer);
|
|
}
|
|
|
|
private static OutputAnalyzer jar(String... args) throws Throwable {
|
|
JDKToolLauncher launcher = JDKToolLauncher.createUsingTestJDK("jar");
|
|
Stream.of(args).forEach(launcher::addToolArg);
|
|
return ProcessTools.executeCommand(launcher.getCommand());
|
|
}
|
|
|
|
private static String platformPath(String p) {
|
|
return p.replace("/", File.separator);
|
|
}
|
|
|
|
private static void createJarSourceFiles() throws IOException {
|
|
for (int ver : versions) {
|
|
Files.find(srcFileRoot, 3, (file, attrs) -> (file.toString().endsWith(".template")))
|
|
.map(srcFileRoot::relativize)
|
|
.map(Path::toString)
|
|
.map(p -> p.replace(".template", ""))
|
|
.forEach(f -> {
|
|
try {
|
|
Path template = srcFileRoot.resolve(f + ".template");
|
|
Path out = genFileRoot.resolve(platformPath("v" + ver + "/" + f));
|
|
Files.createDirectories(out.getParent());
|
|
List<String> lines = Files.lines(template)
|
|
.map(s -> s.replaceAll("\\$version", String.valueOf(ver)))
|
|
.collect(Collectors.toList());
|
|
Files.write(out, lines);
|
|
} catch (IOException x) {
|
|
throw new UncheckedIOException(x);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
private void compile() throws Throwable {
|
|
for (int ver : versions) {
|
|
Path classes = Paths.get(usr, "classes", "v" + ver);
|
|
Files.createDirectories(classes);
|
|
Path source = genFileRoot.resolve("v" + ver);
|
|
assertTrue(CompilerUtils.compile(source, classes));
|
|
Files.copy(source.resolve("versionResource"),
|
|
classes.resolve("versionResource"),
|
|
StandardCopyOption.REPLACE_EXISTING);
|
|
}
|
|
|
|
Path classes = Paths.get(usr, "classes", "test");
|
|
Files.createDirectory(classes);
|
|
Path source = srcFileRoot.resolve("test");
|
|
assertTrue(
|
|
CompilerUtils.compile(source, classes, "-cp", "classes/v" + OLD_RELEASE));
|
|
Files.copy(srcFileRoot.resolve("manifest.txt"),
|
|
Paths.get(usr, "manifest.txt"),
|
|
StandardCopyOption.REPLACE_EXISTING);
|
|
}
|
|
|
|
OutputAnalyzer java(String... args) throws Throwable {
|
|
String java = JDKToolFinder.getJDKTool("java");
|
|
|
|
List<String> commands = new ArrayList<>();
|
|
commands.add(java);
|
|
Stream.of(args).forEach(x -> commands.add(x));
|
|
return ProcessTools.executeCommand(new ProcessBuilder(commands));
|
|
}
|
|
}
|