aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDecebal Suiu <decebal.suiu@gmail.com>2017-02-16 23:06:02 +0200
committerDecebal Suiu <decebal.suiu@gmail.com>2017-02-16 23:06:02 +0200
commite8d5b8eebf2207194685dc66af72917cbba11ecf (patch)
tree552b76e782cc6c2499cbcc0bb3926c3f988efdd3
parent8bb6410957917df04ac4d9a962308b15d2ad7a8a (diff)
downloadpf4j-e8d5b8eebf2207194685dc66af72917cbba11ecf.tar.gz
pf4j-e8d5b8eebf2207194685dc66af72917cbba11ecf.zip
Reactivate protection against the issues similar with #97
-rw-r--r--demo/app/src/main/java/ro/fortsoft/pf4j/demo/Boot.java4
-rw-r--r--demo/app/src/main/resources/log4j.properties2
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/AbstractExtensionFinder.java15
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/util/ClassUtils.java91
4 files changed, 110 insertions, 2 deletions
diff --git a/demo/app/src/main/java/ro/fortsoft/pf4j/demo/Boot.java b/demo/app/src/main/java/ro/fortsoft/pf4j/demo/Boot.java
index fa779e2..c6ecec3 100644
--- a/demo/app/src/main/java/ro/fortsoft/pf4j/demo/Boot.java
+++ b/demo/app/src/main/java/ro/fortsoft/pf4j/demo/Boot.java
@@ -21,6 +21,7 @@ import java.util.Set;
import org.apache.commons.lang.StringUtils;
import ro.fortsoft.pf4j.DefaultPluginManager;
+import ro.fortsoft.pf4j.JarPluginManager;
import ro.fortsoft.pf4j.PluginManager;
import ro.fortsoft.pf4j.PluginWrapper;
import ro.fortsoft.pf4j.demo.api.Greeting;
@@ -37,7 +38,8 @@ public class Boot {
printLogo();
// create the plugin manager
- final PluginManager pluginManager = new DefaultPluginManager();
+// final PluginManager pluginManager = new DefaultPluginManager();
+ final PluginManager pluginManager = new JarPluginManager();
// load the plugins
pluginManager.loadPlugins();
diff --git a/demo/app/src/main/resources/log4j.properties b/demo/app/src/main/resources/log4j.properties
index 0454ba2..692e39b 100644
--- a/demo/app/src/main/resources/log4j.properties
+++ b/demo/app/src/main/resources/log4j.properties
@@ -5,7 +5,7 @@ log4j.rootLogger=DEBUG, Console
#
log4j.logger.ro.fortsoft.pf4j=DEBUG, Console
# !!! Put the bellow classes on level TRACE when you are in trouble
-log4j.logger.ro.fortsoft.pf4j.PluginClassLoader=WARN, Console
+log4j.logger.ro.fortsoft.pf4j.PluginClassLoader=DEBUG, Console
log4j.logger.ro.fortsoft.pf4j.AbstractExtensionFinder=DEBUG, Console
log4j.additivity.ro.fortsoft.pf4j=false
log4j.additivity.ro.fortsoft.pf4j.PluginClassLoader=false
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/AbstractExtensionFinder.java b/pf4j/src/main/java/ro/fortsoft/pf4j/AbstractExtensionFinder.java
index 8332a18..3c2e0c6 100644
--- a/pf4j/src/main/java/ro/fortsoft/pf4j/AbstractExtensionFinder.java
+++ b/pf4j/src/main/java/ro/fortsoft/pf4j/AbstractExtensionFinder.java
@@ -17,6 +17,7 @@ package ro.fortsoft.pf4j;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
+import ro.fortsoft.pf4j.util.ClassUtils;
import java.util.ArrayList;
import java.util.Collections;
@@ -106,6 +107,9 @@ public abstract class AbstractExtensionFinder implements ExtensionFinder, Plugin
log.debug("Added extension '{}' with ordinal {}", className, extensionWrapper.getOrdinal());
} else {
log.trace("'{}' is not an extension for extension point '{}'", className, type.getName());
+ if (RuntimeMode.DEVELOPMENT.equals(pluginManager.getRuntimeMode())) {
+ checkDifferentClassLoaders(type, extensionClass);
+ }
}
} catch (ClassNotFoundException e) {
log.error(e.getMessage(), e);
@@ -229,4 +233,15 @@ public abstract class AbstractExtensionFinder implements ExtensionFinder, Plugin
return extensionWrapper;
}
+ private void checkDifferentClassLoaders(Class<?> type, Class<?> extensionClass) {
+ ClassLoader typeClassLoader = type.getClassLoader(); // class loader of extension point
+ ClassLoader extensionClassLoader = extensionClass.getClassLoader();
+ boolean match = ClassUtils.getAllInterfacesNames(extensionClass).contains(type.getSimpleName());
+ if (match && !extensionClassLoader.equals(typeClassLoader)) {
+ // in this scenario the method 'isAssignableFrom' returns only FALSE
+ // see http://www.coderanch.com/t/557846/java/java/FWIW-FYI-isAssignableFrom-isInstance-differing
+ log.error("Different class loaders: '{}' (E) and '{}' (EP)", extensionClassLoader, typeClassLoader);
+ }
+ }
+
}
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/util/ClassUtils.java b/pf4j/src/main/java/ro/fortsoft/pf4j/util/ClassUtils.java
new file mode 100644
index 0000000..b435c2f
--- /dev/null
+++ b/pf4j/src/main/java/ro/fortsoft/pf4j/util/ClassUtils.java
@@ -0,0 +1,91 @@
+/*
+ * Copyright 2016 Decebal Suiu
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License 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.util;
+
+import java.lang.reflect.Modifier;
+import java.util.ArrayList;
+import java.util.List;
+
+/**
+ * @author Decebal Suiu
+ */
+public class ClassUtils {
+
+ public static List<String> getAllInterfacesNames(Class<?> aClass) {
+ return toString(getAllInterfaces(aClass));
+ }
+
+ public static List<Class<?>> getAllInterfaces(Class<?> aClass) {
+ List<Class<?>> list = new ArrayList<>();
+
+ while (aClass != null) {
+ Class<?>[] interfaces = aClass.getInterfaces();
+ for (Class<?> anInterface : interfaces) {
+ if (!list.contains(anInterface)) {
+ list.add(anInterface);
+ }
+
+ List<Class<?>> superInterfaces = getAllInterfaces(anInterface);
+ for (Class<?> superInterface : superInterfaces) {
+ if (!list.contains(superInterface)) {
+ list.add(superInterface);
+ }
+ }
+ }
+
+ aClass = aClass.getSuperclass();
+ }
+
+ return list;
+ }
+
+ /*
+ public static List<String> getAllAbstractClassesNames(Class<?> aClass) {
+ return toString(getAllInterfaces(aClass));
+ }
+
+ public static List getAllAbstractClasses(Class aClass) {
+ List<Class<?>> list = new ArrayList<>();
+
+ Class<?> superclass = aClass.getSuperclass();
+ while (superclass != null) {
+ if (Modifier.isAbstract(superclass.getModifiers())) {
+ list.add(superclass);
+ }
+ superclass = superclass.getSuperclass();
+ }
+
+ return list;
+ }
+ */
+
+ /**
+ * Uses {@link Class#getSimpleName()} to convert from {@link Class} to {@link String}.
+ *
+ * @param classes
+ * @return
+ */
+ private static List<String> toString(List<Class<?>> classes) {
+ List<String> list = new ArrayList<>();
+
+ for (Class<?> aClass : classes) {
+ list.add(aClass.getSimpleName());
+ }
+
+ return list;
+ }
+
+}