org.gradle.tooling.internal.provider.serialization.ClasspathInferer 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.MapMaker;
import com.google.common.io.ByteStreams;
import net.jcip.annotations.ThreadSafe;
import org.gradle.api.GradleException;
import org.gradle.internal.classloader.ClassLoaderUtils;
import org.gradle.internal.classloader.ClasspathUtil;
import org.objectweb.asm.ClassReader;
import org.objectweb.asm.Type;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.File;
import java.io.InputStream;
import java.net.JarURLConnection;
import java.net.URL;
import java.net.URLConnection;
import java.util.Collection;
import java.util.HashSet;
import java.util.LinkedHashSet;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
@ThreadSafe
public class ClasspathInferer {
private static final Logger LOGGER = LoggerFactory.getLogger(ClasspathInferer.class);
private final Lock lock = new ReentrantLock();
private final Map, Collection> classPathCache;
public ClasspathInferer() {
this.classPathCache = new MapMaker().weakKeys().makeMap();
}
public void getClassPathFor(Class> targetClass, Collection dest) {
lock.lock();
try {
Collection classPath = classPathCache.get(targetClass);
if (classPath == null) {
Set> visited = new HashSet>();
classPath = new LinkedHashSet();
find(targetClass, visited, classPath);
classPathCache.put(targetClass, classPath);
}
dest.addAll(classPath);
} finally {
lock.unlock();
}
}
/**
* Locates the classpath required by the given target class. Traverses the dependency graph of classes used by the specified class and collects the result in the given collection.
*/
private void find(Class> target, Collection> visited, Collection dest) {
ClassLoader targetClassLoader = target.getClassLoader();
if (targetClassLoader == null || targetClassLoader == ClassLoaderUtils.getPlatformClassLoader()) {
// A system class, skip it
return;
}
if (!visited.add(target)) {
// Already seen this class, skip it
return;
}
String resourceName = target.getName().replace('.', '/') + ".class";
URL resource = targetClassLoader.getResource(resourceName);
try {
if (resource == null) {
LOGGER.warn("Could not determine classpath for {}", target);
return;
}
File classPathRoot = ClasspathUtil.getClasspathForClass(target);
dest.add(classPathRoot.toURI().toURL());
// To determine the dependencies of the class, load up the byte code and look for CONSTANT_Class entries in the constant pool
ClassReader reader;
URLConnection urlConnection = resource.openConnection();
if (urlConnection instanceof JarURLConnection) {
// Using the caches for these connections leaves the Jar files open. Don't use the cache, so that the Jar file is closed when the stream is closed below
// There are other options for solving this that may be more performant. However a class is inspected this way once and the result reused, so this approach is probably fine
urlConnection.setUseCaches(false);
}
InputStream inputStream = urlConnection.getInputStream();
try {
reader = new ClassReader(ByteStreams.toByteArray(inputStream));
} finally {
inputStream.close();
}
char[] charBuffer = new char[reader.getMaxStringLength()];
for (int i = 1; i < reader.getItemCount(); i++) {
int itemOffset = reader.getItem(i);
if (itemOffset > 0 && reader.readByte(itemOffset - 1) == 7) {
// A CONSTANT_Class entry, read the class descriptor
String classDescriptor = reader.readUTF8(itemOffset, charBuffer);
Type type = Type.getObjectType(classDescriptor);
while (type.getSort() == Type.ARRAY) {
type = type.getElementType();
}
if (type.getSort() != Type.OBJECT) {
// A primitive type
continue;
}
String className = type.getClassName();
if (className.equals(target.getName())) {
// A reference to this class
continue;
}
Class> cl;
try {
cl = Class.forName(className, false, targetClassLoader);
} catch (ClassNotFoundException e) {
// This is fine, just ignore it
LOGGER.warn("Could not determine classpath for {}", target);
continue;
}
find(cl, visited, dest);
}
}
} catch (Exception e) {
throw new GradleException(String.format("Could not determine the class-path for %s.", target), e);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy