diff options
author | James Moger <james.moger@gitblit.com> | 2014-04-12 12:10:56 -0400 |
---|---|---|
committer | James Moger <james.moger@gitblit.com> | 2014-04-12 19:35:45 -0400 |
commit | eba6d3dc6b83e31e1c98df23a527abed09886ad3 (patch) | |
tree | 488d4bc65e22aff5dafb7f486219370ff88e80b8 /pf4j | |
parent | b5589a7d0ba12a13130c2f62578ad0d88dd5eb7a (diff) | |
download | pf4j-eba6d3dc6b83e31e1c98df23a527abed09886ad3.tar.gz pf4j-eba6d3dc6b83e31e1c98df23a527abed09886ad3.zip |
Add an optional requires manifest value
As integrators of pf4j evovle their extension APIs it will become
a requirement to specify a minimum system version for loading plugins.
Loading & starting a newer plugin on an older system could result in
runtime failures due to method signature changes or other class
differences.
This change adds a manifest attribute to specify a 'requires' version
which is a minmum system version. It also introduces a method to
specify the system version of the plugin manager and logic to disable
plugins on load if the system version is too old. This works for both
loadPlugins() and loadPlugin().
Diffstat (limited to 'pf4j')
5 files changed, 87 insertions, 5 deletions
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/DefaultPluginManager.java b/pf4j/src/main/java/ro/fortsoft/pf4j/DefaultPluginManager.java index c44787e..970ae2d 100644 --- a/pf4j/src/main/java/ro/fortsoft/pf4j/DefaultPluginManager.java +++ b/pf4j/src/main/java/ro/fortsoft/pf4j/DefaultPluginManager.java @@ -90,6 +90,11 @@ public class DefaultPluginManager implements PluginManager { private RuntimeMode runtimeMode; /** + * The system version used for comparisons to the plugin requires attribute. + */ + private PluginVersion systemVersion = PluginVersion.DEFAULT; + + /** * The plugins directory is supplied by System.getProperty("pf4j.pluginsDir", "plugins"). */ public DefaultPluginManager() { @@ -110,6 +115,16 @@ public class DefaultPluginManager implements PluginManager { initialize(); } + @Override + public void setSystemVersion(PluginVersion version) { + systemVersion = version; + } + + @Override + public PluginVersion getSystemVersion() { + return systemVersion; + } + @Override public List<PluginWrapper> getPlugins() { return new ArrayList<PluginWrapper>(plugins.values()); @@ -461,6 +476,16 @@ public class DefaultPluginManager implements PluginManager { return true; } + PluginVersion requires = pluginWrapper.getDescriptor().getRequires(); + PluginVersion system = getSystemVersion(); + if (!system.isDefault() && !system.atLeast(requires)) { + log.warn(String.format("Failed to enable plugin `{}:{}` because it requires a minimum system version of %s", + pluginWrapper.getPluginId(), + pluginWrapper.getDescriptor().getVersion(), + requires)); + return false; + } + try { if (disabledPlugins.remove(pluginId)) { FileUtils.writeLines(disabledPlugins, new File(pluginsDirectory, "disabled.txt")); @@ -724,6 +749,18 @@ public class DefaultPluginManager implements PluginManager { pluginWrapper.setPluginState(PluginState.DISABLED); } + // optionally enforce minimum system version requirements + PluginVersion requires = pluginWrapper.getDescriptor().getRequires(); + PluginVersion system = getSystemVersion(); + if (!system.isDefault() && !system.atLeast(requires)) { + log.warn(String.format("Disabling plugin '%s:%s' because it requires a minimum system version of %s", + pluginWrapper.getPluginId(), + pluginWrapper.getDescriptor().getVersion(), + requires)); + pluginWrapper.setPluginState(PluginState.DISABLED); + disabledPlugins.add(pluginWrapper.getPluginId()); + } + log.debug("Created wrapper '{}' for plugin '{}'", pluginWrapper, pluginPath); String pluginId = pluginDescriptor.getPluginId(); diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/ManifestPluginDescriptorFinder.java b/pf4j/src/main/java/ro/fortsoft/pf4j/ManifestPluginDescriptorFinder.java index 2de42f5..7f802d1 100644 --- a/pf4j/src/main/java/ro/fortsoft/pf4j/ManifestPluginDescriptorFinder.java +++ b/pf4j/src/main/java/ro/fortsoft/pf4j/ManifestPluginDescriptorFinder.java @@ -103,6 +103,11 @@ public class ManifestPluginDescriptorFinder implements PluginDescriptorFinder { String dependencies = attrs.getValue("Plugin-Dependencies"); pluginDescriptor.setDependencies(dependencies); + String requires = attrs.getValue("Plugin-Requires"); + if (StringUtils.isNotEmpty(requires)) { + pluginDescriptor.setRequires(PluginVersion.createVersion(requires)); + } + return pluginDescriptor; } diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/PluginDescriptor.java b/pf4j/src/main/java/ro/fortsoft/pf4j/PluginDescriptor.java index cb91f7d..aa05b52 100644 --- a/pf4j/src/main/java/ro/fortsoft/pf4j/PluginDescriptor.java +++ b/pf4j/src/main/java/ro/fortsoft/pf4j/PluginDescriptor.java @@ -28,10 +28,12 @@ public class PluginDescriptor { private String pluginDescription; private String pluginClass; private PluginVersion version; + private PluginVersion requires; private String provider; private List<PluginDependency> dependencies; public PluginDescriptor() { + requires = PluginVersion.DEFAULT; dependencies = new ArrayList<PluginDependency>(); } @@ -64,6 +66,13 @@ public class PluginDescriptor { } /** + * Returns the requires of this plugin. + */ + public PluginVersion getRequires() { + return requires; + } + + /** * Returns the provider name of this plugin. */ public String getProvider() { @@ -106,6 +115,10 @@ public class PluginDescriptor { this.provider = provider; } + void setRequires(PluginVersion requires) { + this.requires = requires; + } + void setDependencies(String dependencies) { if (dependencies != null) { dependencies = dependencies.trim(); diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/PluginManager.java b/pf4j/src/main/java/ro/fortsoft/pf4j/PluginManager.java index daf6fbf..81abe7c 100644 --- a/pf4j/src/main/java/ro/fortsoft/pf4j/PluginManager.java +++ b/pf4j/src/main/java/ro/fortsoft/pf4j/PluginManager.java @@ -141,4 +141,20 @@ public interface PluginManager { public void removePluginStateListener(PluginStateListener listener); + /** + * Set the system version. This is used to compare against the plugin + * requires attribute. The default system version is 0.0.0 which + * disables all version checking. + * + * @default 0.0.0 + * @param version + */ + public void setSystemVersion(PluginVersion version); + + /** + * Returns the system version. + * + * * @return the system version + */ + public PluginVersion getSystemVersion(); } diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/PluginVersion.java b/pf4j/src/main/java/ro/fortsoft/pf4j/PluginVersion.java index 6ffd284..7cfc94b 100644 --- a/pf4j/src/main/java/ro/fortsoft/pf4j/PluginVersion.java +++ b/pf4j/src/main/java/ro/fortsoft/pf4j/PluginVersion.java @@ -1,22 +1,22 @@ /* * Copyright 2012 Decebal Suiu - * + * * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this work except in compliance with * the License. You may obtain a copy of the License in the LICENSE file, or at: - * + * * http://www.apache.org/licenses/LICENSE-2.0 - * + * * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on * an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the * specific language governing permissions and limitations under the License. */ package ro.fortsoft.pf4j; -import ro.fortsoft.pf4j.util.StringUtils; - import java.util.regex.Matcher; import java.util.regex.Pattern; +import ro.fortsoft.pf4j.util.StringUtils; + /** * Represents the version of a Plugin and allows versions to be compared. * Version following semantic defined by <a href="http://semver.org/">Semantic Versioning</a> document. @@ -33,6 +33,8 @@ import java.util.regex.Pattern; */ public class PluginVersion implements Comparable<PluginVersion> { + public static final PluginVersion DEFAULT = new PluginVersion(0, 0, 0); + private static final String FORMAT = "(\\d+)\\.(\\d+)(?:\\.)?(\\d*)(\\.|-|\\+)?([0-9A-Za-z-.]*)?"; private static final Pattern PATTERN = Pattern.compile(FORMAT); @@ -95,6 +97,7 @@ public class PluginVersion implements Comparable<PluginVersion> { return qualifier; } + @Override public String toString() { StringBuffer sb = new StringBuffer(50); sb.append(major); @@ -135,6 +138,14 @@ public class PluginVersion implements Comparable<PluginVersion> { return 0; } + public boolean isDefault() { + return compareTo(DEFAULT) == 0; + } + + public boolean atLeast(PluginVersion v) { + return compareTo(v) <= 0; + } + // for test only public static void main(String[] args) { PluginVersion v = PluginVersion.createVersion("1.2.3-SNAPSHOT"); |