jdk-24/jdk/test/java/lang/reflect/Layer/BasicLayerTest.java
Alan Bateman db4d383614 8142968: Module System implementation
Initial integration of JEP 200, JEP 260, JEP 261, and JEP 282

Co-authored-by: Alex Buckley <alex.buckley@oracle.com>
Co-authored-by: Jonathan Gibbons <jonathan.gibbons@oracle.com>
Co-authored-by: Karen Kinnear <karen.kinnear@oracle.com>
Co-authored-by: Mandy Chung <mandy.chung@oracle.com>
Co-authored-by: Mark Reinhold <mark.reinhold@oracle.com>
Co-authored-by: Chris Hegarty <chris.hegarty@oracle.com>
Co-authored-by: Alexandr Scherbatiy <alexandr.scherbatiy@oracle.com>
Co-authored-by: Amy Lu <amy.lu@oracle.com>
Co-authored-by: Calvin Cheung <calvin.cheung@oracle.com>
Co-authored-by: Daniel Fuchs <daniel.fuchs@oracle.com>
Co-authored-by: Erik Joelsson <erik.joelsson@oracle.com>
Co-authored-by: Harold Seigel <harold.seigel@oracle.com>
Co-authored-by: Jaroslav Bachorik <jaroslav.bachorik@oracle.com>
Co-authored-by: Jean-Francois Denise <jean-francois.denise@oracle.com>
Co-authored-by: Jan Lahoda <jan.lahoda@oracle.com>
Co-authored-by: James Laskey <james.laskey@oracle.com>
Co-authored-by: Lois Foltan <lois.foltan@oracle.com>
Co-authored-by: Miroslav Kos <miroslav.kos@oracle.com>
Co-authored-by: Huaming Li <huaming.li@oracle.com>
Co-authored-by: Sean Mullan <sean.mullan@oracle.com>
Co-authored-by: Naoto Sato <naoto.sato@oracle.com>
Co-authored-by: Masayoshi Okutsu <masayoshi.okutsu@oracle.com>
Co-authored-by: Peter Levart <peter.levart@gmail.com>
Co-authored-by: Philip Race <philip.race@oracle.com>
Co-authored-by: Claes Redestad <claes.redestad@oracle.com>
Co-authored-by: Sergey Bylokhov <sergey.bylokhov@oracle.com>
Co-authored-by: Alexandre Iline <alexandre.iline@oracle.com>
Co-authored-by: Volker Simonis <volker.simonis@gmail.com>
Co-authored-by: Staffan Larsen <staffan.larsen@oracle.com>
Co-authored-by: Stuart Marks <stuart.marks@oracle.com>
Co-authored-by: Semyon Sadetsky <semyon.sadetsky@oracle.com>
Co-authored-by: Serguei Spitsyn <serguei.spitsyn@oracle.com>
Co-authored-by: Sundararajan Athijegannathan <sundararajan.athijegannathan@oracle.com>
Co-authored-by: Valerie Peng <valerie.peng@oracle.com>
Co-authored-by: Vincent Ryan <vincent.x.ryan@oracle.com>
Co-authored-by: Weijun Wang <weijun.wang@oracle.com>
Co-authored-by: Yuri Nesterenko <yuri.nesterenko@oracle.com>
Co-authored-by: Yekaterina Kantserova <yekaterina.kantserova@oracle.com>
Co-authored-by: Alexander Kulyakthin <alexander.kulyakhtin@oracle.com>
Co-authored-by: Felix Yang <felix.yang@oracle.com>
Co-authored-by: Andrei Eremeev <andrei.eremeev@oracle.com>
Co-authored-by: Frank Yuan <frank.yuan@oracle.com>
Co-authored-by: Sergei Pikalev <sergei.pikalev@oracle.com>
Co-authored-by: Sibabrata Sahoo <sibabrata.sahoo@oracle.com>
Co-authored-by: Tiantian Du <tiantian.du@oracle.com>
Co-authored-by: Sha Jiang <sha.jiang@oracle.com>
Reviewed-by: alanb, mchung, naoto, rriggs, psandoz, plevart, mullan, ascarpino, vinnie, prr, sherman, dfuchs, mhaupt
2016-03-17 19:04:16 +00:00

