mirror of
https://github.com/Minestom/Minestom.git
synced 2025-01-03 23:17:48 +01:00
Mixin modifications between extensions is now possible
This commit is contained in:
parent
47eb0084eb
commit
25cde2cde7
@ -129,6 +129,8 @@ final class DiscoveredExtension {
|
|||||||
MISSING_DEPENDENCIES("Missing dependencies, check your logs."),
|
MISSING_DEPENDENCIES("Missing dependencies, check your logs."),
|
||||||
INVALID_NAME("Invalid name."),
|
INVALID_NAME("Invalid name."),
|
||||||
NO_ENTRYPOINT("No entrypoint specified."),
|
NO_ENTRYPOINT("No entrypoint specified."),
|
||||||
|
FAILED_TO_SETUP_CLASSLOADER("Extension classloader could not be setup."),
|
||||||
|
LOAD_FAILED("Load failed. See logs for more information."),
|
||||||
;
|
;
|
||||||
|
|
||||||
private final String message;
|
private final String message;
|
||||||
|
@ -33,7 +33,7 @@ public class ExtensionManager {
|
|||||||
private final static String INDEV_RESOURCES_FOLDER = "minestom.extension.indevfolder.resources";
|
private final static String INDEV_RESOURCES_FOLDER = "minestom.extension.indevfolder.resources";
|
||||||
private final static Gson GSON = new Gson();
|
private final static Gson GSON = new Gson();
|
||||||
|
|
||||||
private final Map<String, URLClassLoader> extensionLoaders = new HashMap<>();
|
private final Map<String, MinestomExtensionClassLoader> extensionLoaders = new HashMap<>();
|
||||||
private final Map<String, Extension> extensions = new HashMap<>();
|
private final Map<String, Extension> extensions = new HashMap<>();
|
||||||
private final File extensionFolder = new File("extensions");
|
private final File extensionFolder = new File("extensions");
|
||||||
private final File dependenciesFolder = new File(extensionFolder, ".libs");
|
private final File dependenciesFolder = new File(extensionFolder, ".libs");
|
||||||
@ -66,19 +66,46 @@ public class ExtensionManager {
|
|||||||
List<DiscoveredExtension> discoveredExtensions = discoverExtensions();
|
List<DiscoveredExtension> discoveredExtensions = discoverExtensions();
|
||||||
discoveredExtensions = generateLoadOrder(discoveredExtensions);
|
discoveredExtensions = generateLoadOrder(discoveredExtensions);
|
||||||
loadDependencies(discoveredExtensions);
|
loadDependencies(discoveredExtensions);
|
||||||
|
// remove invalid extensions
|
||||||
|
discoveredExtensions.removeIf(ext -> ext.loadStatus != DiscoveredExtension.LoadStatus.LOAD_SUCCESS);
|
||||||
|
|
||||||
|
for(DiscoveredExtension discoveredExtension : discoveredExtensions) {
|
||||||
|
try {
|
||||||
|
setupClassLoader(discoveredExtension);
|
||||||
|
} catch (Exception e) {
|
||||||
|
discoveredExtension.loadStatus = DiscoveredExtension.LoadStatus.FAILED_TO_SETUP_CLASSLOADER;
|
||||||
|
e.printStackTrace();
|
||||||
|
log.error("Failed to load extension {}", discoveredExtension.getName());
|
||||||
|
log.error("Failed to load extension", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// remove invalid extensions
|
// remove invalid extensions
|
||||||
discoveredExtensions.removeIf(ext -> ext.loadStatus != DiscoveredExtension.LoadStatus.LOAD_SUCCESS);
|
discoveredExtensions.removeIf(ext -> ext.loadStatus != DiscoveredExtension.LoadStatus.LOAD_SUCCESS);
|
||||||
setupCodeModifiers(discoveredExtensions);
|
setupCodeModifiers(discoveredExtensions);
|
||||||
|
|
||||||
for (DiscoveredExtension discoveredExtension : discoveredExtensions) {
|
for (DiscoveredExtension discoveredExtension : discoveredExtensions) {
|
||||||
URLClassLoader loader;
|
try {
|
||||||
URL[] urls = discoveredExtension.files.toArray(new URL[0]);
|
attemptSingleLoad(discoveredExtension);
|
||||||
// TODO: Only putting each extension into its own classloader prevents code modifications (via code modifiers or mixins)
|
} catch (Exception e) {
|
||||||
// TODO: If we want modifications to be possible, we need to add these urls to the current classloader
|
discoveredExtension.loadStatus = DiscoveredExtension.LoadStatus.LOAD_FAILED;
|
||||||
// TODO: Indeed, without adding the urls, the classloader is not able to load the bytecode of extension classes
|
e.printStackTrace();
|
||||||
// TODO: Whether we want to allow extensions to modify one-another is our choice now.
|
log.error("Failed to load extension {}", discoveredExtension.getName());
|
||||||
loader = newClassLoader(discoveredExtension, urls);
|
log.error("Failed to load extension", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
extensionList = Collections.unmodifiableList(extensionList);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setupClassLoader(DiscoveredExtension discoveredExtension) {
|
||||||
|
String extensionName = discoveredExtension.getName();
|
||||||
|
MinestomExtensionClassLoader loader;
|
||||||
|
URL[] urls = discoveredExtension.files.toArray(new URL[0]);
|
||||||
|
loader = newClassLoader(discoveredExtension, urls);
|
||||||
|
extensionLoaders.put(extensionName.toLowerCase(), loader);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void attemptSingleLoad(DiscoveredExtension discoveredExtension) {
|
||||||
// Create ExtensionDescription (authors, version etc.)
|
// Create ExtensionDescription (authors, version etc.)
|
||||||
String extensionName = discoveredExtension.getName();
|
String extensionName = discoveredExtension.getName();
|
||||||
String mainClass = discoveredExtension.getEntrypoint();
|
String mainClass = discoveredExtension.getEntrypoint();
|
||||||
@ -88,11 +115,11 @@ public class ExtensionManager {
|
|||||||
Arrays.asList(discoveredExtension.getAuthors())
|
Arrays.asList(discoveredExtension.getAuthors())
|
||||||
);
|
);
|
||||||
|
|
||||||
extensionLoaders.put(extensionName.toLowerCase(), loader);
|
MinestomExtensionClassLoader loader = extensionLoaders.get(extensionName.toLowerCase());
|
||||||
|
|
||||||
if (extensions.containsKey(extensionName.toLowerCase())) {
|
if (extensions.containsKey(extensionName.toLowerCase())) {
|
||||||
log.error("An extension called '{}' has already been registered.", extensionName);
|
log.error("An extension called '{}' has already been registered.", extensionName);
|
||||||
continue;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Class<?> jarClass;
|
Class<?> jarClass;
|
||||||
@ -100,7 +127,7 @@ public class ExtensionManager {
|
|||||||
jarClass = Class.forName(mainClass, true, loader);
|
jarClass = Class.forName(mainClass, true, loader);
|
||||||
} catch (ClassNotFoundException e) {
|
} catch (ClassNotFoundException e) {
|
||||||
log.error("Could not find main class '{}' in extension '{}'.", mainClass, extensionName, e);
|
log.error("Could not find main class '{}' in extension '{}'.", mainClass, extensionName, e);
|
||||||
continue;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Class<? extends Extension> extensionClass;
|
Class<? extends Extension> extensionClass;
|
||||||
@ -108,7 +135,7 @@ public class ExtensionManager {
|
|||||||
extensionClass = jarClass.asSubclass(Extension.class);
|
extensionClass = jarClass.asSubclass(Extension.class);
|
||||||
} catch (ClassCastException e) {
|
} catch (ClassCastException e) {
|
||||||
log.error("Main class '{}' in '{}' does not extend the 'Extension' superclass.", mainClass, extensionName, e);
|
log.error("Main class '{}' in '{}' does not extend the 'Extension' superclass.", mainClass, extensionName, e);
|
||||||
continue;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Constructor<? extends Extension> constructor;
|
Constructor<? extends Extension> constructor;
|
||||||
@ -118,14 +145,14 @@ public class ExtensionManager {
|
|||||||
constructor.setAccessible(true);
|
constructor.setAccessible(true);
|
||||||
} catch (NoSuchMethodException e) {
|
} catch (NoSuchMethodException e) {
|
||||||
log.error("Main class '{}' in '{}' does not define a no-args constructor.", mainClass, extensionName, e);
|
log.error("Main class '{}' in '{}' does not define a no-args constructor.", mainClass, extensionName, e);
|
||||||
continue;
|
return;
|
||||||
}
|
}
|
||||||
Extension extension = null;
|
Extension extension = null;
|
||||||
try {
|
try {
|
||||||
extension = constructor.newInstance();
|
extension = constructor.newInstance();
|
||||||
} catch (InstantiationException e) {
|
} catch (InstantiationException e) {
|
||||||
log.error("Main class '{}' in '{}' cannot be an abstract class.", mainClass, extensionName, e);
|
log.error("Main class '{}' in '{}' cannot be an abstract class.", mainClass, extensionName, e);
|
||||||
continue;
|
return;
|
||||||
} catch (IllegalAccessException ignored) {
|
} catch (IllegalAccessException ignored) {
|
||||||
// We made it accessible, should not occur
|
// We made it accessible, should not occur
|
||||||
} catch (InvocationTargetException e) {
|
} catch (InvocationTargetException e) {
|
||||||
@ -135,7 +162,7 @@ public class ExtensionManager {
|
|||||||
extensionName,
|
extensionName,
|
||||||
e.getTargetException()
|
e.getTargetException()
|
||||||
);
|
);
|
||||||
continue;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set extension description
|
// Set extension description
|
||||||
@ -147,7 +174,7 @@ public class ExtensionManager {
|
|||||||
// We made it accessible, should not occur
|
// We made it accessible, should not occur
|
||||||
} catch (NoSuchFieldException e) {
|
} catch (NoSuchFieldException e) {
|
||||||
log.error("Main class '{}' in '{}' has no description field.", mainClass, extensionName, e);
|
log.error("Main class '{}' in '{}' has no description field.", mainClass, extensionName, e);
|
||||||
continue;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set logger
|
// Set logger
|
||||||
@ -166,8 +193,6 @@ public class ExtensionManager {
|
|||||||
extensionList.add(extension); // add to a list, as lists preserve order
|
extensionList.add(extension); // add to a list, as lists preserve order
|
||||||
extensions.put(extensionName.toLowerCase(), extension);
|
extensions.put(extensionName.toLowerCase(), extension);
|
||||||
}
|
}
|
||||||
extensionList = Collections.unmodifiableList(extensionList);
|
|
||||||
}
|
|
||||||
|
|
||||||
@NotNull
|
@NotNull
|
||||||
private List<DiscoveredExtension> discoverExtensions() {
|
private List<DiscoveredExtension> discoverExtensions() {
|
||||||
@ -334,42 +359,41 @@ public class ExtensionManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: remove if extensions cannot modify one-another
|
|
||||||
// TODO: use if they can
|
|
||||||
private void injectIntoClasspath(URL dependency, DiscoveredExtension extension) {
|
|
||||||
final ClassLoader cl = getClass().getClassLoader();
|
|
||||||
if (!(cl instanceof URLClassLoader)) {
|
|
||||||
throw new IllegalStateException("Current class loader is not a URLClassLoader, but " + cl + ". This prevents adding URLs into the classpath at runtime.");
|
|
||||||
}
|
|
||||||
if(cl instanceof MinestomRootClassLoader) {
|
|
||||||
((MinestomRootClassLoader) cl).addURL(dependency); // no reflection warnings for us!
|
|
||||||
} else {
|
|
||||||
try {
|
|
||||||
Method addURL = URLClassLoader.class.getDeclaredMethod("addURL", URL.class);
|
|
||||||
addURL.setAccessible(true);
|
|
||||||
addURL.invoke(cl, dependency);
|
|
||||||
} catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) {
|
|
||||||
throw new RuntimeException("Failed to inject URL " + dependency + " into classpath. From extension " + extension.getName(), e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void addDependencyFile(URL dependency, DiscoveredExtension extension) {
|
private void addDependencyFile(URL dependency, DiscoveredExtension extension) {
|
||||||
extension.files.add(dependency);
|
extension.files.add(dependency);
|
||||||
log.trace("Added dependency {} to extension {} classpath", dependency.toExternalForm(), extension.getName());
|
log.trace("Added dependency {} to extension {} classpath", dependency.toExternalForm(), extension.getName());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Loads a URL into the classpath.
|
* Creates a new class loader for the given extension.
|
||||||
|
* Will add the new loader as a child of all its dependencies' loaders.
|
||||||
*
|
*
|
||||||
* @param urls {@link URL} (usually a JAR) that should be loaded.
|
* @param urls {@link URL} (usually a JAR) that should be loaded.
|
||||||
*/
|
*/
|
||||||
@NotNull
|
@NotNull
|
||||||
public URLClassLoader newClassLoader(@NotNull DiscoveredExtension extension, @NotNull URL[] urls) {
|
public MinestomExtensionClassLoader newClassLoader(@NotNull DiscoveredExtension extension, @NotNull URL[] urls) {
|
||||||
MinestomRootClassLoader root = MinestomRootClassLoader.getInstance();
|
MinestomRootClassLoader root = MinestomRootClassLoader.getInstance();
|
||||||
MinestomExtensionClassLoader loader = new MinestomExtensionClassLoader(extension.getName(), urls, root);
|
MinestomExtensionClassLoader loader = new MinestomExtensionClassLoader(extension.getName(), urls, root);
|
||||||
// TODO: tree structure
|
if(extension.getDependencies().length == 0) {
|
||||||
|
// orphaned extension, we can insert it directly
|
||||||
root.addChild(loader);
|
root.addChild(loader);
|
||||||
|
} else {
|
||||||
|
// we need to keep track that it has actually been inserted
|
||||||
|
// even though it should always be (due to the order in which extensions are loaders), it is an additional layer of """security"""
|
||||||
|
boolean foundOne = false;
|
||||||
|
for(String dependency : extension.getDependencies()) {
|
||||||
|
if(extensionLoaders.containsKey(dependency.toLowerCase())) {
|
||||||
|
MinestomExtensionClassLoader parentLoader = extensionLoaders.get(dependency.toLowerCase());
|
||||||
|
parentLoader.addChild(loader);
|
||||||
|
foundOne = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!foundOne) {
|
||||||
|
log.error("Could not load extension {}, could not find any parent inside classloader hierarchy.", extension.getName());
|
||||||
|
throw new RuntimeException("Could not load extension "+extension.getName()+", could not find any parent inside classloader hierarchy.");
|
||||||
|
}
|
||||||
|
}
|
||||||
return loader;
|
return loader;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -0,0 +1,36 @@
|
|||||||
|
package net.minestom.server.extras.selfmodification;
|
||||||
|
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.net.URLClassLoader;
|
||||||
|
import java.util.LinkedList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Classloader part of a hierarchy of classloader
|
||||||
|
*/
|
||||||
|
public abstract class HierarchyClassLoader extends URLClassLoader {
|
||||||
|
protected final List<MinestomExtensionClassLoader> children = new LinkedList<>();
|
||||||
|
|
||||||
|
public HierarchyClassLoader(String name, URL[] urls, ClassLoader parent) {
|
||||||
|
super(name, urls, parent);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void addChild(@NotNull MinestomExtensionClassLoader loader) {
|
||||||
|
children.add(loader);
|
||||||
|
}
|
||||||
|
|
||||||
|
public InputStream getResourceAsStreamWithChildren(String name) {
|
||||||
|
InputStream in = getResourceAsStream(name);
|
||||||
|
if(in != null) return in;
|
||||||
|
|
||||||
|
for(MinestomExtensionClassLoader child : children) {
|
||||||
|
InputStream childInput = child.getResourceAsStreamWithChildren(name);
|
||||||
|
if(childInput != null)
|
||||||
|
return childInput;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
@ -3,16 +3,12 @@ package net.minestom.server.extras.selfmodification;
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.net.URLClassLoader;
|
|
||||||
import java.util.LinkedList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
public class MinestomExtensionClassLoader extends URLClassLoader {
|
public class MinestomExtensionClassLoader extends HierarchyClassLoader {
|
||||||
/**
|
/**
|
||||||
* Root ClassLoader, everything goes through it before any attempt at loading is done inside this classloader
|
* Root ClassLoader, everything goes through it before any attempt at loading is done inside this classloader
|
||||||
*/
|
*/
|
||||||
private final MinestomRootClassLoader root;
|
private final MinestomRootClassLoader root;
|
||||||
private final List<MinestomExtensionClassLoader> children = new LinkedList<>();
|
|
||||||
|
|
||||||
public MinestomExtensionClassLoader(String name, URL[] urls, MinestomRootClassLoader root) {
|
public MinestomExtensionClassLoader(String name, URL[] urls, MinestomRootClassLoader root) {
|
||||||
super(name, urls, root);
|
super(name, urls, root);
|
||||||
@ -36,19 +32,12 @@ public class MinestomExtensionClassLoader extends URLClassLoader {
|
|||||||
* @throws ClassNotFoundException if the class is not found inside this classloader
|
* @throws ClassNotFoundException if the class is not found inside this classloader
|
||||||
*/
|
*/
|
||||||
public Class<?> loadClassAsChild(String name, boolean resolve) throws ClassNotFoundException {
|
public Class<?> loadClassAsChild(String name, boolean resolve) throws ClassNotFoundException {
|
||||||
for(MinestomExtensionClassLoader child : children) {
|
|
||||||
try {
|
|
||||||
Class<?> loaded = child.loadClassAsChild(name, resolve);
|
|
||||||
return loaded;
|
|
||||||
} catch (ClassNotFoundException e) {
|
|
||||||
// move on to next child
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Class<?> loadedClass = findLoadedClass(name);
|
Class<?> loadedClass = findLoadedClass(name);
|
||||||
if(loadedClass != null) {
|
if(loadedClass != null) {
|
||||||
return loadedClass;
|
return loadedClass;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
// not in children, attempt load in this classloader
|
// not in children, attempt load in this classloader
|
||||||
String path = name.replace(".", "/") + ".class";
|
String path = name.replace(".", "/") + ".class";
|
||||||
InputStream in = getResourceAsStream(path);
|
InputStream in = getResourceAsStream(path);
|
||||||
@ -66,5 +55,16 @@ public class MinestomExtensionClassLoader extends URLClassLoader {
|
|||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new ClassNotFoundException("Could not load class " + name, e);
|
throw new ClassNotFoundException("Could not load class " + name, e);
|
||||||
}
|
}
|
||||||
|
} catch (ClassNotFoundException e) {
|
||||||
|
for(MinestomExtensionClassLoader child : children) {
|
||||||
|
try {
|
||||||
|
Class<?> loaded = child.loadClassAsChild(name, resolve);
|
||||||
|
return loaded;
|
||||||
|
} catch (ClassNotFoundException e1) {
|
||||||
|
// move on to next child
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -22,7 +22,7 @@ import java.util.Set;
|
|||||||
* Class Loader that can modify class bytecode when they are loaded
|
* Class Loader that can modify class bytecode when they are loaded
|
||||||
*/
|
*/
|
||||||
@Slf4j
|
@Slf4j
|
||||||
public class MinestomRootClassLoader extends URLClassLoader {
|
public class MinestomRootClassLoader extends HierarchyClassLoader {
|
||||||
|
|
||||||
private static MinestomRootClassLoader INSTANCE;
|
private static MinestomRootClassLoader INSTANCE;
|
||||||
|
|
||||||
@ -60,7 +60,6 @@ public class MinestomRootClassLoader extends URLClassLoader {
|
|||||||
|
|
||||||
// TODO: priorities?
|
// TODO: priorities?
|
||||||
private final List<CodeModifier> modifiers = new LinkedList<>();
|
private final List<CodeModifier> modifiers = new LinkedList<>();
|
||||||
private final List<MinestomExtensionClassLoader> children = new LinkedList<>();
|
|
||||||
|
|
||||||
private MinestomRootClassLoader(ClassLoader parent) {
|
private MinestomRootClassLoader(ClassLoader parent) {
|
||||||
super("Minestom Root ClassLoader", extractURLsFromClasspath(), parent);
|
super("Minestom Root ClassLoader", extractURLsFromClasspath(), parent);
|
||||||
@ -187,6 +186,21 @@ public class MinestomRootClassLoader extends URLClassLoader {
|
|||||||
return originalBytes;
|
return originalBytes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public byte[] loadBytesWithChildren(String name, boolean transform) throws IOException, ClassNotFoundException {
|
||||||
|
if (name == null)
|
||||||
|
throw new ClassNotFoundException();
|
||||||
|
String path = name.replace(".", "/") + ".class";
|
||||||
|
InputStream input = getResourceAsStreamWithChildren(path);
|
||||||
|
if(input == null) {
|
||||||
|
throw new ClassNotFoundException("Could not find resource "+path);
|
||||||
|
}
|
||||||
|
byte[] originalBytes = input.readAllBytes();
|
||||||
|
if(transform) {
|
||||||
|
return transformBytes(originalBytes, name);
|
||||||
|
}
|
||||||
|
return originalBytes;
|
||||||
|
}
|
||||||
|
|
||||||
byte[] transformBytes(byte[] classBytecode, String name) {
|
byte[] transformBytes(byte[] classBytecode, String name) {
|
||||||
if (!isProtected(name)) {
|
if (!isProtected(name)) {
|
||||||
ClassReader reader = new ClassReader(classBytecode);
|
ClassReader reader = new ClassReader(classBytecode);
|
||||||
@ -261,8 +275,4 @@ public class MinestomRootClassLoader extends URLClassLoader {
|
|||||||
public List<CodeModifier> getModifiers() {
|
public List<CodeModifier> getModifiers() {
|
||||||
return modifiers;
|
return modifiers;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void addChild(MinestomExtensionClassLoader loader) {
|
|
||||||
children.add(loader);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -26,7 +26,7 @@ public class MinestomBytecodeProvider implements IClassBytecodeProvider {
|
|||||||
ClassNode node = new ClassNode();
|
ClassNode node = new ClassNode();
|
||||||
ClassReader reader;
|
ClassReader reader;
|
||||||
try {
|
try {
|
||||||
reader = new ClassReader(classLoader.loadBytes(name, transform));
|
reader = new ClassReader(classLoader.loadBytesWithChildren(name, transform));
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new ClassNotFoundException("Could not load ClassNode with name " + name, e);
|
throw new ClassNotFoundException("Could not load ClassNode with name " + name, e);
|
||||||
}
|
}
|
||||||
|
@ -65,7 +65,7 @@ public class MixinServiceMinestom extends MixinServiceAbstract {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public InputStream getResourceAsStream(String name) {
|
public InputStream getResourceAsStream(String name) {
|
||||||
return classLoader.getResourceAsStream(name);
|
return classLoader.getResourceAsStreamWithChildren(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
Loading…
Reference in New Issue
Block a user