package org.jboss.ejb.plugins;
import org.jboss.ejb.BeanLock;
import org.jboss.ejb.Container;
import org.jboss.ejb.EntityContainer;
import org.jboss.ejb.EntityEnterpriseContext;
import org.jboss.ejb.InstanceCache;
import org.jboss.ejb.AllowedOperationsAssociation;
import org.jboss.ejb.InstancePool;
import org.jboss.invocation.Invocation;
import org.jboss.util.NestedRuntimeException;
import javax.ejb.NoSuchObjectLocalException;
import javax.ejb.TimedObject;
import javax.ejb.Timer;
import javax.transaction.Transaction;
import java.lang.reflect.Method;
import java.rmi.NoSuchObjectException;
import java.rmi.RemoteException;
public class EntityInstanceInterceptor
extends AbstractInterceptor
{
protected EntityContainer container;
protected static final Method ejbTimeout;
static
{
try
{
ejbTimeout = TimedObject.class.getMethod("ejbTimeout", new Class[]{Timer.class});
}
catch (Exception e)
{
throw new ExceptionInInitializerError(e);
}
}
public void setContainer(Container container)
{
this.container = (EntityContainer) container;
}
public Container getContainer()
{
return container;
}
public Object invokeHome(Invocation mi)
throws Exception
{
EntityContainer container = (EntityContainer) getContainer();
EntityEnterpriseContext ctx = (EntityEnterpriseContext) container.getInstancePool().get();
InstancePool pool = container.getInstancePool();
mi.setEnterpriseContext(ctx);
ctx.setTransaction(mi.getTransaction());
ctx.setPrincipal(mi.getPrincipal());
AllowedOperationsAssociation.pushInMethodFlag(IN_EJB_HOME);
Object obj = null;
Exception exception = null;
try
{
obj = getNext().invokeHome(mi);
if (ctx.getId() != null)
{
BeanLock lock = container.getLockManager().getLock(ctx.getCacheKey());
lock.sync(); try
{
InstanceCache cache = container.getInstanceCache();
cache.remove(ctx.getCacheKey());
cache.insert(ctx);
}
finally
{
lock.releaseSync();
container.getLockManager().removeLockRef(ctx.getCacheKey());
}
return obj;
}
}
catch (Exception e)
{
exception = e;
}
finally
{
AllowedOperationsAssociation.popInMethodFlag();
}
ctx.setTransaction(null);
mi.setEnterpriseContext(null);
if (exception == null)
{
container.getInstancePool().free(ctx);
return obj;
}
if (exception instanceof RuntimeException)
{
pool.discard(ctx);
}
else
{
pool.free(ctx);
}
throw exception;
}
public Object invoke(Invocation mi)
throws Exception
{
boolean trace = log.isTraceEnabled();
Object key = mi.getId();
EntityEnterpriseContext ctx;
try
{
ctx = (EntityEnterpriseContext) container.getInstanceCache().get(key);
}
catch (NoSuchObjectException e)
{
if (mi.isLocal())
throw new NoSuchObjectLocalException(e.getMessage());
else
throw e;
}
if (trace) log.trace("Begin invoke, key=" + key);
Transaction tx = mi.getTransaction();
if (!container.isReadOnly())
{
Method method = mi.getMethod();
if (method == null ||
!container.getBeanMetaData().isMethodReadOnly(method.getName()))
{
ctx.setTransaction(tx);
}
}
ctx.setPrincipal(mi.getPrincipal());
EnterpriseBeanPolicyContextHandler.setEnterpriseBean(ctx.getInstance());
mi.setEnterpriseContext(ctx);
if (ejbTimeout.equals(mi.getMethod()))
AllowedOperationsAssociation.pushInMethodFlag(IN_EJB_TIMEOUT);
else
AllowedOperationsAssociation.pushInMethodFlag(IN_BUSINESS_METHOD);
Throwable exceptionThrown = null;
boolean discardContext = false;
try
{
Object obj = getNext().invoke(mi);
return obj;
}
catch (RemoteException e)
{
exceptionThrown = e;
discardContext = true;
throw e;
}
catch (RuntimeException e)
{
exceptionThrown = e;
discardContext = true;
throw e;
}
catch (Error e)
{
exceptionThrown = e;
discardContext = true;
throw e;
}
catch (Exception e)
{
exceptionThrown = e;
throw e;
}
catch (Throwable e)
{
exceptionThrown = e;
discardContext = true;
throw new NestedRuntimeException(e);
}
finally
{
AllowedOperationsAssociation.popInMethodFlag();
if (exceptionThrown != null && tx != null)
{
Transaction ctxTx = ctx.getTransaction();
if (tx.equals(ctxTx) && ctx.hasTxSynchronization() == false)
ctx.setTransaction(null);
}
if (exceptionThrown != null &&
!ctx.hasTxSynchronization() && discardContext)
{
container.getInstanceCache().remove(key);
if (trace) log.trace("Ending invoke, exceptionThrown, ctx=" + ctx, exceptionThrown);
}
else if (ctx.getId() == null)
{
container.getInstanceCache().remove(key);
if (trace) log.trace("Ending invoke, cache removal, ctx=" + ctx);
}
if (trace) log.trace("End invoke, key=" + key + ", ctx=" + ctx);
} }
}