/*
* JBoss, Home of Professional Open Source
* Copyright 2010, Red Hat Inc., and individual contributors as indicated
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* 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.jboss.as.weld.services.bootstrap;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.security.PrivilegedActionException;
import java.security.PrivilegedExceptionAction;
import java.util.Arrays;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import org.jboss.as.server.moduleservice.ServiceModuleLoader;
import org.jboss.as.weld.logging.WeldLogger;
import org.jboss.as.weld.util.Reflections;
import org.jboss.modules.Module;
import org.jboss.modules.ModuleClassLoader;
import org.jboss.modules.ModuleIdentifier;
import org.jboss.weld.interceptor.proxy.LifecycleMixin;
import org.jboss.weld.logging.BeanLogger;
import org.jboss.weld.serialization.spi.BeanIdentifier;
import org.jboss.weld.serialization.spi.ProxyServices;
import org.wildfly.security.manager.WildFlySecurityManager;
/**
* {@link ProxyServices} implementation that delegates to the module class loader if the bean class loader cannot be determined
*
* @author Stuart Douglas
* @author Jozef Hartinger
*
*/
public class ProxyServicesImpl implements ProxyServices {
private static String[] REQUIRED_WELD_DEPENDENCIES = new String[] {
"org.jboss.weld.core",
"org.jboss.weld.spi"
};
// these are used to check whether a classloader is capable of loading Weld proxies
private static String[] WELD_CLASSES = new String[] {
BeanIdentifier.class.getName(),
LifecycleMixin.class.getName()
};
private final Module module;
private final ConcurrentMap<ModuleIdentifier, Boolean> processedStaticModules = new ConcurrentHashMap<>();
public ProxyServicesImpl(Module module) {
this.module = module;
}
public ClassLoader getClassLoader(final Class<?> proxiedBeanType) {
if (WildFlySecurityManager.isChecking()) {
return AccessController.doPrivileged(new PrivilegedAction<ClassLoader>() {
public ClassLoader run() {
return _getClassLoader(proxiedBeanType);
}
});
} else {
return _getClassLoader(proxiedBeanType);
}
}
private ClassLoader _getClassLoader(Class<?> proxiedBeanType) {
if (proxiedBeanType.getName().startsWith("java")) {
return module.getClassLoader();
} else if(proxiedBeanType.getClassLoader() instanceof ModuleClassLoader) {
final ModuleClassLoader loader = (ModuleClassLoader)proxiedBeanType.getClassLoader();
//even though this is not strictly spec compliant if a class from the app server is
//being proxied we use the deployment CL to prevent a memory leak
//in theory this means that package private methods will not work correctly
//however the application does not have access to package private methods anyway
//as it is in a different class loader
if(loader.getModule().getModuleLoader() instanceof ServiceModuleLoader) {
//this is a dynamic module
//we can use it to load the proxy
return proxiedBeanType.getClassLoader();
} else {
// this class comes from a static module
// first, check if we can use its classloader to load proxy classes
final Module definingModule = loader.getModule();
Boolean hasWeldDependencies = processedStaticModules.get(definingModule.getIdentifier());
boolean logWarning = false; // only log for the first class in the module
if (hasWeldDependencies == null) {
hasWeldDependencies = canLoadWeldProxies(definingModule); // may be run multiple times but that does not matter
logWarning = processedStaticModules.putIfAbsent(definingModule.getIdentifier(), hasWeldDependencies) == null;
}
if (hasWeldDependencies) {
// this module declares weld dependencies - we can use module's classloader to load the proxy class
// pros: package-private members will work fine
// cons: proxy classes will remain loaded by the module's classloader after undeployment (nothing else leaks)
return proxiedBeanType.getClassLoader();
} else {
// no weld dependencies - we use deployment's classloader to load the proxy class
// pros: proxy classes unloaded with undeployment
// cons: package-private methods and constructors will yield IllegalAccessException
if (logWarning) {
WeldLogger.ROOT_LOGGER.loadingProxiesUsingDeploymentClassLoader(definingModule.getIdentifier(), Arrays.toString(REQUIRED_WELD_DEPENDENCIES));
}
return this.module.getClassLoader();
}
}
} else {
return proxiedBeanType.getClassLoader();
}
}
private static boolean canLoadWeldProxies(Module module) {
for (String weldClass : WELD_CLASSES) {
if (!Reflections.isAccessible(weldClass, module.getClassLoader())) {
return false;
}
}
return true;
}
public void cleanup() {
processedStaticModules.clear();
}
public Class<?> loadBeanClass(final String className) {
try {
return (Class<?>) AccessController.doPrivileged(new PrivilegedExceptionAction<Object>() {
public Object run() throws Exception {
return Class.forName(className, true, getClassLoader(this.getClass()));
}
});
} catch (PrivilegedActionException pae) {
throw BeanLogger.LOG.cannotLoadClass(className, pae.getException());
}
}
}