Cleanup and improve extension changes.

This commit is contained in:
Articdive 2020-10-25 10:41:51 +01:00
parent 535e8946b6
commit 5217964259
No known key found for this signature in database
GPG Key ID: B069585F0F7D90DE
3 changed files with 161 additions and 138 deletions

View File

@ -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;
} }
} }

View File

@ -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

View File

@ -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);