2016-04-15 00:51:30 +00:00
|
|
|
/*
|
2018-02-02 18:31:05 +00:00
|
|
|
* Copyright (c) 2016, 2018, Oracle and/or its affiliates. All rights reserved.
|
2016-04-15 00:51:30 +00:00
|
|
|
* 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
|
2018-02-02 18:31:05 +00:00
|
|
|
* @bug 8149757 8144062 8196627
|
2016-04-15 00:51:30 +00:00
|
|
|
* @summary Test that StandardJavaFileManager uses the correct version of a
|
|
|
|
* class from a multi-release jar on classpath
|
|
|
|
* @library /tools/lib
|
|
|
|
* @modules jdk.compiler/com.sun.tools.javac.api
|
|
|
|
* jdk.compiler/com.sun.tools.javac.main
|
|
|
|
* @build toolbox.ToolBox
|
|
|
|
* @run testng MultiReleaseJarAwareSJFM
|
|
|
|
*/
|
|
|
|
|
|
|
|
import org.testng.Assert;
|
|
|
|
import org.testng.annotations.AfterClass;
|
|
|
|
import org.testng.annotations.BeforeClass;
|
|
|
|
import org.testng.annotations.DataProvider;
|
|
|
|
import org.testng.annotations.Test;
|
|
|
|
|
|
|
|
import javax.tools.FileObject;
|
|
|
|
import javax.tools.JavaFileManager;
|
|
|
|
import javax.tools.StandardJavaFileManager;
|
|
|
|
import javax.tools.ToolProvider;
|
|
|
|
import java.io.File;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.InputStream;
|
|
|
|
import java.lang.invoke.MethodHandle;
|
|
|
|
import java.lang.invoke.MethodHandles;
|
|
|
|
import java.lang.invoke.MethodType;
|
|
|
|
import java.util.List;
|
|
|
|
|
|
|
|
import toolbox.JarTask;
|
|
|
|
import toolbox.JavacTask;
|
|
|
|
import toolbox.ToolBox;
|
|
|
|
|
|
|
|
public class MultiReleaseJarAwareSJFM {
|
2018-02-02 18:31:05 +00:00
|
|
|
private static final int CURRENT_VERSION = Runtime.version().major();
|
2016-04-15 00:51:30 +00:00
|
|
|
|
|
|
|
private final String version8 =
|
|
|
|
"package version;\n" +
|
|
|
|
"\n" +
|
|
|
|
"public class Version {\n" +
|
|
|
|
" public int getVersion() {\n" +
|
|
|
|
" return 8;\n" +
|
|
|
|
" }\n" +
|
|
|
|
"}\n";
|
|
|
|
|
|
|
|
private final String version9 =
|
|
|
|
"package version;\n" +
|
|
|
|
"\n" +
|
|
|
|
"public class Version {\n" +
|
|
|
|
" public int getVersion() {\n" +
|
|
|
|
" int version = (new PackagePrivate()).getVersion();\n" +
|
|
|
|
" if (version == 9) return 9;\n" +
|
|
|
|
" return version;\n" +
|
|
|
|
" }\n" +
|
|
|
|
"}\n";
|
|
|
|
|
|
|
|
private final String packagePrivate =
|
|
|
|
"package version;\n" +
|
|
|
|
"\n" +
|
|
|
|
"class PackagePrivate {\n" +
|
|
|
|
" int getVersion() {\n" +
|
|
|
|
" return 9;\n" +
|
|
|
|
" }\n" +
|
|
|
|
"}\n";
|
|
|
|
|
2018-02-02 18:31:05 +00:00
|
|
|
private final String versionCurrent =
|
2016-04-15 00:51:30 +00:00
|
|
|
"package version;\n" +
|
|
|
|
"\n" +
|
|
|
|
"public class Version {\n" +
|
|
|
|
" public int getVersion() {\n" +
|
2018-02-02 18:31:05 +00:00
|
|
|
" return " + CURRENT_VERSION +";\n" +
|
2016-04-15 00:51:30 +00:00
|
|
|
" }\n" +
|
|
|
|
"}\n";
|
|
|
|
|
|
|
|
private final String manifest =
|
|
|
|
"Manifest-Version: 1.0\n" +
|
|
|
|
"Multi-Release: true\n";
|
|
|
|
|
|
|
|
private final ToolBox tb = new ToolBox();
|
|
|
|
|
|
|
|
private final JavaFileManager.Location jloc = new JavaFileManager.Location() {
|
|
|
|
@Override
|
|
|
|
public String getName() {
|
|
|
|
return "Multi-Release Jar";
|
|
|
|
}
|
|
|
|
@Override
|
|
|
|
public boolean isOutputLocation() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
@BeforeClass
|
|
|
|
public void setup() throws Exception {
|
|
|
|
tb.createDirectories("classes",
|
|
|
|
"classes/META-INF/versions/9",
|
2018-02-02 18:31:05 +00:00
|
|
|
"classes/META-INF/versions/" + CURRENT_VERSION);
|
2016-04-15 00:51:30 +00:00
|
|
|
new JavacTask(tb)
|
|
|
|
.outdir("classes")
|
|
|
|
.sources(version8)
|
|
|
|
.run();
|
|
|
|
new JavacTask(tb)
|
|
|
|
.outdir("classes/META-INF/versions/9")
|
|
|
|
.sources(version9, packagePrivate)
|
|
|
|
.run();
|
|
|
|
new JavacTask(tb)
|
2018-02-02 18:31:05 +00:00
|
|
|
.outdir("classes/META-INF/versions/" + CURRENT_VERSION)
|
|
|
|
.sources(versionCurrent)
|
2016-04-15 00:51:30 +00:00
|
|
|
.run();
|
|
|
|
new JarTask(tb, "multi-release.jar")
|
|
|
|
.manifest(manifest)
|
|
|
|
.baseDir("classes")
|
|
|
|
.files("version/Version.class",
|
|
|
|
"META-INF/versions/9/version/Version.class",
|
|
|
|
"META-INF/versions/9/version/PackagePrivate.class",
|
2018-02-02 18:31:05 +00:00
|
|
|
"META-INF/versions/" + CURRENT_VERSION + "/version/Version.class")
|
2016-04-15 00:51:30 +00:00
|
|
|
.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
@AfterClass
|
|
|
|
public void teardown() throws Exception {
|
|
|
|
tb.deleteFiles(
|
2018-02-02 18:31:05 +00:00
|
|
|
"classes/META-INF/versions/" + CURRENT_VERSION + "/version/Version.class",
|
|
|
|
"classes/META-INF/versions/" + CURRENT_VERSION + "/version",
|
|
|
|
"classes/META-INF/versions/" + CURRENT_VERSION,
|
2016-04-15 00:51:30 +00:00
|
|
|
"classes/META-INF/versions/9/version/Version.class",
|
|
|
|
"classes/META-INF/versions/9/version/PackagePrivate.class",
|
|
|
|
"classes/META-INF/versions/9/version",
|
|
|
|
"classes/META-INF/versions/9",
|
|
|
|
"classes/META-INF/versions",
|
|
|
|
"classes/META-INF",
|
|
|
|
"classes/version/Version.class",
|
|
|
|
"classes/version",
|
|
|
|
"classes",
|
|
|
|
"multi-release.jar"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
@DataProvider(name = "versions")
|
|
|
|
public Object[][] data() {
|
|
|
|
return new Object[][] {
|
|
|
|
{"", 8},
|
|
|
|
{"8", 8},
|
|
|
|
{"9", 9},
|
2016-05-19 18:56:38 +00:00
|
|
|
{"runtime", Runtime.version().major()}
|
2016-04-15 00:51:30 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider = "versions")
|
|
|
|
public void test(String version, int expected) throws Throwable {
|
|
|
|
StandardJavaFileManager jfm = ToolProvider.getSystemJavaCompiler().getStandardFileManager(null, null, null);
|
|
|
|
jfm.setLocation(jloc, List.of(new File("multi-release.jar")));
|
|
|
|
|
|
|
|
if (version.length() > 0) {
|
2016-08-31 03:49:41 +00:00
|
|
|
jfm.handleOption("--multi-release", List.of(version).iterator());
|
2016-04-15 00:51:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CustomClassLoader cldr = new CustomClassLoader(jfm);
|
|
|
|
Class<?> versionClass = cldr.loadClass("version.Version");
|
|
|
|
MethodType mt = MethodType.methodType(int.class);
|
|
|
|
MethodHandle mh = MethodHandles.lookup().findVirtual(versionClass, "getVersion", mt);
|
|
|
|
int v = (int)mh.invoke(versionClass.newInstance());
|
|
|
|
Assert.assertEquals(v, expected);
|
|
|
|
|
|
|
|
jfm.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
private class CustomClassLoader extends ClassLoader {
|
|
|
|
private final JavaFileManager jfm;
|
|
|
|
|
|
|
|
public CustomClassLoader(JavaFileManager jfm) {
|
|
|
|
super(null);
|
|
|
|
this.jfm = jfm;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected Class<?> findClass(String name) throws ClassNotFoundException {
|
|
|
|
int n = name.lastIndexOf('.');
|
|
|
|
String pkg = n == -1 ? "" : name.substring(0, n);
|
|
|
|
String cls = name.substring(n + 1) + ".class";
|
|
|
|
byte[] b;
|
|
|
|
try {
|
|
|
|
FileObject obj = jfm.getFileForInput(jloc, pkg, cls);
|
|
|
|
try (InputStream is = obj.openInputStream()) {
|
|
|
|
b = is.readAllBytes();
|
|
|
|
}
|
|
|
|
} catch (IOException x) {
|
|
|
|
throw new ClassNotFoundException(x.getMessage(), x);
|
|
|
|
}
|
|
|
|
return defineClass(name, b, 0, b.length);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|