868 lines
28 KiB
Java

/*
* Copyright (c) 2014, 2016, 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
* @library /lib/testlibrary
* @build BasicLayerTest ModuleUtils
* @compile layertest/Test.java
* @run testng BasicLayerTest
* @summary Basic tests for java.lang.reflect.Layer
*/
import java.lang.module.Configuration;
import java.lang.module.ModuleDescriptor;
import java.lang.module.ModuleFinder;
import static java.lang.module.ModuleFinder.empty;
import java.lang.reflect.Layer;
import java.lang.reflect.LayerInstantiationException;
import java.lang.reflect.Module;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;
import org.testng.annotations.Test;
import static org.testng.Assert.*;
@Test
public class BasicLayerTest {
/**
* Exercise Layer.empty()
*/
public void testEmpty() {
Layer emptyLayer = Layer.empty();
assertFalse(emptyLayer.parent().isPresent());
assertTrue(emptyLayer.configuration() == Configuration.empty());
assertTrue(emptyLayer.modules().isEmpty());
assertFalse(emptyLayer.findModule("java.base").isPresent());
try {
emptyLayer.findLoader("java.base");
assertTrue(false);
} catch (IllegalArgumentException expected) { }
}
/**
* Exercise Layer.boot()
*/
public void testBoot() {
Layer bootLayer = Layer.boot();
// configuration
Configuration cf = bootLayer.configuration();
assertTrue(cf.findModule("java.base").get()
.reference()
.descriptor()
.exports()
.stream().anyMatch(e -> (e.source().equals("java.lang")
&& !e.isQualified())));
// modules
Set<Module> modules = bootLayer.modules();
assertTrue(modules.contains(Object.class.getModule()));
int count = (int) modules.stream().map(Module::getName).count();
assertEquals(count, modules.size()); // module names are unique
// findModule
Module base = Object.class.getModule();
assertTrue(bootLayer.findModule("java.base").get() == base);
assertTrue(base.getLayer() == bootLayer);
// findLoader
assertTrue(bootLayer.findLoader("java.base") == null);
// parent
assertTrue(bootLayer.parent().get() == Layer.empty());
}
/**
* Exercise Layer.create, created on an empty layer
*/
public void testLayerOnEmpty() {
ModuleDescriptor descriptor1
= new ModuleDescriptor.Builder("m1")
.requires("m2")
.exports("p1")
.build();
ModuleDescriptor descriptor2
= new ModuleDescriptor.Builder("m2")
.requires("m3")
.build();
ModuleDescriptor descriptor3
= new ModuleDescriptor.Builder("m3")
.build();
ModuleFinder finder
= ModuleUtils.finderOf(descriptor1, descriptor2, descriptor3);
Configuration cf = resolveRequires(finder, "m1");
// map each module to its own class loader for this test
ClassLoader loader1 = new ClassLoader() { };
ClassLoader loader2 = new ClassLoader() { };
ClassLoader loader3 = new ClassLoader() { };
Map<String, ClassLoader> map = new HashMap<>();
map.put("m1", loader1);
map.put("m2", loader2);
map.put("m3", loader3);
Layer layer = Layer.empty().defineModules(cf, map::get);
// configuration
assertTrue(layer.configuration() == cf);
assertTrue(layer.configuration().modules().size() == 3);
// modules
Set<Module> modules = layer.modules();
assertTrue(modules.size() == 3);
Set<String> names = modules.stream()
.map(Module::getName)
.collect(Collectors.toSet());
assertTrue(names.contains("m1"));
assertTrue(names.contains("m2"));
assertTrue(names.contains("m3"));
// findModule
Module m1 = layer.findModule("m1").get();
Module m2 = layer.findModule("m2").get();
Module m3 = layer.findModule("m3").get();
assertEquals(m1.getName(), "m1");
assertEquals(m2.getName(), "m2");
assertEquals(m3.getName(), "m3");
assertTrue(m1.getDescriptor() == descriptor1);
assertTrue(m2.getDescriptor() == descriptor2);
assertTrue(m3.getDescriptor() == descriptor3);
assertTrue(m1.getLayer() == layer);
assertTrue(m2.getLayer() == layer);
assertTrue(m3.getLayer() == layer);
assertTrue(modules.contains(m1));
assertTrue(modules.contains(m2));
assertTrue(modules.contains(m3));
assertFalse(layer.findModule("godot").isPresent());
// findLoader
assertTrue(layer.findLoader("m1") == loader1);
assertTrue(layer.findLoader("m2") == loader2);
assertTrue(layer.findLoader("m3") == loader3);
try {
ClassLoader loader = layer.findLoader("godot");
assertTrue(false);
} catch (IllegalArgumentException ignore) { }
// parent
assertTrue(layer.parent().get() == Layer.empty());
}
/**
* Exercise Layer.create, created over the boot layer
*/
public void testLayerOnBoot() {
ModuleDescriptor descriptor1
= new ModuleDescriptor.Builder("m1")
.requires("m2")
.requires("java.base")
.exports("p1")
.build();
ModuleDescriptor descriptor2
= new ModuleDescriptor.Builder("m2")
.requires("java.base")
.build();
ModuleFinder finder
= ModuleUtils.finderOf(descriptor1, descriptor2);
Configuration parent = Layer.boot().configuration();
Configuration cf = resolveRequires(parent, finder, "m1");
ClassLoader loader = new ClassLoader() { };
Layer layer = Layer.boot().defineModules(cf, mn -> loader);
// configuration
assertTrue(layer.configuration() == cf);
assertTrue(layer.configuration().modules().size() == 2);
// modules
Set<Module> modules = layer.modules();
assertTrue(modules.size() == 2);
Set<String> names = modules.stream()
.map(Module::getName)
.collect(Collectors.toSet());
assertTrue(names.contains("m1"));
assertTrue(names.contains("m2"));
// findModule
Module m1 = layer.findModule("m1").get();
Module m2 = layer.findModule("m2").get();
assertEquals(m1.getName(), "m1");
assertEquals(m2.getName(), "m2");
assertTrue(m1.getDescriptor() == descriptor1);
assertTrue(m2.getDescriptor() == descriptor2);
assertTrue(m1.getLayer() == layer);
assertTrue(m2.getLayer() == layer);
assertTrue(modules.contains(m1));
assertTrue(modules.contains(m2));
assertTrue(layer.findModule("java.base").get() == Object.class.getModule());
assertFalse(layer.findModule("godot").isPresent());
// findLoader
assertTrue(layer.findLoader("m1") == loader);
assertTrue(layer.findLoader("m2") == loader);
assertTrue(layer.findLoader("java.base") == null);
// parent
assertTrue(layer.parent().get() == Layer.boot());
}
/**
* Layer.create with a configuration of two modules that have the same
* module-private package.
*/
public void testSameConcealedPackage() {
ModuleDescriptor descriptor1
= new ModuleDescriptor.Builder("m1")
.requires("m2")
.conceals("p")
.build();
ModuleDescriptor descriptor2
= new ModuleDescriptor.Builder("m2")
.conceals("p")
.build();
ModuleFinder finder
= ModuleUtils.finderOf(descriptor1, descriptor2);
Configuration cf = resolveRequires(finder, "m1");
assertTrue(cf.modules().size() == 2);
// one loader per module, should be okay
Layer.empty().defineModules(cf, mn -> new ClassLoader() { });
// same class loader
try {
ClassLoader loader = new ClassLoader() { };
Layer.empty().defineModules(cf, mn -> loader);
assertTrue(false);
} catch (LayerInstantiationException expected) { }
}
/**
* Layer.create with a configuration with a partitioned graph. The same
* package is exported in both partitions.
*/
public void testSameExportInPartitionedGraph() {
// m1 reads m2, m2 exports p to m1
ModuleDescriptor descriptor1
= new ModuleDescriptor.Builder("m1")
.requires("m2")
.build();
ModuleDescriptor descriptor2
= new ModuleDescriptor.Builder("m2")
.exports("p", "m1")
.build();
// m3 reads m4, m4 exports p to m3
ModuleDescriptor descriptor3
= new ModuleDescriptor.Builder("m3")
.requires("m4")
.build();
ModuleDescriptor descriptor4
= new ModuleDescriptor.Builder("m4")
.exports("p", "m3")
.build();
ModuleFinder finder
= ModuleUtils.finderOf(descriptor1,
descriptor2,
descriptor3,
descriptor4);
Configuration cf = resolveRequires(finder, "m1", "m3");
assertTrue(cf.modules().size() == 4);
// one loader per module
Layer.empty().defineModules(cf, mn -> new ClassLoader() { });
// m1 & m2 in one loader, m3 & m4 in another loader
ClassLoader loader1 = new ClassLoader() { };
ClassLoader loader2 = new ClassLoader() { };
Map<String, ClassLoader> map = new HashMap<>();
map.put("m1", loader1);
map.put("m2", loader1);
map.put("m3", loader2);
map.put("m3", loader2);
Layer.empty().defineModules(cf, map::get);
// same loader
try {
ClassLoader loader = new ClassLoader() { };
Layer.empty().defineModules(cf, mn -> loader);
assertTrue(false);
} catch (LayerInstantiationException expected) { }
}
/**
* Layer.create with a configuration that contains a module that has a
* concealed package that is the same name as a non-exported package
* in a parent layer.
*/
public void testConcealSamePackageAsBootLayer() {
// check assumption that java.base contains sun.launcher
ModuleDescriptor base = Object.class.getModule().getDescriptor();
assertTrue(base.conceals().contains("sun.launcher"));
ModuleDescriptor descriptor
= new ModuleDescriptor.Builder("m1")
.requires("java.base")
.conceals("sun.launcher")
.build();
ModuleFinder finder = ModuleUtils.finderOf(descriptor);
Configuration parent = Layer.boot().configuration();
Configuration cf = parent.resolveRequires(finder, empty(), Set.of("m1"));
assertTrue(cf.modules().size() == 1);
ClassLoader loader = new ClassLoader() { };
Layer layer = Layer.boot().defineModules(cf, mn -> loader);
assertTrue(layer.modules().size() == 1);
}
/**
* Test layers with implied readability.
*
* The test consists of three configurations:
* - Configuration/layer1: m1, m2 requires public m1
* - Configuration/layer2: m3 requires m1
*/
public void testImpliedReadabilityWithLayers1() {
// cf1: m1 and m2, m2 requires public m1
ModuleDescriptor descriptor1
= new ModuleDescriptor.Builder("m1")
.build();
ModuleDescriptor descriptor2
= new ModuleDescriptor.Builder("m2")
.requires(ModuleDescriptor.Requires.Modifier.PUBLIC, "m1")
.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1, descriptor2);
Configuration cf1 = resolveRequires(finder1, "m2");
ClassLoader cl1 = new ClassLoader() { };
Layer layer1 = Layer.empty().defineModules(cf1, mn -> cl1);
// cf2: m3, m3 requires m2
ModuleDescriptor descriptor3
= new ModuleDescriptor.Builder("m3")
.requires("m2")
.build();
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor3);
Configuration cf2 = resolveRequires(cf1, finder2, "m3");
ClassLoader cl2 = new ClassLoader() { };
Layer layer2 = layer1.defineModules(cf2, mn -> cl2);
assertTrue(layer1.parent().get() == Layer.empty());
assertTrue(layer2.parent().get() == layer1);
Module m1 = layer2.findModule("m1").get();
Module m2 = layer2.findModule("m2").get();
Module m3 = layer2.findModule("m3").get();
assertTrue(m1.getLayer() == layer1);
assertTrue(m2.getLayer() == layer1);
assertTrue(m3.getLayer() == layer2);
assertTrue(m1.getClassLoader() == cl1);
assertTrue(m2.getClassLoader() == cl1);
assertTrue(m3.getClassLoader() == cl2);
assertTrue(m1.canRead(m1));
assertFalse(m1.canRead(m2));
assertFalse(m1.canRead(m3));
assertTrue(m2.canRead(m1));
assertTrue(m2.canRead(m2));
assertFalse(m2.canRead(m3));
assertTrue(m3.canRead(m1));
assertTrue(m3.canRead(m2));
assertTrue(m3.canRead(m3));
}
/**
* Test layers with implied readability.
*
* The test consists of three configurations:
* - Configuration/layer1: m1
* - Configuration/layer2: m2 requires public m3, m3 requires m2
*/
public void testImpliedReadabilityWithLayers2() {
// cf1: m1
ModuleDescriptor descriptor1
= new ModuleDescriptor.Builder("m1")
.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1);
Configuration cf1 = resolveRequires(finder1, "m1");
ClassLoader cl1 = new ClassLoader() { };
Layer layer1 = Layer.empty().defineModules(cf1, mn -> cl1);
// cf2: m2, m3: m2 requires public m1, m3 requires m2
ModuleDescriptor descriptor2
= new ModuleDescriptor.Builder("m2")
.requires(ModuleDescriptor.Requires.Modifier.PUBLIC, "m1")
.build();
ModuleDescriptor descriptor3
= new ModuleDescriptor.Builder("m3")
.requires("m2")
.build();
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor2, descriptor3);
Configuration cf2 = resolveRequires(cf1, finder2, "m3");
ClassLoader cl2 = new ClassLoader() { };
Layer layer2 = layer1.defineModules(cf2, mn -> cl2);
assertTrue(layer1.parent().get() == Layer.empty());
assertTrue(layer2.parent().get() == layer1);
Module m1 = layer2.findModule("m1").get();
Module m2 = layer2.findModule("m2").get();
Module m3 = layer2.findModule("m3").get();
assertTrue(m1.getLayer() == layer1);
assertTrue(m2.getLayer() == layer2);
assertTrue(m3.getLayer() == layer2);
assertTrue(m1.canRead(m1));
assertFalse(m1.canRead(m2));
assertFalse(m1.canRead(m3));
assertTrue(m2.canRead(m1));
assertTrue(m2.canRead(m2));
assertFalse(m2.canRead(m3));
assertTrue(m3.canRead(m1));
assertTrue(m3.canRead(m2));
assertTrue(m3.canRead(m3));
}
/**
* Test layers with implied readability.
*
* The test consists of three configurations:
* - Configuration/layer1: m1
* - Configuration/layer2: m2 requires public m1
* - Configuration/layer3: m3 requires m1
*/
public void testImpliedReadabilityWithLayers3() {
// cf1: m1
ModuleDescriptor descriptor1
= new ModuleDescriptor.Builder("m1")
.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1);
Configuration cf1 = resolveRequires(finder1, "m1");
ClassLoader cl1 = new ClassLoader() { };
Layer layer1 = Layer.empty().defineModules(cf1, mn -> cl1);
// cf2: m2 requires public m1
ModuleDescriptor descriptor2
= new ModuleDescriptor.Builder("m2")
.requires(ModuleDescriptor.Requires.Modifier.PUBLIC, "m1")
.build();
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor2);
Configuration cf2 = resolveRequires(cf1, finder2, "m2");
ClassLoader cl2 = new ClassLoader() { };
Layer layer2 = layer1.defineModules(cf2, mn -> cl2);
// cf3: m3 requires m2
ModuleDescriptor descriptor3
= new ModuleDescriptor.Builder("m3")
.requires("m2")
.build();
ModuleFinder finder3 = ModuleUtils.finderOf(descriptor3);
Configuration cf3 = resolveRequires(cf2, finder3, "m3");
ClassLoader cl3 = new ClassLoader() { };
Layer layer3 = layer2.defineModules(cf3, mn -> cl3);
assertTrue(layer1.parent().get() == Layer.empty());
assertTrue(layer2.parent().get() == layer1);
assertTrue(layer3.parent().get() == layer2);
Module m1 = layer3.findModule("m1").get();
Module m2 = layer3.findModule("m2").get();
Module m3 = layer3.findModule("m3").get();
assertTrue(m1.getLayer() == layer1);
assertTrue(m2.getLayer() == layer2);
assertTrue(m3.getLayer() == layer3);
assertTrue(m1.canRead(m1));
assertFalse(m1.canRead(m2));
assertFalse(m1.canRead(m3));
assertTrue(m2.canRead(m1));
assertTrue(m2.canRead(m2));
assertFalse(m2.canRead(m3));
assertTrue(m3.canRead(m1));
assertTrue(m3.canRead(m2));
assertTrue(m3.canRead(m3));
}
/**
* Test layers with implied readability.
*
* The test consists of two configurations:
* - Configuration/layer1: m1, m2 requires public m1
* - Configuration/layer2: m3 requires public m2, m4 requires m3
*/
public void testImpliedReadabilityWithLayers4() {
// cf1: m1, m2 requires public m1
ModuleDescriptor descriptor1
= new ModuleDescriptor.Builder("m1")
.build();
ModuleDescriptor descriptor2
= new ModuleDescriptor.Builder("m2")
.requires(ModuleDescriptor.Requires.Modifier.PUBLIC, "m1")
.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1, descriptor2);
Configuration cf1 = resolveRequires(finder1, "m2");
ClassLoader cl1 = new ClassLoader() { };
Layer layer1 = Layer.empty().defineModules(cf1, mn -> cl1);
// cf2: m3 requires public m2, m4 requires m3
ModuleDescriptor descriptor3
= new ModuleDescriptor.Builder("m3")
.requires(ModuleDescriptor.Requires.Modifier.PUBLIC, "m2")
.build();
ModuleDescriptor descriptor4
= new ModuleDescriptor.Builder("m4")
.requires("m3")
.build();
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor3, descriptor4);
Configuration cf2 = resolveRequires(cf1, finder2, "m3", "m4");
ClassLoader cl2 = new ClassLoader() { };
Layer layer2 = layer1.defineModules(cf2, mn -> cl2);
assertTrue(layer1.parent().get() == Layer.empty());
assertTrue(layer2.parent().get() == layer1);
Module m1 = layer2.findModule("m1").get();
Module m2 = layer2.findModule("m2").get();
Module m3 = layer2.findModule("m3").get();
Module m4 = layer2.findModule("m4").get();
assertTrue(m1.getLayer() == layer1);
assertTrue(m2.getLayer() == layer1);
assertTrue(m3.getLayer() == layer2);
assertTrue(m4.getLayer() == layer2);
assertTrue(m1.canRead(m1));
assertFalse(m1.canRead(m2));
assertFalse(m1.canRead(m3));
assertFalse(m1.canRead(m4));
assertTrue(m2.canRead(m1));
assertTrue(m2.canRead(m2));
assertFalse(m1.canRead(m3));
assertFalse(m1.canRead(m4));
assertTrue(m3.canRead(m1));
assertTrue(m3.canRead(m2));
assertTrue(m3.canRead(m3));
assertFalse(m3.canRead(m4));
assertTrue(m4.canRead(m1));
assertTrue(m4.canRead(m2));
assertTrue(m4.canRead(m3));
assertTrue(m4.canRead(m4));
}
/**
* Attempt to use Layer.create to create a layer with a module defined to a
* class loader that already has a module of the same name defined to the
* class loader.
*/
@Test(expectedExceptions = { LayerInstantiationException.class })
public void testModuleAlreadyDefinedToLoader() {
ModuleDescriptor md
= new ModuleDescriptor.Builder("m")
.requires("java.base")
.build();
ModuleFinder finder = ModuleUtils.finderOf(md);
Configuration parent = Layer.boot().configuration();
Configuration cf = parent.resolveRequires(finder, empty(), Set.of("m"));
ClassLoader loader = new ClassLoader() { };
Layer.boot().defineModules(cf, mn -> loader);
// should throw LayerInstantiationException as m1 already defined to loader
Layer.boot().defineModules(cf, mn -> loader);
}
/**
* Attempt to use Layer.create to create a Layer with a module containing
* package {@code p} where the class loader already has a module defined
* to it containing package {@code p}.
*/
@Test(expectedExceptions = { LayerInstantiationException.class })
public void testPackageAlreadyInNamedModule() {
ModuleDescriptor md1
= new ModuleDescriptor.Builder("m1")
.conceals("p")
.requires("java.base")
.build();
ModuleDescriptor md2
= new ModuleDescriptor.Builder("m2")
.conceals("p")
.requires("java.base")
.build();
ModuleFinder finder = ModuleUtils.finderOf(md1, md2);
ClassLoader loader = new ClassLoader() { };
// define m1 containing package p to class loader
Configuration parent = Layer.boot().configuration();
Configuration cf1 = parent.resolveRequires(finder, empty(), Set.of("m1"));
Layer layer1 = Layer.boot().defineModules(cf1, mn -> loader);
// attempt to define m2 containing package p to class loader
Configuration cf2 = parent.resolveRequires(finder, empty(), Set.of("m2"));
// should throw exception because p already in m1
Layer layer2 = Layer.boot().defineModules(cf2, mn -> loader);
}
/**
* Attempt to use Layer.create to create a Layer with a module containing
* a package in which a type is already loaded by the class loader.
*/
@Test(expectedExceptions = { LayerInstantiationException.class })
public void testPackageAlreadyInUnnamedModule() throws Exception {
Class<?> c = layertest.Test.class;
assertFalse(c.getModule().isNamed()); // in unnamed module
ModuleDescriptor md
= new ModuleDescriptor.Builder("m")
.conceals(c.getPackageName())
.requires("java.base")
.build();
ModuleFinder finder = ModuleUtils.finderOf(md);
Configuration parent = Layer.boot().configuration();
Configuration cf = parent.resolveRequires(finder, empty(), Set.of("m"));
Layer.boot().defineModules(cf, mn -> c.getClassLoader());
}
/**
* Parent of configuration != configuration of parent Layer
*/
@Test(expectedExceptions = { IllegalArgumentException.class })
public void testIncorrectParent1() {
ModuleDescriptor descriptor1
= new ModuleDescriptor.Builder("m1")
.requires("java.base")
.build();
ModuleFinder finder = ModuleUtils.finderOf(descriptor1);
Configuration parent = Layer.boot().configuration();
Configuration cf = parent.resolveRequires(finder, empty(), Set.of("m1"));
ClassLoader loader = new ClassLoader() { };
Layer.empty().defineModules(cf, mn -> loader);
}
/**
* Parent of configuration != configuration of parent Layer
*/
@Test(expectedExceptions = { IllegalArgumentException.class })
public void testIncorrectParent2() {
ModuleDescriptor descriptor1
= new ModuleDescriptor.Builder("m1")
.build();
ModuleFinder finder = ModuleUtils.finderOf(descriptor1);
Configuration cf = resolveRequires(finder, "m1");
ClassLoader loader = new ClassLoader() { };
Layer.boot().defineModules(cf, mn -> loader);
}
// null handling
@Test(expectedExceptions = { NullPointerException.class })
public void testCreateWithNull1() {
ClassLoader loader = new ClassLoader() { };
Layer.empty().defineModules(null, mn -> loader);
}
@Test(expectedExceptions = { NullPointerException.class })
public void testCreateWithNull2() {
ClassLoader loader = new ClassLoader() { };
Configuration cf = resolveRequires(Layer.boot().configuration(), empty());
Layer.boot().defineModules(cf, null);
}
@Test(expectedExceptions = { NullPointerException.class })
public void testCreateWithNull3() {
ClassLoader scl = ClassLoader.getSystemClassLoader();
Layer.empty().defineModulesWithOneLoader(null, scl);
}
@Test(expectedExceptions = { NullPointerException.class })
public void testCreateWithNull4() {
ClassLoader scl = ClassLoader.getSystemClassLoader();
Layer.empty().defineModulesWithManyLoaders(null, scl);
}
@Test(expectedExceptions = { NullPointerException.class })
public void testFindModuleWithNull() {
Layer.boot().findModule(null);
}
@Test(expectedExceptions = { NullPointerException.class })
public void testFindLoaderWithNull() {
Layer.boot().findLoader(null);
}
// immutable sets
@Test(expectedExceptions = { UnsupportedOperationException.class })
public void testImmutableSet() {
Module base = Object.class.getModule();
Layer.boot().modules().add(base);
}
/**
* Resolve the given modules, by name, and returns the resulting
* Configuration.
*/
private static Configuration resolveRequires(Configuration cf,
ModuleFinder finder,
String... roots) {
return cf.resolveRequires(finder, empty(), Set.of(roots));
}
private static Configuration resolveRequires(ModuleFinder finder,
String... roots) {
return resolveRequires(Configuration.empty(), finder, roots);
}
}