/*
* Copyright (C) 2009 eXo Platform SAS.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.exoplatform.container;
import org.exoplatform.commons.utils.SecurityHelper;
import java.io.IOException;
import java.net.URL;
import java.security.PrivilegedAction;
import java.util.Collections;
import java.util.Enumeration;
import java.util.LinkedHashSet;
import java.util.Set;
/**
* This class is used to merge different {@link ClassLoader} to create one single one.
* This ClassLoader is used only for the resources the class will be loaded from the
* ContextClassLoader. For each resources, it will always consider that the {@link ClassLoader}
* with the highest priority has always right, in other words for example in the method
* getResource, it will try to get the resource in the {@link ClassLoader} of the highest
* priority, if it cans not find it, it will try the {@link ClassLoader} with the second highest
* priority and so on. The priority of the {@link ClassLoader} is the order given in the
* constructor, the last {@link ClassLoader} is the one with the highest priority.
*
* Created by The eXo Platform SAS
* Author : Nicolas Filotto
* nicolas.filotto@exoplatform.com
* 18 sept. 2009
*/
class UnifiedClassLoader extends ClassLoader
{
/**
* The list of all the {@link ClassLoader} to merge together
*/
private final ClassLoader[] cls;
/**
* @param cls the list of all the {@link ClassLoader} to merge ordered by priority. The last
* {@link ClassLoader} has highest priority.
*/
UnifiedClassLoader(ClassLoader... cls)
{
super(Thread.currentThread().getContextClassLoader());
if (cls == null || cls.length == 0)
{
throw new IllegalArgumentException("The array of ClassLoader cannot be empty");
}
this.cls = cls;
}
/**
* Allows to override the list of {@link ClassLoader} if it is a dynamic list
* @return the list of the {@link ClassLoader} to merge
*/
protected ClassLoader[] getClassLoaders()
{
return cls;
}
/**
* {@inheritDoc}
*/
@Override
public URL getResource(String name)
{
final ClassLoader[] cls = getClassLoaders();
for (int i = cls.length - 1; i >= 0; i--)
{
final ClassLoader cl = cls[i];
URL url = cl.getResource(name);
if (url != null)
{
return url;
}
}
return null;
}
/**
* {@inheritDoc}
*/
@Override
public Enumeration<URL> getResources(String name) throws IOException
{
final ClassLoader[] cls = getClassLoaders();
final Set<URL> urls = new LinkedHashSet<URL>();
for (int i = 0, length = cls.length; i < length; i++)
{
Enumeration<URL> eUrls = cls[i].getResources(name);
if (eUrls != null && eUrls.hasMoreElements())
{
// Prevent duplicates
urls.addAll(Collections.list(eUrls));
}
}
return Collections.enumeration(urls);
}
static protected UnifiedClassLoader createUnifiedClassLoaderInPrivilegedMode(final ClassLoader... cls)
{
return SecurityHelper.doPrivilegedAction(new PrivilegedAction<UnifiedClassLoader>()
{
public UnifiedClassLoader run()
{
return new UnifiedClassLoader(cls);
}
});
}
}