org.gradle.api.internal.initialization.DefaultClassLoaderScopeTest.groovy Maven / Gradle / Ivy
/*
* Copyright 2013 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.gradle.api.internal.initialization
import org.gradle.api.internal.initialization.loadercache.DefaultClassLoaderCache
import org.gradle.api.internal.initialization.loadercache.FileClasspathHasher
import org.gradle.initialization.ClassLoaderScopeRegistryListener
import org.gradle.internal.classloader.CachingClassLoader
import org.gradle.internal.classloader.DefaultHashingClassLoaderFactory
import org.gradle.internal.classpath.ClassPath
import org.gradle.internal.classpath.DefaultClassPath
import org.gradle.test.fixtures.file.TestFile
import org.gradle.test.fixtures.file.TestNameTestDirectoryProvider
import org.junit.Rule
import spock.lang.Specification
class DefaultClassLoaderScopeTest extends Specification {
ClassLoaderScope root
ClassLoaderScope scope
def classpathHasher = new FileClasspathHasher()
DefaultClassLoaderCache classLoaderCache = new DefaultClassLoaderCache(new DefaultHashingClassLoaderFactory(classpathHasher), classpathHasher)
@Rule
TestNameTestDirectoryProvider testDirectoryProvider = new TestNameTestDirectoryProvider(getClass())
def setup() {
file("root/root") << "root"
def rootClassLoader = new URLClassLoader(classPath("root").asURLArray, getClass().classLoader.parent)
root = new RootClassLoaderScope("root", rootClassLoader, rootClassLoader, classLoaderCache, Stub(ClassLoaderScopeRegistryListener))
scope = root.createChild("child", null)
}
TestFile file(String path) {
testDirectoryProvider.testDirectory.file(path)
}
ClassPath classPath(String... paths) {
DefaultClassPath.of(paths.collect { file(it).createDir() } as Iterable)
}
def "locked scope with no modifications exports parent"() {
when:
scope.lock()
then:
scope.localClassLoader.is root.exportClassLoader
scope.exportClassLoader.is root.exportClassLoader
}
def "locked empty scope does not define any classes"() {
when:
scope.lock()
then:
!scope.defines(String)
}
def "ignores empty classpaths"() {
when:
scope.export(classPath())
scope.local(classPath())
scope.lock()
then:
scope.localClassLoader.is root.exportClassLoader
scope.exportClassLoader.is root.exportClassLoader
}
def "locked scope with only exports uses same export and local loader"() {
when:
file("export/foo") << "foo"
scope.export(classPath("export")).lock()
then:
scope.exportClassLoader.getResource("root").text == "root"
scope.exportClassLoader.getResource("foo").text == "foo"
scope.exportClassLoader instanceof URLClassLoader
scope.exportClassLoader.parent.is root.exportClassLoader
scope.localClassLoader.is scope.exportClassLoader
}
def "can export more than one classpath"() {
when:
file("export/1/foo") << "foo"
file("export/2/bar") << "bar"
scope.
export(classPath("export/1")).
export(classPath("export/2")).
lock()
then:
scope.exportClassLoader.getResource("root").text == "root"
scope.exportClassLoader.getResource("foo").text == "foo"
scope.exportClassLoader.getResource("bar").text == "bar"
scope.localClassLoader.is scope.exportClassLoader
}
def "locked scope with only local exports parent loader to children"() {
when:
file("local/1/foo") << "foo"
file("local/2/bar") << "bar"
scope.
local(classPath("local/1")).
local(classPath("local/2")).
lock()
then:
scope.localClassLoader.getResource("root").text == "root"
scope.localClassLoader.getResource("foo").text == "foo"
scope.localClassLoader.getResource("bar").text == "bar"
scope.localClassLoader != scope.exportClassLoader
scope.exportClassLoader.is root.exportClassLoader
}
def "locked scope with only one local exports parent loader to children and uses loader as local loader"() {
when:
file("local/foo") << "foo"
scope.local(classPath("local")).lock()
then:
scope.localClassLoader.getResource("root").text == "root"
scope.localClassLoader.getResource("foo").text == "foo"
scope.localClassLoader instanceof URLClassLoader
scope.localClassLoader.parent.is root.exportClassLoader
scope.exportClassLoader.is root.exportClassLoader
}
def "locked scope with local and exports custom ClassLoader to children"() {
when:
file("local/local") << "bar"
file("export/export") << "bar"
scope.
local(classPath("local")).
export(classPath("export")).
lock()
then:
scope.exportClassLoader.getResource("export").text == "bar"
scope.exportClassLoader.getResource("local") == null
scope.exportClassLoader instanceof URLClassLoader
scope.exportClassLoader.parent == root.exportClassLoader
scope.localClassLoader instanceof URLClassLoader
scope.localClassLoader.parent == scope.exportClassLoader
scope.localClassLoader.getResource("export").text == "bar"
scope.localClassLoader.getResource("local").text == "bar"
}
def "locked scope with local and exports defines local and exported classes"() {
when:
copyTo(TestClass1, file("local"))
copyTo(TestClass2, file("export"))
scope.
local(classPath("local")).
export(classPath("export")).
lock()
then:
def local = scope.localClassLoader.loadClass(TestClass1.name)
def exported = scope.exportClassLoader.loadClass(TestClass2.name)
scope.defines(local)
scope.defines(exported)
!scope.defines(TestClass1)
!scope.defines(TestClass2)
}
def "requesting loaders before locking creates pessimistic setup"() {
given:
scope.localClassLoader // trigger
when:
file("local/local") << "bar"
file("export/export") << "bar"
scope.
local(classPath("local")).
export(classPath("export"))
then:
scope.exportClassLoader instanceof CachingClassLoader
scope.exportClassLoader.getResource("root").text == "root"
scope.exportClassLoader.getResource("export").text == "bar"
scope.exportClassLoader.getResource("local") == null
scope.localClassLoader instanceof CachingClassLoader
scope.localClassLoader.getResource("root").text == "root"
scope.localClassLoader.getResource("export").text == "bar"
scope.localClassLoader.getResource("local").text == "bar"
}
def "pessimistic scope with local and exports defines local and exported classes"() {
given:
scope.localClassLoader // trigger
when:
copyTo(TestClass1, file("local"))
copyTo(TestClass2, file("export"))
scope.
local(classPath("local")).
export(classPath("export"))
then:
def local = scope.localClassLoader.loadClass(TestClass1.name)
def exported = scope.exportClassLoader.loadClass(TestClass2.name)
scope.defines(local)
scope.defines(exported)
!scope.defines(TestClass1)
!scope.defines(TestClass2)
}
def "cannot modify after locking"() {
given:
scope.lock()
when:
scope.local(classPath("local"))
then:
thrown IllegalStateException
when:
scope.export(classPath("local"))
then:
thrown IllegalStateException
}
def "child scopes can access exported but not local"() {
when:
file("local/local") << "bar"
file("export/export") << "bar"
scope.local(classPath("local"))
scope.export(classPath("export"))
def child = scope.lock().createChild("child", null).lock()
then:
child.localClassLoader.getResource("root").text == "root"
child.localClassLoader.getResource("export").text == "bar"
child.localClassLoader.getResource("local") == null
child.exportClassLoader.getResource("root").text == "root"
child.exportClassLoader.getResource("export").text == "bar"
child.exportClassLoader.getResource("local") == null
}
def "class loaders are reused"() {
given:
def c1 = classPath("c1")
def c2 = classPath("c2")
when:
def scope1 = root.createChild("child1", null).export(c1).local(c2).lock()
scope1.exportClassLoader
def scope2 = root.createChild("child2", null).export(c1).local(c2).lock()
scope2.exportClassLoader
then:
scope1.exportClassLoader.is scope2.exportClassLoader
when:
def child = scope1.createChild("child", null).export(c1).local(c2).lock()
child.exportClassLoader
then:
child.exportClassLoader != scope.exportClassLoader // classpath is the same, but root is different
}
def "pessimistic structure has parent visibility"() {
expect:
scope.localClassLoader.getResource("root").text == "root"
}
def "optimised structure has parent visibility"() {
expect:
scope.lock().localClassLoader.getResource("root").text == "root"
}
def "manages cache"() {
expect:
classLoaderCache.size() == 0
file("c1/f") << "c1"
file("c2/f") << "c2"
def c1 = classPath("c1")
def c2 = classPath("c2")
when:
root.createChild("c", null).local(c1).export(c2).lock().exportClassLoader
then:
classLoaderCache.size() == 2
when:
root.createChild("d", null).local(c1).export(c2).lock().exportClassLoader
then:
classLoaderCache.size() == 2
when:
root.createChild("c", null).local(c1).lock().exportClassLoader
then:
classLoaderCache.size() == 3 // c has a local, d has export and local
when:
root.createChild("d", null).lock().exportClassLoader
then:
classLoaderCache.size() == 1
when:
root.createChild("c", null).lock().exportClassLoader
then:
classLoaderCache.size() == 0
}
static ClassLoader isolatedLoader(File... paths) {
new URLClassLoader(paths*.toURI()*.toURL() as URL[], ClassLoader.getSystemClassLoader().parent)
}
def "can attach export loader with no extra classpath before realize"() {
when:
def attachLoaderPath = file("attach")
copyTo(TestClass1, attachLoaderPath)
def attachLoader = isolatedLoader(attachLoaderPath)
def exportLoader = scope.export(attachLoader).lock().exportClassLoader
def localLoader = scope.localClassLoader
then:
exportLoader.loadClass(TestClass1.name)
localLoader.loadClass(TestClass1.name)
}
def "can attach export loader with extra classpath before realize"() {
when:
def attachLoaderPath = file("attach")
def exportClasspath = file("path")
copyTo(TestClass1, attachLoaderPath)
copyTo(TestClass2, exportClasspath)
def attachLoader = isolatedLoader(attachLoaderPath)
def exportLoader = scope
.export(classPath("path"))
.export(attachLoader)
.lock()
.exportClassLoader
def localLoader = scope.localClassLoader
then:
exportLoader.loadClass(TestClass1.name)
exportLoader.loadClass(TestClass2.name)
localLoader.loadClass(TestClass1.name)
localLoader.loadClass(TestClass2.name)
when:
exportLoader.loadClass(TestClass2.name).classLoader.loadClass(TestClass1.name)
then:
thrown ClassNotFoundException
when:
exportLoader.loadClass(TestClass1.name).classLoader.loadClass(TestClass2.name)
then:
thrown ClassNotFoundException
and:
!scope.defines(exportLoader.loadClass(TestClass1.name))
scope.defines(exportLoader.loadClass(TestClass2.name))
}
def "can't add exported loader after lock"() {
when:
scope.lock().export(isolatedLoader(file("foo")))
then:
thrown IllegalStateException
}
def "can add export loaders after eagerly creating"() {
when:
def attachLoaderPath = file("attach")
copyTo(TestClass1, attachLoaderPath)
def attachLoader = isolatedLoader(attachLoaderPath)
def exportLoader = scope.export(attachLoader).exportClassLoader
def localLoader = scope.localClassLoader
then:
exportLoader.loadClass(TestClass1.name)
localLoader.loadClass(TestClass1.name)
when:
def second = file("second")
copyTo(TestClass2, second)
scope.export(isolatedLoader(second))
then:
exportLoader.loadClass(TestClass2.name)
localLoader.loadClass(TestClass2.name)
and:
!scope.defines(exportLoader.loadClass(TestClass1.name))
!scope.defines(exportLoader.loadClass(TestClass2.name))
}
void copyTo(Class> clazz, TestFile destDir) {
def fileName = clazz.name.replace('.', '/') + ".class"
def dest = destDir.file(fileName)
def classFile = clazz.classLoader.getResource(fileName)
dest.copyFrom(classFile)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy