From e66b63281dd2319656a3e2039dbae8ff9b1949d2 Mon Sep 17 00:00:00 2001 From: Rsl1122 Date: Mon, 12 Aug 2019 09:31:57 +0300 Subject: [PATCH] Made build in extension registration more resistant: - Attempt to reduce the amount of exceptions that might slip through and fail to start Plan Affects issues: - Close #1134 --- .../extractor/ExtensionExtractor.java | 16 ++++- .../data/plugin/PluginsConfigSection.java | 14 ++++ .../ExtensionServiceImplementation.java | 4 +- .../system/settings/config/ConfigNode.java | 4 ++ .../implementation/ExtensionRegister.java | 70 +++++++++++-------- 5 files changed, 75 insertions(+), 33 deletions(-) diff --git a/Plan/api/src/main/java/com/djrapitops/plan/extension/extractor/ExtensionExtractor.java b/Plan/api/src/main/java/com/djrapitops/plan/extension/extractor/ExtensionExtractor.java index adb234010..4165f5ed3 100644 --- a/Plan/api/src/main/java/com/djrapitops/plan/extension/extractor/ExtensionExtractor.java +++ b/Plan/api/src/main/java/com/djrapitops/plan/extension/extractor/ExtensionExtractor.java @@ -68,8 +68,13 @@ public final class ExtensionExtractor { } } - private Optional getClassAnnotation(Class ofClass) { - return Optional.ofNullable(extension.getClass().getAnnotation(ofClass)); + private static Optional getClassAnnotation(Class from, Class ofClass) { + return Optional.ofNullable(from.getAnnotation(ofClass)); + } + + public static String getPluginName(Class extensionClass) { + return getClassAnnotation(extensionClass, PluginInfo.class).map(PluginInfo::name) + .orElseThrow(() -> new IllegalArgumentException("Given class had no PluginInfo annotation")); } private Method[] getMethods() { @@ -282,8 +287,13 @@ public final class ExtensionExtractor { } } + private Optional getClassAnnotation(Class ofClass) { + return getClassAnnotation(extension.getClass(), ofClass); + } + private void extractPluginInfo() { - pluginInfo = getClassAnnotation(PluginInfo.class).orElseThrow(() -> new IllegalArgumentException("Given class had no PluginInfo annotation")); + pluginInfo = getClassAnnotation(PluginInfo.class) + .orElseThrow(() -> new IllegalArgumentException("Given class had no PluginInfo annotation")); if (pluginInfo.name().length() > 50) { warnings.add(extensionName + " PluginInfo 'name' was over 50 characters."); diff --git a/Plan/common/src/main/java/com/djrapitops/plan/data/plugin/PluginsConfigSection.java b/Plan/common/src/main/java/com/djrapitops/plan/data/plugin/PluginsConfigSection.java index 9223fe375..85602e04c 100644 --- a/Plan/common/src/main/java/com/djrapitops/plan/data/plugin/PluginsConfigSection.java +++ b/Plan/common/src/main/java/com/djrapitops/plan/data/plugin/PluginsConfigSection.java @@ -20,6 +20,8 @@ import com.djrapitops.plan.system.settings.config.ConfigNode; import com.djrapitops.plan.system.settings.config.PlanConfig; import java.io.IOException; +import java.util.HashSet; +import java.util.Set; /** * Class responsible for generating and generating settings for PluginData @@ -74,4 +76,16 @@ public class PluginsConfigSection { ConfigNode section = getPluginsSection(); return section.getBoolean(pluginName + ".Enabled"); } + + public Set getDisabled() { + ConfigNode section = getPluginsSection(); + + Set disabledPlugins = new HashSet<>(); + for (ConfigNode plugin : section.getChildren()) { + if (!plugin.getBoolean("Enabled")) { + disabledPlugins.add(plugin.getKey(false)); + } + } + return disabledPlugins; + } } diff --git a/Plan/common/src/main/java/com/djrapitops/plan/extension/ExtensionServiceImplementation.java b/Plan/common/src/main/java/com/djrapitops/plan/extension/ExtensionServiceImplementation.java index 3f4fb56a5..f61955b15 100644 --- a/Plan/common/src/main/java/com/djrapitops/plan/extension/ExtensionServiceImplementation.java +++ b/Plan/common/src/main/java/com/djrapitops/plan/extension/ExtensionServiceImplementation.java @@ -83,11 +83,11 @@ public class ExtensionServiceImplementation implements ExtensionService { public void register() { try { - extensionRegister.registerBuiltInExtensions(); + extensionRegister.registerBuiltInExtensions(config.getPluginsConfigSection().getDisabled()); if (Check.isBukkitAvailable()) extensionRegister.registerBukkitExtensions(); if (Check.isBungeeAvailable()) extensionRegister.registerBungeeExtensions(); } catch (IllegalStateException failedToRegisterOne) { - logger.warn("One or more extensions failed to register:"); + logger.warn("One or more extensions failed to register, see suppressed exceptions."); errorHandler.log(L.WARN, this.getClass(), failedToRegisterOne); } } diff --git a/Plan/common/src/main/java/com/djrapitops/plan/system/settings/config/ConfigNode.java b/Plan/common/src/main/java/com/djrapitops/plan/system/settings/config/ConfigNode.java index 70ca37829..ea0c9ee4b 100644 --- a/Plan/common/src/main/java/com/djrapitops/plan/system/settings/config/ConfigNode.java +++ b/Plan/common/src/main/java/com/djrapitops/plan/system/settings/config/ConfigNode.java @@ -349,6 +349,10 @@ public class ConfigNode { return nodeOrder; } + public Collection getChildren() { + return childNodes.values(); + } + @Override public boolean equals(Object o) { if (this == o) return true; diff --git a/Plan/extensions/src/main/java/com/djrapitops/plan/extension/implementation/ExtensionRegister.java b/Plan/extensions/src/main/java/com/djrapitops/plan/extension/implementation/ExtensionRegister.java index 43bcdde31..14fb136c9 100644 --- a/Plan/extensions/src/main/java/com/djrapitops/plan/extension/implementation/ExtensionRegister.java +++ b/Plan/extensions/src/main/java/com/djrapitops/plan/extension/implementation/ExtensionRegister.java @@ -19,10 +19,12 @@ package com.djrapitops.plan.extension.implementation; import com.djrapitops.extension.*; import com.djrapitops.plan.extension.DataExtension; import com.djrapitops.plan.extension.ExtensionService; +import com.djrapitops.plan.extension.extractor.ExtensionExtractor; import javax.inject.Inject; import javax.inject.Singleton; import java.util.Optional; +import java.util.Set; import java.util.function.Supplier; /** @@ -34,47 +36,49 @@ import java.util.function.Supplier; public class ExtensionRegister { private IllegalStateException registerException; - + private Set disabledExtensions; + @Inject public ExtensionRegister() { /* Required for dagger injection */ } - public void registerBuiltInExtensions() { + public void registerBuiltInExtensions(Set disabledExtensions) { + this.disabledExtensions = disabledExtensions; // No need to catch exceptions here, // registerBuiltInExtensions method will not be called unless Plan has enabled properly ExtensionService extensionService = ExtensionService.getInstance(); - register(new AACExtensionFactory()::createExtension); - register(new AdvancedAchievementsExtensionFactory()::createExtension); - register(new AdvancedBanExtensionFactory()::createExtension); - register(new ASkyBlockExtensionFactory()::createExtension); - register(new BanManagerExtensionFactory()::createExtension); - register(new CoreProtectExtensionFactory()::createExtension); - register(new DiscordSRVExtensionFactory()::createExtension); + register(new AACExtensionFactory()::createExtension, AACExtensionFactory.class); + register(new AdvancedAchievementsExtensionFactory()::createExtension, AdvancedAchievementsExtensionFactory.class); + register(new AdvancedBanExtensionFactory()::createExtension, AdvancedBanExtensionFactory.class); + register(new ASkyBlockExtensionFactory()::createExtension, ASkyBlockExtensionFactory.class); + register(new BanManagerExtensionFactory()::createExtension, BanManagerExtensionFactory.class); + register(new CoreProtectExtensionFactory()::createExtension, CoreProtectExtensionFactory.class); + register(new DiscordSRVExtensionFactory()::createExtension, DiscordSRVExtensionFactory.class); registerEssentialsExtension(extensionService); - register(new GriefPreventionExtensionFactory()::createExtension); - register(new GriefPreventionSpongeExtensionFactory()::createExtension); - register(new GriefPreventionPlusExtensionFactory()::createExtension); - register(new McMMOExtensionFactory()::createExtension); + register(new GriefPreventionExtensionFactory()::createExtension, GriefPreventionExtensionFactory.class); + register(new GriefPreventionSpongeExtensionFactory()::createExtension, GriefPreventionSpongeExtensionFactory.class); + register(new GriefPreventionPlusExtensionFactory()::createExtension, GriefPreventionPlusExtensionFactory.class); + register(new McMMOExtensionFactory()::createExtension, McMMOExtensionFactory.class); registerMinigameLibExtensions(extensionService); - register(new NucleusExtensionFactory()::createExtension); - register(new NuVotifierExtensionFactory()::createExtension); - register(new ProtocolSupportExtensionFactory()::createExtension); - register(new RedProtectExtensionFactory()::createExtension); - register(new SpongeEconomyExtensionFactory()::createExtension); - register(new SuperbVoteExtensionFactory()::createExtension); - register(new VaultExtensionFactory()::createExtension); + register(new NucleusExtensionFactory()::createExtension, NucleusExtensionFactory.class); + register(new NuVotifierExtensionFactory()::createExtension, NuVotifierExtensionFactory.class); + register(new ProtocolSupportExtensionFactory()::createExtension, ProtocolSupportExtensionFactory.class); + register(new RedProtectExtensionFactory()::createExtension, RedProtectExtensionFactory.class); + register(new SpongeEconomyExtensionFactory()::createExtension, SpongeEconomyExtensionFactory.class); + register(new SuperbVoteExtensionFactory()::createExtension, SuperbVoteExtensionFactory.class); + register(new VaultExtensionFactory()::createExtension, VaultExtensionFactory.class); if (registerException != null) throw registerException; } public void registerBukkitExtensions() { - register(new ViaVersionBukkitExtensionFactory()::createExtension); + register(new ViaVersionBukkitExtensionFactory()::createExtension, ViaVersionBukkitExtensionFactory.class); } public void registerBungeeExtensions() { - register(new ViaVersionBungeeExtensionFactory()::createExtension); + register(new ViaVersionBungeeExtensionFactory()::createExtension, ViaVersionBungeeExtensionFactory.class); } private void registerEssentialsExtension(ExtensionService extensionService) { @@ -90,16 +94,26 @@ public class ExtensionRegister { } } - private void register(Supplier> extension) { + private void register(Supplier> extension, Class factory) { ExtensionService extensionService = ExtensionService.getInstance(); try { - extension.get().ifPresent(extensionService::register); - } catch (IllegalStateException e) { + Optional optional = extension.get(); + if (!optional.isPresent()) return; + DataExtension dataExtension = optional.get(); + + String extensionName = ExtensionExtractor.getPluginName(dataExtension.getClass()); + if (disabledExtensions.contains(extensionName)) return; + + extensionService.register(dataExtension); + } catch (IllegalStateException | NoClassDefFoundError | IncompatibleClassChangeError e) { + // Places all exceptions to one exception with plugin information so that they can be reported. if (registerException == null) { - registerException = e; - } else { - registerException.addSuppressed(e); + registerException = new IllegalStateException("One or more extensions failed to register:"); + registerException.setStackTrace(new StackTraceElement[0]); } + IllegalStateException info = new IllegalStateException(factory.getSimpleName() + " ran into exception when creating Extension", e); + info.setStackTrace(new StackTraceElement[0]); + registerException.addSuppressed(info); } } } \ No newline at end of file