org.scijava.plugin.DefaultPluginService Maven / Gradle / Ivy
Show all versions of scijava-common Show documentation
/*
* #%L
* SciJava Common shared library for SciJava software.
* %%
* Copyright (C) 2009 - 2017 Board of Regents of the University of
* Wisconsin-Madison, Broad Institute of MIT and Harvard, Max Planck
* Institute of Molecular Cell Biology and Genetics, University of
* Konstanz, and KNIME GmbH.
* %%
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
* #L%
*/
package org.scijava.plugin;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import org.scijava.InstantiableException;
import org.scijava.event.EventService;
import org.scijava.log.LogService;
import org.scijava.plugin.event.PluginsAddedEvent;
import org.scijava.plugin.event.PluginsRemovedEvent;
import org.scijava.service.AbstractService;
import org.scijava.service.Service;
import org.scijava.util.ListUtils;
/**
* Default service for keeping track of available plugins.
*
* Available plugins are discovered using indexes generated by using
* scijava-common as annotation processor. Loading of the actual plugin
* classes can be deferred until a particular plugin is actually needed.
*
*
* Plugins are added or removed via the plugin service are reported via the
* event service. (No events are published for plugins directly added to or
* removed from the {@link PluginIndex}.)
*
*
* @author Curtis Rueden
* @author Johannes Schindelin
* @see SciJavaPlugin
* @see Plugin
*/
@Plugin(type = Service.class)
public class DefaultPluginService extends AbstractService implements
PluginService
{
@Parameter
private LogService log;
@Parameter
private EventService eventService;
/** Index of registered plugins. */
private PluginIndex pluginIndex;
// -- PluginService methods --
@Override
public PluginIndex getIndex() {
return pluginIndex;
}
@Override
public void reloadPlugins() {
// clear all old plugins, and notify interested parties
final List> oldPlugins = pluginIndex.getAll();
pluginIndex.clear();
if (oldPlugins.size() > 0) {
eventService.publish(new PluginsRemovedEvent(oldPlugins));
}
// re-discover all available plugins, and notify interested parties
pluginIndex.discover();
final List> newPlugins = pluginIndex.getAll();
if (newPlugins.size() > 0) {
eventService.publish(new PluginsAddedEvent(newPlugins));
}
logExceptions();
}
@Override
public void addPlugin(final PluginInfo> plugin) {
if (pluginIndex.add(plugin)) {
eventService.publish(new PluginsAddedEvent(plugin));
}
}
@Override
public > void
addPlugins(final Collection plugins)
{
if (pluginIndex.addAll(plugins)) {
eventService.publish(new PluginsAddedEvent(plugins));
}
}
@Override
public void removePlugin(final PluginInfo> plugin) {
if (pluginIndex.remove(plugin)) {
eventService.publish(new PluginsRemovedEvent(plugin));
}
}
@Override
public > void removePlugins(
final Collection plugins)
{
if (pluginIndex.removeAll(plugins)) {
eventService.publish(new PluginsRemovedEvent(plugins));
}
}
@Override
public List> getPlugins() {
return pluginIndex.getAll();
}
@Override
public PluginInfo getPlugin(
final Class pluginClass)
{
return ListUtils.first(getPluginsOfClass(pluginClass));
}
@Override
public PluginInfo
getPlugin(final Class pluginClass, final Class type)
{
return ListUtils.first(getPluginsOfClass(pluginClass, type));
}
@Override
public PluginInfo getPlugin(final String className) {
return ListUtils.first(getPluginsOfClass(className));
}
@Override
public List> getPluginsOfType(
final Class type)
{
return pluginIndex.getPlugins(type);
}
@Override
public List>
getPluginsOfClass(final Class pluginClass)
{
// NB: We must scan *all* plugins for a match. In theory, the same plugin
// Class could be associated with multiple PluginInfo entries of differing
// type anyway. If performance of this method is insufficient, the solution
// will be to rework the PluginIndex data structure to include an index on
// plugin class names.
return getPluginsOfClass(pluginClass, SciJavaPlugin.class);
}
@Override
public List>
getPluginsOfClass(final Class pluginClass, final Class type)
{
final ArrayList> result = new ArrayList<>();
findPluginsOfClass(pluginClass, getPluginsOfType(type), result);
filterNonmatchingClasses(pluginClass, result);
return result;
}
@Override
public List> getPluginsOfClass(
final String className)
{
// NB: Since we cannot load the class in question, and cannot know its type
// hierarch(y/ies) even if we did, we must scan *all* plugins for a match.
return getPluginsOfClass(className, SciJavaPlugin.class);
}
@Override
public List>
getPluginsOfClass(final String className, final Class type)
{
final ArrayList> result =
new ArrayList<>();
findPluginsOfClass(className, getPluginsOfType(type), result);
return result;
}
@Override
public List createInstancesOfType(
final Class type)
{
final List> plugins = getPluginsOfType(type);
return createInstances(plugins);
}
@Override
public List createInstances(
final List> infos)
{
final ArrayList list = new ArrayList<>();
for (final PluginInfo extends PT> info : infos) {
final PT p = createInstance(info);
if (p != null) list.add(p);
}
return list;
}
@Override
public PT
createInstance(final PluginInfo info)
{
try {
final PT p = info.createInstance();
context().inject(p);
return p;
}
catch (final Throwable t) {
log.error("Cannot create plugin: " + info, t);
}
return null;
}
// -- Service methods --
@Override
public void initialize() {
pluginIndex = context().getPluginIndex();
log.debug("Found " + pluginIndex.size() + " plugins.");
if (log.isDebug()) {
for (final PluginInfo> info : pluginIndex) {
log.debug("- " + info);
}
}
logExceptions();
super.initialize();
}
// -- Utility methods --
/**
* Transfers plugins of the given class from the source list to the
* destination list. Note that because this method compares class name
* strings, it does not need to actually load the class in question.
*
* @param className The class name of the desired plugins.
* @param srcList The list to scan for matching plugins.
* @param destList The list to which matching plugins are added.
*/
public static > void findPluginsOfClass(
final String className, final List extends PluginInfo>> srcList,
final List destList)
{
for (final PluginInfo> info : srcList) {
if (info.getClassName().equals(className)) {
@SuppressWarnings("unchecked")
final T match = (T) info;
destList.add(match);
}
}
}
/**
* Gets the plugin type of the given plugin class, as declared by its
* {@code @Plugin} annotation (i.e., {@link Plugin#type()}).
*
* @param pluginClass The plugin class whose plugin type is needed.
* @return The plugin type, or null if no {@link Plugin} annotation exists for
* the given class.
*/
public static Class getPluginType(
final Class pluginClass)
{
final Plugin annotation = pluginClass.getAnnotation(Plugin.class);
if (annotation == null) return null;
@SuppressWarnings("unchecked")
final Class type = (Class) annotation.type();
return type;
}
// -- Helper methods --
/**
* Transfers plugins of the given class from the source list to the
* destination list. Note that because this method compares class objects, it
* must load the classes in question.
*
* @param pluginClass The class of the desired plugins.
* @param srcList The list to scan for matching plugins.
* @param destList The list to which matching plugins are added.
*/
private > void findPluginsOfClass(
final Class> pluginClass, final List extends PluginInfo>> srcList,
final List destList)
{
final String className = pluginClass.getName();
for (final PluginInfo> info : srcList) {
try {
final Class> clazz2 = info.getPluginClass();
if (clazz2 == pluginClass ||
(info.getClassName().equals(className) && info.loadClass() == pluginClass))
{
@SuppressWarnings("unchecked")
final T match = (T) info;
destList.add(match);
}
}
catch (InstantiableException exc) {
log.debug("Ignoring plugin: " + info, exc);
}
}
}
/**
* Filters the given list to include only entries with matching
* classes (not just class names).
*/
private void
filterNonmatchingClasses(final Class pluginClass,
final ArrayList> result)
{
for (final Iterator> iter = result.iterator();
iter.hasNext(); )
{
try {
if (iter.next().loadClass() != pluginClass) iter.remove();
}
catch (InstantiableException exc) {
log.debug(exc);
iter.remove();
}
}
}
/** Logs any exceptions that occurred during the last plugin discovery. */
private void logExceptions() {
final Map exceptions = pluginIndex.getExceptions();
final int excCount = exceptions.size();
if (excCount > 0) {
log.warn(excCount + " exceptions occurred during plugin discovery.");
if (log.isDebug()) {
for (final String name : exceptions.keySet()) {
final Throwable t = exceptions.get(name);
log.debug(name, t);
}
}
}
}
}