5ed704484e
Reviewed-by: dfuchs, bpb
174 lines
5.6 KiB
Java
174 lines
5.6 KiB
Java
/*
|
|
* Copyright (c) 2015, 2020, 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 Package object is local to each ClassLoader.
|
|
* There can be one Package object of "foo" name defined by
|
|
* different class loader.
|
|
* @compile Foo.java
|
|
* @run testng GetPackages
|
|
*/
|
|
|
|
import java.io.IOException;
|
|
import java.io.UncheckedIOException;
|
|
import java.nio.file.Files;
|
|
import java.nio.file.Path;
|
|
import java.nio.file.Paths;
|
|
import java.util.Arrays;
|
|
import java.lang.reflect.*;
|
|
|
|
import org.testng.annotations.DataProvider;
|
|
import org.testng.annotations.Test;
|
|
import static org.testng.Assert.*;
|
|
|
|
public class GetPackages {
|
|
final TestClassLoader loader;
|
|
final Class<?> fooClass;
|
|
/*
|
|
* Each TestClassLoader defines a "foo.Foo" class which has a unique
|
|
* Package object of "foo" regardless whether its ancestor class loader
|
|
* defines a package "foo" or not.
|
|
*/
|
|
GetPackages(TestClassLoader loader) throws ClassNotFoundException {
|
|
this.loader = loader;
|
|
this.fooClass = loader.loadClass("foo.Foo");
|
|
}
|
|
|
|
/** For TestNG */
|
|
public GetPackages() {
|
|
loader = null;
|
|
fooClass = null;
|
|
}
|
|
|
|
/*
|
|
* Check package "foo" defined locally in the TestClassLoader
|
|
* as well as its ancestors.
|
|
*/
|
|
void checkPackage() throws ClassNotFoundException {
|
|
// Name of an unnamed package is empty string
|
|
assertEquals(this.getClass().getPackage().getName(), "");
|
|
|
|
assertEquals(fooClass.getClassLoader(), loader);
|
|
|
|
Package p = loader.getDefinedPackage("foo");
|
|
assertEquals(p.getName(), "foo");
|
|
assertEquals(p, loader.getPackage("foo"));
|
|
|
|
if (loader.getParent() != null) {
|
|
Package p2 = ((TestClassLoader)loader.getParent()).getDefinedPackage("foo");
|
|
assertTrue(p != p2);
|
|
}
|
|
|
|
long count = Arrays.stream(loader.getDefinedPackages())
|
|
.map(Package::getName)
|
|
.filter(pn -> pn.equals("foo"))
|
|
.count();
|
|
assertEquals(count, 1);
|
|
}
|
|
|
|
/*
|
|
* Check the number of package "foo" defined to this class loader and
|
|
* its ancestors
|
|
*/
|
|
Package[] getPackagesFromLoader() {
|
|
return loader.packagesInClassLoaderChain();
|
|
}
|
|
|
|
/*
|
|
* Package.getPackages is caller-sensitve. Call through Foo class
|
|
* to find all Packages visible to this TestClassLoader and its ancestors
|
|
*/
|
|
Package[] getPackagesFromFoo() throws Exception {
|
|
Method m = fooClass.getMethod("getPackages");
|
|
return (Package[])m.invoke(null);
|
|
}
|
|
|
|
private static long numFooPackages(Package[] pkgs) throws Exception {
|
|
return Arrays.stream(pkgs)
|
|
.filter(p -> p.getName().equals("foo"))
|
|
.count();
|
|
}
|
|
|
|
@DataProvider(name = "loaders")
|
|
public static Object[][] testLoaders() {
|
|
TestClassLoader loader1 = new TestClassLoader(null);
|
|
TestClassLoader loader2 = new TestClassLoader(loader1);
|
|
TestClassLoader loader3 = new TestClassLoader(loader2);
|
|
|
|
// Verify the number of expected Package object of "foo" visible
|
|
// to the class loader
|
|
return new Object[][] {
|
|
{ loader1, 1 },
|
|
{ loader2, 2 },
|
|
{ loader3, 3 }
|
|
};
|
|
}
|
|
|
|
@Test(dataProvider = "loaders")
|
|
public static void test(TestClassLoader loader, long expected) throws Exception {
|
|
GetPackages test = new GetPackages(loader);
|
|
// check package "foo" existence
|
|
test.checkPackage();
|
|
|
|
assertEquals(numFooPackages(test.getPackagesFromLoader()), expected);
|
|
assertEquals(numFooPackages(test.getPackagesFromFoo()), expected);
|
|
}
|
|
}
|
|
|
|
class TestClassLoader extends ClassLoader {
|
|
public TestClassLoader() {
|
|
super();
|
|
}
|
|
|
|
public TestClassLoader(ClassLoader parent) {
|
|
super(parent);
|
|
}
|
|
|
|
public Package getPackage(String pn) {
|
|
return super.getPackage(pn);
|
|
}
|
|
|
|
public Package[] packagesInClassLoaderChain() {
|
|
return super.getPackages();
|
|
}
|
|
|
|
@Override
|
|
protected Class<?> findClass(String name) throws ClassNotFoundException {
|
|
Path p = Paths.get(System.getProperty("test.classes", "."));
|
|
|
|
try {
|
|
byte[] bb = Files.readAllBytes(p.resolve("foo/Foo.class"));
|
|
return defineClass(name, bb, 0, bb.length);
|
|
} catch (IOException e) {
|
|
throw new UncheckedIOException(e);
|
|
}
|
|
}
|
|
@Override
|
|
protected Class<?> loadClass(String cn, boolean resolve) throws ClassNotFoundException {
|
|
if (!cn.equals("foo.Foo"))
|
|
return super.loadClass(cn, resolve);
|
|
return findClass(cn);
|
|
}
|
|
}
|