2010-10-31 19:22:53 +01:00
|
|
|
/**
|
|
|
|
|
* Copyright (c) 2010, Sebastian Sdorra
|
|
|
|
|
* All rights reserved.
|
|
|
|
|
*
|
|
|
|
|
* 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.
|
|
|
|
|
* 3. Neither the name of SCM-Manager; nor the names of its
|
|
|
|
|
* contributors may be used to endorse or promote products derived from this
|
|
|
|
|
* software without specific prior written permission.
|
|
|
|
|
*
|
|
|
|
|
* 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 REGENTS 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.
|
|
|
|
|
*
|
|
|
|
|
* http://bitbucket.org/sdorra/scm-manager
|
|
|
|
|
*
|
2010-10-12 09:16:40 +02:00
|
|
|
*/
|
|
|
|
|
|
2010-12-01 14:26:29 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
package sonia.scm.plugin;
|
2010-10-12 09:16:40 +02:00
|
|
|
|
|
|
|
|
//~--- non-JDK imports --------------------------------------------------------
|
|
|
|
|
|
2014-03-28 22:49:55 +01:00
|
|
|
import com.google.common.base.Throwables;
|
2013-01-18 09:39:32 +01:00
|
|
|
import com.google.common.collect.ImmutableSet;
|
2014-03-28 22:49:55 +01:00
|
|
|
import com.google.common.collect.ImmutableSet.Builder;
|
|
|
|
|
import com.google.common.collect.Iterables;
|
2013-01-18 09:39:32 +01:00
|
|
|
import com.google.common.collect.Sets;
|
|
|
|
|
import com.google.inject.Binder;
|
|
|
|
|
import com.google.inject.Module;
|
|
|
|
|
|
2010-10-16 11:03:54 +02:00
|
|
|
import org.slf4j.Logger;
|
|
|
|
|
import org.slf4j.LoggerFactory;
|
|
|
|
|
|
2010-10-12 09:16:40 +02:00
|
|
|
//~--- JDK imports ------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
import java.io.IOException;
|
|
|
|
|
|
|
|
|
|
import java.net.URL;
|
|
|
|
|
|
2010-12-01 14:26:29 +01:00
|
|
|
import java.util.Collection;
|
2010-10-12 09:16:40 +02:00
|
|
|
import java.util.Enumeration;
|
|
|
|
|
import java.util.Set;
|
|
|
|
|
|
2014-08-27 21:01:05 +02:00
|
|
|
import javax.servlet.ServletContext;
|
|
|
|
|
|
2014-03-28 22:49:55 +01:00
|
|
|
import javax.xml.bind.JAXBContext;
|
|
|
|
|
import javax.xml.bind.JAXBException;
|
2010-10-12 09:16:40 +02:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
*
|
|
|
|
|
* @author Sebastian Sdorra
|
|
|
|
|
*/
|
2010-12-13 18:59:00 +01:00
|
|
|
public class DefaultPluginLoader implements PluginLoader
|
2010-10-12 09:16:40 +02:00
|
|
|
{
|
|
|
|
|
|
2010-12-21 17:08:41 +01:00
|
|
|
/** Field description */
|
2014-03-28 22:49:55 +01:00
|
|
|
public static final String PATH_MODULECONFIG = "META-INF/scm/module.xml";
|
2013-01-18 09:39:32 +01:00
|
|
|
|
2010-10-12 09:16:40 +02:00
|
|
|
/** Field description */
|
2010-12-01 14:26:29 +01:00
|
|
|
public static final String PATH_PLUGINCONFIG = "META-INF/scm/plugin.xml";
|
|
|
|
|
|
2010-12-13 18:59:00 +01:00
|
|
|
/** the logger for DefaultPluginLoader */
|
2010-10-12 09:16:40 +02:00
|
|
|
private static final Logger logger =
|
2010-12-13 18:59:00 +01:00
|
|
|
LoggerFactory.getLogger(DefaultPluginLoader.class);
|
2010-10-12 09:16:40 +02:00
|
|
|
|
2010-12-01 14:26:29 +01:00
|
|
|
//~--- constructors ---------------------------------------------------------
|
2010-10-12 09:16:40 +02:00
|
|
|
|
|
|
|
|
/**
|
2010-12-01 14:26:29 +01:00
|
|
|
* Constructs ...
|
2010-10-12 09:16:40 +02:00
|
|
|
*
|
2014-08-27 21:01:05 +02:00
|
|
|
* @param servletContext
|
2014-06-06 08:57:41 +02:00
|
|
|
* @param parent
|
2014-08-27 21:01:05 +02:00
|
|
|
* @param installedPlugins
|
2010-10-12 09:16:40 +02:00
|
|
|
*/
|
2014-08-27 21:01:05 +02:00
|
|
|
public DefaultPluginLoader(ServletContext servletContext, ClassLoader parent,
|
|
|
|
|
Set<PluginWrapper> installedPlugins)
|
2010-10-12 09:16:40 +02:00
|
|
|
{
|
2014-08-27 21:01:05 +02:00
|
|
|
this.installedPlugins = installedPlugins;
|
|
|
|
|
this.uberClassLoader = new UberClassLoader(parent, installedPlugins);
|
|
|
|
|
this.uberWebResourceLoader =
|
|
|
|
|
new DefaultUberWebResourceLoader(servletContext, installedPlugins);
|
2010-10-12 09:16:40 +02:00
|
|
|
|
2010-12-01 14:26:29 +01:00
|
|
|
try
|
2010-10-12 09:16:40 +02:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
JAXBContext context = JAXBContext.newInstance(ScmModule.class,
|
|
|
|
|
Plugin.class);
|
2010-10-12 09:16:40 +02:00
|
|
|
|
2014-06-06 08:57:41 +02:00
|
|
|
modules = getInstalled(parent, context, PATH_MODULECONFIG);
|
|
|
|
|
|
2014-03-28 22:49:55 +01:00
|
|
|
appendExtensions(multiple, single, extensions, modules);
|
2014-08-27 21:01:05 +02:00
|
|
|
appendExtensions(multiple, single, extensions, unwrap());
|
2013-01-18 09:39:32 +01:00
|
|
|
}
|
2014-06-06 08:57:41 +02:00
|
|
|
catch (IOException | JAXBException ex)
|
2013-01-18 09:39:32 +01:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
throw Throwables.propagate(ex);
|
2013-01-18 09:39:32 +01:00
|
|
|
}
|
2013-01-15 16:55:07 +01:00
|
|
|
}
|
|
|
|
|
|
2014-03-28 22:49:55 +01:00
|
|
|
//~--- methods --------------------------------------------------------------
|
|
|
|
|
|
2013-01-18 09:39:32 +01:00
|
|
|
/**
|
|
|
|
|
* Method description
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* @param binder
|
|
|
|
|
*/
|
2014-03-28 22:49:55 +01:00
|
|
|
@Override
|
2013-01-18 09:39:32 +01:00
|
|
|
public void processExtensions(Binder binder)
|
|
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
logger.info("start processing extensions");
|
2013-01-18 09:39:32 +01:00
|
|
|
|
2014-03-28 22:49:55 +01:00
|
|
|
if (logger.isInfoEnabled())
|
2012-04-11 17:53:39 +02:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
logger.info(
|
|
|
|
|
"found {} extensions for {} multiple and {} single extension points",
|
|
|
|
|
extensions.size(), multiple.size(), single.size());
|
2010-10-12 09:16:40 +02:00
|
|
|
}
|
2014-03-28 22:49:55 +01:00
|
|
|
|
|
|
|
|
new ExtensionBinder(binder).bind(multiple, single, extensions);
|
2010-10-12 09:16:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//~--- get methods ----------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Method description
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
2010-12-01 14:26:29 +01:00
|
|
|
@Override
|
2014-03-28 22:49:55 +01:00
|
|
|
public Set<Module> getInjectionModules()
|
2010-10-12 09:16:40 +02:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
return ImmutableSet.copyOf(injectionModules);
|
2010-10-12 09:16:40 +02:00
|
|
|
}
|
|
|
|
|
|
2013-01-18 09:39:32 +01:00
|
|
|
/**
|
|
|
|
|
* Method description
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
2014-03-28 22:49:55 +01:00
|
|
|
@Override
|
|
|
|
|
public Collection<ScmModule> getInstalledModules()
|
2010-12-21 17:08:41 +01:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
return modules;
|
2010-12-21 17:08:41 +01:00
|
|
|
}
|
|
|
|
|
|
2010-12-01 14:26:29 +01:00
|
|
|
/**
|
|
|
|
|
* Method description
|
|
|
|
|
*
|
|
|
|
|
*
|
2013-01-18 09:39:32 +01:00
|
|
|
* @return
|
2010-12-01 14:26:29 +01:00
|
|
|
*/
|
2014-03-28 22:49:55 +01:00
|
|
|
@Override
|
2014-08-27 21:01:05 +02:00
|
|
|
public Collection<PluginWrapper> getInstalledPlugins()
|
2013-01-18 09:39:32 +01:00
|
|
|
{
|
2014-08-27 21:01:05 +02:00
|
|
|
return installedPlugins;
|
2013-01-18 09:39:32 +01:00
|
|
|
}
|
|
|
|
|
|
2014-06-06 08:57:41 +02:00
|
|
|
/**
|
|
|
|
|
* Method description
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
public ClassLoader getUberClassLoader()
|
|
|
|
|
{
|
|
|
|
|
return uberClassLoader;
|
|
|
|
|
}
|
|
|
|
|
|
2014-08-27 21:01:05 +02:00
|
|
|
/**
|
|
|
|
|
* Method description
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
public UberWebResourceLoader getUberWebResourceLoader()
|
|
|
|
|
{
|
|
|
|
|
return uberWebResourceLoader;
|
|
|
|
|
}
|
|
|
|
|
|
2014-03-28 22:49:55 +01:00
|
|
|
//~--- methods --------------------------------------------------------------
|
2013-01-18 09:39:32 +01:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Method description
|
|
|
|
|
*
|
|
|
|
|
*
|
2014-03-28 22:49:55 +01:00
|
|
|
* @param multiple
|
|
|
|
|
* @param single
|
|
|
|
|
* @param extensions
|
|
|
|
|
* @param mods
|
2013-01-18 09:39:32 +01:00
|
|
|
*/
|
2014-03-28 22:49:55 +01:00
|
|
|
private void appendExtensions(Set<Class> multiple, Set<Class> single,
|
|
|
|
|
Set<Class> extensions, Iterable<? extends ScmModule> mods)
|
2013-01-18 09:39:32 +01:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
for (ScmModule mod : mods)
|
2013-01-18 09:39:32 +01:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
for (ExtensionPointElement epe : mod.getExtensionPoints())
|
2013-01-18 09:39:32 +01:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
if (epe.isMultiple())
|
2013-01-18 09:39:32 +01:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
multiple.add(epe.getClazz());
|
2013-01-18 09:39:32 +01:00
|
|
|
}
|
2014-03-28 22:49:55 +01:00
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
single.add(epe.getClazz());
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-01-18 09:39:32 +01:00
|
|
|
|
2014-03-28 22:49:55 +01:00
|
|
|
for (Class extensionClass : mod.getExtensions())
|
|
|
|
|
{
|
|
|
|
|
if (Module.class.isAssignableFrom(extensionClass))
|
2013-01-18 09:39:32 +01:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
injectionModules.add((Module) extensionClass.newInstance());
|
|
|
|
|
}
|
2014-08-19 19:30:13 +02:00
|
|
|
catch (IllegalAccessException | InstantiationException ex)
|
2014-03-28 22:49:55 +01:00
|
|
|
{
|
|
|
|
|
logger.error("could not create instance of module", ex);
|
|
|
|
|
}
|
2013-01-18 09:39:32 +01:00
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
extensions.add(extensionClass);
|
2013-01-18 09:39:32 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-03 19:25:57 +02:00
|
|
|
Iterables.addAll(extensions, mod.getRestProviders());
|
|
|
|
|
Iterables.addAll(extensions, mod.getRestResources());
|
2014-03-28 22:49:55 +01:00
|
|
|
}
|
2013-01-18 09:39:32 +01:00
|
|
|
}
|
|
|
|
|
|
2014-08-27 21:01:05 +02:00
|
|
|
/**
|
|
|
|
|
* Method description
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
|
|
|
|
private Iterable<Plugin> unwrap()
|
|
|
|
|
{
|
|
|
|
|
return PluginsInternal.unwrap(installedPlugins);
|
|
|
|
|
}
|
|
|
|
|
|
2014-03-28 22:49:55 +01:00
|
|
|
//~--- get methods ----------------------------------------------------------
|
|
|
|
|
|
2013-01-18 09:39:32 +01:00
|
|
|
/**
|
|
|
|
|
* Method description
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* @param classLoader
|
2014-03-28 22:49:55 +01:00
|
|
|
* @param context
|
|
|
|
|
* @param path
|
|
|
|
|
* @param <T>
|
|
|
|
|
*
|
|
|
|
|
* @return
|
2013-01-18 09:39:32 +01:00
|
|
|
*
|
|
|
|
|
* @throws IOException
|
2014-03-28 22:49:55 +01:00
|
|
|
* @throws JAXBException
|
2013-01-18 09:39:32 +01:00
|
|
|
*/
|
2014-08-19 21:45:23 +02:00
|
|
|
@SuppressWarnings("unchecked")
|
2014-03-28 22:49:55 +01:00
|
|
|
private <T> Set<T> getInstalled(ClassLoader classLoader, JAXBContext context,
|
|
|
|
|
String path)
|
|
|
|
|
throws IOException, JAXBException
|
2013-01-18 09:39:32 +01:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
Builder<T> builder = ImmutableSet.builder();
|
|
|
|
|
Enumeration<URL> urls = classLoader.getResources(path);
|
2013-01-18 09:39:32 +01:00
|
|
|
|
2014-03-28 22:49:55 +01:00
|
|
|
while (urls.hasMoreElements())
|
2013-01-18 09:39:32 +01:00
|
|
|
{
|
2014-03-28 22:49:55 +01:00
|
|
|
URL url = urls.nextElement();
|
|
|
|
|
T module = (T) context.createUnmarshaller().unmarshal(url);
|
|
|
|
|
|
|
|
|
|
builder.add(module);
|
2013-01-18 09:39:32 +01:00
|
|
|
}
|
2014-03-28 22:49:55 +01:00
|
|
|
|
|
|
|
|
return builder.build();
|
2013-01-18 09:39:32 +01:00
|
|
|
}
|
|
|
|
|
|
2010-10-12 09:16:40 +02:00
|
|
|
//~--- fields ---------------------------------------------------------------
|
|
|
|
|
|
2014-06-06 08:57:41 +02:00
|
|
|
/** Field description */
|
|
|
|
|
private final ClassLoader uberClassLoader;
|
|
|
|
|
|
2013-01-15 16:55:07 +01:00
|
|
|
/** Field description */
|
2014-08-27 21:01:05 +02:00
|
|
|
private final UberWebResourceLoader uberWebResourceLoader;
|
2013-01-18 09:39:32 +01:00
|
|
|
|
|
|
|
|
/** Field description */
|
2014-08-27 21:01:05 +02:00
|
|
|
private Set<PluginWrapper> installedPlugins;
|
2013-01-18 09:39:32 +01:00
|
|
|
|
|
|
|
|
/** Field description */
|
2014-08-27 21:01:05 +02:00
|
|
|
private Set<ScmModule> modules;
|
|
|
|
|
|
|
|
|
|
/** Field description */
|
|
|
|
|
private Set<Class> multiple = Sets.newHashSet();
|
2013-01-18 09:39:32 +01:00
|
|
|
|
|
|
|
|
/** Field description */
|
2014-03-28 22:49:55 +01:00
|
|
|
private Set<Class> single = Sets.newHashSet();
|
2013-01-18 09:39:32 +01:00
|
|
|
|
2010-10-12 09:16:40 +02:00
|
|
|
/** Field description */
|
2014-03-28 22:49:55 +01:00
|
|
|
private Set<Module> injectionModules = Sets.newHashSet();
|
2013-01-18 09:39:32 +01:00
|
|
|
|
|
|
|
|
/** Field description */
|
2014-03-28 22:49:55 +01:00
|
|
|
private Set<Class> extensions = Sets.newHashSet();
|
2010-10-12 09:16:40 +02:00
|
|
|
}
|