mirror of
https://github.com/Minestom/Minestom.git
synced 2025-01-19 06:32:03 +01:00
Cleanup and improve extension changes.
This commit is contained in:
parent
535e8946b6
commit
5217964259
@ -2,13 +2,114 @@ package net.minestom.server.extensions;
|
|||||||
|
|
||||||
import lombok.extern.slf4j.Slf4j;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
import org.jetbrains.annotations.Nullable;
|
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
|
|
||||||
@Slf4j
|
@Slf4j(topic = "minestom-extensions")
|
||||||
class DiscoveredExtension {
|
final class DiscoveredExtension {
|
||||||
private static String NAME_REGEX = "[A-Za-z][_A-Za-z0-9]+";
|
public static final String NAME_REGEX = "[A-Za-z][_A-Za-z0-9]+";
|
||||||
|
private String name;
|
||||||
|
private String entrypoint;
|
||||||
|
private String version;
|
||||||
|
private String mixinConfig;
|
||||||
|
private String[] authors;
|
||||||
|
private String[] codeModifiers;
|
||||||
|
private Dependencies dependencies;
|
||||||
|
transient File[] files = new File[0];
|
||||||
|
transient LoadStatus loadStatus = LoadStatus.LOAD_SUCCESS;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
public String[] getCodeModifiers() {
|
||||||
|
if (codeModifiers == null) {
|
||||||
|
codeModifiers = new String[0];
|
||||||
|
}
|
||||||
|
return codeModifiers;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
public String getMixinConfig() {
|
||||||
|
return mixinConfig;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
public String[] getAuthors() {
|
||||||
|
return authors;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
public String getVersion() {
|
||||||
|
return version;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
public String getEntrypoint() {
|
||||||
|
return entrypoint;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
public Dependencies getDependencies() {
|
||||||
|
return dependencies;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void verifyIntegrity(@NotNull DiscoveredExtension extension) {
|
||||||
|
if (extension.name == null) {
|
||||||
|
StringBuilder fileList = new StringBuilder();
|
||||||
|
for (File f : extension.files) {
|
||||||
|
fileList.append(f.getAbsolutePath()).append(", ");
|
||||||
|
}
|
||||||
|
log.error("Extension with no name. (at {}})", fileList);
|
||||||
|
log.error("Extension at ({}) will not be loaded.", fileList);
|
||||||
|
extension.loadStatus = DiscoveredExtension.LoadStatus.INVALID_NAME;
|
||||||
|
|
||||||
|
// To ensure @NotNull: name = INVALID_NAME
|
||||||
|
extension.name = extension.loadStatus.name();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!extension.name.matches(NAME_REGEX)) {
|
||||||
|
log.error("Extension '{}' specified an invalid name.", extension.name);
|
||||||
|
log.error("Extension '{}' will not be loaded.", extension.name);
|
||||||
|
extension.loadStatus = DiscoveredExtension.LoadStatus.INVALID_NAME;
|
||||||
|
|
||||||
|
// To ensure @NotNull: name = INVALID_NAME
|
||||||
|
extension.name = extension.loadStatus.name();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (extension.entrypoint == null) {
|
||||||
|
log.error("Extension '{}' did not specify an entry point (via 'entrypoint').", extension.name);
|
||||||
|
log.error("Extension '{}' will not be loaded.", extension.name);
|
||||||
|
extension.loadStatus = DiscoveredExtension.LoadStatus.NO_ENTRYPOINT;
|
||||||
|
|
||||||
|
// To ensure @NotNull: entrypoint = NO_ENTRYPOINT
|
||||||
|
extension.entrypoint = extension.loadStatus.name();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Handle defaults
|
||||||
|
// If we reach this code, then the extension will most likely be loaded:
|
||||||
|
if (extension.version == null) {
|
||||||
|
log.warn("Extension '{}' did not specify a version.", extension.name);
|
||||||
|
log.warn("Extension '{}' will continue to load but should specify a plugin version.", extension.name);
|
||||||
|
extension.version = "Unspecified";
|
||||||
|
}
|
||||||
|
if (extension.mixinConfig == null) {
|
||||||
|
extension.mixinConfig = "";
|
||||||
|
}
|
||||||
|
if (extension.authors == null) {
|
||||||
|
extension.authors = new String[0];
|
||||||
|
}
|
||||||
|
if (extension.codeModifiers == null) {
|
||||||
|
extension.codeModifiers = new String[0];
|
||||||
|
}
|
||||||
|
// No dependencies were specified
|
||||||
|
if (extension.dependencies == null) {
|
||||||
|
extension.dependencies = new Dependencies();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
enum LoadStatus {
|
enum LoadStatus {
|
||||||
LOAD_SUCCESS("Actually, it did not fail. This message should not have been printed."),
|
LOAD_SUCCESS("Actually, it did not fail. This message should not have been printed."),
|
||||||
@ -19,102 +120,23 @@ class DiscoveredExtension {
|
|||||||
|
|
||||||
private final String message;
|
private final String message;
|
||||||
|
|
||||||
LoadStatus(String message) {
|
LoadStatus(@NotNull String message) {
|
||||||
this.message = message;
|
this.message = message;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
public String getMessage() {
|
public String getMessage() {
|
||||||
return message;
|
return message;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static class Dependencies {
|
static final class Dependencies {
|
||||||
|
Repository[] repositories = new Repository[0];
|
||||||
|
String[] artifacts = new String[0];
|
||||||
|
|
||||||
static class Repository {
|
static class Repository {
|
||||||
String name;
|
String name = "";
|
||||||
String url;
|
String url = "";
|
||||||
}
|
}
|
||||||
|
|
||||||
Repository[] repositories;
|
|
||||||
String[] artifacts;
|
|
||||||
}
|
|
||||||
|
|
||||||
transient File[] files = new File[0];
|
|
||||||
transient LoadStatus loadStatus = LoadStatus.LOAD_SUCCESS;
|
|
||||||
|
|
||||||
private String[] codeModifiers;
|
|
||||||
private String[] authors;
|
|
||||||
private String mixinConfig;
|
|
||||||
private String name;
|
|
||||||
private String version;
|
|
||||||
private String entrypoint;
|
|
||||||
private Dependencies dependencies;
|
|
||||||
|
|
||||||
void checkIntegrity() {
|
|
||||||
if(name == null) {
|
|
||||||
StringBuilder fileList = new StringBuilder();
|
|
||||||
for(File f : files) {
|
|
||||||
fileList.append(f.getAbsolutePath()).append(", ");
|
|
||||||
}
|
|
||||||
log.error("Extension with no name. (at {}})", fileList);
|
|
||||||
log.error("Extension at ({}) will not be loaded.", fileList);
|
|
||||||
loadStatus = LoadStatus.INVALID_NAME;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(!name.matches(NAME_REGEX)) {
|
|
||||||
log.error("Extension '{}' specified an invalid name.", name);
|
|
||||||
log.error("Extension '{}' will not be loaded.", name);
|
|
||||||
loadStatus = LoadStatus.INVALID_NAME;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(entrypoint == null) {
|
|
||||||
log.error("Extension '{}' did not specify an entry point (via 'entrypoint').", name);
|
|
||||||
log.error("Extension '{}' will not be loaded.", name);
|
|
||||||
loadStatus = LoadStatus.NO_ENTRYPOINT;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(codeModifiers == null) {
|
|
||||||
codeModifiers = new String[0];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@NotNull
|
|
||||||
public String getName() {
|
|
||||||
if(name == null) {
|
|
||||||
throw new IllegalStateException("Missing extension name");
|
|
||||||
}
|
|
||||||
return name;
|
|
||||||
}
|
|
||||||
|
|
||||||
@NotNull
|
|
||||||
public String[] getCodeModifiers() {
|
|
||||||
if(codeModifiers == null) {
|
|
||||||
codeModifiers = new String[0];
|
|
||||||
}
|
|
||||||
return codeModifiers;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public String getMixinConfig() {
|
|
||||||
return mixinConfig;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public String[] getAuthors() {
|
|
||||||
return authors;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public String getVersion() {
|
|
||||||
return version;
|
|
||||||
}
|
|
||||||
|
|
||||||
@NotNull
|
|
||||||
public String getEntrypoint() {
|
|
||||||
return entrypoint;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public Dependencies getDependencies() {
|
|
||||||
return dependencies;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,14 +1,16 @@
|
|||||||
package net.minestom.server.extensions;
|
package net.minestom.server.extensions;
|
||||||
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
import org.jetbrains.annotations.Nullable;
|
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public abstract class Extension {
|
public abstract class Extension {
|
||||||
|
// Set by reflection
|
||||||
|
@SuppressWarnings("unused")
|
||||||
private ExtensionDescription description;
|
private ExtensionDescription description;
|
||||||
|
// Set by reflection
|
||||||
|
@SuppressWarnings("unused")
|
||||||
private Logger logger;
|
private Logger logger;
|
||||||
|
|
||||||
protected Extension() {
|
protected Extension() {
|
||||||
@ -35,23 +37,25 @@ public abstract class Extension {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
public ExtensionDescription getDescription() {
|
public ExtensionDescription getDescription() {
|
||||||
return description;
|
return description;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
protected Logger getLogger() {
|
protected Logger getLogger() {
|
||||||
return logger;
|
return logger;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected static class ExtensionDescription {
|
public static class ExtensionDescription {
|
||||||
private final String name;
|
private final String name;
|
||||||
private final String version;
|
private final String version;
|
||||||
private final List<String> authors;
|
private final List<String> authors;
|
||||||
|
|
||||||
protected ExtensionDescription(@NotNull String name, @NotNull String version, @Nullable List<String> authors) {
|
ExtensionDescription(@NotNull String name, @NotNull String version, @NotNull List<String> authors) {
|
||||||
this.name = name;
|
this.name = name;
|
||||||
this.version = version;
|
this.version = version;
|
||||||
this.authors = authors != null ? authors : new ArrayList<>();
|
this.authors = authors;
|
||||||
}
|
}
|
||||||
|
|
||||||
@NotNull
|
@NotNull
|
||||||
|
@ -22,10 +22,15 @@ import java.lang.reflect.Method;
|
|||||||
import java.net.MalformedURLException;
|
import java.net.MalformedURLException;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.net.URLClassLoader;
|
import java.net.URLClassLoader;
|
||||||
import java.util.*;
|
import java.util.ArrayList;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.LinkedList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
import java.util.zip.ZipFile;
|
import java.util.zip.ZipFile;
|
||||||
|
|
||||||
@Slf4j
|
@Slf4j(topic = "Minestom-Extensions")
|
||||||
public class ExtensionManager {
|
public class ExtensionManager {
|
||||||
|
|
||||||
private final static String INDEV_CLASSES_FOLDER = "minestom.extension.indevfolder.classes";
|
private final static String INDEV_CLASSES_FOLDER = "minestom.extension.indevfolder.classes";
|
||||||
@ -78,28 +83,14 @@ public class ExtensionManager {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get 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();
|
||||||
Extension.ExtensionDescription extensionDescription;
|
Extension.ExtensionDescription extensionDescription = new Extension.ExtensionDescription(
|
||||||
{
|
extensionName,
|
||||||
String version;
|
discoveredExtension.getVersion(),
|
||||||
if (discoveredExtension.getVersion() == null) {
|
Arrays.asList(discoveredExtension.getAuthors())
|
||||||
log.warn("Extension '{}' did not specify a version.", extensionName);
|
);
|
||||||
log.warn("Extension '{}' will continue to load but should specify a plugin version.", extensionName);
|
|
||||||
version = "Not Specified";
|
|
||||||
} else {
|
|
||||||
version = discoveredExtension.getVersion();
|
|
||||||
}
|
|
||||||
List<String> authors;
|
|
||||||
if (discoveredExtension.getAuthors() == null) {
|
|
||||||
authors = new ArrayList<>();
|
|
||||||
} else {
|
|
||||||
authors = Arrays.asList(discoveredExtension.getAuthors());
|
|
||||||
}
|
|
||||||
|
|
||||||
extensionDescription = new Extension.ExtensionDescription(extensionName, version, authors);
|
|
||||||
}
|
|
||||||
|
|
||||||
extensionLoaders.put(extensionName.toLowerCase(), loader);
|
extensionLoaders.put(extensionName.toLowerCase(), loader);
|
||||||
|
|
||||||
@ -165,9 +156,9 @@ public class ExtensionManager {
|
|||||||
|
|
||||||
// Set logger
|
// Set logger
|
||||||
try {
|
try {
|
||||||
Field descriptionField = extensionClass.getSuperclass().getDeclaredField("logger");
|
Field loggerField = extensionClass.getSuperclass().getDeclaredField("logger");
|
||||||
descriptionField.setAccessible(true);
|
loggerField.setAccessible(true);
|
||||||
descriptionField.set(extension, LoggerFactory.getLogger(extensionClass));
|
loggerField.set(extension, LoggerFactory.getLogger(extensionClass));
|
||||||
} catch (IllegalAccessException e) {
|
} catch (IllegalAccessException e) {
|
||||||
// We made it accessible, should not occur
|
// We made it accessible, should not occur
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
@ -181,29 +172,29 @@ public class ExtensionManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void loadDependencies(List<DiscoveredExtension> extensions) {
|
private void loadDependencies(List<DiscoveredExtension> extensions) {
|
||||||
for(DiscoveredExtension ext : extensions) {
|
for (DiscoveredExtension ext : extensions) {
|
||||||
try {
|
try {
|
||||||
DependencyGetter getter = new DependencyGetter();
|
DependencyGetter getter = new DependencyGetter();
|
||||||
DiscoveredExtension.Dependencies dependencies = ext.getDependencies();
|
DiscoveredExtension.Dependencies dependencies = ext.getDependencies();
|
||||||
if(dependencies.repositories == null) {
|
|
||||||
throw new IllegalStateException("Missing 'repositories' array.");
|
|
||||||
}
|
|
||||||
if(dependencies.artifacts == null) {
|
|
||||||
throw new IllegalStateException("Missing 'artifacts' array.");
|
|
||||||
}
|
|
||||||
List<MavenRepository> repoList = new LinkedList<>();
|
List<MavenRepository> repoList = new LinkedList<>();
|
||||||
for(var repository : dependencies.repositories) {
|
for (var repository : dependencies.repositories) {
|
||||||
if(repository.name == null) {
|
if (repository.name == null) {
|
||||||
throw new IllegalStateException("Missing 'name' element in repository object.");
|
throw new IllegalStateException("Missing 'name' element in repository object.");
|
||||||
}
|
}
|
||||||
if(repository.url == null) {
|
if (repository.name.isEmpty()) {
|
||||||
|
throw new IllegalStateException("Invalid 'name' element in repository object.");
|
||||||
|
}
|
||||||
|
if (repository.url == null) {
|
||||||
throw new IllegalStateException("Missing 'url' element in repository object.");
|
throw new IllegalStateException("Missing 'url' element in repository object.");
|
||||||
}
|
}
|
||||||
|
if (repository.url.isEmpty()) {
|
||||||
|
throw new IllegalStateException("Invalid 'url' element in repository object.");
|
||||||
|
}
|
||||||
repoList.add(new MavenRepository(repository.name, repository.url));
|
repoList.add(new MavenRepository(repository.name, repository.url));
|
||||||
}
|
}
|
||||||
getter.addMavenResolver(repoList);
|
getter.addMavenResolver(repoList);
|
||||||
|
|
||||||
for(var artifact : dependencies.artifacts) {
|
for (var artifact : dependencies.artifacts) {
|
||||||
var resolved = getter.get(artifact, dependenciesFolder);
|
var resolved = getter.get(artifact, dependenciesFolder);
|
||||||
injectIntoClasspath(resolved.getContentsLocation(), ext);
|
injectIntoClasspath(resolved.getContentsLocation(), ext);
|
||||||
log.trace("Dependency of extension {}: {}", ext.getName(), resolved);
|
log.trace("Dependency of extension {}: {}", ext.getName(), resolved);
|
||||||
@ -227,7 +218,7 @@ public class ExtensionManager {
|
|||||||
addURL.setAccessible(true);
|
addURL.setAccessible(true);
|
||||||
addURL.invoke(cl, dependency);
|
addURL.invoke(cl, dependency);
|
||||||
} catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) {
|
} catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) {
|
||||||
throw new RuntimeException("Failed to inject URL "+dependency+" into classpath. From extension "+extension.getName(), e);
|
throw new RuntimeException("Failed to inject URL " + dependency + " into classpath. From extension " + extension.getName(), e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -246,8 +237,11 @@ public class ExtensionManager {
|
|||||||
|
|
||||||
DiscoveredExtension extension = GSON.fromJson(reader, DiscoveredExtension.class);
|
DiscoveredExtension extension = GSON.fromJson(reader, DiscoveredExtension.class);
|
||||||
extension.files = new File[]{file};
|
extension.files = new File[]{file};
|
||||||
extension.checkIntegrity();
|
|
||||||
if(extension.loadStatus == DiscoveredExtension.LoadStatus.LOAD_SUCCESS) {
|
// Verify integrity and ensure defaults
|
||||||
|
DiscoveredExtension.verifyIntegrity(extension);
|
||||||
|
|
||||||
|
if (extension.loadStatus == DiscoveredExtension.LoadStatus.LOAD_SUCCESS) {
|
||||||
extensions.add(extension);
|
extensions.add(extension);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -263,8 +257,11 @@ public class ExtensionManager {
|
|||||||
try (InputStreamReader reader = new InputStreamReader(new FileInputStream(new File(extensionResources, "extension.json")))) {
|
try (InputStreamReader reader = new InputStreamReader(new FileInputStream(new File(extensionResources, "extension.json")))) {
|
||||||
DiscoveredExtension extension = GSON.fromJson(reader, DiscoveredExtension.class);
|
DiscoveredExtension extension = GSON.fromJson(reader, DiscoveredExtension.class);
|
||||||
extension.files = new File[]{new File(extensionClasses), new File(extensionResources)};
|
extension.files = new File[]{new File(extensionClasses), new File(extensionResources)};
|
||||||
extension.checkIntegrity();
|
|
||||||
if(extension.loadStatus == DiscoveredExtension.LoadStatus.LOAD_SUCCESS) {
|
// Verify integrity and ensure defaults
|
||||||
|
DiscoveredExtension.verifyIntegrity(extension);
|
||||||
|
|
||||||
|
if (extension.loadStatus == DiscoveredExtension.LoadStatus.LOAD_SUCCESS) {
|
||||||
extensions.add(extension);
|
extensions.add(extension);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -320,7 +317,7 @@ public class ExtensionManager {
|
|||||||
for (String codeModifierClass : extension.getCodeModifiers()) {
|
for (String codeModifierClass : extension.getCodeModifiers()) {
|
||||||
modifiableClassLoader.loadModifier(extension.files, codeModifierClass);
|
modifiableClassLoader.loadModifier(extension.files, codeModifierClass);
|
||||||
}
|
}
|
||||||
if (extension.getMixinConfig() != null) {
|
if (!extension.getMixinConfig().isEmpty()) {
|
||||||
final String mixinConfigFile = extension.getMixinConfig();
|
final String mixinConfigFile = extension.getMixinConfig();
|
||||||
Mixins.addConfiguration(mixinConfigFile);
|
Mixins.addConfiguration(mixinConfigFile);
|
||||||
log.info("Found mixin in extension " + extension.getName() + ": " + mixinConfigFile);
|
log.info("Found mixin in extension " + extension.getName() + ": " + mixinConfigFile);
|
||||||
|
Loading…
Reference in New Issue
Block a user