forked from External/mage
128 lines
5.4 KiB
Java
128 lines
5.4 KiB
Java
/*
|
|
* Copyright 2010 BetaSteward_at_googlemail.com. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without modification, are
|
|
* permitted provided that the following conditions are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright notice, this list of
|
|
* conditions and the following disclaimer.
|
|
*
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice, this list
|
|
* of conditions and the following disclaimer in the documentation and/or other materials
|
|
* provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY BetaSteward_at_googlemail.com ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
|
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
* FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BetaSteward_at_googlemail.com OR
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* The views and conclusions contained in the software and documentation are those of the
|
|
* authors and should not be interpreted as representing official policies, either expressed
|
|
* or implied, of BetaSteward_at_googlemail.com.
|
|
*/
|
|
package mage.util;
|
|
|
|
import java.io.File;
|
|
import java.io.FileInputStream;
|
|
import java.io.IOException;
|
|
import java.net.URL;
|
|
import java.net.URLDecoder;
|
|
import java.util.*;
|
|
import java.util.jar.JarEntry;
|
|
import java.util.jar.JarInputStream;
|
|
|
|
/**
|
|
*
|
|
* @author North
|
|
*/
|
|
public final class ClassScanner {
|
|
|
|
private static void checkClassForInclusion(List<Class> cards, Class type, String name, ClassLoader cl) {
|
|
try {
|
|
Class clazz = Class.forName(name, true, cl);
|
|
if (clazz.getEnclosingClass() == null && type.isAssignableFrom(clazz)) {
|
|
cards.add(clazz);
|
|
}
|
|
} catch (ClassNotFoundException ex) {
|
|
// ignored
|
|
}
|
|
}
|
|
|
|
public static List<Class> findClasses(ClassLoader classLoader, List<String> packages, Class<?> type) {
|
|
List<Class> cards = new ArrayList<>();
|
|
try {
|
|
if(classLoader == null) classLoader = Thread.currentThread().getContextClassLoader();
|
|
assert classLoader != null;
|
|
|
|
HashMap<String, String> dirs = new HashMap<>();
|
|
TreeSet<String> jars = new TreeSet<>();
|
|
for (String packageName : packages) {
|
|
String path = packageName.replace('.', '/');
|
|
Enumeration<URL> resources = classLoader.getResources(path);
|
|
|
|
while (resources.hasMoreElements()) {
|
|
URL resource = resources.nextElement();
|
|
String filePath = resource.getFile();
|
|
if (filePath.startsWith("file:")) {
|
|
filePath = filePath.substring("file:".length(), filePath.lastIndexOf('!'));
|
|
jars.add(filePath);
|
|
} else {
|
|
dirs.put(filePath, packageName);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (String filePath : dirs.keySet()) {
|
|
cards.addAll(findClasses(classLoader, new File(filePath), dirs.get(filePath), type));
|
|
}
|
|
|
|
for (String filePath : jars) {
|
|
File file = new File(URLDecoder.decode(filePath, "UTF-8"));
|
|
cards.addAll(findClassesInJar(classLoader, file, packages, type));
|
|
}
|
|
} catch (IOException ex) {
|
|
}
|
|
return cards;
|
|
}
|
|
|
|
private static List<Class> findClasses(ClassLoader classLoader, File directory, String packageName, Class<?> type) {
|
|
List<Class> cards = new ArrayList<>();
|
|
if (!directory.exists()) return cards;
|
|
|
|
for (File file : directory.listFiles()) {
|
|
if (file.getName().endsWith(".class")) {
|
|
String name = packageName + '.' + file.getName().substring(0, file.getName().length() - 6);
|
|
checkClassForInclusion(cards, type, name, classLoader);
|
|
}
|
|
}
|
|
return cards;
|
|
}
|
|
|
|
private static List<Class> findClassesInJar(ClassLoader classLoader, File file, List<String> packages, Class<?> type) {
|
|
List<Class> cards = new ArrayList<>();
|
|
if (!file.exists()) return cards;
|
|
|
|
|
|
try(JarInputStream jarFile = new JarInputStream(new FileInputStream(file))) {
|
|
while (true) {
|
|
JarEntry jarEntry = jarFile.getNextJarEntry();
|
|
if (jarEntry == null) {
|
|
break;
|
|
}
|
|
if (jarEntry.getName().endsWith(".class")) {
|
|
String className = jarEntry.getName().replace(".class", "").replace('/', '.');
|
|
int packageNameEnd = className.lastIndexOf('.');
|
|
String packageName = packageNameEnd != -1 ? className.substring(0, packageNameEnd) : "";
|
|
if (packages.contains(packageName)) checkClassForInclusion(cards, type, className, classLoader);
|
|
}
|
|
}
|
|
} catch (IOException ex) {
|
|
}
|
|
return cards;
|
|
}
|
|
}
|