/* * JBoss, Home of Professional Open Source. * Copyright 2006, Red Hat Middleware LLC, and individual contributors * as indicated by the @author tags. See the copyright.txt file 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.ejb; import java.lang.reflect.Constructor; import java.lang.reflect.Method; import java.net.URL; import java.net.URLClassLoader; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.LinkedList; import java.util.HashMap; import java.util.Iterator; import java.util.Map; import java.util.ListIterator; import java.util.Set; import java.util.HashSet; import javax.ejb.EJBLocalHome; import javax.management.ObjectName; import javax.naming.InitialContext; import javax.naming.NamingException; import javax.security.jacc.PolicyConfiguration; import javax.security.jacc.PolicyConfigurationFactory; import javax.security.jacc.EJBMethodPermission; import javax.security.jacc.PolicyContextException; import javax.security.jacc.EJBRoleRefPermission; import javax.transaction.TransactionManager; import org.jboss.deployment.DeploymentException; import org.jboss.deployment.DeploymentInfo; import org.jboss.deployment.EARDeployerMBean; import org.jboss.logging.Logger; import org.jboss.metadata.ApplicationMetaData; import org.jboss.metadata.BeanMetaData; import org.jboss.metadata.ConfigurationMetaData; import org.jboss.metadata.EntityMetaData; import org.jboss.metadata.InvokerProxyBindingMetaData; import org.jboss.metadata.MetaData; import org.jboss.metadata.SessionMetaData; import org.jboss.metadata.XmlLoadable; import org.jboss.metadata.MethodMetaData; import org.jboss.metadata.SecurityRoleRefMetaData; import org.jboss.mx.loading.RepositoryClassLoader; import org.jboss.ejb.plugins.SecurityProxyInterceptor; import org.jboss.ejb.plugins.StatefulSessionInstancePool; import org.jboss.security.AuthenticationManager; import org.jboss.security.RealmMapping; import org.jboss.system.Registry; import org.jboss.system.ServiceControllerMBean; import org.jboss.system.ServiceMBeanSupport; import org.jboss.mx.util.MBeanProxyExt; import org.jboss.mx.util.ObjectNameFactory; import org.jboss.util.loading.DelegatingClassLoader; import org.jboss.web.WebClassLoader; import org.jboss.web.WebServiceMBean; import org.jboss.invocation.InvocationType; import org.w3c.dom.Element; /** * An EjbModule represents a collection of beans that are deployed as a * unit. * *
The beans may use the EjbModule to access other beans within the same
* deployment unit.
*
* @see Container
* @see EJBDeployer
*
* @author Rickard Ãberg
* @author David Jencks
* @author Francisco Reverbel
* @author Adrian.Brock
* @author Scott Stark
* @version $Revision: 60959 $
*
* @jmx:mbean extends="org.jboss.system.ServiceMBean"
*/
public class EjbModule
extends ServiceMBeanSupport
implements EjbModuleMBean
{
public static final String BASE_EJB_MODULE_NAME = "jboss.j2ee:service=EjbModule";
public static final ObjectName EJB_MODULE_QUERY_NAME = ObjectNameFactory.create(BASE_EJB_MODULE_NAME + ",*");
public static String DEFAULT_STATELESS_CONFIGURATION = "Default Stateless SessionBean";
public static String DEFAULT_STATEFUL_CONFIGURATION = "Default Stateful SessionBean";
public static String DEFAULT_ENTITY_BMP_CONFIGURATION = "Default BMP EntityBean";
public static String DEFAULT_ENTITY_CMP_CONFIGURATION = "Default CMP EntityBean";
public static String DEFAULT_MESSAGEDRIVEN_CONFIGURATION = "Default MesageDriven Bean";
// Constants uses with container interceptor configurations
public static final int BMT = 1;
public static final int CMT = 2;
public static final int ANY = 3;
static final String BMT_VALUE = "Bean";
static final String CMT_VALUE = "Container";
static final String ANY_VALUE = "Both";
/** Class logger. */
private static final Logger log = Logger.getLogger(EjbModule.class);
// Constants -----------------------------------------------------
// Attributes ----------------------------------------------------
/** HashMapstart
method calls
* the start method on each contatiner, then the init method on each
* container. Conversion to a different registration system with one-phase
* startup is conceivable.
*
* @exception Exception if an error occurs
*/
protected void startService() throws Exception
{
// before EntityContainer returns from the startService, its PM should be usable
ListIterator iter = containerOrdering.listIterator();
while( iter.hasNext() )
{
Container con = (Container) iter.next();
if(con.getBeanMetaData().isEntity())
{
ClassLoader oldCl = SecurityActions.getContextClassLoader();
SecurityActions.setContextClassLoader(con.getClassLoader());
try
{
((EntityContainer)con).getPersistenceManager().start();
}
finally
{
// Reset classloader
SecurityActions.setContextClassLoader(oldCl);
}
}
}
iter = containerOrdering.listIterator();
while( iter.hasNext() )
{
Container con = (Container) iter.next();
log.debug("startService, starting container: " + con.getBeanMetaData().getEjbName());
serviceController.start(con.getJmxName());
}
}
/**
* Stops all the containers of this application.
*/
protected void stopService() throws Exception
{
ListIterator iter = containerOrdering.listIterator(containerOrdering.size());
while( iter.hasPrevious() )
{
Container con = (Container) iter.previous();
try
{
// The container may already be destroyed so validate metaData
BeanMetaData metaData = con.getBeanMetaData();
String ejbName = metaData != null ? metaData.getEjbName() : "Unknown";
log.debug("stopService, stopping container: " + ejbName);
serviceController.stop(con.getJmxName());
}
catch (Exception e)
{
log.error("unexpected exception stopping Container: " + con.getJmxName(), e);
} // end of try-catch
}
}
protected void destroyService() throws Exception
{
WebServiceMBean webServer = null;
if (webServiceName != null)
{
webServer =
(WebServiceMBean) MBeanProxyExt.create(WebServiceMBean.class,
webServiceName);
}
ListIterator iter = containerOrdering.listIterator(containerOrdering.size());
// Unegister the permissions with the JACC layer
String contextID = deploymentInfo.shortName;
PolicyConfigurationFactory pcFactory = PolicyConfigurationFactory.getPolicyConfigurationFactory();
PolicyConfiguration pc = pcFactory.getPolicyConfiguration(contextID, true);
pc.delete();
while ( iter.hasPrevious() )
{
Container con = (Container) iter.previous();
ObjectName jmxName = con.getJmxName();
int conState = con.getState();
boolean destroyContainer = conState == CREATED || conState == STOPPED
|| conState == FAILED;
log.debug("Looking to destroy container: " + jmxName
+ ", state: " + con.getStateString() + ", destroy: " + destroyContainer);
// always unregister from Registry
int jmxHash = jmxName.hashCode();
Registry.unbind(new Integer(jmxHash));
// Unregister the web classloader
//Removing the wcl should probably be done in stop of the container,
// but I don't want to look for errors today.
if (webServer != null)
{
ClassLoader wcl = con.getWebClassLoader();
if (wcl != null)
{
try
{
webServer.removeClassLoader(wcl);
}
catch (Throwable e)
{
log.warn("Failed to unregister webClassLoader", e);
}
}
}
// Only destroy containers that have been created or started
if( destroyContainer )
{
try
{
serviceController.destroy(jmxName);
log.info("Undeployed " + con.getBeanMetaData().getEjbName());
}
catch (Throwable e)
{
log.error("unexpected exception destroying Container: " + jmxName, e);
} // end of try-catch
}
// If the container was registered with the mbeanserver, remove it
if (conState != UNREGISTERED)
{
try
{
serviceController.remove(jmxName);
}
catch (Throwable e)
{
log.error("unexpected exception removing Container: " + jmxName, e);
} // end of try-catch
}
// cleanup container
con.setBeanMetaData(null);
con.setWebClassLoader(null);
con.setClassLoader(null);
con.setLocalClassLoader(null);
con.setEjbModule(null);
con.setDeploymentInfo(null);
con.setTransactionManager(null);
con.setSecurityManager(null);
con.setRealmMapping(null);
con.setSecurityProxy(null);
con.proxyFactories.clear();
}
this.containers.clear();
this.localHomes.clear();
this.containerOrdering.clear();
this.moduleData.clear();
this.serviceController = null;
}
// ******************
// Container Creation
// ******************
private Container createContainer(BeanMetaData bean, DeploymentInfo sdi)
throws Exception
{
ClassLoader cl = sdi.ucl;
ClassLoader localCl = sdi.localCl;
Container container = null;
// Added message driven deployment
if (bean.isMessageDriven())
{
container = createMessageDrivenContainer(bean, cl, localCl);
}
else if (bean.isSession()) // Is session?
{
if (((SessionMetaData) bean).isStateless()) // Is stateless?
{
container = createStatelessSessionContainer((SessionMetaData) bean, cl, localCl);
}
else // Stateful
{
container = createStatefulSessionContainer((SessionMetaData) bean, cl, localCl);
}
}
else // Entity
{
container = createEntityContainer(bean, cl, localCl);
}
return container;
}
private MessageDrivenContainer createMessageDrivenContainer(BeanMetaData bean,
ClassLoader cl,
ClassLoader localCl)
throws Exception
{
// get the container configuration for this bean
// a default configuration is now always provided
ConfigurationMetaData conf = bean.getContainerConfiguration();
// Stolen from Stateless deploy
// Create container
MessageDrivenContainer container = new MessageDrivenContainer();
int transType = bean.isContainerManagedTx() ? CMT : BMT;
initializeContainer(container, conf, bean, transType, cl, localCl);
createProxyFactories(bean, container, cl);
container.setInstancePool(createInstancePool(conf, cl));
return container;
}
private StatelessSessionContainer createStatelessSessionContainer(SessionMetaData bean,
ClassLoader cl,
ClassLoader localCl)
throws Exception
{
// get the container configuration for this bean
// a default configuration is now always provided
ConfigurationMetaData conf = bean.getContainerConfiguration();
// Create container
StatelessSessionContainer container = new StatelessSessionContainer();
int transType = bean.isContainerManagedTx() ? CMT : BMT;
initializeContainer(container, conf, bean, transType, cl, localCl);
if (bean.getHome() != null || bean.getServiceEndpoint()!=null)
{
createProxyFactories(bean, container, cl);
}
container.setInstancePool(createInstancePool(conf, cl));
return container;
}
private StatefulSessionContainer createStatefulSessionContainer(SessionMetaData bean,
ClassLoader cl,
ClassLoader localCl)
throws Exception
{
// get the container configuration for this bean
// a default configuration is now always provided
ConfigurationMetaData conf = bean.getContainerConfiguration();
// Create container
StatefulSessionContainer container = new StatefulSessionContainer();
int transType = bean.isContainerManagedTx() ? CMT : BMT;
initializeContainer(container, conf, bean, transType, cl, localCl);
if (bean.getHome() != null || bean.getServiceEndpoint()!=null)
{
createProxyFactories(bean, container, cl);
}
container.setInstanceCache(createInstanceCache(conf, cl));
// No real instance pool, use the shadow class
StatefulSessionInstancePool ip = new StatefulSessionInstancePool();
ip.importXml(conf.getContainerPoolConf());
container.setInstancePool(ip);
// Set persistence manager
container.setPersistenceManager((StatefulSessionPersistenceManager) cl.loadClass(conf.getPersistenceManager()).newInstance());
//Set the bean Lock Manager
container.setLockManager(createBeanLockManager(container, false, conf.getLockClass(), cl));
return container;
}
private EntityContainer createEntityContainer(BeanMetaData bean,
ClassLoader cl,
ClassLoader localCl)
throws Exception
{
// get the container configuration for this bean
// a default configuration is now always provided
ConfigurationMetaData conf = bean.getContainerConfiguration();
// Create container
EntityContainer container = new EntityContainer();
int transType = CMT;
initializeContainer(container, conf, bean, transType, cl, localCl);
if (bean.getHome() != null)
{
createProxyFactories(bean, container, cl);
}
container.setInstanceCache(createInstanceCache(conf, cl));
container.setInstancePool(createInstancePool(conf, cl));
//Set the bean Lock Manager
boolean reentrant = ((EntityMetaData) bean).isReentrant();
BeanLockManager lockMgr = createBeanLockManager(container, reentrant, conf.getLockClass(), cl);
container.setLockManager(lockMgr);
// Set persistence manager
if (((EntityMetaData) bean).isBMP())
{
Class pmClass = cl.loadClass(conf.getPersistenceManager());
EntityPersistenceManager pm = (EntityPersistenceManager) pmClass.newInstance();
container.setPersistenceManager(pm);
}
else
{
// CMP takes a manager and a store
org.jboss.ejb.plugins.CMPPersistenceManager persistenceManager =
new org.jboss.ejb.plugins.CMPPersistenceManager();
//Load the store from configuration
Class pmClass = cl.loadClass(conf.getPersistenceManager());
EntityPersistenceStore pm = (EntityPersistenceStore) pmClass.newInstance();
persistenceManager.setPersistenceStore(pm);
// Set the manager on the container
container.setPersistenceManager(persistenceManager);
}
return container;
}
// **************
// Helper Methods
// **************
/**
* Perform the common steps to initializing a container.
*/
private void initializeContainer(Container container,
ConfigurationMetaData conf,
BeanMetaData bean,
int transType,
ClassLoader cl,
ClassLoader localCl)
throws NamingException, DeploymentException
{
// Create local classloader for this container
// For loading resources that must come from the local jar. Not for loading classes!
container.setLocalClassLoader(new URLClassLoader(new URL[0], localCl));
// Set metadata (do it *before* creating the container's WebClassLoader)
container.setEjbModule(this);
container.setBeanMetaData(bean);
// Create the container's WebClassLoader
// and register it with the web service.
String webClassLoaderName = getWebClassLoader(conf, bean);
log.debug("Creating WebClassLoader of class " + webClassLoaderName);
WebClassLoader wcl = null;
try
{
Class clazz = cl.loadClass(webClassLoaderName);
Constructor constructor = clazz.getConstructor(
new Class[]{ObjectName.class, RepositoryClassLoader.class});
wcl = (WebClassLoader) constructor.newInstance(
new Object[]{container.getJmxName(), cl});
}
catch (Exception e)
{
throw new DeploymentException(
"Failed to create WebClassLoader of class "
+ webClassLoaderName + ": ", e);
}
if (webServiceName != null)
{
WebServiceMBean webServer =
(WebServiceMBean) MBeanProxyExt.create(WebServiceMBean.class,
webServiceName);
URL[] codebase = {webServer.addClassLoader(wcl)};
wcl.setWebURLs(codebase);
} // end of if ()
container.setWebClassLoader(wcl);
// Create classloader for this container
// Only used to unique the bean ENC and does not augment class loading
container.setClassLoader(new DelegatingClassLoader(wcl));
// Set transaction manager
InitialContext iniCtx = new InitialContext();
container.setTransactionManager(tm);
// Set security domain manager
String securityDomain = bean.getApplicationMetaData().getSecurityDomain();
String confSecurityDomain = conf.getSecurityDomain();
// Default the config security to the application security manager
if (confSecurityDomain == null)
confSecurityDomain = securityDomain;
// Check for an empty confSecurityDomain which signifies to disable security
if (confSecurityDomain != null && confSecurityDomain.length() == 0)
confSecurityDomain = null;
if (confSecurityDomain != null)
{ // Either the application has a security domain or the container has security setup
try
{
log.debug("Setting security domain from: " + confSecurityDomain);
Object securityMgr = iniCtx.lookup(confSecurityDomain);
AuthenticationManager ejbS = (AuthenticationManager) securityMgr;
RealmMapping rM = (RealmMapping) securityMgr;
container.setSecurityManager(ejbS);
container.setRealmMapping(rM);
}
catch (NamingException e)
{
throw new DeploymentException("Could not find the security-domain, name=" + confSecurityDomain, e);
}
catch (Exception e)
{
throw new DeploymentException("Invalid security-domain specified, name=" + confSecurityDomain, e);
}
}
// Load the security proxy instance if one was configured
String securityProxyClassName = bean.getSecurityProxy();
if (securityProxyClassName != null)
{
try
{
Class proxyClass = cl.loadClass(securityProxyClassName);
Object proxy = proxyClass.newInstance();
container.setSecurityProxy(proxy);
log.debug("setSecurityProxy, " + proxy);
}
catch (Exception e)
{
throw new DeploymentException("Failed to create SecurityProxy of type: " +
securityProxyClassName, e);
}
}
// Install the container interceptors based on the configuration
addInterceptors(container, transType, conf.getContainerInterceptorsConf());
}
/**
* Return the name of the WebClassLoader class for this ejb.
*/
private static String getWebClassLoader(ConfigurationMetaData conf,
BeanMetaData bmd)
throws DeploymentException
{
String webClassLoader = null;
Iterator it = bmd.getInvokerBindings();
int count = 0;
while (it.hasNext())
{
String invoker = (String) it.next();
ApplicationMetaData amd = bmd.getApplicationMetaData();
InvokerProxyBindingMetaData imd =
amd.getInvokerProxyBindingMetaDataByName(invoker);
if (imd == null)
{
String msg = "Failed to find InvokerProxyBindingMetaData for: '"
+ invoker + "'. Check the invoker-proxy-binding-name to "
+ "invoker-proxy-binding/name mappings in jboss.xml";
throw new DeploymentException(msg);
}
Element proxyFactoryConfig = imd.getProxyFactoryConfig();
String webCL = MetaData.getOptionalChildContent(proxyFactoryConfig,
"web-class-loader");
if (webCL != null)
{
log.debug("Invoker " + invoker
+ " specified WebClassLoader class" + webCL);
webClassLoader = webCL;
count++;
}
}
if (count > 1)
{
log.warn(count + " invokers have WebClassLoader specifications.");
log.warn("Using the last specification seen ("
+ webClassLoader + ").");
}
else if (count == 0)
{
webClassLoader = conf.getWebClassLoader();
}
return webClassLoader;
}
/**
* Given a container-interceptors element of a container-configuration,
* add the indicated interceptors to the container depending on the container
* transcation type and metricsEnabled flag.
*
*
* @todo marcf: frankly the transaction type stuff makes no sense to me, we have externalized
* the container stack construction in jbossxml and I don't see why or why there would be a
* type missmatch on the transaction
*
* @param container the container instance to setup.
* @param transType one of the BMT, CMT or ANY constants.
* @param element the container-interceptors element from the
* container-configuration.
*/
private void addInterceptors(Container container,
int transType,
Element element)
throws DeploymentException
{
// Get the interceptor stack(either jboss.xml or standardjboss.xml)
Iterator interceptorElements = MetaData.getChildrenByTagName(element, "interceptor");
String transTypeString = stringTransactionValue(transType);
ClassLoader loader = container.getClassLoader();
/* First build the container interceptor stack from interceptorElements
match transType and metricsEnabled values
*/
ArrayList istack = new ArrayList();
while (interceptorElements != null && interceptorElements.hasNext())
{
Element ielement = (Element) interceptorElements.next();
/* Check that the interceptor is configured for the transaction mode of the bean
by comparing its 'transaction' attribute to the string representation
of transType
FIXME: marcf, WHY???????
*/
String transAttr = ielement.getAttribute("transaction");
if (transAttr == null || transAttr.length() == 0)
transAttr = ANY_VALUE;
if (transAttr.equalsIgnoreCase(ANY_VALUE) || transAttr.equalsIgnoreCase(transTypeString))
{ // The transaction type matches the container bean trans type, check the metricsEnabled
String metricsAttr = ielement.getAttribute("metricsEnabled");
boolean metricsInterceptor = metricsAttr.equalsIgnoreCase("true");
try
{
boolean metricsEnabled = ((Boolean) server.getAttribute(EJBDeployerMBean.OBJECT_NAME,
"MetricsEnabled")).booleanValue();
if (metricsEnabled == false && metricsInterceptor == true)
{
continue;
}
}
catch (Exception e)
{
throw new DeploymentException("couldn't contace EJBDeployer!", e);
} // end of try-catch
String className = null;
try
{
className = MetaData.getFirstElementContent(ielement, null);
Class clazz = loader.loadClass(className);
Interceptor interceptor = (Interceptor) clazz.newInstance();
if (interceptor instanceof XmlLoadable)
{
((XmlLoadable)interceptor).importXml(ielement);
}
istack.add(interceptor);
}
catch (ClassNotFoundException e)
{
log.warn("Could not load the " + className + " interceptor", e);
}
catch (Exception e)
{
log.warn("Could not load the " + className + " interceptor for this container", e);
}
}
}
if (istack.size() == 0)
log.warn("There are no interceptors configured. Check the standardjboss.xml file");
// Now add the interceptors to the container
for (int i = 0; i < istack.size(); i++)
{
Interceptor interceptor = (Interceptor) istack.get(i);
container.addInterceptor(interceptor);
}
/* If there is a security proxy associated with the container add its
interceptor just before the container interceptor
*/
if (container.getSecurityProxy() != null)
container.addInterceptor(new SecurityProxyInterceptor());
// Finally we add the last interceptor from the container
container.addInterceptor(container.createContainerInterceptor());
}
private void createPermissions(BeanMetaData bean, PolicyConfiguration pc)
throws PolicyContextException
{
// Process the method-permission MethodMetaData
Iterator iter = bean.getPermissionMethods();
while( iter.hasNext() )
{
MethodMetaData mmd = (MethodMetaData) iter.next();
String[] params = null;
if( mmd.isParamGiven() )
params = mmd.getMethodParams();
String methodName = mmd.getMethodName();
if( methodName != null && methodName.equals("*") )
methodName = null;
EJBMethodPermission p = new EJBMethodPermission(mmd.getEjbName(),
methodName, mmd.getInterfaceType(), params);
if( mmd.isUnchecked() )
{
pc.addToUncheckedPolicy(p);
}
else
{
Set roles = mmd.getRoles();
Iterator riter = roles.iterator();
while( riter.hasNext() )
{
String role = (String) riter.next();
pc.addToRole(role, p);
}
}
}
// Process the exclude-list MethodMetaData
iter = bean.getExcludedMethods();
while( iter.hasNext() )
{
MethodMetaData mmd = (MethodMetaData) iter.next();
String[] params = null;
if( mmd.isParamGiven() )
params = mmd.getMethodParams();
EJBMethodPermission p = new EJBMethodPermission(mmd.getEjbName(),
mmd.getMethodName(), mmd.getInterfaceType(), params);
pc.addToExcludedPolicy(p);
}
// Process the security-role-ref SecurityRoleRefMetaData
iter = bean.getSecurityRoleReferences();
while( iter.hasNext() )
{
SecurityRoleRefMetaData srrmd = (SecurityRoleRefMetaData) iter.next();
EJBRoleRefPermission p = new EJBRoleRefPermission(bean.getEjbName(), srrmd.getName());
pc.addToRole(srrmd.getLink(), p);
}
/* Special handling of stateful session bean getEJBObject due how the
stateful session handles acquire the proxy by sending an invocation to
the ejb container.
*/
/*if( bean instanceof SessionMetaData )
{
SessionMetaData smd = (SessionMetaData) bean;
if( smd.isStateful() )
{
EJBMethodPermission p = new EJBMethodPermission(bean.getEjbName(),
"getEJBObject", "Home", null);
pc.addToUncheckedPolicy(p);
}
}*/
}
/** Create any JACC permissions for the ejb methods that were not explicitly
* assigned method-permission or exclude-list mappings.
* @param con - the ejb container
* @param bean - the bean metadata
* @throws ClassNotFoundException
* @throws PolicyContextException
*/
void createMissingPermissions(Container con, BeanMetaData bean)
throws ClassNotFoundException, PolicyContextException
{
String contextID = con.getJaccContextID();
PolicyConfigurationFactory pcFactory = PolicyConfigurationFactory.getPolicyConfigurationFactory();
PolicyConfiguration pc = pcFactory.getPolicyConfiguration(contextID, false);
Class clazz = con.getHomeClass();
// If there is no security domain mark all methods as unchecked
boolean hasSecurityDomain = con.getSecurityManager() != null;
boolean exclude = hasSecurityDomain ? bean.isExcludeMissingMethods() : false;
if( clazz != null )
{
addMissingMethodPermissions(bean, exclude, clazz, InvocationType.HOME, pc);
}
clazz = con.getLocalHomeClass();
if( clazz != null )
{
addMissingMethodPermissions(bean, exclude, clazz, InvocationType.LOCALHOME, pc);
}
clazz = con.getLocalClass();
if( clazz != null )
{
addMissingMethodPermissions(bean, exclude, clazz, InvocationType.LOCAL, pc);
}
clazz = con.getRemoteClass();
if( clazz != null )
{
addMissingMethodPermissions(bean, exclude, clazz, InvocationType.REMOTE, pc);
}
pc.commit();
}
private void getInterfaces(Class iface, HashSet tmp)
{
tmp.add(iface);
Class[] ifaces = iface.getInterfaces();
for(int n = 0; n < ifaces.length; n ++)
{
Class iface2 = ifaces[n];
tmp.add(iface2);
getInterfaces(iface2, tmp);
}
}
private void addMissingMethodPermissions(BeanMetaData bean, boolean exclude,
Class iface, InvocationType type, PolicyConfiguration pc)
throws PolicyContextException
{
String ejbName = bean.getEjbName();
HashSet tmp = new HashSet();
getInterfaces(iface, tmp);
Class[] ifaces = new Class[tmp.size()];
tmp.toArray(ifaces);
for(int n = 0; n < ifaces.length; n ++)
{
Class c = ifaces[n];
Method[] methods = c.getDeclaredMethods();
for(int m = 0; m < methods.length; m ++)
{
String methodName = methods[m].getName();
Class[] params = methods[m].getParameterTypes();
// See if there is a method-permission
if( bean.hasMethodPermission(methodName, params, type) )
continue;
// Create a permission for the missing method-permission
EJBMethodPermission p = new EJBMethodPermission(ejbName,
type.toInterfaceString(), methods[m]);
if( exclude )
pc.addToExcludedPolicy(p);
else
pc.addToUncheckedPolicy(p);
}
}
}
private static String stringTransactionValue(int transType)
{
String transaction = ANY_VALUE;
switch (transType)
{
case BMT:
transaction = BMT_VALUE;
break;
case CMT:
transaction = CMT_VALUE;
break;
}
return transaction;
}
/**
* Create all proxy factories for this ejb
*/
private static void createProxyFactories(BeanMetaData conf, Container container,
ClassLoader cl)
throws Exception
{
Iterator it = conf.getInvokerBindings();
boolean foundOne=false;
while (it.hasNext())
{
String invoker = (String) it.next();
String jndiBinding = conf.getInvokerBinding(invoker);
log.debug("creating binding for " + jndiBinding + ":" + invoker);
InvokerProxyBindingMetaData imd = conf.getApplicationMetaData().getInvokerProxyBindingMetaDataByName(invoker);
EJBProxyFactory ci = null;
// create a ProxyFactory instance
try
{
ci = (EJBProxyFactory) cl.loadClass(imd.getProxyFactory()).newInstance();
ci.setContainer(container);
ci.setInvokerMetaData(imd);
ci.setInvokerBinding(jndiBinding);
if (ci instanceof XmlLoadable)
{
// the container invoker can load its configuration from the jboss.xml element
((XmlLoadable) ci).importXml(imd.getProxyFactoryConfig());
}
container.addProxyFactory(invoker, ci);
foundOne=true;
}
catch (Exception e)
{
log.warn("The Container Invoker "+invoker+" (in jboss.xml or standardjboss.xml) could not be created because of "+e+
" We will ignore this error, but you may miss a transport for this bean.");
}
}
if(!foundOne) {
throw new DeploymentException("Missing or invalid Container Invokers (in jboss.xml or standardjboss.xml).");
}
}
private static BeanLockManager createBeanLockManager(Container container, boolean reentrant, String beanLock,
ClassLoader cl)
throws Exception
{
// The bean lock manager
BeanLockManager lockManager = new BeanLockManager(container);
Class lockClass = null;
try
{
lockClass = cl.loadClass(beanLock);
}
catch (Exception e)
{
throw new DeploymentException("Missing or invalid lock class (in jboss.xml or standardjboss.xml): " + beanLock + " - " + e);
}
lockManager.setLockCLass(lockClass);
lockManager.setReentrant(reentrant);
return lockManager;
}
private static InstancePool createInstancePool(ConfigurationMetaData conf,
ClassLoader cl)
throws Exception
{
// Set instance pool
InstancePool ip = null;
try
{
ip = (InstancePool) cl.loadClass(conf.getInstancePool()).newInstance();
}
catch (Exception e)
{
throw new DeploymentException("Missing or invalid Instance Pool (in jboss.xml or standardjboss.xml)", e);
}
if (ip instanceof XmlLoadable)
((XmlLoadable) ip).importXml(conf.getContainerPoolConf());
return ip;
}
private static InstanceCache createInstanceCache(ConfigurationMetaData conf,
ClassLoader cl)
throws Exception
{
// Set instance cache
InstanceCache ic = null;
try
{
ic = (InstanceCache) cl.loadClass(conf.getInstanceCache()).newInstance();
}
catch (Exception e)
{
throw new DeploymentException("Missing or invalid Instance Cache (in jboss.xml or standardjboss.xml)", e);
}
if (ic instanceof XmlLoadable)
((XmlLoadable) ic).importXml(conf.getContainerCacheConf());
return ic;
}
}
/*
vim:ts=3:sw=3:et
*/