blob: d56ff6918b14117461ac404cfe53db5282d90486 [file] [log] [blame]
/*
* 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);
}
}