/* * 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); } }