Fix dependency loading, use a ClassLoader instance which shared among all the plugins

This commit is contained in:
ceze88 2024-02-07 14:46:51 +01:00
parent 66a2abda32
commit cc59a825be

View File

@ -2,6 +2,7 @@ package com.craftaro.core.dependency;
import com.craftaro.core.CraftaroCoreConstants; import com.craftaro.core.CraftaroCoreConstants;
import com.craftaro.core.SongodaCore; import com.craftaro.core.SongodaCore;
import com.georgev22.api.libraryloader.ClassLoaderAccess;
import com.georgev22.api.libraryloader.LibraryLoader; import com.georgev22.api.libraryloader.LibraryLoader;
import com.georgev22.api.libraryloader.exceptions.InvalidDependencyException; import com.georgev22.api.libraryloader.exceptions.InvalidDependencyException;
import com.georgev22.api.libraryloader.exceptions.UnknownDependencyException; import com.georgev22.api.libraryloader.exceptions.UnknownDependencyException;
@ -12,7 +13,10 @@ import org.bukkit.plugin.Plugin;
import java.io.File; import java.io.File;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.net.URL; import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.Files; import java.nio.file.Files;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collection; import java.util.Collection;
@ -25,13 +29,16 @@ public class DependencyLoader {
private static final int DEPENDENCY_VERSION = 1; private static final int DEPENDENCY_VERSION = 1;
private final LibraryLoader libraryLoader; private final LibraryLoader libraryLoader;
private final ClassLoaderAccess parentClassLoaderAccess;
public DependencyLoader(Plugin plugin) { public DependencyLoader(Plugin plugin) {
//Bind loaded dependencies to the plugin's parent class loader so classes could be accessed across plugins
URLClassLoader parentClassLoader = (URLClassLoader) plugin.getClass().getClassLoader().getParent();
this.libraryLoader = new LibraryLoader( this.libraryLoader = new LibraryLoader(
DependencyLoader.class.getClassLoader(), parentClassLoader,
new File(plugin.getDataFolder().getParentFile(), CraftaroCoreConstants.getProjectName() + "/dependencies/v" + DEPENDENCY_VERSION), new File(plugin.getDataFolder().getParentFile(), CraftaroCoreConstants.getProjectName() + "/dependencies/v" + DEPENDENCY_VERSION),
SongodaCore.getLogger() SongodaCore.getLogger()
); );
this.parentClassLoaderAccess = new ClassLoaderAccess(parentClassLoader);
} }
public void loadDependencies(Collection<Dependency> dependencies) throws IOException { public void loadDependencies(Collection<Dependency> dependencies) throws IOException {
@ -45,7 +52,7 @@ public class DependencyLoader {
File outputFile = new File(this.libraryLoader.getLibFolder(), dependency.getGroupId().replace(".", File.separator) + File.separator + dependency.getArtifactId().replace(".", File.separator) + File.separator + dependency.getVersion() + File.separator + "raw-" + name + ".jar"); File outputFile = new File(this.libraryLoader.getLibFolder(), dependency.getGroupId().replace(".", File.separator) + File.separator + dependency.getArtifactId().replace(".", File.separator) + File.separator + dependency.getVersion() + File.separator + "raw-" + name + ".jar");
File relocatedFile = new File(outputFile.getParentFile(), name.replace("raw-", "") + ".jar"); File relocatedFile = new File(outputFile.getParentFile(), name.replace("raw-", "") + ".jar");
if (relocatedFile.exists()) { if (relocatedFile.exists()) {
if (isLoaded(relocatedFile)) { if (isJarLoaded(relocatedFile)) {
return; return;
} }
@ -94,11 +101,11 @@ public class DependencyLoader {
} }
try { try {
this.libraryLoader.load(new LibraryLoader.Dependency(dependency.getGroupId(), dependency.getArtifactId(), dependency.getVersion(), dependency.getRepositoryUrl()), true); //Do not check path here, it uses the original non relocated paths. Use isJarLoaded instead
} catch (InvalidDependencyException ignored) { this.libraryLoader.load(new LibraryLoader.Dependency(dependency.getGroupId(), dependency.getArtifactId(), dependency.getVersion(), dependency.getRepositoryUrl()), false);
// Already loaded } catch (Exception e) {
} catch (UnknownDependencyException ex) { // Something went wrong
throw new RuntimeException(ex); e.printStackTrace();
} }
SongodaCore.getLogger().info("----------------------------"); SongodaCore.getLogger().info("----------------------------");
} }
@ -120,31 +127,13 @@ public class DependencyLoader {
/** /**
* Finds the first .class file in the jar and check if it's loaded * Finds the first .class file in the jar and check if it's loaded
*/ */
private boolean isLoaded(File jarFile) throws IOException { private boolean isJarLoaded(File jarFile) throws IOException {
try (ZipFile zipFile = new ZipFile(jarFile)) { URL jarFileURL = jarFile.toURI().toURL();
Enumeration<? extends ZipEntry> zipEntries = zipFile.entries(); try {
while (zipEntries.hasMoreElements()) { return this.parentClassLoaderAccess.getPathURLs().stream().anyMatch(url -> url.getFile().equals(jarFileURL.getFile()));
ZipEntry zipEntry = zipEntries.nextElement(); } catch (Exception e) {
if (zipEntry.getName().startsWith("META-INF")) { throw new RuntimeException(e);
continue;
}
if (!zipEntry.getName().endsWith(".class")) {
continue;
}
String className = zipEntry.getName()
.substring(0, zipEntry.getName().length() - ".class".length())
.replace("/", ".");
try {
Class.forName(className);
return true;
} catch (Throwable th) {
return false;
}
}
} }
return false;
} }
public static int getDependencyVersion() { public static int getDependencyVersion() {