org.gradle.tooling.internal.provider.serialization.DefaultPayloadClassLoaderRegistry Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
/*
* Copyright 2016 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.tooling.internal.provider.serialization;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import net.jcip.annotations.ThreadSafe;
import org.gradle.api.Transformer;
import org.gradle.internal.classloader.ClassLoaderSpec;
import org.gradle.internal.classloader.ClassLoaderVisitor;
import org.gradle.internal.classloader.SystemClassLoaderSpec;
import org.gradle.internal.classloader.VisitableURLClassLoader;
import org.gradle.internal.reflect.JavaReflectionUtil;
import org.gradle.util.CollectionUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.UUID;
/**
* A {@link PayloadClassLoaderRegistry} that maps classes loaded by a set of ClassLoaders that it manages. For ClassLoaders owned by this JVM, inspects the ClassLoader to determine a ClassLoader spec to send across to the peer JVM. For classes serialized from the peer, maintains a set of cached ClassLoaders created using the ClassLoader specs received from the peer.
*/
@ThreadSafe
public class DefaultPayloadClassLoaderRegistry implements PayloadClassLoaderRegistry {
private static final Logger LOGGER = LoggerFactory.getLogger(DefaultPayloadClassLoaderRegistry.class);
private final PayloadClassLoaderFactory classLoaderFactory;
private final ClassLoaderCache cache;
private final ClassLoaderToDetailsTransformer detailsToClassLoader = new ClassLoaderToDetailsTransformer();
private final DetailsToClassLoaderTransformer classLoaderToDetails = new DetailsToClassLoaderTransformer();
public DefaultPayloadClassLoaderRegistry(ClassLoaderCache cache, PayloadClassLoaderFactory payloadClassLoaderFactory) {
this.cache = cache;
this.classLoaderFactory = payloadClassLoaderFactory;
}
public SerializeMap newSerializeSession() {
return new SerializeMap() {
final Map classLoaderIds = new HashMap();
final Map classLoaderDetails = new HashMap();
public short visitClass(Class> target) {
ClassLoader classLoader = target.getClassLoader();
Short id = classLoaderIds.get(classLoader);
if (id != null) {
return id;
}
if (classLoaderIds.size() == Short.MAX_VALUE) {
throw new UnsupportedOperationException();
}
ClassLoaderDetails details = getDetails(classLoader);
id = (short) (classLoaderIds.size() + 1);
classLoaderIds.put(classLoader, id);
classLoaderDetails.put(id, details);
return id;
}
@Override
public void collectClassLoaderDefinitions(Map details) {
details.putAll(classLoaderDetails);
}
};
}
public DeserializeMap newDeserializeSession() {
return new DeserializeMap() {
public Class> resolveClass(ClassLoaderDetails classLoaderDetails, String className) throws ClassNotFoundException {
ClassLoader classLoader = getClassLoader(classLoaderDetails);
return Class.forName(className, false, classLoader);
}
};
}
private ClassLoader getClassLoader(ClassLoaderDetails details) {
ClassLoader classLoader = cache.getClassLoader(details, detailsToClassLoader);
if (details.spec instanceof ClientOwnedClassLoaderSpec) {
ClientOwnedClassLoaderSpec spec = (ClientOwnedClassLoaderSpec) details.spec;
VisitableURLClassLoader urlClassLoader = (VisitableURLClassLoader) classLoader;
Set currentClassPath = ImmutableSet.copyOf(urlClassLoader.getURLs());
for (URL url : spec.getClasspath()) {
if (!currentClassPath.contains(url)) {
JavaReflectionUtil.method(URLClassLoader.class, Void.class, "addURL", URL.class).invoke(urlClassLoader, url);
}
}
}
return classLoader;
}
private ClassLoaderDetails getDetails(ClassLoader classLoader) {
return cache.getDetails(classLoader, classLoaderToDetails);
}
private static class ClassLoaderSpecVisitor extends ClassLoaderVisitor {
final ClassLoader classLoader;
final List parents = new ArrayList();
ClassLoaderSpec spec;
URL[] classPath;
public ClassLoaderSpecVisitor(ClassLoader classLoader) {
this.classLoader = classLoader;
}
@Override
public void visit(ClassLoader candidate) {
if (candidate == classLoader) {
super.visit(candidate);
} else {
parents.add(candidate);
}
}
@Override
public void visitClassPath(URL[] classPath) {
this.classPath = classPath;
}
@Override
public void visitSpec(ClassLoaderSpec spec) {
this.spec = spec;
}
}
private class ClassLoaderToDetailsTransformer implements Transformer {
public ClassLoader transform(ClassLoaderDetails details) {
List parents = new ArrayList();
for (ClassLoaderDetails parentDetails : details.parents) {
parents.add(getClassLoader(parentDetails));
}
if (parents.isEmpty()) {
parents.add(classLoaderFactory.getClassLoaderFor(SystemClassLoaderSpec.INSTANCE, ImmutableList.of()));
}
LOGGER.info("Creating ClassLoader {} from {} and {}.", details.uuid, details.spec, parents);
return classLoaderFactory.getClassLoaderFor(details.spec, parents);
}
}
private class DetailsToClassLoaderTransformer implements Transformer {
public ClassLoaderDetails transform(ClassLoader classLoader) {
ClassLoaderSpecVisitor visitor = new ClassLoaderSpecVisitor(classLoader);
visitor.visit(classLoader);
if (visitor.spec == null) {
if (visitor.classPath == null) {
visitor.spec = SystemClassLoaderSpec.INSTANCE;
} else {
visitor.spec = new VisitableURLClassLoader.Spec(CollectionUtils.toList(visitor.classPath));
}
}
UUID uuid = UUID.randomUUID();
ClassLoaderDetails details = new ClassLoaderDetails(uuid, visitor.spec);
for (ClassLoader parent : visitor.parents) {
details.parents.add(getDetails(parent));
}
return details;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy