aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDecebal Suiu <decebal.suiu@gmail.com>2016-01-13 11:10:43 +0200
committerDecebal Suiu <decebal.suiu@gmail.com>2016-01-13 11:10:43 +0200
commit96a649f4989b6db1c3f14cc5feb3475297372079 (patch)
treed92b89fc2a3eec09bfc5d9281b057793ef1dd473
parent03ac519e7cff4dfa941e7ffafa85d24c3b879758 (diff)
parentea9521ed586a058c4fe991c51aa498f083384d2c (diff)
downloadpf4j-96a649f4989b6db1c3f14cc5feb3475297372079.tar.gz
pf4j-96a649f4989b6db1c3f14cc5feb3475297372079.zip
Merge pull request #85 from decebals/service_provider
Extension storage based on Java Service Provider (META-INf/services)
-rw-r--r--demo/app/src/main/java/ro/fortsoft/pf4j/demo/HowdyGreeting.java32
-rw-r--r--demo/app/src/main/resources/META-INF/services/ro.fortsoft.pf4j.demo.api.Greeting1
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/AbstractExtensionFinder.java137
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/DefaultExtensionFinder.java195
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/ExtensionsIndexer.java138
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/LegacyExtensionFinder.java122
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/PluginClassLoader.java5
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/ServiceProviderExtensionFinder.java132
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/processor/ExtensionAnnotationProcessor.java198
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/processor/ExtensionStorage.java73
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/processor/LegacyExtensionStorage.java101
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/processor/ServiceProviderExtensionStorage.java114
-rw-r--r--pf4j/src/main/java/ro/fortsoft/pf4j/util/CompoundClassLoader.java79
-rw-r--r--pf4j/src/main/resources/META-INF/services/javax.annotation.processing.Processor2
14 files changed, 938 insertions, 391 deletions
diff --git a/demo/app/src/main/java/ro/fortsoft/pf4j/demo/HowdyGreeting.java b/demo/app/src/main/java/ro/fortsoft/pf4j/demo/HowdyGreeting.java
new file mode 100644
index 0000000..3d7d24b
--- /dev/null
+++ b/demo/app/src/main/java/ro/fortsoft/pf4j/demo/HowdyGreeting.java
@@ -0,0 +1,32 @@
+/*
+ * Copyright 2015 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.demo;
+
+import ro.fortsoft.pf4j.demo.api.Greeting;
+
+/**
+ * A Service Implementation (no @Extension) declared via Java Service Provider mechanism (using META-INF/services).
+ *
+ * @author Decebal Suiu
+ */
+public class HowdyGreeting implements Greeting {
+
+ @Override
+ public String getGreeting() {
+ return "Howdy";
+ }
+
+}
diff --git a/demo/app/src/main/resources/META-INF/services/ro.fortsoft.pf4j.demo.api.Greeting b/demo/app/src/main/resources/META-INF/services/ro.fortsoft.pf4j.demo.api.Greeting
new file mode 100644
index 0000000..4f8f4ad
--- /dev/null
+++ b/demo/app/src/main/resources/META-INF/services/ro.fortsoft.pf4j.demo.api.Greeting
@@ -0,0 +1 @@
+ro.fortsoft.pf4j.demo.HowdyGreeting
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/AbstractExtensionFinder.java b/pf4j/src/main/java/ro/fortsoft/pf4j/AbstractExtensionFinder.java
new file mode 100644
index 0000000..3cefea9
--- /dev/null
+++ b/pf4j/src/main/java/ro/fortsoft/pf4j/AbstractExtensionFinder.java
@@ -0,0 +1,137 @@
+/*
+ * Copyright 2015 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;
+
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+
+/**
+ * @author Decebal Suiu
+ */
+public abstract class AbstractExtensionFinder implements ExtensionFinder, PluginStateListener {
+
+ protected static final Logger log = LoggerFactory.getLogger(AbstractExtensionFinder.class);
+
+ protected PluginManager pluginManager;
+ protected volatile Map<String, Set<String>> entries; // cache by pluginId
+
+ public AbstractExtensionFinder(PluginManager pluginManager) {
+ this.pluginManager = pluginManager;
+ }
+
+ public abstract Map<String, Set<String>> readPluginsStorages();
+
+ public abstract Map<String, Set<String>> readClasspathStorages();
+
+ @Override
+ public <T> List<ExtensionWrapper<T>> find(Class<T> type) {
+ log.debug("Finding extensions for extension point '{}'", type.getName());
+ Map<String, Set<String>> entries = getEntries();
+
+ List<ExtensionWrapper<T>> result = new ArrayList<>();
+ for (Map.Entry<String, Set<String>> entry : entries.entrySet()) {
+ String pluginId = entry.getKey();
+
+ if (pluginId != null) {
+ PluginWrapper pluginWrapper = pluginManager.getPlugin(pluginId);
+ if (PluginState.STARTED != pluginWrapper.getPluginState()) {
+ continue;
+ }
+ }
+
+ for (String className : entry.getValue()) {
+ try {
+ ClassLoader classLoader;
+ if (pluginId != null) {
+ classLoader = pluginManager.getPluginClassLoader(pluginId);
+ } else {
+ classLoader = getClass().getClassLoader();
+ }
+ log.debug("Loading class '{}' using class loader '{}'", className, classLoader);
+ Class<?> extensionClass = classLoader.loadClass(className);
+
+ log.debug("Checking extension type '{}'", className);
+ if (type.isAssignableFrom(extensionClass)) {
+ ExtensionDescriptor descriptor = new ExtensionDescriptor();
+ int ordinal = 0;
+ if (extensionClass.isAnnotationPresent(Extension.class)) {
+ ordinal = extensionClass.getAnnotation(Extension.class).ordinal();
+ }
+ descriptor.setOrdinal(ordinal);
+ descriptor.setExtensionClass(extensionClass);
+
+ ExtensionWrapper extensionWrapper = new ExtensionWrapper<>(descriptor);
+ extensionWrapper.setExtensionFactory(pluginManager.getExtensionFactory());
+ result.add(extensionWrapper);
+ log.debug("Added extension '{}' with ordinal {}", className, ordinal);
+ } else {
+ log.debug("'{}' is not an extension for extension point '{}'", className, type.getName());
+ }
+ } catch (ClassNotFoundException e) {
+ log.error(e.getMessage(), e);
+ }
+ }
+ }
+
+ if (entries.isEmpty()) {
+ log.debug("No extensions found for extension point '{}'", type.getName());
+ } else {
+ log.debug("Found {} extensions for extension point '{}'", result.size(), type.getName());
+ }
+
+ // sort by "ordinal" property
+ Collections.sort(result);
+
+ return result;
+ }
+
+ @Override
+ public Set<String> findClassNames(String pluginId) {
+ return getEntries().get(pluginId);
+ }
+
+ @Override
+ public void pluginStateChanged(PluginStateEvent event) {
+ // TODO optimize (do only for some transitions)
+ // clear cache
+ entries = null;
+ }
+
+ private Map<String, Set<String>> readStorages() {
+ Map<String, Set<String>> result = new LinkedHashMap<>();
+
+ result.putAll(readClasspathStorages());
+ result.putAll(readPluginsStorages());
+
+ return result;
+ }
+
+ private Map<String, Set<String>> getEntries() {
+ if (entries == null) {
+ entries = readStorages();
+ }
+
+ return entries;
+ }
+
+}
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/DefaultExtensionFinder.java b/pf4j/src/main/java/ro/fortsoft/pf4j/DefaultExtensionFinder.java
index 9ff571a..cf8c87b 100644
--- a/pf4j/src/main/java/ro/fortsoft/pf4j/DefaultExtensionFinder.java
+++ b/pf4j/src/main/java/ro/fortsoft/pf4j/DefaultExtensionFinder.java
@@ -15,208 +15,57 @@
*/
package ro.fortsoft.pf4j;
-import org.slf4j.Logger;
-import org.slf4j.LoggerFactory;
-
-import java.io.IOException;
-import java.io.InputStreamReader;
-import java.io.Reader;
-import java.net.URL;
import java.util.ArrayList;
-import java.util.Collections;
-import java.util.Enumeration;
import java.util.HashSet;
-import java.util.LinkedHashMap;
import java.util.List;
-import java.util.Map;
import java.util.Set;
/**
* The default implementation for ExtensionFinder.
- * All extensions declared in a plugin are indexed in a file "META-INF/extensions.idx".
- * This class lookup extensions in all extensions index files "META-INF/extensions.idx".
+ * It's a compound ExtensionFinder.
*
* @author Decebal Suiu
*/
public class DefaultExtensionFinder implements ExtensionFinder, PluginStateListener {
- private static final Logger log = LoggerFactory.getLogger(DefaultExtensionFinder.class);
-
- protected PluginManager pluginManager;
- protected volatile Map<String, Set<String>> entries; // cache by pluginId
+ protected List<ExtensionFinder> finders = new ArrayList<>();
public DefaultExtensionFinder(PluginManager pluginManager) {
- this.pluginManager = pluginManager;
- }
+ addDefaults(pluginManager);
+ }
@Override
- public <T> List<ExtensionWrapper<T>> find(Class<T> type) {
- log.debug("Checking extension point '{}'", type.getName());
- if (!isExtensionPoint(type)) {
- log.warn("'{}' is not an extension point", type.getName());
-
- return Collections.emptyList(); // or return null ?!
- }
-
- log.debug("Finding extensions for extension point '{}'", type.getName());
- Map<String, Set<String>> entries = getEntries();
-
- List<ExtensionWrapper<T>> result = new ArrayList<>();
- for (Map.Entry<String, Set<String>> entry : entries.entrySet()) {
- String pluginId = entry.getKey();
-
- if (pluginId != null) {
- PluginWrapper pluginWrapper = pluginManager.getPlugin(pluginId);
- if (PluginState.STARTED != pluginWrapper.getPluginState()) {
- continue;
- }
- }
-
- for (String className : entry.getValue()) {
- try {
- ClassLoader classLoader;
- if (pluginId != null) {
- classLoader = pluginManager.getPluginClassLoader(pluginId);
- } else {
- classLoader = getClass().getClassLoader();
- }
- log.debug("Loading class '{}' using class loader '{}'", className, classLoader);
- Class<?> extensionClass = classLoader.loadClass(className);
-
- log.debug("Checking extension type '{}'", className);
- if (type.isAssignableFrom(extensionClass) && extensionClass.isAnnotationPresent(Extension.class)) {
- Extension extension = extensionClass.getAnnotation(Extension.class);
- ExtensionDescriptor descriptor = new ExtensionDescriptor();
- descriptor.setOrdinal(extension.ordinal());
- descriptor.setExtensionClass(extensionClass);
-
- ExtensionWrapper extensionWrapper = new ExtensionWrapper<>(descriptor);
- extensionWrapper.setExtensionFactory(pluginManager.getExtensionFactory());
- result.add(extensionWrapper);
- log.debug("Added extension '{}' with ordinal {}", className, extension.ordinal());
- } else {
- log.debug("'{}' is not an extension for extension point '{}'", className, type.getName());
- }
- } catch (ClassNotFoundException e) {
- log.error(e.getMessage(), e);
- }
- }
- }
-
- if (entries.isEmpty()) {
- log.debug("No extensions found for extension point '{}'", type.getName());
- } else {
- log.debug("Found {} extensions for extension point '{}'", result.size(), type.getName());
+ public <T> List<ExtensionWrapper<T>> find(Class<T> type) {
+ List<ExtensionWrapper<T>> extensions = new ArrayList<>();
+ for (ExtensionFinder finder : finders) {
+ extensions.addAll(finder.find(type));
}
- // sort by "ordinal" property
- Collections.sort(result);
-
- return result;
- }
-
- @Override
- public Set<String> findClassNames(String pluginId) {
- return getEntries().get(pluginId);
+ return extensions;
}
@Override
- public void pluginStateChanged(PluginStateEvent event) {
- // TODO optimize (do only for some transitions)
- // clear cache
- entries = null;
- }
-
- protected Map<String, Set<String>> readIndexFiles() {
- Map<String, Set<String>> result = new LinkedHashMap<>();
-
- result.putAll(readClasspathIndexFiles());
- result.putAll(readPluginsIndexFiles());
-
- return result;
- }
-
- private Map<String, Set<String>> readClasspathIndexFiles() {
- log.debug("Reading extensions index files from classpath");
-
- Map<String, Set<String>> result = new LinkedHashMap<>();
-
- Set<String> bucket = new HashSet<>();
- try {
- Enumeration<URL> urls = getClass().getClassLoader().getResources(ExtensionsIndexer.EXTENSIONS_RESOURCE);
- while (urls.hasMoreElements()) {
- URL url = urls.nextElement();
- log.debug("Read '{}'", url.getFile());
- Reader reader = new InputStreamReader(url.openStream(), "UTF-8");
- ExtensionsIndexer.readIndex(reader, bucket);
- }
-
- if (bucket.isEmpty()) {
- log.debug("No extensions found");
- } else {
- log.debug("Found possible {} extensions:", bucket.size());
- for (String entry : bucket) {
- log.debug(" " + entry);
- }
- }
-
- result.put(null, bucket);
- } catch (IOException e) {
- log.error(e.getMessage(), e);
+ public Set<String> findClassNames(String pluginId) {
+ Set<String> classNames = new HashSet<>();
+ for (ExtensionFinder finder : finders) {
+ classNames.addAll(finder.findClassNames(pluginId));
}
- return result;
+ return classNames;
}
- private Map<String, Set<String>> readPluginsIndexFiles() {
- log.debug("Reading extensions index files from plugins");
-
- Map<String, Set<String>> result = new LinkedHashMap<>();
-
- List<PluginWrapper> plugins = pluginManager.getPlugins();
- for (PluginWrapper plugin : plugins) {
- String pluginId = plugin.getDescriptor().getPluginId();
- log.debug("Reading extensions index file for plugin '{}'", pluginId);
- Set<String> bucket = new HashSet<>();
-
- try {
- URL url = plugin.getPluginClassLoader().getResource(ExtensionsIndexer.EXTENSIONS_RESOURCE);
- if (url != null) {
- log.debug("Read '{}'", url.getFile());
- Reader reader = new InputStreamReader(url.openStream(), "UTF-8");
- ExtensionsIndexer.readIndex(reader, bucket);
- } else {
- log.debug("Cannot find '{}'", ExtensionsIndexer.EXTENSIONS_RESOURCE);
- }
-
- if (bucket.isEmpty()) {
- log.debug("No extensions found");
- } else {
- log.debug("Found possible {} extensions:", bucket.size());
- for (String entry : bucket) {
- log.debug(" " + entry);
- }
- }
-
- result.put(pluginId, bucket);
- } catch (IOException e) {
- log.error(e.getMessage(), e);
+ @Override
+ public void pluginStateChanged(PluginStateEvent event) {
+ for (ExtensionFinder finder : finders) {
+ if (finder instanceof PluginStateListener) {
+ ((PluginStateListener) finder).pluginStateChanged(event);
}
}
-
- return result;
- }
-
- private boolean isExtensionPoint(Class<?> type) {
- return ExtensionPoint.class.isAssignableFrom(type);
}
- private Map<String, Set<String>> getEntries() {
- if (entries == null) {
- entries = readIndexFiles();
- }
-
- return entries;
+ protected void addDefaults(PluginManager pluginManager) {
+ finders.add(new ServiceProviderExtensionFinder(pluginManager));
+ finders.add(new LegacyExtensionFinder(pluginManager));
}
}
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/ExtensionsIndexer.java b/pf4j/src/main/java/ro/fortsoft/pf4j/ExtensionsIndexer.java
deleted file mode 100644
index d06dbba..0000000
--- a/pf4j/src/main/java/ro/fortsoft/pf4j/ExtensionsIndexer.java
+++ /dev/null
@@ -1,138 +0,0 @@
-/*
- * Copyright 2013 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;
-
-import java.io.BufferedReader;
-import java.io.FileNotFoundException;
-import java.io.IOException;
-import java.io.Reader;
-import java.io.Writer;
-import java.util.ArrayList;
-import java.util.HashSet;
-import java.util.List;
-import java.util.Set;
-
-import javax.annotation.processing.AbstractProcessor;
-import javax.annotation.processing.RoundEnvironment;
-import javax.lang.model.SourceVersion;
-import javax.lang.model.element.Element;
-import javax.lang.model.element.TypeElement;
-import javax.tools.Diagnostic.Kind;
-import javax.tools.FileObject;
-import javax.tools.StandardLocation;
-
-/**
- * @author Decebal Suiu
- */
-public class ExtensionsIndexer extends AbstractProcessor {
-
- public static final String EXTENSIONS_RESOURCE = "META-INF/extensions.idx";
-
- private List<TypeElement> extensions = new ArrayList<>();
-
- @Override
- public SourceVersion getSupportedSourceVersion() {
- return SourceVersion.latest();
- }
-
- @Override
- public Set<String> getSupportedAnnotationTypes() {
- Set<String> annotationTypes = new HashSet<>();
- annotationTypes.add(Extension.class.getName());
-
- return annotationTypes;
- }
-
- @Override
- public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
- if (roundEnv.processingOver()) {
- return false;
- }
-
- for (Element element : roundEnv.getElementsAnnotatedWith(Extension.class)) {
- if (!(element instanceof TypeElement)) {
- continue;
- }
-
- TypeElement typeElement = (TypeElement) element;
- String message = "Extension found in " + processingEnv.getElementUtils().getBinaryName(typeElement).toString();
- processingEnv.getMessager().printMessage(Kind.NOTE, message);
- extensions.add(typeElement);
- }
-
- /*
- if (!roundEnv.processingOver()) {
- return false;
- }
- */
-
- write();
-
- return false;
-// return true; // no further processing of this annotation type
- }
-
- private void write() {
- Set<String> entries = new HashSet<>();
- for (TypeElement typeElement : extensions) {
- entries.add(processingEnv.getElementUtils().getBinaryName(typeElement).toString());
- }
-
- read(entries); // read old entries
- write(entries); // write entries
- }
-
- private void write(Set<String> entries) {
- try {
- FileObject file = processingEnv.getFiler().createResource(StandardLocation.CLASS_OUTPUT, "", EXTENSIONS_RESOURCE);
- Writer writer = file.openWriter();
- for (String entry : entries) {
- writer.write(entry);
- writer.write("\n");
- }
- writer.close();
- } catch (FileNotFoundException e) {
- // it's the first time, create the file
- } catch (IOException e) {
- processingEnv.getMessager().printMessage(Kind.ERROR, e.toString());
- }
- }
-
- private void read(Set<String> entries) {
- try {
- FileObject file = processingEnv.getFiler().getResource(StandardLocation.CLASS_OUTPUT, "", EXTENSIONS_RESOURCE);
- readIndex(file.openReader(true), entries);
- } catch (FileNotFoundException e) {
- } catch (IOException e) {
- // thrown by Eclipse JDT when not found
- } catch (UnsupportedOperationException e) {
- // java6 does not support reading old index files
- }
- }
-
- public static void readIndex(Reader reader, Set<String> entries) throws IOException {
- BufferedReader bufferedReader = new BufferedReader(reader);
-
- String line;
- while ((line = bufferedReader.readLine()) != null) {
- entries.add(line);
- }
-
- reader.close();
- }
-
-}
-
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/LegacyExtensionFinder.java b/pf4j/src/main/java/ro/fortsoft/pf4j/LegacyExtensionFinder.java
new file mode 100644
index 0000000..03a6bab
--- /dev/null
+++ b/pf4j/src/main/java/ro/fortsoft/pf4j/LegacyExtensionFinder.java
@@ -0,0 +1,122 @@
+/*
+ * Copyright 2013 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;
+
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import ro.fortsoft.pf4j.processor.LegacyExtensionStorage;
+
+import java.io.IOException;
+import java.io.InputStreamReader;
+import java.io.Reader;
+import java.net.URL;
+import java.util.Enumeration;
+import java.util.HashSet;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+
+/**
+ * All extensions declared in a plugin are indexed in a file "META-INF/extensions.idx".
+ * This class lookup extensions in all extensions index files "META-INF/extensions.idx".
+ *
+ * @author Decebal Suiu
+ */
+public class LegacyExtensionFinder extends AbstractExtensionFinder {
+
+ private static final Logger log = LoggerFactory.getLogger(LegacyExtensionFinder.class);
+
+ public LegacyExtensionFinder(PluginManager pluginManager) {
+ super(pluginManager);
+ }
+
+ @Override
+ public Map<String, Set<String>> readClasspathStorages() {
+ log.debug("Reading extensions storages from classpath");
+ Map<String, Set<String>> result = new LinkedHashMap<>();
+
+ Set<String> bucket = new HashSet<>();
+ try {
+ Enumeration<URL> urls = getClass().getClassLoader().getResources(getExtensionsResource());
+ while (urls.hasMoreElements()) {
+ URL url = urls.nextElement();
+ log.debug("Read '{}'", url.getFile());
+ Reader reader = new InputStreamReader(url.openStream(), "UTF-8");
+ LegacyExtensionStorage.read(reader, bucket);
+ }
+
+ if (bucket.isEmpty()) {
+ log.debug("No extensions found");
+ } else {
+ log.debug("Found possible {} extensions:", bucket.size());
+ for (String entry : bucket) {
+ log.debug(" " + entry);
+ }
+ }
+
+ result.put(null, bucket);
+ } catch (IOException e) {
+ log.error(e.getMessage(), e);
+ }
+
+ return result;
+ }
+
+ @Override
+ public Map<String, Set<String>> readPluginsStorages() {
+ log.debug("Reading extensions storages from plugins");
+ Map<String, Set<String>> result = new LinkedHashMap<>();
+
+ List<PluginWrapper> plugins = pluginManager.getPlugins();
+ for (PluginWrapper plugin : plugins) {
+ String pluginId = plugin.getDescriptor().getPluginId();
+ log.debug("Reading extensions storage for plugin '{}'", pluginId);
+ Set<String> bucket = new HashSet<>();
+
+ try {
+ URL url = ((PluginClassLoader) plugin.getPluginClassLoader()).findResource(getExtensionsResource());
+ if (url != null) {
+ log.debug("Read '{}'", url.getFile());
+ Reader reader = new InputStreamReader(url.openStream(), "UTF-8");
+ LegacyExtensionStorage.read(reader, bucket);
+ } else {
+ log.debug("Cannot find '{}'", getExtensionsResource());
+ }
+
+ if (bucket.isEmpty()) {
+ log.debug("No extensions found");
+ } else {
+ log.debug("Found possible {} extensions:", bucket.size());
+ for (String entry : bucket) {
+ log.debug(" " + entry);
+ }
+ }
+
+ result.put(pluginId, bucket);
+ } catch (IOException e) {
+ log.error(e.getMessage(), e);
+ }
+ }
+
+ return result;
+ }
+
+ private static String getExtensionsResource() {
+ return LegacyExtensionStorage.EXTENSIONS_RESOURCE;
+ }
+
+}
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/PluginClassLoader.java b/pf4j/src/main/java/ro/fortsoft/pf4j/PluginClassLoader.java
index c2f81df..b231ffd 100644
--- a/pf4j/src/main/java/ro/fortsoft/pf4j/PluginClassLoader.java
+++ b/pf4j/src/main/java/ro/fortsoft/pf4j/PluginClassLoader.java
@@ -129,6 +129,11 @@ public class PluginClassLoader extends URLClassLoader {
return super.getResource(name);
}
+ @Override
+ public URL findResource(String name) {
+ return super.findResource(name);
+ }
+
/**
* Release all resources acquired by this class loader.
* The current implementation is incomplete.
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/ServiceProviderExtensionFinder.java b/pf4j/src/main/java/ro/fortsoft/pf4j/ServiceProviderExtensionFinder.java
new file mode 100644
index 0000000..05c9cb4
--- /dev/null
+++ b/pf4j/src/main/java/ro/fortsoft/pf4j/ServiceProviderExtensionFinder.java
@@ -0,0 +1,132 @@
+/*
+ * Copyright 2015 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;
+
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import ro.fortsoft.pf4j.processor.ServiceProviderExtensionStorage;
+
+import java.io.File;
+import java.io.FileReader;
+import java.io.IOException;
+import java.io.Reader;
+import java.net.URISyntaxException;
+import java.net.URL;
+import java.util.HashSet;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+
+/**
+ * The ServiceLoader base implementation for ExtensionFinder.
+ * This class lookup extensions in all extensions index files "META-INF/services".
+ *
+ * @author Decebal Suiu
+ */
+public class ServiceProviderExtensionFinder extends AbstractExtensionFinder {
+
+ private static final Logger log = LoggerFactory.getLogger(ServiceProviderExtensionFinder.class);
+
+ public ServiceProviderExtensionFinder(PluginManager pluginManager) {
+ super(pluginManager);
+ }
+
+ @Override
+ public Map<String, Set<String>> readClasspathStorages() {
+ log.debug("Reading extensions storages from classpath");
+ Map<String, Set<String>> result = new LinkedHashMap<>();
+
+ Set<String> bucket = new HashSet<>();
+ try {
+ URL url = getClass().getClassLoader().getResource(getExtensionsResource());
+ if (url != null) {
+ File[] files = new File(url.toURI()).listFiles();
+ if (files != null) {
+ for (File file : files) {
+ log.debug("Read '{}'", file);
+ Reader reader = new FileReader(file);
+ ServiceProviderExtensionStorage.read(reader, bucket);
+ }
+ }
+ }
+
+ if (bucket.isEmpty()) {
+ log.debug("No extensions found");
+ } else {
+ log.debug("Found possible {} extensions:", bucket.size());
+ for (String entry : bucket) {
+ log.debug(" " + entry);
+ }
+ }
+
+ result.put(null, bucket);
+ } catch (IOException | URISyntaxException e) {
+ log.error(e.getMessage(), e);
+ }
+
+ return result;
+ }
+
+ @Override
+ public Map<String, Set<String>> readPluginsStorages() {
+ log.debug("Reading extensions storages from plugins");
+ Map<String, Set<String>> result = new LinkedHashMap<>();
+
+ List<PluginWrapper> plugins = pluginManager.getPlugins();
+ for (PluginWrapper plugin : plugins) {
+ String pluginId = plugin.getDescriptor().getPluginId();
+ log.debug("Reading extensions storages for plugin '{}'", pluginId);
+ Set<String> bucket = new HashSet<>();
+
+ try {
+ URL url = ((PluginClassLoader) plugin.getPluginClassLoader()).findResource(getExtensionsResource());
+ if (url != null) {
+ File[] files = new File(url.toURI()).listFiles();
+ if (files != null) {
+ for (File file : files) {
+ log.debug("Read '{}'", file);
+ Reader reader = new FileReader(file);
+ ServiceProviderExtensionStorage.read(reader, bucket);
+ }
+ }
+ } else {
+ log.debug("Cannot find '{}'", getExtensionsResource());
+ }
+
+ if (bucket.isEmpty()) {
+ log.debug("No extensions found");
+ } else {
+ log.debug("Found possible {} extensions:", bucket.size());
+ for (String entry : bucket) {
+ log.debug(" " + entry);
+ }
+ }
+
+ result.put(pluginId, bucket);
+ } catch (IOException | URISyntaxException e) {
+ log.error(e.getMessage(), e);
+ }
+ }
+
+ return result;
+ }
+
+ private static String getExtensionsResource() {
+ return ServiceProviderExtensionStorage.EXTENSIONS_RESOURCE;
+ }
+
+}
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/processor/ExtensionAnnotationProcessor.java b/pf4j/src/main/java/ro/fortsoft/pf4j/processor/ExtensionAnnotationProcessor.java
new file mode 100644
index 0000000..881c05f
--- /dev/null
+++ b/pf4j/src/main/java/ro/fortsoft/pf4j/processor/ExtensionAnnotationProcessor.java
@@ -0,0 +1,198 @@
+/*
+ * Copyright 2013 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.processor;
+
+import ro.fortsoft.pf4j.Extension;
+import ro.fortsoft.pf4j.ExtensionPoint;
+
+import javax.annotation.processing.AbstractProcessor;
+import javax.annotation.processing.ProcessingEnvironment;
+import javax.annotation.processing.RoundEnvironment;
+import javax.lang.model.SourceVersion;
+import javax.lang.model.element.Element;
+import javax.lang.model.element.TypeElement;
+import javax.lang.model.type.DeclaredType;
+import javax.lang.model.type.TypeKind;
+import javax.lang.model.type.TypeMirror;
+import javax.tools.Diagnostic;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+import java.util.TreeSet;
+
+/**
+ * @author Decebal Suiu
+ */
+public class ExtensionAnnotationProcessor extends AbstractProcessor {
+
+ private Map<String, Set<String>> extensions = new HashMap<>(); // the key is the extension point
+ private Map<String, Set<String>> oldExtensions = new HashMap<>(); // the key is the extension point
+
+ private ExtensionStorage storage;
+
+ @Override
+ public synchronized void init(ProcessingEnvironment processingEnv) {
+ super.init(processingEnv);
+
+ storage = new LegacyExtensionStorage(this);
+// storage = new ServiceProviderExtensionStorage(this);
+ }
+
+ @Override
+ public SourceVersion getSupportedSourceVersion() {
+ return SourceVersion.latest();
+ }
+
+ @Override
+ public Set<String> getSupportedAnnotationTypes() {
+ Set<String> annotationTypes = new HashSet<>();
+ annotationTypes.add(Extension.class.getName());
+
+ return annotationTypes;
+ }
+
+ @Override
+ public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
+ if (roundEnv.processingOver()) {
+ return false;
+ }
+
+ for (Element element : roundEnv.getElementsAnnotatedWith(Extension.class)) {
+ // check if @Extension is put on class and not on method or constructor
+ if (!(element instanceof TypeElement)) {
+ continue;
+ }
+
+ // check if class extends/implements an extension point
+ if (!isExtension(element.asType())) {
+ continue;
+ }
+
+ TypeElement extensionElement = (TypeElement) element;
+// Extension annotation = element.getAnnotation(Extension.class);
+ List<TypeElement> extensionPointElements = findExtensionPoints(extensionElement);
+ if (extensionPointElements.isEmpty()) {
+ // TODO throw error ?
+ continue;
+ }
+
+ String extension = getBinaryName(extensionElement);
+ for (TypeElement extensionPointElement : extensionPointElements) {
+ String extensionPoint = getBinaryName(extensionPointElement);
+ Set<String> extensionPoints = extensions.get(extensionPoint);
+ if (extensionPoints == null) {
+ extensionPoints = new TreeSet<>();
+ extensions.put(extensionPoint, extensionPoints);
+ }
+ extensionPoints.add(extension);
+ }
+ }
+
+ /*
+ if (!roundEnv.processingOver()) {
+ return false;
+ }
+ */
+
+ // read old extensions
+ oldExtensions = storage.read();
+ for (Map.Entry<String, Set<String>> entry : oldExtensions.entrySet()) {
+ String extensionPoint = entry.getKey();
+ if (extensions.containsKey(extensionPoint)) {
+ extensions.get(extensionPoint).addAll(entry.getValue());
+ } else {
+ extensions.put(extensionPoint, entry.getValue());
+ }
+ }
+
+ // write extensions
+ storage.write(extensions);
+
+ return false;
+// return true; // no further processing of this annotation type
+ }
+
+ public ProcessingEnvironment getProcessingEnvironment() {
+ return processingEnv;
+ }
+
+ public void error(String message, Object... args) {
+ processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR, String.format(message, args));
+ }
+
+ public void error(Element element, String message, Object... args) {
+ processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR, String.format(message, args), element);
+ }
+
+ public void note(String message, Object... args) {
+ processingEnv.getMessager().printMessage(Diagnostic.Kind.NOTE, String.format(message, args));
+ }
+
+ public void note(Element element, String message, Object... args) {
+ processingEnv.getMessager().printMessage(Diagnostic.Kind.NOTE, String.format(message, args), element);
+ }
+
+ public String getBinaryName(TypeElement element) {
+ return processingEnv.getElementUtils().getBinaryName(element).toString();
+ }
+
+ public Map<String, Set<String>> getExtensions() {
+ return extensions;
+ }
+
+ public Map<String, Set<String>> getOldExtensions() {
+ return oldExtensions;
+ }
+
+ private List<TypeElement> findExtensionPoints(TypeElement extensionElement) {
+ List<TypeElement> extensionPointElements = new ArrayList<>();
+
+ // search in interfaces
+ for (TypeMirror item : extensionElement.getInterfaces()) {
+ boolean isExtensionPoint = processingEnv.getTypeUtils().isSubtype(item, getExtensionPointType());
+ if (isExtensionPoint) {
+ TypeElement extensionPointElement = (TypeElement) ((DeclaredType) item).asElement();
+ extensionPointElements.add(extensionPointElement);
+ }
+ }
+
+ // search in superclass
+ TypeMirror superclass = extensionElement.getSuperclass();
+ if (superclass.getKind() != TypeKind.NONE) {
+ boolean isExtensionPoint = processingEnv.getTypeUtils().isSubtype(superclass, getExtensionPointType());
+ if (isExtensionPoint) {
+ TypeElement extensionPointElement = (TypeElement) ((DeclaredType) superclass).asElement();
+ extensionPointElements.add(extensionPointElement);
+ }
+ }
+
+ return extensionPointElements;
+ }
+
+ private boolean isExtension(TypeMirror typeMirror) {
+ return processingEnv.getTypeUtils().isAssignable(typeMirror, getExtensionPointType());
+ }
+
+
+ private TypeMirror getExtensionPointType() {
+ return processingEnv.getElementUtils().getTypeElement(ExtensionPoint.class.getName()).asType();
+ }
+
+}
+
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/processor/ExtensionStorage.java b/pf4j/src/main/java/ro/fortsoft/pf4j/processor/ExtensionStorage.java
new file mode 100644
index 0000000..9aeec14
--- /dev/null
+++ b/pf4j/src/main/java/ro/fortsoft/pf4j/processor/ExtensionStorage.java
@@ -0,0 +1,73 @@
+/*
+ * Copyright 2015 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.processor;
+
+import javax.annotation.processing.Filer;
+import javax.lang.model.element.Element;
+import java.util.Map;
+import java.util.Set;
+
+/**
+ * @author Decebal Suiu
+ */
+public abstract class ExtensionStorage {
+
+ protected final ExtensionAnnotationProcessor processor;
+
+ public ExtensionStorage(ExtensionAnnotationProcessor processor) {
+ this.processor = processor;
+ }
+
+ public abstract Map<String, Set<String>> read();
+
+ public abstract void write(Map<String, Set<String>> extensions);
+
+ /**
+ * Helper method.
+ */
+ protected Filer getFiler() {
+ return processor.getProcessingEnvironment().getFiler();
+ }
+
+ /**
+ * Helper method.
+ */
+ protected void error(String message, Object... args) {
+ processor.error(message, args);
+ }
+
+ /**
+ * Helper method.
+ */
+ protected void error(Element element, String message, Object... args) {
+ processor.error(element, message, args);
+ }
+
+ /**
+ * Helper method.
+ */
+ protected void note(String message, Object... args) {
+ processor.note(message, args);
+ }
+
+ /**
+ * Helper method.
+ */
+ protected void note(Element element, String message, Object... args) {
+ processor.note(element, message, args);
+ }
+
+}
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/processor/LegacyExtensionStorage.java b/pf4j/src/main/java/ro/fortsoft/pf4j/processor/LegacyExtensionStorage.java
new file mode 100644
index 0000000..e742178
--- /dev/null
+++ b/pf4j/src/main/java/ro/fortsoft/pf4j/processor/LegacyExtensionStorage.java
@@ -0,0 +1,101 @@
+/*
+ * Copyright 2015 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.processor;
+
+import javax.tools.FileObject;
+import javax.tools.StandardLocation;
+import java.io.BufferedReader;
+import java.io.BufferedWriter;
+import java.io.FileNotFoundException;
+import java.io.IOException;
+import java.io.Reader;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.Map;
+import java.util.Set;
+import java.util.regex.Pattern;
+
+/**
+ * @author Decebal Suiu
+ */
+public class LegacyExtensionStorage extends ExtensionStorage {
+
+ public static final String EXTENSIONS_RESOURCE = "META-INF/extensions.idx";
+
+ private static final Pattern COMMENT = Pattern.compile("#.*");
+ private static final Pattern WHITESPACE = Pattern.compile("\\s+");
+
+ public LegacyExtensionStorage(ExtensionAnnotationProcessor processor) {
+ super(processor);
+ }
+
+ public static void read(Reader reader, Set<String> entries) throws IOException {
+ BufferedReader bufferedReader = new BufferedReader(reader);
+
+ String line;
+ while ((line = bufferedReader.readLine()) != null) {
+ line = COMMENT.matcher(line).replaceFirst("");
+ line = WHITESPACE.matcher(line).replaceAll("");
+ if (line.length() > 0) {
+ entries.add(line);
+ }
+ }
+
+ bufferedReader.close();
+ }
+
+ @Override
+ public Map<String, Set<String>> read() {
+ Map<String, Set<String>> extensions = new HashMap<>();
+
+ try {
+ FileObject file = getFiler().getResource(StandardLocation.CLASS_OUTPUT, "", EXTENSIONS_RESOURCE);
+ // TODO try to calculate the extension point
+ Set<String> entries = new HashSet<>();
+ read(file.openReader(true), entries);
+ extensions.put(null, entries);
+ } catch (FileNotFoundException e) {
+ // ignore
+ } catch (IOException e) {
+ error(e.getMessage());
+ }
+
+ return extensions;
+ }
+
+ @Override
+ public void write(Map<String, Set<String>> extensions) {
+ try {
+ FileObject file = getFiler().createResource(StandardLocation.CLASS_OUTPUT, "", EXTENSIONS_RESOURCE);
+ BufferedWriter writer = new BufferedWriter(file.openWriter());
+ writer.write("# Generated by PF4J"); // write header
+ writer.newLine();
+ for (Map.Entry<String, Set<String>> entry : extensions.entrySet()) {
+ for (String extension : entry.getValue()) {
+ writer.write(extension);
+ writer.newLine();
+ }
+ }
+
+ writer.close();
+ } catch (FileNotFoundException e) {
+ // it's the first time, create the file
+ } catch (IOException e) {
+ error(e.toString());
+ }
+ }
+
+}
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/processor/ServiceProviderExtensionStorage.java b/pf4j/src/main/java/ro/fortsoft/pf4j/processor/ServiceProviderExtensionStorage.java
new file mode 100644
index 0000000..8fb6b35
--- /dev/null
+++ b/pf4j/src/main/java/ro/fortsoft/pf4j/processor/ServiceProviderExtensionStorage.java
@@ -0,0 +1,114 @@
+/*
+ * Copyright 2015 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.processor;
+
+import javax.tools.FileObject;
+import javax.tools.StandardLocation;
+import java.io.BufferedReader;
+import java.io.BufferedWriter;
+import java.io.FileNotFoundException;
+import java.io.IOException;
+import java.io.Reader;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.Map;
+import java.util.Set;
+import java.util.regex.Pattern;
+
+/**
+ * @author Decebal Suiu
+ */
+public class ServiceProviderExtensionStorage extends ExtensionStorage {
+
+ public static final String EXTENSIONS_RESOURCE = "META-INF/services";
+
+ private static final Pattern COMMENT = Pattern.compile("#.*");
+ private static final Pattern WHITESPACE = Pattern.compile("\\s+");
+
+ public ServiceProviderExtensionStorage(ExtensionAnnotationProcessor processor) {
+ super(processor);
+ }
+
+ public static void read(Reader reader, Set<String> entries) throws IOException {
+ BufferedReader bufferedReader = new BufferedReader(reader);
+
+ String line;
+ while ((line = bufferedReader.readLine()) != null) {
+ line = COMMENT.matcher(line).replaceFirst("");
+ line = WHITESPACE.matcher(line).replaceAll("");
+ if (line.length() > 0) {
+ entries.add(line);
+ }
+ }
+
+ bufferedReader.close();
+ }
+
+ @Override
+ public Map<String, Set<String>> read() {
+ Map<String, Set<String>> extensions = new HashMap<>();
+
+ for (String extensionPoint : processor.getExtensions().keySet()) {
+ try {
+ FileObject file = getFiler().getResource(StandardLocation.CLASS_OUTPUT, "", EXTENSIONS_RESOURCE
+ + "/" + extensionPoint);
+ Set<String> entries = new HashSet<>();
+ read(file.openReader(true), entries);
+ extensions.put(extensionPoint, entries);
+ } catch (FileNotFoundException e) {
+ // doesn't exist, ignore
+ } catch (IOException e) {
+ error(e.getMessage());
+ }
+ }
+
+ return extensions;
+ }
+
+ @Override
+ public void write(Map<String, Set<String>> extensions) {
+ for (Map.Entry<String, Set<String>> entry : extensions.entrySet()) {
+ String extensionPoint = entry.getKey();
+ try {
+ FileObject file = getFiler().createResource(StandardLocation.CLASS_OUTPUT, "", EXTENSIONS_RESOURCE
+ + "/" + extensionPoint);
+ BufferedWriter writer = new BufferedWriter(file.openWriter());
+ // write header
+ writer.write("# Generated by PF4J"); // write header
+ writer.newLine();
+ // write extensions
+ for (String extension : entry.getValue()) {
+ writer.write(extension);
+ if (!isExtensionOld(extensionPoint, extension)) {
+ writer.write(" # pf4j extension");
+ }
+ writer.newLine();
+ }
+ writer.close();
+ } catch (FileNotFoundException e) {
+ // it's the first time, create the file
+ } catch (IOException e) {
+ error(e.toString());
+ }
+ }
+ }
+
+ private boolean isExtensionOld(String extensionPoint, String extension) {
+ return processor.getOldExtensions().containsKey(extensionPoint)
+ && processor.getOldExtensions().get(extensionPoint).contains(extension);
+ }
+
+}
diff --git a/pf4j/src/main/java/ro/fortsoft/pf4j/util/CompoundClassLoader.java b/pf4j/src/main/java/ro/fortsoft/pf4j/util/CompoundClassLoader.java
deleted file mode 100644
index d61ade5..0000000
--- a/pf4j/src/main/java/ro/fortsoft/pf4j/util/CompoundClassLoader.java
+++ /dev/null
@@ -1,79 +0,0 @@
-/*
- * Copyright 2012 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.io.IOException;
-import java.net.URL;
-import java.util.ArrayList;
-import java.util.Collections;
-import java.util.Enumeration;
-import java.util.HashSet;
-import java.util.List;
-import java.util.Set;
-
-/**
- * A class loader that has multiple loaders and uses them for loading classes and resources.
- *
- * @author Decebal Suiu
- */
-public class CompoundClassLoader extends ClassLoader {
-
- private Set<ClassLoader> loaders = new HashSet<>();
-
- public void addLoader(ClassLoader loader) {
- loaders.add(loader);
- }
-
- public void removeLoader(ClassLoader loader) {
- loaders.remove(loader);
- }
-
- @Override
- public Class<?> findClass(String name) throws ClassNotFoundException {
- for (ClassLoader loader : loaders) {
- try {
- return loader.loadClass(name);
- } catch (ClassNotFoundException e) {
- // try next
- }
- }
-
- throw new ClassNotFoundException(name);
- }
-
- @Override
- public URL findResource(String name) {
- for (ClassLoader loader : loaders) {
- URL url = loader.getResource(name);
- if (url != null) {
- return url;
- }
- }
-
- return null;
- }
-
- @Override
- protected Enumeration<URL> findResources(String name) throws IOException {
- List<URL> resources = new ArrayList<>();
- for (ClassLoader loader : loaders) {
- resources.addAll(Collections.list(loader.getResources(name)));
- }
-
- return Collections.enumeration(resources);
- }
-
-}
diff --git a/pf4j/src/main/resources/META-INF/services/javax.annotation.processing.Processor b/pf4j/src/main/resources/META-INF/services/javax.annotation.processing.Processor
index a944d7c..c616e6e 100644
--- a/pf4j/src/main/resources/META-INF/services/javax.annotation.processing.Processor
+++ b/pf4j/src/main/resources/META-INF/services/javax.annotation.processing.Processor
@@ -1 +1 @@
-ro.fortsoft.pf4j.ExtensionsIndexer
+ro.fortsoft.pf4j.processor.ExtensionAnnotationProcessor